@nitra/cursor 12.9.0 → 12.10.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.
Files changed (168) hide show
  1. package/.claude-template/settings.template.json +1 -23
  2. package/CHANGELOG.md +6 -0
  3. package/bin/n-cursor.js +8 -43
  4. package/docs/stryker.config.md +0 -2
  5. package/lib/docs/llm.md +16 -21
  6. package/lib/docs/omlx.md +20 -25
  7. package/lib/llm.mjs +25 -7
  8. package/lib/omlx.mjs +10 -3
  9. package/package.json +1 -1
  10. package/rules/abie/docs/main.md +0 -2
  11. package/rules/abie/lib/docs/http-route.md +0 -2
  12. package/rules/abie/main.mdc +0 -22
  13. package/rules/adr/docs/main.md +0 -2
  14. package/rules/adr/main.mdc +0 -9
  15. package/rules/bun/docs/main.md +0 -2
  16. package/rules/bun/main.mdc +1 -15
  17. package/rules/capacitor/docs/main.md +0 -2
  18. package/rules/capacitor/main.mdc +0 -6
  19. package/rules/changelog/docs/main.md +0 -2
  20. package/rules/changelog/js/agent-workflow.mdc +1 -1
  21. package/rules/changelog/js/consistency.mjs +3 -3
  22. package/rules/changelog/js/docs/consistency.md +18 -23
  23. package/rules/changelog/main.mdc +0 -5
  24. package/rules/ci4/docs/main.md +0 -2
  25. package/rules/ci4/main.mdc +0 -5
  26. package/rules/doc-files/docs/main.md +0 -2
  27. package/rules/doc-files/js/docs/docgen-crc.md +0 -2
  28. package/rules/doc-files/js/docs/docgen-extract.md +0 -2
  29. package/rules/doc-files/js/docs/docgen-files-batch.md +0 -2
  30. package/rules/doc-files/js/docs/docgen-gen.md +0 -2
  31. package/rules/doc-files/js/docs/docgen-judge-measure.md +0 -2
  32. package/rules/doc-files/js/docs/docgen-judge.md +0 -2
  33. package/rules/doc-files/js/docs/docgen-scan.md +0 -2
  34. package/rules/doc-files/js/docs/run-lint.md +0 -2
  35. package/rules/docker/docs/main.md +0 -2
  36. package/rules/docker/js/docs/lint.md +0 -2
  37. package/rules/docker/lib/docs/docker-hadolint.md +0 -2
  38. package/rules/docker/main.mdc +1 -21
  39. package/rules/efes/docs/main.md +0 -2
  40. package/rules/efes/main.mdc +0 -1
  41. package/rules/feedback/docs/main.md +0 -2
  42. package/rules/ga/docs/main.md +0 -2
  43. package/rules/ga/js/docs/index.md +0 -1
  44. package/rules/ga/main.mdc +1 -31
  45. package/rules/graphql/docs/main.md +0 -2
  46. package/rules/graphql/main.mdc +0 -5
  47. package/rules/hasura/docs/main.md +0 -2
  48. package/rules/hasura/js/docs/index.md +3 -3
  49. package/rules/hasura/js/docs/migrations.md +0 -2
  50. package/rules/hasura/main.mdc +1 -11
  51. package/rules/image-avif/docs/main.md +0 -2
  52. package/rules/image-avif/main.mdc +1 -9
  53. package/rules/image-compress/docs/main.md +0 -2
  54. package/rules/image-compress/js/docs/index.md +0 -1
  55. package/rules/image-compress/main.mdc +1 -9
  56. package/rules/js/docs/main.md +0 -2
  57. package/rules/js/js/dep-policy.mjs +8 -4
  58. package/rules/js/js/docs/check.md +0 -2
  59. package/rules/js/js/docs/dep-policy.md +10 -12
  60. package/rules/js/js/docs/index.md +5 -5
  61. package/rules/js/js/docs/tooling.md +0 -2
  62. package/rules/js/js/docs/utils_imports.md +0 -2
  63. package/rules/js/main.mdc +0 -31
  64. package/rules/js-bun-db/docs/main.md +0 -2
  65. package/rules/js-bun-db/js/docs/safety.md +18 -23
  66. package/rules/js-bun-db/js/safety.mjs +31 -3
  67. package/rules/js-bun-db/lib/bun-sql-scan.mjs +123 -0
  68. package/rules/js-bun-db/lib/docs/bun-sql-scan.md +37 -331
  69. package/rules/js-bun-db/main.mdc +1 -23
  70. package/rules/js-bun-redis/docs/main.md +0 -2
  71. package/rules/js-bun-redis/main.mdc +0 -5
  72. package/rules/js-mssql/docs/main.md +0 -2
  73. package/rules/js-mssql/main.mdc +0 -12
  74. package/rules/js-run/docs/main.md +0 -2
  75. package/rules/js-run/js/docs/runtime.md +15 -13
  76. package/rules/js-run/js/runtime.mjs +48 -4
  77. package/rules/js-run/main.mdc +0 -25
  78. package/rules/k8s/docs/main.md +0 -2
  79. package/rules/k8s/main.mdc +0 -45
  80. package/rules/nginx-default-tpl/docs/main.md +0 -2
  81. package/rules/nginx-default-tpl/main.mdc +0 -13
  82. package/rules/npm-module/docs/main.md +0 -2
  83. package/rules/npm-module/js/docs/header_doc_pointer.md +0 -2
  84. package/rules/npm-module/js/docs/rule_meta.md +0 -2
  85. package/rules/npm-module/js/docs/skill_meta.md +0 -2
  86. package/rules/npm-module/main.mdc +1 -15
  87. package/rules/php/docs/main.md +0 -2
  88. package/rules/php/js/docs/index.md +0 -1
  89. package/rules/php/main.mdc +1 -9
  90. package/rules/python/docs/main.md +0 -2
  91. package/rules/python/js/docs/index.md +0 -1
  92. package/rules/python/main.mdc +1 -13
  93. package/rules/rego/docs/main.md +0 -2
  94. package/rules/rego/js/docs/index.md +2 -2
  95. package/rules/rego/js/docs/tooling.md +0 -2
  96. package/rules/rego/js/tooling.mdc +14 -0
  97. package/rules/rego/main.mdc +0 -9
  98. package/rules/rego/policy/package_json/package_json.mdc +12 -0
  99. package/rules/release/docs/main.md +0 -2
  100. package/rules/release/main.mdc +2 -2
  101. package/rules/rust/docs/main.md +0 -2
  102. package/rules/rust/js/docs/index.md +0 -1
  103. package/rules/rust/main.mdc +1 -11
  104. package/rules/rust/policy/package_json/package_json.mdc +12 -0
  105. package/rules/security/docs/main.md +0 -2
  106. package/rules/security/js/docs/index.md +0 -1
  107. package/rules/security/main.mdc +0 -13
  108. package/rules/style/docs/main.md +0 -2
  109. package/rules/style/js/docs/index.md +2 -2
  110. package/rules/style/js/docs/tooling.md +0 -2
  111. package/rules/style/main.mdc +1 -23
  112. package/rules/tauri/docs/main.md +0 -2
  113. package/rules/tauri/main.mdc +1 -11
  114. package/rules/test/docs/main.md +0 -2
  115. package/rules/test/js/docs/no-console-store-restore.md +0 -2
  116. package/rules/test/js/docs/sandbox-aware-test.md +0 -2
  117. package/rules/test/js/docs/stryker_config.md +0 -2
  118. package/rules/test/js/docs/vitest-config-pool-forks.md +0 -2
  119. package/rules/test/main.mdc +1 -21
  120. package/rules/text/docs/main.md +0 -2
  121. package/rules/text/js/docs/cspell-fix.md +0 -2
  122. package/rules/text/js/docs/run-dotenv-linter.md +0 -2
  123. package/rules/text/js/docs/run-shellcheck.md +0 -2
  124. package/rules/text/js/docs/run-v8r.md +0 -2
  125. package/rules/text/main.mdc +0 -33
  126. package/rules/tool-surface/docs/main.md +0 -2
  127. package/rules/vue/docs/main.md +0 -2
  128. package/rules/vue/main.mdc +0 -22
  129. package/rules/worktree/docs/main.md +0 -2
  130. package/scripts/docs/auto-rules.md +0 -2
  131. package/scripts/docs/auto-skills.md +0 -2
  132. package/scripts/docs/hook.md +13 -12
  133. package/scripts/docs/post-tool-use-check.md +0 -2
  134. package/scripts/docs/sync-claude-config.md +1 -3
  135. package/scripts/docs/sync-setup-bun-deps-action.md +0 -2
  136. package/scripts/hook.mjs +3 -4
  137. package/scripts/lib/docs/check-mdc-template-refs.md +0 -2
  138. package/scripts/lib/docs/index.md +35 -35
  139. package/scripts/lib/docs/inline-template-links.md +6 -8
  140. package/scripts/lib/docs/list-project-rules-mdc.md +0 -2
  141. package/scripts/lib/docs/list-rule-ids.md +0 -2
  142. package/scripts/lib/docs/mirror-parity.md +8 -10
  143. package/scripts/lib/docs/read-n-cursor-config-lite.md +0 -2
  144. package/scripts/lib/docs/rule-meta.md +0 -2
  145. package/scripts/lib/docs/run-lint.md +0 -2
  146. package/scripts/lib/docs/run-rule-cli.md +0 -2
  147. package/scripts/lib/docs/run-rule.md +0 -2
  148. package/scripts/lib/docs/run-standard-lint.md +0 -2
  149. package/scripts/lib/docs/run-standard-rule.md +0 -2
  150. package/scripts/lib/docs/skill-meta.md +0 -2
  151. package/scripts/lib/docs/timing-summary.md +0 -2
  152. package/scripts/lib/docs/worktree-notice.md +0 -2
  153. package/scripts/lib/fix/docs/analyze-escalation.md +0 -2
  154. package/scripts/lib/fix/docs/index.md +10 -9
  155. package/scripts/lib/fix/docs/llm-worker.md +18 -8
  156. package/scripts/lib/fix/docs/orchestrator.md +44 -20
  157. package/scripts/lib/fix/docs/run-conformance-check.md +0 -2
  158. package/scripts/lib/fix/docs/t0.md +0 -2
  159. package/scripts/lib/fix/docs/verbose-block.md +27 -0
  160. package/scripts/lib/fix/llm-worker.mjs +75 -22
  161. package/scripts/lib/fix/orchestrator.mjs +13 -3
  162. package/scripts/lib/fix/verbose-block.mjs +82 -0
  163. package/scripts/lib/inline-template-links.mjs +32 -22
  164. package/scripts/lib/mirror-parity.mjs +2 -2
  165. package/scripts/sync-claude-config.mjs +7 -4
  166. package/scripts/utils/docs/resolve-js-root.md +0 -2
  167. package/skills/doc-files/SKILL.md +9 -24
  168. package/skills/llm-patch/SKILL.md +4 -4
