obsidian-dev-utils 52.0.3 → 52.1.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.
@@ -125,15 +125,37 @@ class SuccessTaskResult extends CliTaskResult {
125
125
  }
126
126
  }
127
127
  function toCommandLine(args) {
128
- return args.map((arg) => {
129
- if (/[\s"\n]/.test(arg)) {
130
- let escapedArg = arg;
131
- escapedArg = replaceAll(escapedArg, /"/g, '\\"');
132
- escapedArg = replaceAll(escapedArg, /\n/g, "\\n");
133
- return `"${escapedArg}"`;
134
- }
128
+ return args.map((arg) => argvQuote(arg)).join(" ");
129
+ }
130
+ function argvQuote(arg) {
131
+ if (arg.length > 0 && !/[\s\t\n\v"]/.test(arg)) {
135
132
  return arg;
136
- }).join(" ");
133
+ }
134
+ const BACKSLASH_ESCAPE_FACTOR = 2;
135
+ let result = '"';
136
+ for (let i = 0; i < arg.length; i++) {
137
+ let numBackslashes = 0;
138
+ while (i < arg.length && arg[i] === "\\") {
139
+ i++;
140
+ numBackslashes++;
141
+ }
142
+ if (i === arg.length) {
143
+ result += "\\".repeat(numBackslashes * BACKSLASH_ESCAPE_FACTOR);
144
+ break;
145
+ }
146
+ const ch = arg.charAt(i);
147
+ if (ch === '"') {
148
+ result += `${"\\".repeat(numBackslashes * BACKSLASH_ESCAPE_FACTOR + 1)}"`;
149
+ } else {
150
+ result += "\\".repeat(numBackslashes) + ch;
151
+ }
152
+ }
153
+ result += '"';
154
+ return result;
155
+ }
156
+ const CMD_META_RE = /[()%!^"<>&|]/g;
157
+ function cmdEscapeCommandLine(commandLine) {
158
+ return replaceAll(commandLine, CMD_META_RE, "^$&");
137
159
  }
138
160
  async function wrapCliTask(taskFn) {
139
161
  enableLibraryDebuggers();
@@ -150,7 +172,8 @@ async function wrapResult(taskFn) {
150
172
  }
151
173
  export {
152
174
  CliTaskResult,
175
+ cmdEscapeCommandLine,
153
176
  toCommandLine,
154
177
  wrapCliTask
155
178
  };
156
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/script-utils/cli-utils.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility classes and functions for managing task results, including\n * success, exit codes, and chaining multiple tasks.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport process from 'node:process';\n\nimport type { MaybeReturn } from '../type.ts';\n\nimport { enableLibraryDebuggers } from '../debug.ts';\nimport { printError } from '../error.ts';\nimport { noop } from '../function.ts';\nimport { replaceAll } from '../string.ts';\n\n/**\n * Abstract class representing the result of a task. Includes methods for handling success,\n * exit codes, and chaining tasks.\n */\nexport abstract class CliTaskResult {\n  /**\n   * Chains multiple tasks together, executing them sequentially until one fails.\n   *\n   * @param tasks - An array of task functions that return a {@link CliTaskResult} or `void`.\n   * @returns A {@link Promise} that resolves with the first failed {@link CliTaskResult} or a success result.\n   */\n  public static async chain(tasks: (() => Promisable<MaybeReturn<CliTaskResult>>)[]): Promise<CliTaskResult> {\n    for (const task of tasks) {\n      const result = await wrapResult(task);\n      if (!result.isSuccessful()) {\n        return result;\n      }\n    }\n\n    return CliTaskResult.Success();\n  }\n\n  /**\n   * Creates a {@link CliTaskResult} that does not exit the process.\n   *\n   * @returns A {@link CliTaskResult} that does not exit the process.\n   */\n  public static DoNotExit(): CliTaskResult {\n    return new DoNotExitTaskResult();\n  }\n\n  /**\n   * A failure result of a CLI task.\n   *\n   * @returns The failure result.\n   */\n  public static Failure(): CliTaskResult {\n    return this.Success(false);\n  }\n\n  /**\n   * Creates a {@link CliTaskResult} based on an exit code.\n   *\n   * @param exitCode - The exit code to represent.\n   * @returns A {@link CliTaskResult} representing the exit code.\n   */\n  public static FromExitCode(exitCode: number): CliTaskResult {\n    return new ExitCodeTaskResult(exitCode);\n  }\n\n  /**\n   * Creates a CliTaskResult representing a successful task result.\n   *\n   * @param isSuccess - A boolean indicating whether the task was successful. Default is `true`.\n   * @returns A CliTaskResult object representing a successful task result.\n   */\n  public static Success(isSuccess = true): CliTaskResult {\n    return new SuccessTaskResult(isSuccess);\n  }\n\n  /**\n   * Exits the process based on the task result.\n   */\n  public abstract exit(): void;\n\n  /**\n   * Throws an error if the task was not successful.\n   */\n  public throwOnFailure(): void {\n    if (!this.isSuccessful()) {\n      throw new Error('Task failed');\n    }\n  }\n\n  /**\n   * Determines if the task was successful.\n   *\n   * @returns `true` if the task was successful, otherwise `false`.\n   */\n  protected abstract isSuccessful(): boolean;\n}\n\n/**\n * A task result that does not exit the process.\n */\nclass DoNotExitTaskResult extends CliTaskResult {\n  /**\n   * Does not exit the process.\n   */\n  public override exit(): void {\n    noop();\n  }\n\n  protected override isSuccessful(): boolean {\n    return true;\n  }\n}\n\n/**\n * A task result based on an exit code.\n */\nclass ExitCodeTaskResult extends CliTaskResult {\n  public constructor(private readonly exitCode: number) {\n    super();\n  }\n\n  /**\n   * Exits the process with the specified exit code.\n   */\n  public override exit(): void {\n    process.exit(this.exitCode);\n  }\n\n  protected override isSuccessful(): boolean {\n    return this.exitCode === 0;\n  }\n}\n\n/**\n * A task result based on success or failure.\n */\nclass SuccessTaskResult extends CliTaskResult {\n  public constructor(private readonly _isSuccessful: boolean) {\n    super();\n  }\n\n  /**\n   * Exits the process based on the success of the task.\n   */\n  public override exit(): void {\n    process.exit(this._isSuccessful ? 0 : 1);\n  }\n\n  protected override isSuccessful(): boolean {\n    return this._isSuccessful;\n  }\n}\n\n/**\n * Converts an array of command-line arguments into a single command-line string.\n * Handles escaping of special characters such as spaces, quotes, and newlines.\n *\n * @param args - The array of command-line arguments to convert.\n * @returns A string representing the command-line invocation.\n */\nexport function toCommandLine(args: string[]): string {\n  return args\n    .map((arg) => {\n      if (/[\\s\"\\n]/.test(arg)) {\n        let escapedArg = arg;\n        escapedArg = replaceAll(escapedArg, /\"/g, '\\\\\"');\n        escapedArg = replaceAll(escapedArg, /\\n/g, '\\\\n');\n        return `\"${escapedArg}\"`;\n      }\n      return arg;\n    })\n    .join(' ');\n}\n\n/**\n * Wraps a CLI task function to ensure it runs safely and handles its {@link CliTaskResult}.\n *\n * @param taskFn - The task function to execute, which may return a {@link CliTaskResult} or `void`.\n * @returns A {@link Promise} that resolves when the task is completed and exits with the appropriate status.\n */\nexport async function wrapCliTask(taskFn: () => Promisable<MaybeReturn<CliTaskResult>>): Promise<void> {\n  enableLibraryDebuggers();\n  const result = await wrapResult(taskFn);\n  result.exit();\n}\n\n/**\n * Safely executes a task function and returns a {@link CliTaskResult}. If the task function throws an error,\n * An error is caught, and a failure {@link CliTaskResult} is returned.\n *\n * @param taskFn - The task function to execute.\n * @returns A {@link Promise} that resolves with a {@link CliTaskResult} representing the outcome of the task.\n */\nasync function wrapResult(taskFn: () => Promisable<MaybeReturn<CliTaskResult>>): Promise<CliTaskResult> {\n  try {\n    return (await taskFn()) as CliTaskResult | undefined ?? CliTaskResult.Success();\n  } catch (error) {\n    printError(new Error('An error occurred during task execution', { cause: error }));\n    return CliTaskResult.Failure();\n  }\n}\n"],
  "mappings": ";;;;;;;;;;;;;;;;;;;;;AASA,OAAO,aAAa;AAIpB,SAAS,8BAA8B;AACvC,SAAS,kBAAkB;AAC3B,SAAS,YAAY;AACrB,SAAS,kBAAkB;AAMpB,MAAe,cAAc;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOlC,aAAoB,MAAM,OAAiF;AACzG,eAAW,QAAQ,OAAO;AACxB,YAAM,SAAS,MAAM,WAAW,IAAI;AACpC,UAAI,CAAC,OAAO,aAAa,GAAG;AAC1B,eAAO;AAAA,MACT;AAAA,IACF;AAEA,WAAO,cAAc,QAAQ;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAc,YAA2B;AACvC,WAAO,IAAI,oBAAoB;AAAA,EACjC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,OAAc,UAAyB;AACrC,WAAO,KAAK,QAAQ,KAAK;AAAA,EAC3B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,OAAc,aAAa,UAAiC;AAC1D,WAAO,IAAI,mBAAmB,QAAQ;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,OAAc,QAAQ,YAAY,MAAqB;AACrD,WAAO,IAAI,kBAAkB,SAAS;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAUO,iBAAuB;AAC5B,QAAI,CAAC,KAAK,aAAa,GAAG;AACxB,YAAM,IAAI,MAAM,aAAa;AAAA,IAC/B;AAAA,EACF;AAQF;AAKA,MAAM,4BAA4B,cAAc;AAAA;AAAA;AAAA;AAAA,EAI9B,OAAa;AAC3B,SAAK;AAAA,EACP;AAAA,EAEmB,eAAwB;AACzC,WAAO;AAAA,EACT;AACF;AAKA,MAAM,2BAA2B,cAAc;AAAA,EACtC,YAA6B,UAAkB;AACpD,UAAM;AAD4B;AAAA,EAEpC;AAAA;AAAA;AAAA;AAAA,EAKgB,OAAa;AAC3B,YAAQ,KAAK,KAAK,QAAQ;AAAA,EAC5B;AAAA,EAEmB,eAAwB;AACzC,WAAO,KAAK,aAAa;AAAA,EAC3B;AACF;AAKA,MAAM,0BAA0B,cAAc;AAAA,EACrC,YAA6B,eAAwB;AAC1D,UAAM;AAD4B;AAAA,EAEpC;AAAA;AAAA;AAAA;AAAA,EAKgB,OAAa;AAC3B,YAAQ,KAAK,KAAK,gBAAgB,IAAI,CAAC;AAAA,EACzC;AAAA,EAEmB,eAAwB;AACzC,WAAO,KAAK;AAAA,EACd;AACF;AASO,SAAS,cAAc,MAAwB;AACpD,SAAO,KACJ,IAAI,CAAC,QAAQ;AACZ,QAAI,UAAU,KAAK,GAAG,GAAG;AACvB,UAAI,aAAa;AACjB,mBAAa,WAAW,YAAY,MAAM,KAAK;AAC/C,mBAAa,WAAW,YAAY,OAAO,KAAK;AAChD,aAAO,IAAI,UAAU;AAAA,IACvB;AACA,WAAO;AAAA,EACT,CAAC,EACA,KAAK,GAAG;AACb;AAQA,eAAsB,YAAY,QAAqE;AACrG,yBAAuB;AACvB,QAAM,SAAS,MAAM,WAAW,MAAM;AACtC,SAAO,KAAK;AACd;AASA,eAAe,WAAW,QAA8E;AACtG,MAAI;AACF,WAAQ,MAAM,OAAO,KAAmC,cAAc,QAAQ;AAAA,EAChF,SAAS,OAAO;AACd,eAAW,IAAI,MAAM,2CAA2C,EAAE,OAAO,MAAM,CAAC,CAAC;AACjF,WAAO,cAAc,QAAQ;AAAA,EAC/B;AACF;",
  "names": []
}

179
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/script-utils/cli-utils.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility classes and functions for managing task results, including\n * success, exit codes, and chaining multiple tasks.\n */\n\nimport type { Promisable } from 'type-fest';\n\nimport process from 'node:process';\n\nimport type { MaybeReturn } from '../type.ts';\n\nimport { enableLibraryDebuggers } from '../debug.ts';\nimport { printError } from '../error.ts';\nimport { noop } from '../function.ts';\nimport { replaceAll } from '../string.ts';\n\n/**\n * Abstract class representing the result of a task. Includes methods for handling success,\n * exit codes, and chaining tasks.\n */\nexport abstract class CliTaskResult {\n  /**\n   * Chains multiple tasks together, executing them sequentially until one fails.\n   *\n   * @param tasks - An array of task functions that return a {@link CliTaskResult} or `void`.\n   * @returns A {@link Promise} that resolves with the first failed {@link CliTaskResult} or a success result.\n   */\n  public static async chain(tasks: (() => Promisable<MaybeReturn<CliTaskResult>>)[]): Promise<CliTaskResult> {\n    for (const task of tasks) {\n      const result = await wrapResult(task);\n      if (!result.isSuccessful()) {\n        return result;\n      }\n    }\n\n    return CliTaskResult.Success();\n  }\n\n  /**\n   * Creates a {@link CliTaskResult} that does not exit the process.\n   *\n   * @returns A {@link CliTaskResult} that does not exit the process.\n   */\n  public static DoNotExit(): CliTaskResult {\n    return new DoNotExitTaskResult();\n  }\n\n  /**\n   * A failure result of a CLI task.\n   *\n   * @returns The failure result.\n   */\n  public static Failure(): CliTaskResult {\n    return this.Success(false);\n  }\n\n  /**\n   * Creates a {@link CliTaskResult} based on an exit code.\n   *\n   * @param exitCode - The exit code to represent.\n   * @returns A {@link CliTaskResult} representing the exit code.\n   */\n  public static FromExitCode(exitCode: number): CliTaskResult {\n    return new ExitCodeTaskResult(exitCode);\n  }\n\n  /**\n   * Creates a CliTaskResult representing a successful task result.\n   *\n   * @param isSuccess - A boolean indicating whether the task was successful. Default is `true`.\n   * @returns A CliTaskResult object representing a successful task result.\n   */\n  public static Success(isSuccess = true): CliTaskResult {\n    return new SuccessTaskResult(isSuccess);\n  }\n\n  /**\n   * Exits the process based on the task result.\n   */\n  public abstract exit(): void;\n\n  /**\n   * Throws an error if the task was not successful.\n   */\n  public throwOnFailure(): void {\n    if (!this.isSuccessful()) {\n      throw new Error('Task failed');\n    }\n  }\n\n  /**\n   * Determines if the task was successful.\n   *\n   * @returns `true` if the task was successful, otherwise `false`.\n   */\n  protected abstract isSuccessful(): boolean;\n}\n\n/**\n * A task result that does not exit the process.\n */\nclass DoNotExitTaskResult extends CliTaskResult {\n  /**\n   * Does not exit the process.\n   */\n  public override exit(): void {\n    noop();\n  }\n\n  protected override isSuccessful(): boolean {\n    return true;\n  }\n}\n\n/**\n * A task result based on an exit code.\n */\nclass ExitCodeTaskResult extends CliTaskResult {\n  public constructor(private readonly exitCode: number) {\n    super();\n  }\n\n  /**\n   * Exits the process with the specified exit code.\n   */\n  public override exit(): void {\n    process.exit(this.exitCode);\n  }\n\n  protected override isSuccessful(): boolean {\n    return this.exitCode === 0;\n  }\n}\n\n/**\n * A task result based on success or failure.\n */\nclass SuccessTaskResult extends CliTaskResult {\n  public constructor(private readonly _isSuccessful: boolean) {\n    super();\n  }\n\n  /**\n   * Exits the process based on the success of the task.\n   */\n  public override exit(): void {\n    process.exit(this._isSuccessful ? 0 : 1);\n  }\n\n  protected override isSuccessful(): boolean {\n    return this._isSuccessful;\n  }\n}\n\n/**\n * Converts an array of command-line arguments into a single command-line string\n * using the `CommandLineToArgvW` convention (the standard used by the\n * Microsoft C runtime and most Windows programs).\n *\n * Implements the ArgvQuote algorithm from\n * {@link https://learn.microsoft.com/archive/blogs/twistylittlepassagesallalike/everyone-quotes-command-line-arguments-the-wrong-way | Everyone quotes command line arguments the wrong way}:\n * backslashes before quotes and at the end of a quoted argument are doubled.\n *\n * This produces a shell-agnostic command line. Callers that route through\n * a specific shell (cmd.exe, PowerShell, sh) must apply shell-specific\n * escaping on top \u2014 see {@link cmdEscapeCommandLine}.\n *\n * @param args - The array of command-line arguments to convert.\n * @returns A string representing the command-line invocation.\n */\nexport function toCommandLine(args: string[]): string {\n  return args.map((arg) => argvQuote(arg)).join(' ');\n}\n\n/**\n * Quotes a single argument so that `CommandLineToArgvW` will decode it\n * unchanged. Implements the ArgvQuote algorithm from\n * {@link https://learn.microsoft.com/archive/blogs/twistylittlepassagesallalike/everyone-quotes-command-line-arguments-the-wrong-way | Everyone quotes command line arguments the wrong way}.\n *\n * @param arg - The raw argument string.\n * @returns The quoted argument string.\n */\nfunction argvQuote(arg: string): string {\n  if (arg.length > 0 && !/[\\s\\t\\n\\v\"]/.test(arg)) {\n    return arg;\n  }\n\n  const BACKSLASH_ESCAPE_FACTOR = 2;\n  let result = '\"';\n  for (let i = 0; i < arg.length; i++) {\n    let numBackslashes = 0;\n    while (i < arg.length && arg[i] === '\\\\') {\n      i++;\n      numBackslashes++;\n    }\n\n    if (i === arg.length) {\n      result += '\\\\'.repeat(numBackslashes * BACKSLASH_ESCAPE_FACTOR);\n      break;\n    }\n\n    const ch = arg.charAt(i);\n    if (ch === '\"') {\n      result += `${'\\\\'.repeat(numBackslashes * BACKSLASH_ESCAPE_FACTOR + 1)}\"`;\n    } else {\n      result += '\\\\'.repeat(numBackslashes) + ch;\n    }\n  }\n\n  result += '\"';\n  return result;\n}\n\n/**\n * Matches `cmd.exe` metacharacters that must be `^`-escaped.\n */\nconst CMD_META_RE = /[()%!^\"<>&|]/g;\n\n/**\n * Escapes `cmd.exe` metacharacters with `^` so that `cmd.exe` passes them\n * through literally. This is necessary because `cmd.exe`'s `\"` handling\n * differs from `CommandLineToArgvW` and cannot be relied upon.\n *\n * Apply this to a command line string that will be executed via `cmd.exe`\n * (e.g., `spawn(cmd, [], { shell: true })` on Windows).\n *\n * @param commandLine - The already-quoted command line string.\n * @returns The string with all cmd metacharacters `^`-escaped.\n */\nexport function cmdEscapeCommandLine(commandLine: string): string {\n  return replaceAll(commandLine, CMD_META_RE, '^$&');\n}\n\n/**\n * Wraps a CLI task function to ensure it runs safely and handles its {@link CliTaskResult}.\n *\n * @param taskFn - The task function to execute, which may return a {@link CliTaskResult} or `void`.\n * @returns A {@link Promise} that resolves when the task is completed and exits with the appropriate status.\n */\nexport async function wrapCliTask(taskFn: () => Promisable<MaybeReturn<CliTaskResult>>): Promise<void> {\n  enableLibraryDebuggers();\n  const result = await wrapResult(taskFn);\n  result.exit();\n}\n\n/**\n * Safely executes a task function and returns a {@link CliTaskResult}. If the task function throws an error,\n * An error is caught, and a failure {@link CliTaskResult} is returned.\n *\n * @param taskFn - The task function to execute.\n * @returns A {@link Promise} that resolves with a {@link CliTaskResult} representing the outcome of the task.\n */\nasync function wrapResult(taskFn: () => Promisable<MaybeReturn<CliTaskResult>>): Promise<CliTaskResult> {\n  try {\n    return (await taskFn()) as CliTaskResult | undefined ?? CliTaskResult.Success();\n  } catch (error) {\n    printError(new Error('An error occurred during task execution', { cause: error }));\n    return CliTaskResult.Failure();\n  }\n}\n"],
  "mappings": "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",
  "names": []
}

@@ -24,14 +24,27 @@ import process from "node:process";
24
24
  import { getLibDebugger } from "../debug.mjs";
25
25
  import { trimEnd } from "../string.mjs";
26
26
  import { assertNonNullable } from "../type-guards.mjs";
27
- import { toCommandLine } from "./cli-utils.mjs";
27
+ import {
28
+ cmdEscapeCommandLine,
29
+ toCommandLine
30
+ } from "./cli-utils.mjs";
28
31
  function exec(command, options = {}) {
29
32
  if (Array.isArray(command)) {
30
33
  const batchResult = handleBatchedCommand(command, options);
31
34
  if (batchResult) {
32
35
  return batchResult;
33
36
  }
34
- command = toCommandLine(command.filter((part) => typeof part === "string"));
37
+ const args = command.filter((part) => typeof part === "string");
38
+ const commandLine = toCommandLine(args);
39
+ const maxCommandLength2 = getMaxCommandLength();
40
+ if (commandLine.length > maxCommandLength2) {
41
+ return Promise.reject(
42
+ new Error(
43
+ `Command line is too long (${String(commandLine.length)} chars, max ${String(maxCommandLength2)} on ${process.platform}). Consider splitting into smaller batches or use ExecArg.`
44
+ )
45
+ );
46
+ }
47
+ return execString(commandLine, options, args);
35
48
  }
36
49
  const maxCommandLength = getMaxCommandLength();
37
50
  if (command.length > maxCommandLength) {
@@ -43,7 +56,7 @@ function exec(command, options = {}) {
43
56
  }
44
57
  return execString(command, options);
45
58
  }
46
- function execString(command, options = {}) {
59
+ function execString(command, options = {}, rawArgs) {
47
60
  const {
48
61
  cwd = process.cwd(),
49
62
  isQuiet: quiet = false,
@@ -53,15 +66,7 @@ function execString(command, options = {}) {
53
66
  } = options;
54
67
  return new Promise((resolve, reject) => {
55
68
  getLibDebugger("Exec")(`Executing command: ${command}`);
56
- const child = spawn(command, [], {
57
- cwd,
58
- env: {
59
- DEBUG_COLORS: "1",
60
- ...process.env
61
- },
62
- shell: true,
63
- stdio: "pipe"
64
- });
69
+ const child = spawnViaShell(command, cwd, rawArgs);
65
70
  let stdout = "";
66
71
  let stderr = "";
67
72
  child.stdin.write(stdin);
@@ -119,6 +124,10 @@ ${stderr}`));
119
124
  });
120
125
  });
121
126
  }
127
+ const CHILD_ENV = {
128
+ DEBUG_COLORS: "1",
129
+ ...process.env
130
+ };
122
131
  async function executeBatches(baseCommand, batches, options) {
123
132
  const results = [];
124
133
  for (const batch of batches) {
@@ -181,7 +190,28 @@ function handleBatchedCommand(parts, options) {
181
190
  function isExecArg(part) {
182
191
  return typeof part === "object" && "batchedArgs" in part;
183
192
  }
193
+ function spawnViaShell(command, cwd, rawArgs) {
194
+ if (process.platform === "win32" && command.includes("\n")) {
195
+ if (!rawArgs) {
196
+ throw new Error("Commands containing newlines cannot be executed through cmd.exe on Windows. Pass an argument array instead of a string.");
197
+ }
198
+ const [program, ...args] = rawArgs;
199
+ assertNonNullable(program, "Command array must not be empty");
200
+ return spawn(program, args, {
201
+ cwd,
202
+ env: CHILD_ENV,
203
+ stdio: "pipe"
204
+ });
205
+ }
206
+ const shellCommand = process.platform === "win32" ? cmdEscapeCommandLine(command) : command;
207
+ return spawn(shellCommand, [], {
208
+ cwd,
209
+ env: CHILD_ENV,
210
+ shell: true,
211
+ stdio: "pipe"
212
+ });
213
+ }
184
214
  export {
185
215
  exec
186
216
  };
187
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/script-utils/exec.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for executing commands.\n */\n\nimport { spawn } from 'node:child_process';\nimport process from 'node:process';\n\nimport { getLibDebugger } from '../debug.ts';\nimport { trimEnd } from '../string.ts';\nimport { assertNonNullable } from '../type-guards.ts';\nimport { toCommandLine } from './cli-utils.ts';\n\n/**\n * A command part: either a plain string or an {@link ExecArg} with batched arguments.\n */\nexport type CommandPart = ExecArg | string;\n\n/**\n * A command argument that contains a list of args to be batched.\n * If the expanded command exceeds the platform's max command length,\n * the batched args are split into sequential executions.\n */\nexport interface ExecArg {\n  /**\n   * The arguments to batch.\n   */\n  batchedArgs: string[];\n}\n\n/**\n * Options for {@link exec} that return detailed results.\n */\nexport interface ExecDetailedOptions extends ExecOption {\n  /**\n   * Must be `true` to receive detailed results.\n   */\n  withDetails: true;\n}\n\n/**\n * Options for executing a command.\n */\nexport interface ExecOption {\n  /**\n   * A current working folder for the command execution.\n   */\n  readonly cwd?: string;\n\n  /**\n   * If `true`, suppresses the output of the command.\n   */\n  readonly isQuiet?: boolean;\n\n  /**\n   * If `true`, throws an error if the command fails.\n   */\n  readonly shouldFailIfCalledFromOutsideRoot?: boolean;\n\n  /**\n   * If `true`, ignores the exit code of the command.\n   */\n  readonly shouldIgnoreExitCode?: boolean;\n\n  /**\n   * If `false`, only returns the output of the command.\n   */\n  readonly shouldIncludeDetails?: boolean;\n\n  /**\n   * An input to be passed to the command.\n   */\n  readonly stdin?: string;\n}\n\n/**\n * A result of {@link exec}.\n */\nexport interface ExecResult {\n  /**\n   * An exit code of the command. A value of `null` indicates that the process did not exit normally.\n   */\n  exitCode: null | number;\n\n  /**\n   * A signal that caused the process to be terminated. A value of `null` indicates that no signal was received.\n   */\n  exitSignal: NodeJS.Signals | null;\n\n  /**\n   * A standard error output from the command.\n   */\n  stderr: string;\n\n  /**\n   * A standard output from the command.\n   */\n  stdout: string;\n}\n\n/**\n * Options for {@link exec} that return only stdout.\n */\nexport interface ExecSimpleOptions extends ExecOption {\n  /**\n   * Must be `false` or omitted to receive only stdout.\n   */\n  withDetails?: false;\n}\n\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with the output of the command.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport async function exec(command: CommandPart[] | string, options?: ExecSimpleOptions): Promise<string>;\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with ExecResult object.\n *          The ExecResult object contains the exit code, exit signal, stderr, and stdout.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport function exec(command: CommandPart[] | string, options: ExecDetailedOptions): Promise<ExecResult>;\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with the output of the command or an ExecResult object.\n *          The ExecResult object contains the exit code, exit signal, stderr, and stdout.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport function exec(command: CommandPart[] | string, options: ExecOption = {}): Promise<ExecResult | string> {\n  if (Array.isArray(command)) {\n    const batchResult = handleBatchedCommand(command, options);\n    if (batchResult) {\n      return batchResult;\n    }\n    command = toCommandLine(command.filter((part): part is string => typeof part === 'string'));\n  }\n\n  const maxCommandLength = getMaxCommandLength();\n  if (command.length > maxCommandLength) {\n    return Promise.reject(\n      new Error(\n        `Command line is too long (${String(command.length)} chars, max ${\n          String(maxCommandLength)\n        } on ${process.platform}). Consider splitting into smaller batches or use ExecArg.`\n      )\n    );\n  }\n\n  return execString(command, options);\n}\n\n/**\n * Executes a single string command.\n *\n * @param command - The command string.\n * @param options - The exec options.\n * @returns A Promise resolving to the result.\n */\nfunction execString(command: string, options: ExecOption = {}): Promise<ExecResult | string> {\n  const {\n    cwd = process.cwd(),\n    isQuiet: quiet = false,\n    shouldIgnoreExitCode: ignoreExitCode = false,\n    shouldIncludeDetails: withDetails = false,\n    stdin = ''\n  } = options;\n\n  return new Promise((resolve, reject) => {\n    getLibDebugger('Exec')(`Executing command: ${command}`);\n\n    const child = spawn(command, [], {\n      cwd,\n      env: {\n        DEBUG_COLORS: '1',\n        ...process.env\n      },\n      shell: true,\n      stdio: 'pipe'\n    });\n\n    let stdout = '';\n    let stderr = '';\n\n    child.stdin.write(stdin);\n    child.stdin.end();\n\n    child.stdout.on('data', (data: Buffer) => {\n      if (!quiet) {\n        process.stdout.write(data);\n      }\n      stdout += data.toString('utf-8');\n    });\n\n    child.stdout.on('end', () => {\n      stdout = trimEnd(stdout, '\\n');\n    });\n\n    child.stderr.on('data', (data: Buffer) => {\n      if (!quiet) {\n        process.stderr.write(data);\n      }\n      stderr += data.toString('utf-8');\n    });\n\n    child.stderr.on('end', () => {\n      stderr = trimEnd(stderr, '\\n');\n    });\n\n    child.on('close', (exitCode, exitSignal) => {\n      if (exitCode !== 0 && !ignoreExitCode) {\n        reject(new Error(`Command failed with exit code ${exitCode ? String(exitCode) : '(null)'}\\n${stderr}`));\n        return;\n      }\n\n      if (!withDetails) {\n        resolve(stdout);\n        return;\n      }\n      resolve({\n        exitCode,\n        exitSignal,\n        stderr,\n        stdout\n      });\n    });\n\n    child.on('error', (err) => {\n      if (!ignoreExitCode) {\n        reject(err);\n        return;\n      }\n\n      if (!withDetails) {\n        resolve(stdout);\n        return;\n      }\n\n      resolve({\n        exitCode: null,\n        exitSignal: null,\n        stderr,\n        stdout\n      });\n    });\n  });\n}\n\n/**\n * Executes batched commands sequentially and concatenates stdout.\n *\n * @param baseCommand - The base command without batched args.\n * @param batches - The batches of args.\n * @param options - The exec options.\n * @returns A Promise resolving to the concatenated result.\n */\nasync function executeBatches(baseCommand: string, batches: string[][], options: ExecOption): Promise<ExecResult | string> {\n  const results: string[] = [];\n\n  for (const batch of batches) {\n    const batchCommand = `${baseCommand} ${batch.join(' ')}`;\n    const result = await execString(batchCommand, options);\n    if (typeof result === 'string') {\n      results.push(result);\n    }\n  }\n\n  if (options.shouldIncludeDetails) {\n    return { exitCode: 0, exitSignal: null, stderr: '', stdout: results.join('\\n') };\n  }\n\n  return results.join('\\n');\n}\n\n/**\n * Returns the platform-specific max command line length.\n *\n * @returns The max command length in characters.\n */\nfunction getMaxCommandLength(): number {\n  const WINDOWS_MAX_COMMAND_LENGTH = 8191;\n  const UNIX_MAX_COMMAND_LENGTH = 131072;\n  return process.platform === 'win32' ? WINDOWS_MAX_COMMAND_LENGTH : UNIX_MAX_COMMAND_LENGTH;\n}\n\n/**\n * Handles a command array that may contain an {@link ExecArg}.\n * Returns a Promise if batching is needed, or `undefined` if the command\n * has no ExecArg and should be processed normally.\n *\n * @param parts - The command parts.\n * @param options - The exec options.\n * @returns A Promise if batching is handled, or `undefined`.\n */\nfunction handleBatchedCommand(parts: CommandPart[], options: ExecOption): Promise<ExecResult | string> | undefined {\n  const execArgs = parts.filter(isExecArg);\n  if (execArgs.length === 0) {\n    return undefined;\n  }\n  if (execArgs.length > 1) {\n    return Promise.reject(new Error('Only one ExecArg with batchedArgs is allowed per command'));\n  }\n\n  const execArg = execArgs[0];\n  assertNonNullable(execArg);\n  const staticParts = parts.filter((part): part is string => typeof part === 'string');\n  const baseCommand = toCommandLine(staticParts);\n  const maxCommandLength = getMaxCommandLength();\n\n  // Try expanding all args inline\n  const fullCommand = `${baseCommand} ${execArg.batchedArgs.join(' ')}`;\n  if (fullCommand.length <= maxCommandLength) {\n    return execString(fullCommand, options);\n  }\n\n  // Split into batches\n  const batches: string[][] = [];\n  let currentBatch: string[] = [];\n\n  for (const arg of execArg.batchedArgs) {\n    const tentative = `${baseCommand} ${[...currentBatch, arg].join(' ')}`;\n    if (tentative.length > maxCommandLength) {\n      if (currentBatch.length === 0) {\n        return Promise.reject(\n          new Error(\n            `Cannot split command into batches: a single argument (${String(arg.length)} chars) plus the base command (${\n              String(baseCommand.length)\n            } chars) exceeds the max command length (${String(maxCommandLength)}).`\n          )\n        );\n      }\n      batches.push(currentBatch);\n      currentBatch = [arg];\n    } else {\n      currentBatch.push(arg);\n    }\n  }\n  /* v8 ignore start -- Always true after the loop; batchedArgs is non-empty at this point. */\n  if (currentBatch.length > 0) {\n    /* v8 ignore stop */\n    batches.push(currentBatch);\n  }\n\n  return executeBatches(baseCommand, batches, options);\n}\n\n/**\n * Checks if a command part is an {@link ExecArg}.\n *\n * @param part - The command part to check.\n * @returns Whether the part is an ExecArg.\n */\nfunction isExecArg(part: CommandPart): part is ExecArg {\n  return typeof part === 'object' && 'batchedArgs' in part;\n}\n"],
  "mappings": "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",
  "names": []
}

217
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../src/script-utils/exec.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * Contains utility functions for executing commands.\n */\n\nimport type { ChildProcessWithoutNullStreams } from 'node:child_process';\n\nimport { spawn } from 'node:child_process';\nimport process from 'node:process';\n\nimport { getLibDebugger } from '../debug.ts';\nimport { trimEnd } from '../string.ts';\nimport { assertNonNullable } from '../type-guards.ts';\nimport {\n  cmdEscapeCommandLine,\n  toCommandLine\n} from './cli-utils.ts';\n\n/**\n * A command part: either a plain string or an {@link ExecArg} with batched arguments.\n */\nexport type CommandPart = ExecArg | string;\n\n/**\n * A command argument that contains a list of args to be batched.\n * If the expanded command exceeds the platform's max command length,\n * the batched args are split into sequential executions.\n */\nexport interface ExecArg {\n  /**\n   * The arguments to batch.\n   */\n  batchedArgs: string[];\n}\n\n/**\n * Options for {@link exec} that return detailed results.\n */\nexport interface ExecDetailedOptions extends ExecOption {\n  /**\n   * Must be `true` to receive detailed results.\n   */\n  withDetails: true;\n}\n\n/**\n * Options for executing a command.\n */\nexport interface ExecOption {\n  /**\n   * A current working folder for the command execution.\n   */\n  readonly cwd?: string;\n\n  /**\n   * If `true`, suppresses the output of the command.\n   */\n  readonly isQuiet?: boolean;\n\n  /**\n   * If `true`, throws an error if the command fails.\n   */\n  readonly shouldFailIfCalledFromOutsideRoot?: boolean;\n\n  /**\n   * If `true`, ignores the exit code of the command.\n   */\n  readonly shouldIgnoreExitCode?: boolean;\n\n  /**\n   * If `false`, only returns the output of the command.\n   */\n  readonly shouldIncludeDetails?: boolean;\n\n  /**\n   * An input to be passed to the command.\n   */\n  readonly stdin?: string;\n}\n\n/**\n * A result of {@link exec}.\n */\nexport interface ExecResult {\n  /**\n   * An exit code of the command. A value of `null` indicates that the process did not exit normally.\n   */\n  exitCode: null | number;\n\n  /**\n   * A signal that caused the process to be terminated. A value of `null` indicates that no signal was received.\n   */\n  exitSignal: NodeJS.Signals | null;\n\n  /**\n   * A standard error output from the command.\n   */\n  stderr: string;\n\n  /**\n   * A standard output from the command.\n   */\n  stdout: string;\n}\n\n/**\n * Options for {@link exec} that return only stdout.\n */\nexport interface ExecSimpleOptions extends ExecOption {\n  /**\n   * Must be `false` or omitted to receive only stdout.\n   */\n  withDetails?: false;\n}\n\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with the output of the command.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport async function exec(command: CommandPart[] | string, options?: ExecSimpleOptions): Promise<string>;\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with ExecResult object.\n *          The ExecResult object contains the exit code, exit signal, stderr, and stdout.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport function exec(command: CommandPart[] | string, options: ExecDetailedOptions): Promise<ExecResult>;\n/**\n * Executes a command.\n *\n * @param command - The command to execute. It can be a string or an array of strings.\n * @param options - The options for the execution.\n * @returns A {@link Promise} that resolves with the output of the command or an ExecResult object.\n *          The ExecResult object contains the exit code, exit signal, stderr, and stdout.\n * @throws If the command fails with a non-zero exit code and ignoreExitCode is `false`.\n *         The error message includes the exit code and stderr.\n *         If an error occurs during the execution and ignoreExitCode is `true`,\n *         the error is resolved with the stdout and stderr.\n */\nexport function exec(command: CommandPart[] | string, options: ExecOption = {}): Promise<ExecResult | string> {\n  if (Array.isArray(command)) {\n    const batchResult = handleBatchedCommand(command, options);\n    if (batchResult) {\n      return batchResult;\n    }\n    const args = command.filter((part): part is string => typeof part === 'string');\n    const commandLine = toCommandLine(args);\n\n    const maxCommandLength = getMaxCommandLength();\n    if (commandLine.length > maxCommandLength) {\n      return Promise.reject(\n        new Error(\n          `Command line is too long (${String(commandLine.length)} chars, max ${\n            String(maxCommandLength)\n          } on ${process.platform}). Consider splitting into smaller batches or use ExecArg.`\n        )\n      );\n    }\n\n    return execString(commandLine, options, args);\n  }\n\n  const maxCommandLength = getMaxCommandLength();\n  if (command.length > maxCommandLength) {\n    return Promise.reject(\n      new Error(\n        `Command line is too long (${String(command.length)} chars, max ${\n          String(maxCommandLength)\n        } on ${process.platform}). Consider splitting into smaller batches or use ExecArg.`\n      )\n    );\n  }\n\n  return execString(command, options);\n}\n\n/**\n * Executes a single string command.\n *\n * @param command - The command string.\n * @param options - The exec options.\n * @param rawArgs - The original argument array (if available), used by the PowerShell\n *   fallback path to quote arguments with PowerShell-native single quotes.\n * @returns A Promise resolving to the result.\n */\nfunction execString(command: string, options: ExecOption = {}, rawArgs?: string[]): Promise<ExecResult | string> {\n  const {\n    cwd = process.cwd(),\n    isQuiet: quiet = false,\n    shouldIgnoreExitCode: ignoreExitCode = false,\n    shouldIncludeDetails: withDetails = false,\n    stdin = ''\n  } = options;\n\n  return new Promise((resolve, reject) => {\n    getLibDebugger('Exec')(`Executing command: ${command}`);\n\n    const child = spawnViaShell(command, cwd, rawArgs);\n\n    let stdout = '';\n    let stderr = '';\n\n    child.stdin.write(stdin);\n    child.stdin.end();\n\n    child.stdout.on('data', (data: Buffer) => {\n      if (!quiet) {\n        process.stdout.write(data);\n      }\n      stdout += data.toString('utf-8');\n    });\n\n    child.stdout.on('end', () => {\n      stdout = trimEnd(stdout, '\\n');\n    });\n\n    child.stderr.on('data', (data: Buffer) => {\n      if (!quiet) {\n        process.stderr.write(data);\n      }\n      stderr += data.toString('utf-8');\n    });\n\n    child.stderr.on('end', () => {\n      stderr = trimEnd(stderr, '\\n');\n    });\n\n    child.on('close', (exitCode, exitSignal) => {\n      if (exitCode !== 0 && !ignoreExitCode) {\n        reject(new Error(`Command failed with exit code ${exitCode ? String(exitCode) : '(null)'}\\n${stderr}`));\n        return;\n      }\n\n      if (!withDetails) {\n        resolve(stdout);\n        return;\n      }\n      resolve({\n        exitCode,\n        exitSignal,\n        stderr,\n        stdout\n      });\n    });\n\n    child.on('error', (err) => {\n      if (!ignoreExitCode) {\n        reject(err);\n        return;\n      }\n\n      if (!withDetails) {\n        resolve(stdout);\n        return;\n      }\n\n      resolve({\n        exitCode: null,\n        exitSignal: null,\n        stderr,\n        stdout\n      });\n    });\n  });\n}\n\n/**\n * Default environment variables passed to child processes.\n */\nconst CHILD_ENV = {\n  DEBUG_COLORS: '1',\n  ...process.env\n};\n\n/**\n * Executes batched commands sequentially and concatenates stdout.\n *\n * @param baseCommand - The base command without batched args.\n * @param batches - The batches of args.\n * @param options - The exec options.\n * @returns A Promise resolving to the concatenated result.\n */\nasync function executeBatches(baseCommand: string, batches: string[][], options: ExecOption): Promise<ExecResult | string> {\n  const results: string[] = [];\n\n  for (const batch of batches) {\n    const batchCommand = `${baseCommand} ${batch.join(' ')}`;\n    const result = await execString(batchCommand, options);\n    if (typeof result === 'string') {\n      results.push(result);\n    }\n  }\n\n  if (options.shouldIncludeDetails) {\n    return { exitCode: 0, exitSignal: null, stderr: '', stdout: results.join('\\n') };\n  }\n\n  return results.join('\\n');\n}\n\n/**\n * Returns the platform-specific max command line length.\n *\n * @returns The max command length in characters.\n */\nfunction getMaxCommandLength(): number {\n  const WINDOWS_MAX_COMMAND_LENGTH = 8191;\n  const UNIX_MAX_COMMAND_LENGTH = 131072;\n  return process.platform === 'win32' ? WINDOWS_MAX_COMMAND_LENGTH : UNIX_MAX_COMMAND_LENGTH;\n}\n\n/**\n * Handles a command array that may contain an {@link ExecArg}.\n * Returns a Promise if batching is needed, or `undefined` if the command\n * has no ExecArg and should be processed normally.\n *\n * @param parts - The command parts.\n * @param options - The exec options.\n * @returns A Promise if batching is handled, or `undefined`.\n */\nfunction handleBatchedCommand(parts: CommandPart[], options: ExecOption): Promise<ExecResult | string> | undefined {\n  const execArgs = parts.filter(isExecArg);\n  if (execArgs.length === 0) {\n    return undefined;\n  }\n  if (execArgs.length > 1) {\n    return Promise.reject(new Error('Only one ExecArg with batchedArgs is allowed per command'));\n  }\n\n  const execArg = execArgs[0];\n  assertNonNullable(execArg);\n  const staticParts = parts.filter((part): part is string => typeof part === 'string');\n  const baseCommand = toCommandLine(staticParts);\n  const maxCommandLength = getMaxCommandLength();\n\n  // Try expanding all args inline\n  const fullCommand = `${baseCommand} ${execArg.batchedArgs.join(' ')}`;\n  if (fullCommand.length <= maxCommandLength) {\n    return execString(fullCommand, options);\n  }\n\n  // Split into batches\n  const batches: string[][] = [];\n  let currentBatch: string[] = [];\n\n  for (const arg of execArg.batchedArgs) {\n    const tentative = `${baseCommand} ${[...currentBatch, arg].join(' ')}`;\n    if (tentative.length > maxCommandLength) {\n      if (currentBatch.length === 0) {\n        return Promise.reject(\n          new Error(\n            `Cannot split command into batches: a single argument (${String(arg.length)} chars) plus the base command (${\n              String(baseCommand.length)\n            } chars) exceeds the max command length (${String(maxCommandLength)}).`\n          )\n        );\n      }\n      batches.push(currentBatch);\n      currentBatch = [arg];\n    } else {\n      currentBatch.push(arg);\n    }\n  }\n  /* v8 ignore start -- Always true after the loop; batchedArgs is non-empty at this point. */\n  if (currentBatch.length > 0) {\n    /* v8 ignore stop */\n    batches.push(currentBatch);\n  }\n\n  return executeBatches(baseCommand, batches, options);\n}\n\n/**\n * Checks if a command part is an {@link ExecArg}.\n *\n * @param part - The command part to check.\n * @returns Whether the part is an ExecArg.\n */\nfunction isExecArg(part: CommandPart): part is ExecArg {\n  return typeof part === 'object' && 'batchedArgs' in part;\n}\n\n/**\n * Spawns a child process via the appropriate shell.\n *\n * On Windows, if the command contains newlines (which `cmd.exe` cannot handle)\n * and the raw args array is available, spawns the process directly without\n * any shell \u2014 passing args via `CreateProcess`, which avoids all quoting issues.\n *\n * On Windows (cmd.exe path), applies `^`-escaping for cmd metacharacters.\n *\n * @param command - The command string to execute.\n * @param cwd - The working directory.\n * @param rawArgs - The original argument array (if available).\n * @returns The spawned child process.\n */\nfunction spawnViaShell(command: string, cwd: string, rawArgs?: string[]): ChildProcessWithoutNullStreams {\n  if (process.platform === 'win32' && command.includes('\\n')) {\n    if (!rawArgs) {\n      throw new Error('Commands containing newlines cannot be executed through cmd.exe on Windows. Pass an argument array instead of a string.');\n    }\n    const [program, ...args] = rawArgs;\n    assertNonNullable(program, 'Command array must not be empty');\n    return spawn(program, args, {\n      cwd,\n      env: CHILD_ENV,\n      stdio: 'pipe'\n    });\n  }\n\n  const shellCommand = process.platform === 'win32' ? cmdEscapeCommandLine(command) : command;\n  return spawn(shellCommand, [], {\n    cwd,\n    env: CHILD_ENV,\n    shell: true,\n    stdio: 'pipe'\n  });\n}\n"],
  "mappings": "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",
  "names": ["maxCommandLength"]
}

@@ -437,6 +437,7 @@ function getStylisticConfigs(context) {
437
437
  ],
438
438
  files: context.allFiles(),
439
439
  rules: {
440
+ "@stylistic/generator-star-spacing": "off",
440
441
  "@stylistic/indent": "off",
441
442
  "@stylistic/indent-binary-ops": "off",
442
443
  "@stylistic/jsx-one-expression-per-line": "off",
@@ -533,4 +534,4 @@ export {
533
534
  EslintConfigContext,
534
535
  defineEslintConfigs
535
536
  };
536
- //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/script-utils/linters/eslint-config.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * ESLint configuration for TypeScript projects with various plugins.\n *\n * This module exports ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n * such as `@typescript-eslint/parser`, `@typescript-eslint/eslint-plugin`,\n * `eslint-plugin-modules-newlines`, `@stylistic/eslint-plugin`.\n * It sets up parsers, plugins, and rules for maintaining code quality and consistency.\n */\n\n/* v8 ignore start -- Declarative ESLint rule/plugin configuration; correctness is verified by running ESLint, not unit tests. */\n\nimport type {\n  ESLint,\n  Linter\n} from 'eslint';\n\n/* eslint-disable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\nimport commentsConfigs from '@eslint-community/eslint-plugin-eslint-comments/configs';\nimport { includeIgnoreFile } from '@eslint/compat';\nimport eslint from '@eslint/js';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport stylistic from '@stylistic/eslint-plugin';\nimport { createTypeScriptImportResolver } from 'eslint-import-resolver-typescript';\nimport { flatConfigs as eslintPluginImportXFlatConfigs } from 'eslint-plugin-import-x';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport obsidianmd from 'eslint-plugin-obsidianmd';\nimport { configs as perfectionistConfigs } from 'eslint-plugin-perfectionist';\nimport { defineConfig } from 'eslint/config';\nimport globals from 'globals';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `_default` is too confusing.\nimport tseslint from 'typescript-eslint';\n\nimport { ObsidianPluginRepoPaths } from '../../obsidian/plugin/obsidian-plugin-repo-paths.ts';\nimport { join } from '../../path.ts';\nimport { getRootFolder } from '../root.ts';\nimport { obsidianDevUtilsPlugin } from './eslint-rules/obsidian-dev-utils-plugin.ts';\n\n/**\n * The parameters for defining ESLint configurations.\n */\nexport interface DefineEslintConfigsParams {\n  /**\n   * A function that builds custom ESLint configurations.\n   *\n   * @param context - The ESLint configuration context.\n   * @returns The custom ESLint configurations.\n   */\n  customConfigs?(context: EslintConfigContext): Linter.Config[];\n\n  /**\n   * A function that edits the ESLint configuration context.\n   *\n   * @param context - The ESLint configuration context.\n   */\n  editContext?(context: EslintConfigContext): void;\n}\n\n/**\n * The context for defining ESLint configurations.\n */\nexport class EslintConfigContext {\n  /**\n   * The root configuration files.\n   */\n  public readonly rootConfigFiles: string[] = [];\n\n  /**\n   * The script files.\n   */\n  public readonly scriptFiles: string[] = [];\n\n  /**\n   * The source files.\n   */\n  public readonly sourceFiles: string[] = [];\n\n  /**\n   * The test files.\n   */\n  public readonly testFiles: string[] = [];\n\n  /**\n   * The all files.\n   *\n   * @returns The all files.\n   */\n  public allFiles(): string[] {\n    return [...this.testFiles, ...this.scriptFiles, ...this.rootConfigFiles, ...this.sourceFiles];\n  }\n}\n\n/**\n * Build ESLint configurations.\n *\n * This function builds ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n *\n * @param params - The parameters for defining ESLint configurations.\n * @returns The ESLint configurations.\n */\nexport function defineEslintConfigs(params: DefineEslintConfigsParams = {}): Linter.Config[] {\n  const context = new EslintConfigContext();\n  context.rootConfigFiles.push(\n    ObsidianPluginRepoPaths.CommitlintConfigTs,\n    ObsidianPluginRepoPaths.EslintConfigMts,\n    ObsidianPluginRepoPaths.VitestConfigTs\n  );\n  context.scriptFiles.push(\n    join(ObsidianPluginRepoPaths.Scripts, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs)\n  );\n  context.sourceFiles.push(\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTsx)\n  );\n  context.testFiles.push(\n    join(ObsidianPluginRepoPaths.Tests, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Mocks, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTestTs)\n  );\n\n  if (params.editContext) {\n    params.editContext(context);\n  }\n\n  const customConfigs = params.customConfigs?.(context) ?? [];\n\n  return defineConfig(\n    includeIgnoreFile(join(getRootFolder() ?? '', '.gitignore')),\n    ...getEslintConfigs(context),\n    ...getTseslintConfigs(context),\n    ...getStylisticConfigs(context),\n    ...getObsidianLintConfigs(context),\n    ...getImportXConfigs(context),\n    ...getPerfectionistConfigs(context),\n    ...getEslintImportResolverTypescriptConfigs(),\n    ...getEslintCommentsConfigs(context),\n    ...getObsidianDevUtilsPluginConfigs(context),\n    ...customConfigs\n  );\n}\n\nfunction getEslintCommentsConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `recommended` is too confusing.\n      extends: [commentsConfigs.recommended],\n      files: context.allFiles(),\n      rules: {\n        '@eslint-community/eslint-comments/require-description': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [eslint.configs.recommended],\n      files: context.allFiles(),\n      rules: {\n        'accessor-pairs': 'error',\n        'array-callback-return': 'error',\n        'camelcase': 'error',\n        'capitalized-comments': ['error', 'always', { block: { ignorePattern: 'v8' } }],\n        'complexity': 'error',\n        'consistent-this': 'error',\n        'curly': 'error',\n        'default-case': 'error',\n        'default-case-last': 'error',\n        'default-param-last': 'error',\n        'eqeqeq': 'error',\n        'func-name-matching': 'error',\n        'func-names': 'error',\n        'func-style': [\n          'error',\n          'declaration',\n          {\n            allowArrowFunctions: false\n          }\n        ],\n        'grouped-accessor-pairs': [\n          'error',\n          'getBeforeSet'\n        ],\n        'guard-for-in': 'error',\n        'no-alert': 'error',\n        'no-array-constructor': 'error',\n        'no-bitwise': 'error',\n        'no-caller': 'error',\n        'no-console': [\n          'error',\n          {\n            allow: [\n              'warn',\n              'error'\n            ]\n          }\n        ],\n        'no-constructor-return': 'error',\n        'no-div-regex': 'error',\n        'no-else-return': [\n          'error',\n          {\n            allowElseIf: false\n          }\n        ],\n        'no-empty-function': 'error',\n        'no-extend-native': 'error',\n        'no-extra-bind': 'error',\n        'no-extra-label': 'error',\n        'no-implicit-coercion': [\n          'error',\n          {\n            allow: [\n              '!!'\n            ]\n          }\n        ],\n        'no-implied-eval': 'error',\n        'no-inner-declarations': 'error',\n        'no-iterator': 'error',\n        'no-label-var': 'error',\n        'no-labels': 'error',\n        'no-lone-blocks': 'error',\n        'no-lonely-if': 'error',\n        'no-loop-func': 'error',\n        'no-magic-numbers': [\n          'error',\n          {\n            detectObjects: true,\n            enforceConst: true,\n            ignore: [\n              -1,\n              0,\n              1\n            ]\n          }\n        ],\n        'no-multi-assign': 'error',\n        'no-multi-str': 'error',\n        'no-negated-condition': 'error',\n        'no-nested-ternary': 'error',\n        'no-new-func': 'error',\n        'no-new-wrappers': 'error',\n        'no-object-constructor': 'error',\n        'no-octal-escape': 'error',\n        'no-promise-executor-return': 'error',\n        'no-proto': 'error',\n        'no-restricted-syntax': [\n          'error',\n          {\n            message: 'Do not use definite assignment assertions (!). Initialize the field or make it optional.',\n            selector: 'PropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use definite assignment assertions (!) on abstract fields.',\n            selector: 'TSAbstractPropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function parameters. Define a named interface instead.',\n            selector: ':function > Identifier TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function return types. Define a named interface instead.',\n            selector: ':function > TSTypeAnnotation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in interface/method signatures. Define a named interface instead.',\n            selector: 'TSMethodSignature TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types as type arguments. Define a named interface instead.',\n            selector: 'TSTypeParameterInstantiation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in type annotations. Define a named interface instead.',\n            selector: 'TSTypeAnnotation TSTypeLiteral'\n          }\n        ],\n        'no-return-assign': 'error',\n        'no-script-url': 'error',\n        'no-self-compare': 'error',\n        'no-sequences': 'error',\n        'no-shadow': 'error',\n        'no-template-curly-in-string': 'error',\n        'no-throw-literal': 'error',\n        'no-unmodified-loop-condition': 'error',\n        'no-unneeded-ternary': 'error',\n        'no-unreachable-loop': 'error',\n        'no-unused-expressions': 'error',\n        'no-useless-assignment': 'error',\n        'no-useless-call': 'error',\n        'no-useless-computed-key': 'error',\n        'no-useless-concat': 'error',\n        'no-useless-constructor': 'error',\n        'no-useless-rename': 'error',\n        'no-useless-return': 'error',\n        'no-var': 'error',\n        'no-void': 'error',\n        'object-shorthand': 'error',\n        'operator-assignment': 'error',\n        'prefer-arrow-callback': 'error',\n        'prefer-const': 'error',\n        'prefer-exponentiation-operator': 'error',\n        'prefer-named-capture-group': 'error',\n        'prefer-numeric-literals': 'error',\n        'prefer-object-has-own': 'error',\n        'prefer-object-spread': 'error',\n        'prefer-promise-reject-errors': 'error',\n        'prefer-regex-literals': 'error',\n        'prefer-rest-params': 'error',\n        'prefer-spread': 'error',\n        'prefer-template': 'error',\n        'radix': 'error',\n        'require-atomic-updates': 'error',\n        'require-await': 'error',\n        'symbol-description': 'error',\n        'unicode-bom': 'error',\n        'vars-on-top': 'error',\n        'yoda': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintImportResolverTypescriptConfigs(): Linter.Config[] {\n  return defineConfig([\n    {\n      settings: {\n        'import-x/resolver-next': [\n          createTypeScriptImportResolver({\n            alwaysTryTypes: true\n          })\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getImportXConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        eslintPluginImportXFlatConfigs.recommended as Linter.Config,\n        eslintPluginImportXFlatConfigs.typescript as Linter.Config,\n        eslintPluginImportXFlatConfigs.errors as Linter.Config,\n        eslintPluginImportXFlatConfigs.warnings as Linter.Config\n      ],\n      files: context.allFiles(),\n      rules: {\n        'import-x/consistent-type-specifier-style': 'error',\n        'import-x/extensions': ['error', 'ignorePackages'],\n        'import-x/first': 'error',\n        'import-x/imports-first': 'error',\n        'import-x/newline-after-import': 'error',\n        'import-x/no-absolute-path': 'error',\n        'import-x/no-amd': 'error',\n        'import-x/no-anonymous-default-export': 'error',\n        'import-x/no-commonjs': 'error',\n        'import-x/no-cycle': 'error',\n        'import-x/no-default-export': 'error',\n        'import-x/no-deprecated': 'error',\n        'import-x/no-duplicates': 'error',\n        'import-x/no-dynamic-require': 'error',\n        'import-x/no-empty-named-blocks': 'error',\n        'import-x/no-extraneous-dependencies': 'error',\n        'import-x/no-import-module-exports': 'error',\n        'import-x/no-mutable-exports': 'error',\n        'import-x/no-named-default': 'error',\n        'import-x/no-namespace': 'error',\n        'import-x/no-nodejs-modules': 'error',\n        'import-x/no-relative-packages': 'error',\n        'import-x/no-restricted-paths': 'error',\n        'import-x/no-self-import': 'error',\n        'import-x/no-unassigned-import': [\n          'error',\n          {\n            allow: [\n              '**/*.css',\n              '**/*.sass',\n              '**/*.scss'\n            ]\n          }\n        ],\n        'import-x/no-unused-modules': 'off',\n        'import-x/no-useless-path-segments': 'error',\n        'import-x/no-webpack-loader-syntax': 'error'\n      }\n    },\n    {\n      files: context.scriptFiles,\n      rules: {\n        'import-x/no-nodejs-modules': 'off'\n      }\n    },\n    {\n      files: [\n        ...context.rootConfigFiles,\n        join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.MainTs)\n      ],\n      rules: {\n        'import-x/no-default-export': 'off'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianDevUtilsPluginConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      files: context.allFiles(),\n      plugins: {\n        'obsidian-dev-utils': obsidianDevUtilsPlugin\n      },\n      rules: {\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianLintConfigs(context: EslintConfigContext): Linter.Config[] {\n  const obsidianRecommendedConfigs = Array.from(obsidianmd.configs?.['recommended'] as Iterable<Linter.Config>);\n\n  const scopedObsidianRecommendedConfigs = obsidianRecommendedConfigs.map((config) => {\n    if (config.files?.includes('package.json')) {\n      return config;\n    }\n\n    return {\n      ...config,\n      files: context.sourceFiles\n    };\n  });\n\n  return defineConfig([\n    ...scopedObsidianRecommendedConfigs,\n    {\n      plugins: {\n        obsidianmd: obsidianmd as ESLint.Plugin\n      }\n    },\n    {\n      languageOptions: {\n        globals: {\n          ...globals.browser,\n          ...globals.node,\n          activeDocument: 'readonly',\n          activeWindow: 'readonly',\n          ajax: 'readonly',\n          ajaxPromise: 'readonly',\n          createDiv: 'readonly',\n          createEl: 'readonly',\n          createFragment: 'readonly',\n          createSpan: 'readonly',\n          createSvg: 'readonly',\n          DomElementInfo: 'readonly',\n          fish: 'readonly',\n          fishAll: 'readonly',\n          isBoolean: 'readonly',\n          nextFrame: 'readonly',\n          NodeJS: 'readonly',\n          ready: 'readonly',\n          sleep: 'readonly'\n        }\n      }\n    }\n  ]);\n}\n\nfunction getPerfectionistConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([{\n    extends: [perfectionistConfigs['recommended-alphabetical']],\n    files: context.allFiles()\n  }]);\n}\n\nfunction getStylisticConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        stylistic.configs.recommended,\n        stylistic.configs.customize({\n          arrowParens: true,\n          braceStyle: '1tbs',\n          commaDangle: 'never',\n          semi: true\n        })\n      ],\n      files: context.allFiles(),\n      rules: {\n        '@stylistic/indent': 'off',\n        '@stylistic/indent-binary-ops': 'off',\n        '@stylistic/jsx-one-expression-per-line': 'off',\n        '@stylistic/no-extra-semi': 'error',\n        '@stylistic/object-curly-newline': [\n          'error',\n          {\n            ExportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            },\n            ImportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            }\n          }\n        ],\n        '@stylistic/operator-linebreak': [\n          'error',\n          'before',\n          {\n            overrides: {\n              '=': 'after'\n            }\n          }\n        ],\n        '@stylistic/quotes': [\n          'error',\n          'single',\n          {\n            allowTemplateLiterals: 'never'\n          }\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getTseslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.strictTypeChecked,\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.stylisticTypeChecked\n      ],\n      files: context.allFiles(),\n      languageOptions: {\n        parserOptions: {\n          ecmaFeatures: {\n            jsx: true\n          },\n          projectService: true,\n          tsconfigRootDir: getRootFolder() ?? ''\n        }\n      },\n      rules: {\n        '@typescript-eslint/explicit-function-return-type': 'error',\n        '@typescript-eslint/explicit-member-accessibility': 'error',\n        '@typescript-eslint/no-invalid-void-type': ['error', {\n          allowAsThisParameter: true\n        }],\n        '@typescript-eslint/no-this-alias': ['error', {\n          allowedNames: [\n            'that'\n          ]\n        }],\n        '@typescript-eslint/no-unused-vars': [\n          'error',\n          {\n            args: 'all',\n            argsIgnorePattern: '^_',\n            caughtErrors: 'all',\n            caughtErrorsIgnorePattern: '^_',\n            destructuredArrayIgnorePattern: '^_',\n            ignoreRestSiblings: true,\n            varsIgnorePattern: '^_'\n          }\n        ],\n        '@typescript-eslint/prefer-readonly': 'error',\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    },\n    {\n      settings: {\n        react: {\n          version: 'detect'\n        }\n      }\n    }\n  ]);\n}\n\n/* eslint-enable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}

537
+ //# sourceMappingURL=data:application/json;base64,{
  "version": 3,
  "sources": ["../../../../../src/script-utils/linters/eslint-config.ts"],
  "sourcesContent": ["/**\n * @packageDocumentation\n *\n * ESLint configuration for TypeScript projects with various plugins.\n *\n * This module exports ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n * such as `@typescript-eslint/parser`, `@typescript-eslint/eslint-plugin`,\n * `eslint-plugin-modules-newlines`, `@stylistic/eslint-plugin`.\n * It sets up parsers, plugins, and rules for maintaining code quality and consistency.\n */\n\n/* v8 ignore start -- Declarative ESLint rule/plugin configuration; correctness is verified by running ESLint, not unit tests. */\n\nimport type {\n  ESLint,\n  Linter\n} from 'eslint';\n\n/* eslint-disable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\nimport commentsConfigs from '@eslint-community/eslint-plugin-eslint-comments/configs';\nimport { includeIgnoreFile } from '@eslint/compat';\nimport eslint from '@eslint/js';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport stylistic from '@stylistic/eslint-plugin';\nimport { createTypeScriptImportResolver } from 'eslint-import-resolver-typescript';\nimport { flatConfigs as eslintPluginImportXFlatConfigs } from 'eslint-plugin-import-x';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `plugin` is too confusing.\nimport obsidianmd from 'eslint-plugin-obsidianmd';\nimport { configs as perfectionistConfigs } from 'eslint-plugin-perfectionist';\nimport { defineConfig } from 'eslint/config';\nimport globals from 'globals';\n// eslint-disable-next-line import-x/no-rename-default -- The default export name `_default` is too confusing.\nimport tseslint from 'typescript-eslint';\n\nimport { ObsidianPluginRepoPaths } from '../../obsidian/plugin/obsidian-plugin-repo-paths.ts';\nimport { join } from '../../path.ts';\nimport { getRootFolder } from '../root.ts';\nimport { obsidianDevUtilsPlugin } from './eslint-rules/obsidian-dev-utils-plugin.ts';\n\n/**\n * The parameters for defining ESLint configurations.\n */\nexport interface DefineEslintConfigsParams {\n  /**\n   * A function that builds custom ESLint configurations.\n   *\n   * @param context - The ESLint configuration context.\n   * @returns The custom ESLint configurations.\n   */\n  customConfigs?(context: EslintConfigContext): Linter.Config[];\n\n  /**\n   * A function that edits the ESLint configuration context.\n   *\n   * @param context - The ESLint configuration context.\n   */\n  editContext?(context: EslintConfigContext): void;\n}\n\n/**\n * The context for defining ESLint configurations.\n */\nexport class EslintConfigContext {\n  /**\n   * The root configuration files.\n   */\n  public readonly rootConfigFiles: string[] = [];\n\n  /**\n   * The script files.\n   */\n  public readonly scriptFiles: string[] = [];\n\n  /**\n   * The source files.\n   */\n  public readonly sourceFiles: string[] = [];\n\n  /**\n   * The test files.\n   */\n  public readonly testFiles: string[] = [];\n\n  /**\n   * The all files.\n   *\n   * @returns The all files.\n   */\n  public allFiles(): string[] {\n    return [...this.testFiles, ...this.scriptFiles, ...this.rootConfigFiles, ...this.sourceFiles];\n  }\n}\n\n/**\n * Build ESLint configurations.\n *\n * This function builds ESLint configurations for TypeScript projects, integrating multiple ESLint plugins\n *\n * @param params - The parameters for defining ESLint configurations.\n * @returns The ESLint configurations.\n */\nexport function defineEslintConfigs(params: DefineEslintConfigsParams = {}): Linter.Config[] {\n  const context = new EslintConfigContext();\n  context.rootConfigFiles.push(\n    ObsidianPluginRepoPaths.CommitlintConfigTs,\n    ObsidianPluginRepoPaths.EslintConfigMts,\n    ObsidianPluginRepoPaths.VitestConfigTs\n  );\n  context.scriptFiles.push(\n    join(ObsidianPluginRepoPaths.Scripts, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs)\n  );\n  context.sourceFiles.push(\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTsx)\n  );\n  context.testFiles.push(\n    join(ObsidianPluginRepoPaths.Tests, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Mocks, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTs),\n    join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.AnyPath, ObsidianPluginRepoPaths.AnyTestTs)\n  );\n\n  if (params.editContext) {\n    params.editContext(context);\n  }\n\n  const customConfigs = params.customConfigs?.(context) ?? [];\n\n  return defineConfig(\n    includeIgnoreFile(join(getRootFolder() ?? '', '.gitignore')),\n    ...getEslintConfigs(context),\n    ...getTseslintConfigs(context),\n    ...getStylisticConfigs(context),\n    ...getObsidianLintConfigs(context),\n    ...getImportXConfigs(context),\n    ...getPerfectionistConfigs(context),\n    ...getEslintImportResolverTypescriptConfigs(),\n    ...getEslintCommentsConfigs(context),\n    ...getObsidianDevUtilsPluginConfigs(context),\n    ...customConfigs\n  );\n}\n\nfunction getEslintCommentsConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `recommended` is too confusing.\n      extends: [commentsConfigs.recommended],\n      files: context.allFiles(),\n      rules: {\n        '@eslint-community/eslint-comments/require-description': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [eslint.configs.recommended],\n      files: context.allFiles(),\n      rules: {\n        'accessor-pairs': 'error',\n        'array-callback-return': 'error',\n        'camelcase': 'error',\n        'capitalized-comments': ['error', 'always', { block: { ignorePattern: 'v8' } }],\n        'complexity': 'error',\n        'consistent-this': 'error',\n        'curly': 'error',\n        'default-case': 'error',\n        'default-case-last': 'error',\n        'default-param-last': 'error',\n        'eqeqeq': 'error',\n        'func-name-matching': 'error',\n        'func-names': 'error',\n        'func-style': [\n          'error',\n          'declaration',\n          {\n            allowArrowFunctions: false\n          }\n        ],\n        'grouped-accessor-pairs': [\n          'error',\n          'getBeforeSet'\n        ],\n        'guard-for-in': 'error',\n        'no-alert': 'error',\n        'no-array-constructor': 'error',\n        'no-bitwise': 'error',\n        'no-caller': 'error',\n        'no-console': [\n          'error',\n          {\n            allow: [\n              'warn',\n              'error'\n            ]\n          }\n        ],\n        'no-constructor-return': 'error',\n        'no-div-regex': 'error',\n        'no-else-return': [\n          'error',\n          {\n            allowElseIf: false\n          }\n        ],\n        'no-empty-function': 'error',\n        'no-extend-native': 'error',\n        'no-extra-bind': 'error',\n        'no-extra-label': 'error',\n        'no-implicit-coercion': [\n          'error',\n          {\n            allow: [\n              '!!'\n            ]\n          }\n        ],\n        'no-implied-eval': 'error',\n        'no-inner-declarations': 'error',\n        'no-iterator': 'error',\n        'no-label-var': 'error',\n        'no-labels': 'error',\n        'no-lone-blocks': 'error',\n        'no-lonely-if': 'error',\n        'no-loop-func': 'error',\n        'no-magic-numbers': [\n          'error',\n          {\n            detectObjects: true,\n            enforceConst: true,\n            ignore: [\n              -1,\n              0,\n              1\n            ]\n          }\n        ],\n        'no-multi-assign': 'error',\n        'no-multi-str': 'error',\n        'no-negated-condition': 'error',\n        'no-nested-ternary': 'error',\n        'no-new-func': 'error',\n        'no-new-wrappers': 'error',\n        'no-object-constructor': 'error',\n        'no-octal-escape': 'error',\n        'no-promise-executor-return': 'error',\n        'no-proto': 'error',\n        'no-restricted-syntax': [\n          'error',\n          {\n            message: 'Do not use definite assignment assertions (!). Initialize the field or make it optional.',\n            selector: 'PropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use definite assignment assertions (!) on abstract fields.',\n            selector: 'TSAbstractPropertyDefinition[definite=true]'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function parameters. Define a named interface instead.',\n            selector: ':function > Identifier TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in function return types. Define a named interface instead.',\n            selector: ':function > TSTypeAnnotation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in interface/method signatures. Define a named interface instead.',\n            selector: 'TSMethodSignature TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types as type arguments. Define a named interface instead.',\n            selector: 'TSTypeParameterInstantiation TSTypeLiteral'\n          },\n          {\n            message: 'Do not use anonymous inline object types in type annotations. Define a named interface instead.',\n            selector: 'TSTypeAnnotation TSTypeLiteral'\n          }\n        ],\n        'no-return-assign': 'error',\n        'no-script-url': 'error',\n        'no-self-compare': 'error',\n        'no-sequences': 'error',\n        'no-shadow': 'error',\n        'no-template-curly-in-string': 'error',\n        'no-throw-literal': 'error',\n        'no-unmodified-loop-condition': 'error',\n        'no-unneeded-ternary': 'error',\n        'no-unreachable-loop': 'error',\n        'no-unused-expressions': 'error',\n        'no-useless-assignment': 'error',\n        'no-useless-call': 'error',\n        'no-useless-computed-key': 'error',\n        'no-useless-concat': 'error',\n        'no-useless-constructor': 'error',\n        'no-useless-rename': 'error',\n        'no-useless-return': 'error',\n        'no-var': 'error',\n        'no-void': 'error',\n        'object-shorthand': 'error',\n        'operator-assignment': 'error',\n        'prefer-arrow-callback': 'error',\n        'prefer-const': 'error',\n        'prefer-exponentiation-operator': 'error',\n        'prefer-named-capture-group': 'error',\n        'prefer-numeric-literals': 'error',\n        'prefer-object-has-own': 'error',\n        'prefer-object-spread': 'error',\n        'prefer-promise-reject-errors': 'error',\n        'prefer-regex-literals': 'error',\n        'prefer-rest-params': 'error',\n        'prefer-spread': 'error',\n        'prefer-template': 'error',\n        'radix': 'error',\n        'require-atomic-updates': 'error',\n        'require-await': 'error',\n        'symbol-description': 'error',\n        'unicode-bom': 'error',\n        'vars-on-top': 'error',\n        'yoda': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getEslintImportResolverTypescriptConfigs(): Linter.Config[] {\n  return defineConfig([\n    {\n      settings: {\n        'import-x/resolver-next': [\n          createTypeScriptImportResolver({\n            alwaysTryTypes: true\n          })\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getImportXConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        eslintPluginImportXFlatConfigs.recommended as Linter.Config,\n        eslintPluginImportXFlatConfigs.typescript as Linter.Config,\n        eslintPluginImportXFlatConfigs.errors as Linter.Config,\n        eslintPluginImportXFlatConfigs.warnings as Linter.Config\n      ],\n      files: context.allFiles(),\n      rules: {\n        'import-x/consistent-type-specifier-style': 'error',\n        'import-x/extensions': ['error', 'ignorePackages'],\n        'import-x/first': 'error',\n        'import-x/imports-first': 'error',\n        'import-x/newline-after-import': 'error',\n        'import-x/no-absolute-path': 'error',\n        'import-x/no-amd': 'error',\n        'import-x/no-anonymous-default-export': 'error',\n        'import-x/no-commonjs': 'error',\n        'import-x/no-cycle': 'error',\n        'import-x/no-default-export': 'error',\n        'import-x/no-deprecated': 'error',\n        'import-x/no-duplicates': 'error',\n        'import-x/no-dynamic-require': 'error',\n        'import-x/no-empty-named-blocks': 'error',\n        'import-x/no-extraneous-dependencies': 'error',\n        'import-x/no-import-module-exports': 'error',\n        'import-x/no-mutable-exports': 'error',\n        'import-x/no-named-default': 'error',\n        'import-x/no-namespace': 'error',\n        'import-x/no-nodejs-modules': 'error',\n        'import-x/no-relative-packages': 'error',\n        'import-x/no-restricted-paths': 'error',\n        'import-x/no-self-import': 'error',\n        'import-x/no-unassigned-import': [\n          'error',\n          {\n            allow: [\n              '**/*.css',\n              '**/*.sass',\n              '**/*.scss'\n            ]\n          }\n        ],\n        'import-x/no-unused-modules': 'off',\n        'import-x/no-useless-path-segments': 'error',\n        'import-x/no-webpack-loader-syntax': 'error'\n      }\n    },\n    {\n      files: context.scriptFiles,\n      rules: {\n        'import-x/no-nodejs-modules': 'off'\n      }\n    },\n    {\n      files: [\n        ...context.rootConfigFiles,\n        join(ObsidianPluginRepoPaths.Src, ObsidianPluginRepoPaths.MainTs)\n      ],\n      rules: {\n        'import-x/no-default-export': 'off'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianDevUtilsPluginConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      files: context.allFiles(),\n      plugins: {\n        'obsidian-dev-utils': obsidianDevUtilsPlugin\n      },\n      rules: {\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    }\n  ]);\n}\n\nfunction getObsidianLintConfigs(context: EslintConfigContext): Linter.Config[] {\n  const obsidianRecommendedConfigs = Array.from(obsidianmd.configs?.['recommended'] as Iterable<Linter.Config>);\n\n  const scopedObsidianRecommendedConfigs = obsidianRecommendedConfigs.map((config) => {\n    if (config.files?.includes('package.json')) {\n      return config;\n    }\n\n    return {\n      ...config,\n      files: context.sourceFiles\n    };\n  });\n\n  return defineConfig([\n    ...scopedObsidianRecommendedConfigs,\n    {\n      plugins: {\n        obsidianmd: obsidianmd as ESLint.Plugin\n      }\n    },\n    {\n      languageOptions: {\n        globals: {\n          ...globals.browser,\n          ...globals.node,\n          activeDocument: 'readonly',\n          activeWindow: 'readonly',\n          ajax: 'readonly',\n          ajaxPromise: 'readonly',\n          createDiv: 'readonly',\n          createEl: 'readonly',\n          createFragment: 'readonly',\n          createSpan: 'readonly',\n          createSvg: 'readonly',\n          DomElementInfo: 'readonly',\n          fish: 'readonly',\n          fishAll: 'readonly',\n          isBoolean: 'readonly',\n          nextFrame: 'readonly',\n          NodeJS: 'readonly',\n          ready: 'readonly',\n          sleep: 'readonly'\n        }\n      }\n    }\n  ]);\n}\n\nfunction getPerfectionistConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([{\n    extends: [perfectionistConfigs['recommended-alphabetical']],\n    files: context.allFiles()\n  }]);\n}\n\nfunction getStylisticConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        stylistic.configs.recommended,\n        stylistic.configs.customize({\n          arrowParens: true,\n          braceStyle: '1tbs',\n          commaDangle: 'never',\n          semi: true\n        })\n      ],\n      files: context.allFiles(),\n      rules: {\n        '@stylistic/generator-star-spacing': 'off',\n        '@stylistic/indent': 'off',\n        '@stylistic/indent-binary-ops': 'off',\n        '@stylistic/jsx-one-expression-per-line': 'off',\n        '@stylistic/no-extra-semi': 'error',\n        '@stylistic/object-curly-newline': [\n          'error',\n          {\n            ExportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            },\n            ImportDeclaration: {\n              minProperties: 2,\n              multiline: true\n            }\n          }\n        ],\n        '@stylistic/operator-linebreak': [\n          'error',\n          'before',\n          {\n            overrides: {\n              '=': 'after'\n            }\n          }\n        ],\n        '@stylistic/quotes': [\n          'error',\n          'single',\n          {\n            allowTemplateLiterals: 'never'\n          }\n        ]\n      }\n    }\n  ]);\n}\n\nfunction getTseslintConfigs(context: EslintConfigContext): Linter.Config[] {\n  return defineConfig([\n    {\n      extends: [\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.strictTypeChecked,\n        // eslint-disable-next-line import-x/no-named-as-default-member -- The default export name `_default` is too confusing.\n        ...tseslint.configs.stylisticTypeChecked\n      ],\n      files: context.allFiles(),\n      languageOptions: {\n        parserOptions: {\n          ecmaFeatures: {\n            jsx: true\n          },\n          projectService: true,\n          tsconfigRootDir: getRootFolder() ?? ''\n        }\n      },\n      rules: {\n        '@typescript-eslint/explicit-function-return-type': 'error',\n        '@typescript-eslint/explicit-member-accessibility': 'error',\n        '@typescript-eslint/no-invalid-void-type': ['error', {\n          allowAsThisParameter: true\n        }],\n        '@typescript-eslint/no-this-alias': ['error', {\n          allowedNames: [\n            'that'\n          ]\n        }],\n        '@typescript-eslint/no-unused-vars': [\n          'error',\n          {\n            args: 'all',\n            argsIgnorePattern: '^_',\n            caughtErrors: 'all',\n            caughtErrorsIgnorePattern: '^_',\n            destructuredArrayIgnorePattern: '^_',\n            ignoreRestSiblings: true,\n            varsIgnorePattern: '^_'\n          }\n        ],\n        '@typescript-eslint/prefer-readonly': 'error',\n        'obsidian-dev-utils/no-used-underscore-params': 'error'\n      }\n    },\n    {\n      settings: {\n        react: {\n          version: 'detect'\n        }\n      }\n    }\n  ]);\n}\n\n/* eslint-enable no-magic-numbers -- We disabled magic numbers because they are used all over the configs. */\n\n/* v8 ignore stop */\n"],
  "mappings": "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",
  "names": []
}

@@ -19,11 +19,12 @@ if you want to view the source, please visit the github repository of this plugi
19
19
  globalThis.process = browserProcess;
20
20
  })();
21
21
 
22
+ import { getFunctionExpressionString } from "../function.mjs";
22
23
  import { trimStart } from "../string.mjs";
23
24
  import { exec } from "./exec.mjs";
24
25
  async function evalObsidianCli(params) {
25
26
  const { args, fn, vaultPath } = params;
26
- const fnString = fn.toString();
27
+ const fnString = getFunctionExpressionString(fn);
27
28
  const argsStr = args && args.length > 0 ? `, ...${JSON.stringify(args)}` : "";
28
29
  const expression = `await (${fnString})(app${argsStr})`;
29
30
  const resultStr = await exec(["obsidian", "eval", `code=(async () => JSON.stringify(${expression}))()`, vaultPath], { cwd: vaultPath, isQuiet: true });
@@ -33,4 +34,4 @@ async function evalObsidianCli(params) {
33
34
  export {
34
35
  evalObsidianCli
35
36
  };
36
- //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL3NjcmlwdC11dGlscy9vYnNpZGlhbi1jbGkudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgdXRpbGl0aWVzIGZvciBpbnRlcmFjdGluZyB3aXRoIHRoZSBPYnNpZGlhbiBDTEkuXG4gKi9cblxuLyogdjggaWdub3JlIHN0YXJ0IC0tIFJlcXVpcmVzIE9ic2lkaWFuIENMSSB0byBiZSBpbnN0YWxsZWQgYW5kIHJ1bm5pbmcuICovXG5cbmltcG9ydCB0eXBlIHsgQXBwIH0gZnJvbSAnb2JzaWRpYW4nO1xuaW1wb3J0IHR5cGUgeyBQcm9taXNhYmxlIH0gZnJvbSAndHlwZS1mZXN0JztcblxuaW1wb3J0IHsgdHJpbVN0YXJ0IH0gZnJvbSAnLi4vc3RyaW5nLnRzJztcbmltcG9ydCB7IGV4ZWMgfSBmcm9tICcuL2V4ZWMudHMnO1xuXG4vKipcbiAqIFBhcmFtZXRlcnMgZm9yIHtAbGluayBldmFsT2JzaWRpYW5DbGl9LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEV2YWxPYnNpZGlhbkNsaVBhcmFtczxBcmdzIGV4dGVuZHMgdW5rbm93bltdLCBSZXN1bHQ+IHtcbiAgLyoqXG4gICAqIEFkZGl0aW9uYWwgYXJndW1lbnRzIHRvIHBhc3MgYWZ0ZXIgYGFwcGAuIE11c3QgYmUgSlNPTi1zZXJpYWxpemFibGUuXG4gICAqL1xuICBhcmdzPzogQXJncztcblxuICAvKipcbiAgICogVGhlIGZ1bmN0aW9uIHRvIGV2YWx1YXRlIGluIHRoZSBPYnNpZGlhbiBjb250ZXh0LlxuICAgKi9cbiAgZm46IChhcHA6IEFwcCwgLi4uYXJnczogQXJncykgPT4gUHJvbWlzYWJsZTxSZXN1bHQ+O1xuXG4gIC8qKlxuICAgKiBUaGUgcGF0aCB0byB0aGUgT2JzaWRpYW4gdmF1bHQuXG4gICAqL1xuICB2YXVsdFBhdGg6IHN0cmluZztcbn1cblxuLyoqXG4gKiBFdmFsdWF0ZXMgYSBmdW5jdGlvbiBpbnNpZGUgdGhlIHJ1bm5pbmcgT2JzaWRpYW4gaW5zdGFuY2VcbiAqIHZpYSB0aGUgT2JzaWRpYW4gQ0xJIGFuZCByZXR1cm5zIHRoZSBwYXJzZWQgcmVzdWx0LlxuICpcbiAqIFRoZSBmdW5jdGlvbiByZWNlaXZlcyBgYXBwYCBhcyBpdHMgZmlyc3QgYXJndW1lbnQsIGZvbGxvd2VkIGJ5IGFueSBhZGRpdGlvbmFsIGBhcmdzYC5cbiAqIEl0IGlzIHNlcmlhbGl6ZWQgdmlhIGB0b1N0cmluZygpYCBhbmQgaW52b2tlZCBhcyBhbiBJSUZFLlxuICogVGhlIGZ1bmN0aW9uIG11c3QgYmUgc2VsZi1jb250YWluZWQgXHUyMDE0IGNsb3N1cmVzIG92ZXIgbG9jYWwgdmFyaWFibGVzIHdpbGwgbm90IHdvcmsuXG4gKiBQYXNzIGFueSBuZWVkZWQgdmFsdWVzIGFzIGBhcmdzYCBcdTIwMTQgdGhleSBhcmUgSlNPTi1zZXJpYWxpemVkIGFuZCBkZXNlcmlhbGl6ZWQgb24gdGhlIE9ic2lkaWFuIHNpZGUuXG4gKlxuICogVGhlIHJlc3VsdCBpcyBgSlNPTi5zdHJpbmdpZnlgJ2Qgb24gdGhlIE9ic2lkaWFuIHNpZGUgYW5kIHBhcnNlZCBiYWNrLlxuICpcbiAqIEBwYXJhbSBwYXJhbXMgLSBUaGUgcGFyYW1ldGVycyBmb3IgdGhlIGZ1bmN0aW9uIHRvIGV2YWx1YXRlLlxuICogQHJldHVybnMgQSB7QGxpbmsgUHJvbWlzZX0gdGhhdCByZXNvbHZlcyB0byB0aGUgcmV0dXJuIHZhbHVlIG9mIGBmbmAuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBldmFsT2JzaWRpYW5DbGk8QXJncyBleHRlbmRzIHVua25vd25bXSwgUmVzdWx0PihwYXJhbXM6IEV2YWxPYnNpZGlhbkNsaVBhcmFtczxBcmdzLCBSZXN1bHQ+KTogUHJvbWlzZTxSZXN1bHQ+IHtcbiAgY29uc3QgeyBhcmdzLCBmbiwgdmF1bHRQYXRoIH0gPSBwYXJhbXM7XG4gIGNvbnN0IGZuU3RyaW5nID0gZm4udG9TdHJpbmcoKTtcbiAgY29uc3QgYXJnc1N0ciA9IGFyZ3MgJiYgYXJncy5sZW5ndGggPiAwID8gYCwgLi4uJHtKU09OLnN0cmluZ2lmeShhcmdzKSBhcyBzdHJpbmd9YCA6ICcnO1xuICBjb25zdCBleHByZXNzaW9uID0gYGF3YWl0ICgke2ZuU3RyaW5nfSkoYXBwJHthcmdzU3RyfSlgO1xuICBjb25zdCByZXN1bHRTdHIgPSBhd2FpdCBleGVjKFsnb2JzaWRpYW4nLCAnZXZhbCcsIGBjb2RlPShhc3luYyAoKSA9PiBKU09OLnN0cmluZ2lmeSgke2V4cHJlc3Npb259KSkoKWAsIHZhdWx0UGF0aF0sIHsgY3dkOiB2YXVsdFBhdGgsIGlzUXVpZXQ6IHRydWUgfSk7XG4gIGNvbnN0IHJlc3VsdEpzb24gPSB0cmltU3RhcnQocmVzdWx0U3RyLCAnPT4gJyk7XG4gIHJldHVybiBKU09OLnBhcnNlKHJlc3VsdEpzb24pIGFzIFJlc3VsdDtcbn1cblxuLyogdjggaWdub3JlIHN0b3AgKi9cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQVdBLFNBQVMsaUJBQWlCO0FBQzFCLFNBQVMsWUFBWTtBQW9DckIsZUFBc0IsZ0JBQWdELFFBQThEO0FBQ2xJLFFBQU0sRUFBRSxNQUFNLElBQUksVUFBVSxJQUFJO0FBQ2hDLFFBQU0sV0FBVyxHQUFHLFNBQVM7QUFDN0IsUUFBTSxVQUFVLFFBQVEsS0FBSyxTQUFTLElBQUksUUFBUSxLQUFLLFVBQVUsSUFBSSxDQUFXLEtBQUs7QUFDckYsUUFBTSxhQUFhLFVBQVUsUUFBUSxRQUFRLE9BQU87QUFDcEQsUUFBTSxZQUFZLE1BQU0sS0FBSyxDQUFDLFlBQVksUUFBUSxvQ0FBb0MsVUFBVSxRQUFRLFNBQVMsR0FBRyxFQUFFLEtBQUssV0FBVyxTQUFTLEtBQUssQ0FBQztBQUNySixRQUFNLGFBQWEsVUFBVSxXQUFXLEtBQUs7QUFDN0MsU0FBTyxLQUFLLE1BQU0sVUFBVTtBQUM5QjsiLAogICJuYW1lcyI6IFtdCn0K
37
+ //# sourceMappingURL=data:application/json;base64,ewogICJ2ZXJzaW9uIjogMywKICAic291cmNlcyI6IFsiLi4vLi4vLi4vLi4vc3JjL3NjcmlwdC11dGlscy9vYnNpZGlhbi1jbGkudHMiXSwKICAic291cmNlc0NvbnRlbnQiOiBbIi8qKlxuICogQHBhY2thZ2VEb2N1bWVudGF0aW9uXG4gKlxuICogVGhpcyBtb2R1bGUgcHJvdmlkZXMgdXRpbGl0aWVzIGZvciBpbnRlcmFjdGluZyB3aXRoIHRoZSBPYnNpZGlhbiBDTEkuXG4gKi9cblxuLyogdjggaWdub3JlIHN0YXJ0IC0tIFJlcXVpcmVzIE9ic2lkaWFuIENMSSB0byBiZSBpbnN0YWxsZWQgYW5kIHJ1bm5pbmcuICovXG5cbmltcG9ydCB0eXBlIHsgQXBwIH0gZnJvbSAnb2JzaWRpYW4nO1xuaW1wb3J0IHR5cGUgeyBQcm9taXNhYmxlIH0gZnJvbSAndHlwZS1mZXN0JztcblxuaW1wb3J0IHsgZ2V0RnVuY3Rpb25FeHByZXNzaW9uU3RyaW5nIH0gZnJvbSAnLi4vZnVuY3Rpb24udHMnO1xuaW1wb3J0IHsgdHJpbVN0YXJ0IH0gZnJvbSAnLi4vc3RyaW5nLnRzJztcbmltcG9ydCB7IGV4ZWMgfSBmcm9tICcuL2V4ZWMudHMnO1xuXG4vKipcbiAqIFBhcmFtZXRlcnMgZm9yIHtAbGluayBldmFsT2JzaWRpYW5DbGl9LlxuICovXG5leHBvcnQgaW50ZXJmYWNlIEV2YWxPYnNpZGlhbkNsaVBhcmFtczxBcmdzIGV4dGVuZHMgdW5rbm93bltdLCBSZXN1bHQ+IHtcbiAgLyoqXG4gICAqIEFkZGl0aW9uYWwgYXJndW1lbnRzIHRvIHBhc3MgYWZ0ZXIgYGFwcGAuIE11c3QgYmUgSlNPTi1zZXJpYWxpemFibGUuXG4gICAqL1xuICBhcmdzPzogQXJncztcblxuICAvKipcbiAgICogVGhlIGZ1bmN0aW9uIHRvIGV2YWx1YXRlIGluIHRoZSBPYnNpZGlhbiBjb250ZXh0LlxuICAgKi9cbiAgZm46IChhcHA6IEFwcCwgLi4uYXJnczogQXJncykgPT4gUHJvbWlzYWJsZTxSZXN1bHQ+O1xuXG4gIC8qKlxuICAgKiBUaGUgcGF0aCB0byB0aGUgT2JzaWRpYW4gdmF1bHQuXG4gICAqL1xuICB2YXVsdFBhdGg6IHN0cmluZztcbn1cblxuLyoqXG4gKiBFdmFsdWF0ZXMgYSBmdW5jdGlvbiBpbnNpZGUgdGhlIHJ1bm5pbmcgT2JzaWRpYW4gaW5zdGFuY2VcbiAqIHZpYSB0aGUgT2JzaWRpYW4gQ0xJIGFuZCByZXR1cm5zIHRoZSBwYXJzZWQgcmVzdWx0LlxuICpcbiAqIFRoZSBmdW5jdGlvbiByZWNlaXZlcyBgYXBwYCBhcyBpdHMgZmlyc3QgYXJndW1lbnQsIGZvbGxvd2VkIGJ5IGFueSBhZGRpdGlvbmFsIGBhcmdzYC5cbiAqIEl0IGlzIHNlcmlhbGl6ZWQgdmlhIGB0b1N0cmluZygpYCBhbmQgaW52b2tlZCBhcyBhbiBJSUZFLlxuICogVGhlIGZ1bmN0aW9uIG11c3QgYmUgc2VsZi1jb250YWluZWQgXHUyMDE0IGNsb3N1cmVzIG92ZXIgbG9jYWwgdmFyaWFibGVzIHdpbGwgbm90IHdvcmsuXG4gKiBQYXNzIGFueSBuZWVkZWQgdmFsdWVzIGFzIGBhcmdzYCBcdTIwMTQgdGhleSBhcmUgSlNPTi1zZXJpYWxpemVkIGFuZCBkZXNlcmlhbGl6ZWQgb24gdGhlIE9ic2lkaWFuIHNpZGUuXG4gKlxuICogVGhlIHJlc3VsdCBpcyBgSlNPTi5zdHJpbmdpZnlgJ2Qgb24gdGhlIE9ic2lkaWFuIHNpZGUgYW5kIHBhcnNlZCBiYWNrLlxuICpcbiAqIEBwYXJhbSBwYXJhbXMgLSBUaGUgcGFyYW1ldGVycyBmb3IgdGhlIGZ1bmN0aW9uIHRvIGV2YWx1YXRlLlxuICogQHJldHVybnMgQSB7QGxpbmsgUHJvbWlzZX0gdGhhdCByZXNvbHZlcyB0byB0aGUgcmV0dXJuIHZhbHVlIG9mIGBmbmAuXG4gKi9cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBldmFsT2JzaWRpYW5DbGk8QXJncyBleHRlbmRzIHVua25vd25bXSwgUmVzdWx0PihwYXJhbXM6IEV2YWxPYnNpZGlhbkNsaVBhcmFtczxBcmdzLCBSZXN1bHQ+KTogUHJvbWlzZTxSZXN1bHQ+IHtcbiAgY29uc3QgeyBhcmdzLCBmbiwgdmF1bHRQYXRoIH0gPSBwYXJhbXM7XG4gIGNvbnN0IGZuU3RyaW5nID0gZ2V0RnVuY3Rpb25FeHByZXNzaW9uU3RyaW5nKGZuKTtcbiAgY29uc3QgYXJnc1N0ciA9IGFyZ3MgJiYgYXJncy5sZW5ndGggPiAwID8gYCwgLi4uJHtKU09OLnN0cmluZ2lmeShhcmdzKSBhcyBzdHJpbmd9YCA6ICcnO1xuICBjb25zdCBleHByZXNzaW9uID0gYGF3YWl0ICgke2ZuU3RyaW5nfSkoYXBwJHthcmdzU3RyfSlgO1xuICBjb25zdCByZXN1bHRTdHIgPSBhd2FpdCBleGVjKFsnb2JzaWRpYW4nLCAnZXZhbCcsIGBjb2RlPShhc3luYyAoKSA9PiBKU09OLnN0cmluZ2lmeSgke2V4cHJlc3Npb259KSkoKWAsIHZhdWx0UGF0aF0sIHsgY3dkOiB2YXVsdFBhdGgsIGlzUXVpZXQ6IHRydWUgfSk7XG4gIGNvbnN0IHJlc3VsdEpzb24gPSB0cmltU3RhcnQocmVzdWx0U3RyLCAnPT4gJyk7XG4gIHJldHVybiBKU09OLnBhcnNlKHJlc3VsdEpzb24pIGFzIFJlc3VsdDtcbn1cblxuLyogdjggaWdub3JlIHN0b3AgKi9cbiJdLAogICJtYXBwaW5ncyI6ICI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztBQVdBLFNBQVMsbUNBQW1DO0FBQzVDLFNBQVMsaUJBQWlCO0FBQzFCLFNBQVMsWUFBWTtBQW9DckIsZUFBc0IsZ0JBQWdELFFBQThEO0FBQ2xJLFFBQU0sRUFBRSxNQUFNLElBQUksVUFBVSxJQUFJO0FBQ2hDLFFBQU0sV0FBVyw0QkFBNEIsRUFBRTtBQUMvQyxRQUFNLFVBQVUsUUFBUSxLQUFLLFNBQVMsSUFBSSxRQUFRLEtBQUssVUFBVSxJQUFJLENBQVcsS0FBSztBQUNyRixRQUFNLGFBQWEsVUFBVSxRQUFRLFFBQVEsT0FBTztBQUNwRCxRQUFNLFlBQVksTUFBTSxLQUFLLENBQUMsWUFBWSxRQUFRLG9DQUFvQyxVQUFVLFFBQVEsU0FBUyxHQUFHLEVBQUUsS0FBSyxXQUFXLFNBQVMsS0FBSyxDQUFDO0FBQ3JKLFFBQU0sYUFBYSxVQUFVLFdBQVcsS0FBSztBQUM3QyxTQUFPLEtBQUssTUFBTSxVQUFVO0FBQzlCOyIsCiAgIm5hbWVzIjogW10KfQo=
@@ -4,31 +4,42 @@
4
4
  * This module provides functions for running tests using the Vitest framework.
5
5
  */
6
6
  /**
7
- * Options for running tests with coverage.
7
+ * Parameters for running tests with coverage.
8
8
  */
9
- export interface TestCoverageOptions {
9
+ export interface TestCoverageParams extends TestParams {
10
10
  /**
11
11
  * Minimum coverage percentage required. If the actual coverage falls below
12
12
  * this threshold, the process exits with a non-zero code.
13
13
  */
14
- minCoverage?: number;
14
+ minCoverageInPercents?: number;
15
+ }
16
+ /**
17
+ * Parameters for running tests.
18
+ */
19
+ export interface TestParams {
20
+ /**
21
+ * The projects to run.
22
+ */
23
+ projects?: string[];
15
24
  }
16
25
  /**
17
26
  * Runs the test suite.
18
27
  *
28
+ * @param params - The parameters for the test.
19
29
  * @returns A {@link Promise} that resolves when the tests have completed.
20
30
  */
21
- export declare function test(): Promise<void>;
31
+ export declare function test(params?: TestParams): Promise<void>;
22
32
  /**
23
33
  * Runs the test suite with coverage.
24
34
  *
25
- * @param options - Optional coverage configuration.
35
+ * @param params - Optional coverage configuration.
26
36
  * @returns A {@link Promise} that resolves when the tests have completed.
27
37
  */
28
- export declare function testCoverage(options?: TestCoverageOptions): Promise<void>;
38
+ export declare function testCoverage(params?: TestCoverageParams): Promise<void>;
29
39
  /**
30
40
  * Runs the test suite in watch mode.
31
41
  *
42
+ * @param params - The parameters for the test.
32
43
  * @returns A {@link Promise} that resolves when the tests have completed.
33
44
  */
34
- export declare function testWatch(): Promise<void>;
45
+ export declare function testWatch(params?: TestParams): Promise<void>;