@@ -8,8 +8,6 @@ docgen:
8
8
  score: 100
9
9
  ---
10
10
 
11
- ## Огляд
12
-
13
11
  Модуль сканує монорепозиторій для пошуку каталогів `utils` та аналізу їхнього вмісту. Аналіз файлів JS/TS у цих каталогах здійснюється на відповідність шаблону забороненого відносного імпорту, що базується на конфігурації, визначеній у `.n-cursor.json`. При виявленні порушень, система генерує повідомлення, позначене як (js.mdc).
14
12
 
15
13
  ## Поведінка
package/rules/js/main.mdc CHANGED
@@ -7,38 +7,7 @@ version: '1.30'
7
7
 
8
8
  **oxlint**, **ESLint**, **jscpd**, **knip**. Запуск — **`n-cursor lint js`** (локально; у CI — `--read-only`, без **`--fix`** для oxlint/eslint). Без **prettier** і **@nitra/prettier-config**.
9
9
 
10
- [js-file-extensions](./js/file-extensions.mdc)
11
-
12
- [js-package-json](./js/package-json.mdc)
13
-
14
- [js-eslint-config](./js/eslint-config.mdc)
15
-
16
- [js-oxlintrc](./js/oxlintrc.mdc)
17
-
18
- [js-extensions](./js/extensions.mdc)
19
-
20
- [js-jscpd](./js/jscpd.mdc)
21
-
22
- [js-knip](./js/knip.mdc)
23
-
24
- [js-dep-policy](./js/dep-policy.mdc)
25
-
26
- [js-lint-js-workflow](./js/lint-js-workflow.mdc)
27
-
28
- [js-utils-lib-structure](./js/utils-lib-structure.mdc)
29
-
30
- [js-for-in](./js/for-in.mdc)
31
-
32
- [js-tests](./js/tests.mdc)
33
-
34
10
  ## Швидкий gate через conftest
35
11
 
36
12
  Rego-пакети у `policy/` — запускаються `npx @nitra/cursor fix js` або `conftest`:
37
13
 
38
- [js-policy-package_json](./policy/package_json/package_json.mdc)
39
-
40
- [js-policy-jscpd](./policy/jscpd/jscpd.mdc)
41
-
42
- [js-policy-lint_js_yml](./policy/lint_js_yml/lint_js_yml.mdc)
43
-
44
- [js-policy-vscode_extensions](./policy/vscode_extensions/vscode_extensions.mdc)
@@ -8,8 +8,6 @@ docgen:
8
8
  score: 90
9
9
  ---
10
10
 
11
- ## Огляд
12
-
13
11
  Модуль виконує перевірку, застосовуючи правила та політики. При виклику публічної функції `run` він завантажує конфігурації, зокрема `meta.json`, застосовує білі списки та підбиває підсумки. Модуль є Read-only, тобто не пише у файлову систему чи базу даних. Під час роботи відбувається кешування даних у межах одного прогону. Результат перевірки визначає код виходу процесу.
14
12
 
15
13
  ## Поведінка
@@ -3,39 +3,34 @@ type: JS Module
3
3
  title: safety.mjs
4
4
  resource: npm/rules/js-bun-db/js/safety.mjs
5
5
  docgen:
6
- crc: eaeb52bc
6
+ crc: e828ee4a
7
+ model: omlx/gemma-4-e4b-it-OptiQ-4bit
7
8
  score: 100
8
9
  ---
9
10
 
10
- Огляд
11
+ Модуль сканує репозиторій для валідації безпечності використання Bun SQL. Він шукає всі файли `package.json` та JS/TS джерела, щоб перевірити відповідність патернів Bun SQL правилам, визначеним у конфігурації, що спирається на `package.json`.
11
12
 
12
- Файл перевіряє шляхи до файлів та залежностей, пов'язаних з Bun SQL та бібліотекою `pg`. Файл збирає метадані про використання `pg` та механізми LISTEN/NOTIFY. (js-bun-db.mdc)
13
+ Поведінка:
14
+
15
+ - Перевіряє відповідність патернів Bun SQL встановленим правилам (js-bun-db.mdc).
13
16
 
14
17
  ## Поведінка
15
18
 
16
- 1. Завантажити шляхи до файлів з коду
17
- 2. Просканувати джерела за патернами Bun SQL
18
- 3. Зібрати метадані про використання `pg` та LISTEN/NOTIFY
19
- 4. Перевірити залежності `pg` та використання LISTEN/NOTIFY
20
- 5. Перевірити імпорти `pg` та використання LISTEN/NOTIFY
21
- 6. Перевірити наявність `package.json` для залежностей `pg`
22
- 7. Перевірити використання `import { sql|SQL } from 'bun'`
23
- 8. Перевірити відсутність створення `new SQL` всередині функцій
24
- 9. Перевірити використання `sql.unsafe` без маркерів дозволу
25
- 10. Перевірити використання `sql.unsafe` з template-літералами
26
- 11. Перевірити використання `pg-leftover` викликів
27
- 12. Перевірити використання `findBunSqlUnsafeBunSqlDynamicSqlListInText`
28
- 13. Перевірити використання `findUnsafeBunSqlInListMissingEmptyGuardInText`
29
- 14. Перевірити використання `findPgFormatShimDefinitionInText`
30
- 15. Перевірити використання `findPgFormatLikeQueryWrapperInText`
31
- 16. Перевірити наявність використання `import { sql } from 'bun'`
19
+ 1. Викликається `check` для початку перевірки.
20
+ 2. Перевіряється наявність `package.json` у корені репозиторію. Якщо відсутній, перевірка припиняється.
21
+ 3. Завантажуються конфігураційні шляхи, які ігноруються під час обходу.
22
+ 4. Знаходяться всі файли `package.json` у репозиторії. Якщо жоден не знайдено, перевірка припиняється.
23
+ 5. Знаходяться всі JS/TS джерела для сканування патернів Bun SQL. Якщо жоден не знайдено, перевірка припиняється.
24
+ 6. Скануються знайдені JS/TS джерела на небезпечні патерни Bun SQL. Збираються метадані про використання Bun SQL та про використання бібліотеки `pg` (LISTEN/NOTIFY).
25
+ 7. Перевіряється залежність `pg` у знайдених `package.json` на відповідність правилу: якщо `pg` є залежністю, у коді має бути хоча б одне використання LISTEN/NOTIFY.
26
+ 8. Перевіряється кожен файл, що імпортує `pg`. Якщо у ньому немає використання LISTEN/NOTIFY, реєструється порушення.
27
+ 9. Після завершення сканування та перевірок, реєструються всі знайдені порушення, пов'язані з Bun SQL (наприклад, створення `new SQL` всередині функцій, використання `sql.unsafe` без маркерів, проблеми з динамічними списками тощо).
28
+ 10. Якщо всі перевірки пройдені успішно, реєструється повідомлення про відповідність правилу `js-bun-db.mdc`.
32
29
 
33
30
  ## Публічний API
34
31
 
35
- check — Перевіряє відповідність проєкту правилу js-bun-db.mdc
32
+ check — порівнює структуру проєкту з вимогами, описаними у js-bun-db.mdc
36
33
 
37
34
  ## Гарантії поведінки
38
35
 
39
- - Read-only: файл не виконує операцій запису у файлову систему.
40
- - Перехоплює помилки і не пропускає винятків назовні (fail-safe).
41
- - Не звертається до мережі.
36
+ - Read-only: не виконує операцій запису (ФС/БД).
@@ -9,10 +9,12 @@ import {
9
9
  findBunSqlPgLeftoverCallInText,
10
10
  findBunSqlUnsafeUseWithoutAllowMarkerInText,
11
11
  findBunSqlUnsafeWithInterpolatedTemplateInText,
12
+ findJsonStringifyBeforeJsonbInText,
12
13
  findPgFormatLikeQueryWrapperInText,
13
14
  findPgFormatShimDefinitionInText,
14
15
  findPgLibImportInText,
15
16
  findPgListenNotifyUsageInText,
17
+ findSqlArrayWithoutTypeArgInText,
16
18
  findUnsafeBunSqlDynamicSqlListInText,
17
19
  findUnsafeBunSqlInListMissingEmptyGuardInText,
18
20
  isBunSqlScanSourceFile,
@@ -83,7 +85,9 @@ async function scanSourcesForBunSqlPatterns(sourcePaths, repoRoot, reporter) {
83
85
  inListGuard: 0,
84
86
  pgLeftover: 0,
85
87
  pgFormatShim: 0,
86
- queryWrapper: 0
88
+ queryWrapper: 0,
89
+ jsonStringifyJsonb: 0,
90
+ sqlArrayNoType: 0
87
91
  }
88
92
  let hasBunSqlImport = false
89
93
  /** @type {{ rel: string, imports: { line: number, snippet: string }[], listenNotify: { line: number, snippet: string, kind: string }[] }[]} */
@@ -127,7 +131,7 @@ function collectPgUsageForFile(content, rel, pgUsage) {
127
131
  * @param {string} content вміст файлу
128
132
  * @param {string} rel posix-шлях відносно `repoRoot`
129
133
  * @param {(msg: string) => void} fail callback при помилці
130
- * @param {{ perRequest: number, unsafeCall: number, dynamicList: number, inListGuard: number, pgLeftover: number, pgFormatShim: number, queryWrapper: number }} counts акумулятори
134
+ * @param {{ perRequest: number, unsafeCall: number, unsafeTemplateInterp: number, dynamicList: number, inListGuard: number, pgLeftover: number, pgFormatShim: number, queryWrapper: number, jsonStringifyJsonb: number, sqlArrayNoType: number }} counts акумулятори
131
135
  * @returns {void}
132
136
  */
133
137
  function scanFileForBunSqlPatterns(content, rel, fail, counts) {
@@ -202,6 +206,22 @@ function scanFileForBunSqlPatterns(content, rel, fail, counts) {
202
206
  `sql\`...\${value}...\` (js-bun-db.mdc): ${v.snippet}`
203
207
  )
204
208
  }
209
+ for (const v of findJsonStringifyBeforeJsonbInText(content, rel)) {
210
+ counts.jsonStringifyJsonb++
211
+ fail(
212
+ `js-bun-db: ${rel}:${v.line} — JSON.stringify(...) перед ::jsonb зайвий: Bun SQL серіалізує ` +
213
+ `об'єкти/масиви у JSON автоматично, явний stringify призводить до подвійної серіалізації ` +
214
+ `(js-bun-db.mdc query-safety): ${v.snippet}`
215
+ )
216
+ }
217
+ for (const v of findSqlArrayWithoutTypeArgInText(content, rel)) {
218
+ counts.sqlArrayNoType++
219
+ fail(
220
+ `js-bun-db: ${rel}:${v.line} — sql.array(arr) без другого аргументу типу — ` +
221
+ `вкажи явний pg-тип: sql.array(arr, 'int8') / sql.array(arr, 'uuid') тощо ` +
222
+ `(js-bun-db.mdc sql-array): ${v.snippet}`
223
+ )
224
+ }
205
225
  }
206
226
 
207
227
  /**
@@ -338,7 +358,9 @@ export async function check(cwd = process.cwd()) {
338
358
  inListGuard,
339
359
  pgLeftover,
340
360
  pgFormatShim,
341
- queryWrapper
361
+ queryWrapper,
362
+ jsonStringifyJsonb,
363
+ sqlArrayNoType
342
364
  } = await scanSourcesForBunSqlPatterns(sourcePaths, repoRoot, reporter)
343
365
 
344
366
  const { pgDepFails, pgImportFails, pgDepsFound, listenNotifyEvidence } = await checkPgDependencyAndUsage(
@@ -396,6 +418,12 @@ export async function check(cwd = process.cwd()) {
396
418
  if (queryWrapper === 0) {
397
419
  pass('js-bun-db: немає query(text, params)-обгорток над unsafe(...) у файлах з Bun SQL')
398
420
  }
421
+ if (jsonStringifyJsonb === 0) {
422
+ pass('js-bun-db: немає JSON.stringify(...) перед ::jsonb — Bun SQL серіалізує автоматично')
423
+ }
424
+ if (sqlArrayNoType === 0) {
425
+ pass('js-bun-db: усі sql.array() мають явний аргумент типу')
426
+ }
399
427
 
400
428
  return reporter.getExitCode()
401
429
  }
@@ -927,3 +927,126 @@ function kindFromListenNotifyMatch(text) {
927
927
  export function isBunSqlScanSourceFile(relativePathPosix) {
928
928
  return SOURCE_FILE_RE.test(relativePathPosix) && !relativePathPosix.endsWith('.d.ts')
929
929
  }
930
+
931
+ // Імена відомих SQL-інстансів, для яких перевіряємо .array() без типу.
932
+ const SQL_INSTANCE_NAMES = new Set(['sql', 'pgWrite', 'pgRead'])
933
+
934
+ /**
935
+ * Чи це виклик `JSON.stringify(...)` (JSON.stringify через MemberExpression).
936
+ * @param {unknown} node AST node
937
+ * @returns {boolean}
938
+ */
939
+ function isJsonStringifyCall(node) {
940
+ if (!node || typeof node !== 'object' || node.type !== 'CallExpression') return false
941
+ const callee = node.callee
942
+ if (!callee || callee.type !== 'MemberExpression' || callee.computed) return false
943
+ const obj = callee.object
944
+ const prop = callee.property
945
+ return (
946
+ !!obj &&
947
+ obj.type === 'Identifier' &&
948
+ obj.name === 'JSON' &&
949
+ !!prop &&
950
+ prop.type === 'Identifier' &&
951
+ prop.name === 'stringify'
952
+ )
953
+ }
954
+
955
+ /**
956
+ * Знаходить виклики `JSON.stringify(...)::jsonb` всередині SQL template literal-ів.
957
+ * Bun SQL серіалізує об'єкти/масиви у JSON автоматично — явний `JSON.stringify`
958
+ * перед `::jsonb` призводить до подвійної серіалізації (js-bun-db.mdc).
959
+ * @param {string} content вихідний код
960
+ * @param {string} [virtualPath] шлях для вибору lang
961
+ * @returns {{ line: number, snippet: string }[]} список порушень
962
+ */
963
+ export function findJsonStringifyBeforeJsonbInText(content, virtualPath = 'scan.ts') {
964
+ const program = parseProgramOrNull(content, virtualPath)
965
+ if (!program) return []
966
+
967
+ /** @type {{ line: number, snippet: string }[]} */
968
+ const out = []
969
+
970
+ walkAstWithAncestors(program, [], node => {
971
+ let template = null
972
+ if (node.type === 'TemplateLiteral') template = node
973
+ else if (node.type === 'TaggedTemplateExpression') template = node.quasi
974
+ if (!template || typeof template !== 'object' || template.type !== 'TemplateLiteral') return
975
+
976
+ const expressions = template.expressions
977
+ const quasis = template.quasis
978
+ if (!Array.isArray(expressions) || !Array.isArray(quasis)) return
979
+
980
+ for (const [i, expr] of expressions.entries()) {
981
+ // Перевіряємо прямий JSON.stringify(...) і JSON.stringify всередині sql.array(...)
982
+ const isDirectStringify = isJsonStringifyCall(expr)
983
+ // sql.array(batch.map(r => JSON.stringify(...)), 'jsonb')
984
+ const hasSqlArrayStringify =
985
+ !isDirectStringify &&
986
+ expr.type === 'CallExpression' &&
987
+ Array.isArray(expr.arguments) &&
988
+ expr.arguments.some(arg => {
989
+ if (isJsonStringifyCall(arg)) return true
990
+ // .map(r => JSON.stringify(...))
991
+ if (arg.type === 'CallExpression' && Array.isArray(arg.arguments)) {
992
+ const cb = arg.arguments[0]
993
+ if (!cb) return false
994
+ const body = cb.type === 'ArrowFunctionExpression' || cb.type === 'FunctionExpression' ? cb.body : null
995
+ if (body && isJsonStringifyCall(body)) return true
996
+ }
997
+ return false
998
+ })
999
+
1000
+ if (!isDirectStringify && !hasSqlArrayStringify) continue
1001
+
1002
+ // Quasi після expr (quasi[i+1]) — текст одразу після закриваючого }
1003
+ const nextQuasi = quasis[i + 1]
1004
+ const rawAfter =
1005
+ nextQuasi && typeof nextQuasi === 'object' && nextQuasi.value && typeof nextQuasi.value.raw === 'string'
1006
+ ? nextQuasi.value.raw
1007
+ : ''
1008
+
1009
+ if (/^\s*::jsonb/u.test(rawAfter) || hasSqlArrayStringify) {
1010
+ out.push({
1011
+ line: offsetToLine(content, expr.start),
1012
+ snippet: normalizeSnippet(content.slice(expr.start, expr.end))
1013
+ })
1014
+ }
1015
+ }
1016
+ })
1017
+ return out
1018
+ }
1019
+
1020
+ /**
1021
+ * Знаходить виклики `sql.array(arr)` / `pgWrite.array(arr)` / `pgRead.array(arr)` без
1022
+ * обов'язкового другого аргументу (типу pg-елемента). Без типу Bun не може вивести
1023
+ * pg-тип, що призводить до mismatch (js-bun-db.mdc).
1024
+ * @param {string} content вихідний код
1025
+ * @param {string} [virtualPath] шлях для вибору lang
1026
+ * @returns {{ line: number, snippet: string }[]} список порушень
1027
+ */
1028
+ export function findSqlArrayWithoutTypeArgInText(content, virtualPath = 'scan.ts') {
1029
+ const program = parseProgramOrNull(content, virtualPath)
1030
+ if (!program) return []
1031
+
1032
+ /** @type {{ line: number, snippet: string }[]} */
1033
+ const out = []
1034
+
1035
+ walkAstWithAncestors(program, [], node => {
1036
+ if (!node || node.type !== 'CallExpression') return
1037
+ const callee = node.callee
1038
+ if (!callee || callee.type !== 'MemberExpression' || callee.computed) return
1039
+ const prop = callee.property
1040
+ if (!prop || prop.type !== 'Identifier' || prop.name !== 'array') return
1041
+ const obj = callee.object
1042
+ if (!obj || obj.type !== 'Identifier' || !SQL_INSTANCE_NAMES.has(obj.name)) return
1043
+ const args = node.arguments
1044
+ if (!Array.isArray(args) || args.length !== 1) return
1045
+
1046
+ out.push({
1047
+ line: offsetToLine(content, node.start),
1048
+ snippet: normalizeSnippet(content.slice(node.start, node.end))
1049
+ })
1050
+ })
1051
+ return out
1052
+ }