@shapecraft/core 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (584) hide show
  1. package/dist/cjs/common/array.d.ts +3 -0
  2. package/dist/cjs/common/array.d.ts.map +1 -0
  3. package/dist/cjs/common/array.js +17 -0
  4. package/dist/cjs/common/array.js.map +1 -0
  5. package/dist/cjs/common/index.d.ts +6 -0
  6. package/dist/cjs/common/index.d.ts.map +1 -0
  7. package/dist/cjs/common/index.js +22 -0
  8. package/dist/cjs/common/index.js.map +1 -0
  9. package/dist/cjs/common/is.d.ts +5 -0
  10. package/dist/cjs/common/is.d.ts.map +1 -0
  11. package/dist/cjs/common/is.js +36 -0
  12. package/dist/cjs/common/is.js.map +1 -0
  13. package/dist/cjs/common/object.d.ts +4 -0
  14. package/dist/cjs/common/object.d.ts.map +1 -0
  15. package/dist/cjs/common/object.js +20 -0
  16. package/dist/cjs/common/object.js.map +1 -0
  17. package/dist/cjs/common/phantom.d.ts +2 -0
  18. package/dist/cjs/common/phantom.d.ts.map +1 -0
  19. package/dist/cjs/common/phantom.js +3 -0
  20. package/dist/cjs/common/phantom.js.map +1 -0
  21. package/dist/cjs/common/result.d.ts +22 -0
  22. package/dist/cjs/common/result.d.ts.map +1 -0
  23. package/dist/cjs/common/result.js +28 -0
  24. package/dist/cjs/common/result.js.map +1 -0
  25. package/dist/cjs/common/string.d.ts +3 -0
  26. package/dist/cjs/common/string.d.ts.map +1 -0
  27. package/dist/cjs/common/string.js +33 -0
  28. package/dist/cjs/common/string.js.map +1 -0
  29. package/dist/cjs/common/types.d.ts +14 -0
  30. package/dist/cjs/common/types.d.ts.map +1 -0
  31. package/dist/cjs/common/types.js +3 -0
  32. package/dist/cjs/common/types.js.map +1 -0
  33. package/dist/cjs/index.d.ts +2 -0
  34. package/dist/cjs/index.d.ts.map +1 -0
  35. package/dist/cjs/index.js +18 -0
  36. package/dist/cjs/index.js.map +1 -0
  37. package/dist/cjs/shape/annotate.d.ts +57 -0
  38. package/dist/cjs/shape/annotate.d.ts.map +1 -0
  39. package/dist/cjs/shape/annotate.js +37 -0
  40. package/dist/cjs/shape/annotate.js.map +1 -0
  41. package/dist/cjs/shape/annotation.d.ts +39 -0
  42. package/dist/cjs/shape/annotation.d.ts.map +1 -0
  43. package/dist/cjs/shape/annotation.js +3 -0
  44. package/dist/cjs/shape/annotation.js.map +1 -0
  45. package/dist/cjs/shape/base.d.ts +33 -0
  46. package/dist/cjs/shape/base.d.ts.map +1 -0
  47. package/dist/cjs/shape/base.js +26 -0
  48. package/dist/cjs/shape/base.js.map +1 -0
  49. package/dist/cjs/shape/builder.d.ts +168 -0
  50. package/dist/cjs/shape/builder.d.ts.map +1 -0
  51. package/dist/cjs/shape/builder.js +156 -0
  52. package/dist/cjs/shape/builder.js.map +1 -0
  53. package/dist/cjs/shape/error.d.ts +5 -0
  54. package/dist/cjs/shape/error.d.ts.map +1 -0
  55. package/dist/cjs/shape/error.js +3 -0
  56. package/dist/cjs/shape/error.js.map +1 -0
  57. package/dist/cjs/shape/index.d.ts +4 -0
  58. package/dist/cjs/shape/index.d.ts.map +1 -0
  59. package/dist/cjs/shape/index.js +20 -0
  60. package/dist/cjs/shape/index.js.map +1 -0
  61. package/dist/cjs/shape/number.d.ts +8 -0
  62. package/dist/cjs/shape/number.d.ts.map +1 -0
  63. package/dist/cjs/shape/number.js +123 -0
  64. package/dist/cjs/shape/number.js.map +1 -0
  65. package/dist/cjs/shape/shape.d.ts +150 -0
  66. package/dist/cjs/shape/shape.d.ts.map +1 -0
  67. package/dist/cjs/shape/shape.js +158 -0
  68. package/dist/cjs/shape/shape.js.map +1 -0
  69. package/dist/cjs/shape/tags.d.ts +4 -0
  70. package/dist/cjs/shape/tags.d.ts.map +1 -0
  71. package/dist/cjs/shape/tags.js +3 -0
  72. package/dist/cjs/shape/tags.js.map +1 -0
  73. package/dist/cjs/shape/transforms/index.d.ts +4 -0
  74. package/dist/cjs/shape/transforms/index.d.ts.map +1 -0
  75. package/dist/cjs/shape/transforms/index.js +20 -0
  76. package/dist/cjs/shape/transforms/index.js.map +1 -0
  77. package/dist/cjs/shape/transforms/json-schema/index.d.ts +3 -0
  78. package/dist/cjs/shape/transforms/json-schema/index.d.ts.map +1 -0
  79. package/dist/cjs/shape/transforms/json-schema/index.js +19 -0
  80. package/dist/cjs/shape/transforms/json-schema/index.js.map +1 -0
  81. package/dist/cjs/shape/transforms/json-schema/transform.d.ts +5 -0
  82. package/dist/cjs/shape/transforms/json-schema/transform.d.ts.map +1 -0
  83. package/dist/cjs/shape/transforms/json-schema/transform.js +739 -0
  84. package/dist/cjs/shape/transforms/json-schema/transform.js.map +1 -0
  85. package/dist/cjs/shape/transforms/json-schema/types.d.ts +87 -0
  86. package/dist/cjs/shape/transforms/json-schema/types.d.ts.map +1 -0
  87. package/dist/cjs/shape/transforms/json-schema/types.js +3 -0
  88. package/dist/cjs/shape/transforms/json-schema/types.js.map +1 -0
  89. package/dist/cjs/shape/transforms/sql/dialects/dialect.d.ts +51 -0
  90. package/dist/cjs/shape/transforms/sql/dialects/dialect.d.ts.map +1 -0
  91. package/dist/cjs/shape/transforms/sql/dialects/dialect.js +8 -0
  92. package/dist/cjs/shape/transforms/sql/dialects/dialect.js.map +1 -0
  93. package/dist/cjs/shape/transforms/sql/dialects/index.d.ts +7 -0
  94. package/dist/cjs/shape/transforms/sql/dialects/index.d.ts.map +1 -0
  95. package/dist/cjs/shape/transforms/sql/dialects/index.js +19 -0
  96. package/dist/cjs/shape/transforms/sql/dialects/index.js.map +1 -0
  97. package/dist/cjs/shape/transforms/sql/dialects/postgres.d.ts +8 -0
  98. package/dist/cjs/shape/transforms/sql/dialects/postgres.d.ts.map +1 -0
  99. package/dist/cjs/shape/transforms/sql/dialects/postgres.js +374 -0
  100. package/dist/cjs/shape/transforms/sql/dialects/postgres.js.map +1 -0
  101. package/dist/cjs/shape/transforms/sql/dialects/sqlite.d.ts +3 -0
  102. package/dist/cjs/shape/transforms/sql/dialects/sqlite.d.ts.map +1 -0
  103. package/dist/cjs/shape/transforms/sql/dialects/sqlite.js +318 -0
  104. package/dist/cjs/shape/transforms/sql/dialects/sqlite.js.map +1 -0
  105. package/dist/cjs/shape/transforms/sql/from-sql.d.ts +5 -0
  106. package/dist/cjs/shape/transforms/sql/from-sql.d.ts.map +1 -0
  107. package/dist/cjs/shape/transforms/sql/from-sql.js +201 -0
  108. package/dist/cjs/shape/transforms/sql/from-sql.js.map +1 -0
  109. package/dist/cjs/shape/transforms/sql/index.d.ts +4 -0
  110. package/dist/cjs/shape/transforms/sql/index.d.ts.map +1 -0
  111. package/dist/cjs/shape/transforms/sql/index.js +20 -0
  112. package/dist/cjs/shape/transforms/sql/index.js.map +1 -0
  113. package/dist/cjs/shape/transforms/sql/options.d.ts +6 -0
  114. package/dist/cjs/shape/transforms/sql/options.d.ts.map +1 -0
  115. package/dist/cjs/shape/transforms/sql/options.js +6 -0
  116. package/dist/cjs/shape/transforms/sql/options.js.map +1 -0
  117. package/dist/cjs/shape/transforms/sql/parser/check-decoder.d.ts +25 -0
  118. package/dist/cjs/shape/transforms/sql/parser/check-decoder.d.ts.map +1 -0
  119. package/dist/cjs/shape/transforms/sql/parser/check-decoder.js +417 -0
  120. package/dist/cjs/shape/transforms/sql/parser/check-decoder.js.map +1 -0
  121. package/dist/cjs/shape/transforms/sql/parser/create-domain.d.ts +11 -0
  122. package/dist/cjs/shape/transforms/sql/parser/create-domain.d.ts.map +1 -0
  123. package/dist/cjs/shape/transforms/sql/parser/create-domain.js +95 -0
  124. package/dist/cjs/shape/transforms/sql/parser/create-domain.js.map +1 -0
  125. package/dist/cjs/shape/transforms/sql/parser/create-table.d.ts +28 -0
  126. package/dist/cjs/shape/transforms/sql/parser/create-table.d.ts.map +1 -0
  127. package/dist/cjs/shape/transforms/sql/parser/create-table.js +744 -0
  128. package/dist/cjs/shape/transforms/sql/parser/create-table.js.map +1 -0
  129. package/dist/cjs/shape/transforms/sql/parser/create-type.d.ts +23 -0
  130. package/dist/cjs/shape/transforms/sql/parser/create-type.d.ts.map +1 -0
  131. package/dist/cjs/shape/transforms/sql/parser/create-type.js +76 -0
  132. package/dist/cjs/shape/transforms/sql/parser/create-type.js.map +1 -0
  133. package/dist/cjs/shape/transforms/sql/parser/cursor.d.ts +32 -0
  134. package/dist/cjs/shape/transforms/sql/parser/cursor.d.ts.map +1 -0
  135. package/dist/cjs/shape/transforms/sql/parser/cursor.js +168 -0
  136. package/dist/cjs/shape/transforms/sql/parser/cursor.js.map +1 -0
  137. package/dist/cjs/shape/transforms/sql/parser/default-decoder.d.ts +10 -0
  138. package/dist/cjs/shape/transforms/sql/parser/default-decoder.d.ts.map +1 -0
  139. package/dist/cjs/shape/transforms/sql/parser/default-decoder.js +125 -0
  140. package/dist/cjs/shape/transforms/sql/parser/default-decoder.js.map +1 -0
  141. package/dist/cjs/shape/transforms/sql/parser/lexer.d.ts +14 -0
  142. package/dist/cjs/shape/transforms/sql/parser/lexer.d.ts.map +1 -0
  143. package/dist/cjs/shape/transforms/sql/parser/lexer.js +259 -0
  144. package/dist/cjs/shape/transforms/sql/parser/lexer.js.map +1 -0
  145. package/dist/cjs/shape/transforms/sql/parser/pg-types.d.ts +21 -0
  146. package/dist/cjs/shape/transforms/sql/parser/pg-types.d.ts.map +1 -0
  147. package/dist/cjs/shape/transforms/sql/parser/pg-types.js +213 -0
  148. package/dist/cjs/shape/transforms/sql/parser/pg-types.js.map +1 -0
  149. package/dist/cjs/shape/transforms/sql/parser/sqlite-types.d.ts +8 -0
  150. package/dist/cjs/shape/transforms/sql/parser/sqlite-types.d.ts.map +1 -0
  151. package/dist/cjs/shape/transforms/sql/parser/sqlite-types.js +93 -0
  152. package/dist/cjs/shape/transforms/sql/parser/sqlite-types.js.map +1 -0
  153. package/dist/cjs/shape/transforms/sql/parser/statements.d.ts +9 -0
  154. package/dist/cjs/shape/transforms/sql/parser/statements.d.ts.map +1 -0
  155. package/dist/cjs/shape/transforms/sql/parser/statements.js +158 -0
  156. package/dist/cjs/shape/transforms/sql/parser/statements.js.map +1 -0
  157. package/dist/cjs/shape/transforms/sql/parser/type-spec.d.ts +12 -0
  158. package/dist/cjs/shape/transforms/sql/parser/type-spec.d.ts.map +1 -0
  159. package/dist/cjs/shape/transforms/sql/parser/type-spec.js +136 -0
  160. package/dist/cjs/shape/transforms/sql/parser/type-spec.js.map +1 -0
  161. package/dist/cjs/shape/transforms/sql/transform.d.ts +5 -0
  162. package/dist/cjs/shape/transforms/sql/transform.d.ts.map +1 -0
  163. package/dist/cjs/shape/transforms/sql/transform.js +262 -0
  164. package/dist/cjs/shape/transforms/sql/transform.js.map +1 -0
  165. package/dist/cjs/shape/transforms/typescript/index.d.ts +2 -0
  166. package/dist/cjs/shape/transforms/typescript/index.d.ts.map +1 -0
  167. package/dist/cjs/shape/transforms/typescript/index.js +18 -0
  168. package/dist/cjs/shape/transforms/typescript/index.js.map +1 -0
  169. package/dist/cjs/shape/transforms/typescript/transform.d.ts +3 -0
  170. package/dist/cjs/shape/transforms/typescript/transform.d.ts.map +1 -0
  171. package/dist/cjs/shape/transforms/typescript/transform.js +190 -0
  172. package/dist/cjs/shape/transforms/typescript/transform.js.map +1 -0
  173. package/dist/cjs/shape/validate.d.ts +7 -0
  174. package/dist/cjs/shape/validate.d.ts.map +1 -0
  175. package/dist/cjs/shape/validate.js +360 -0
  176. package/dist/cjs/shape/validate.js.map +1 -0
  177. package/dist/esm/common/array.d.ts +3 -0
  178. package/dist/esm/common/array.d.ts.map +1 -0
  179. package/dist/esm/common/array.js +12 -0
  180. package/dist/esm/common/array.js.map +1 -0
  181. package/dist/esm/common/index.d.ts +6 -0
  182. package/dist/esm/common/index.d.ts.map +1 -0
  183. package/dist/esm/common/index.js +6 -0
  184. package/dist/esm/common/index.js.map +1 -0
  185. package/dist/esm/common/is.d.ts +5 -0
  186. package/dist/esm/common/is.d.ts.map +1 -0
  187. package/dist/esm/common/is.js +29 -0
  188. package/dist/esm/common/is.js.map +1 -0
  189. package/dist/esm/common/object.d.ts +4 -0
  190. package/dist/esm/common/object.d.ts.map +1 -0
  191. package/dist/esm/common/object.js +15 -0
  192. package/dist/esm/common/object.js.map +1 -0
  193. package/dist/esm/common/phantom.d.ts +2 -0
  194. package/dist/esm/common/phantom.d.ts.map +1 -0
  195. package/dist/esm/common/phantom.js +2 -0
  196. package/dist/esm/common/phantom.js.map +1 -0
  197. package/dist/esm/common/result.d.ts +22 -0
  198. package/dist/esm/common/result.d.ts.map +1 -0
  199. package/dist/esm/common/result.js +25 -0
  200. package/dist/esm/common/result.js.map +1 -0
  201. package/dist/esm/common/string.d.ts +3 -0
  202. package/dist/esm/common/string.d.ts.map +1 -0
  203. package/dist/esm/common/string.js +28 -0
  204. package/dist/esm/common/string.js.map +1 -0
  205. package/dist/esm/common/types.d.ts +14 -0
  206. package/dist/esm/common/types.d.ts.map +1 -0
  207. package/dist/esm/common/types.js +2 -0
  208. package/dist/esm/common/types.js.map +1 -0
  209. package/dist/esm/index.d.ts +2 -0
  210. package/dist/esm/index.d.ts.map +1 -0
  211. package/dist/esm/index.js +2 -0
  212. package/dist/esm/index.js.map +1 -0
  213. package/dist/esm/shape/annotate.d.ts +57 -0
  214. package/dist/esm/shape/annotate.d.ts.map +1 -0
  215. package/dist/esm/shape/annotate.js +34 -0
  216. package/dist/esm/shape/annotate.js.map +1 -0
  217. package/dist/esm/shape/annotation.d.ts +39 -0
  218. package/dist/esm/shape/annotation.d.ts.map +1 -0
  219. package/dist/esm/shape/annotation.js +2 -0
  220. package/dist/esm/shape/annotation.js.map +1 -0
  221. package/dist/esm/shape/base.d.ts +33 -0
  222. package/dist/esm/shape/base.d.ts.map +1 -0
  223. package/dist/esm/shape/base.js +20 -0
  224. package/dist/esm/shape/base.js.map +1 -0
  225. package/dist/esm/shape/builder.d.ts +168 -0
  226. package/dist/esm/shape/builder.d.ts.map +1 -0
  227. package/dist/esm/shape/builder.js +146 -0
  228. package/dist/esm/shape/builder.js.map +1 -0
  229. package/dist/esm/shape/error.d.ts +5 -0
  230. package/dist/esm/shape/error.d.ts.map +1 -0
  231. package/dist/esm/shape/error.js +2 -0
  232. package/dist/esm/shape/error.js.map +1 -0
  233. package/dist/esm/shape/index.d.ts +4 -0
  234. package/dist/esm/shape/index.d.ts.map +1 -0
  235. package/dist/esm/shape/index.js +4 -0
  236. package/dist/esm/shape/index.js.map +1 -0
  237. package/dist/esm/shape/number.d.ts +8 -0
  238. package/dist/esm/shape/number.d.ts.map +1 -0
  239. package/dist/esm/shape/number.js +114 -0
  240. package/dist/esm/shape/number.js.map +1 -0
  241. package/dist/esm/shape/shape.d.ts +150 -0
  242. package/dist/esm/shape/shape.d.ts.map +1 -0
  243. package/dist/esm/shape/shape.js +121 -0
  244. package/dist/esm/shape/shape.js.map +1 -0
  245. package/dist/esm/shape/tags.d.ts +4 -0
  246. package/dist/esm/shape/tags.d.ts.map +1 -0
  247. package/dist/esm/shape/tags.js +2 -0
  248. package/dist/esm/shape/tags.js.map +1 -0
  249. package/dist/esm/shape/transforms/index.d.ts +4 -0
  250. package/dist/esm/shape/transforms/index.d.ts.map +1 -0
  251. package/dist/esm/shape/transforms/index.js +4 -0
  252. package/dist/esm/shape/transforms/index.js.map +1 -0
  253. package/dist/esm/shape/transforms/json-schema/index.d.ts +3 -0
  254. package/dist/esm/shape/transforms/json-schema/index.d.ts.map +1 -0
  255. package/dist/esm/shape/transforms/json-schema/index.js +3 -0
  256. package/dist/esm/shape/transforms/json-schema/index.js.map +1 -0
  257. package/dist/esm/shape/transforms/json-schema/transform.d.ts +5 -0
  258. package/dist/esm/shape/transforms/json-schema/transform.d.ts.map +1 -0
  259. package/dist/esm/shape/transforms/json-schema/transform.js +734 -0
  260. package/dist/esm/shape/transforms/json-schema/transform.js.map +1 -0
  261. package/dist/esm/shape/transforms/json-schema/types.d.ts +87 -0
  262. package/dist/esm/shape/transforms/json-schema/types.d.ts.map +1 -0
  263. package/dist/esm/shape/transforms/json-schema/types.js +2 -0
  264. package/dist/esm/shape/transforms/json-schema/types.js.map +1 -0
  265. package/dist/esm/shape/transforms/sql/dialects/dialect.d.ts +51 -0
  266. package/dist/esm/shape/transforms/sql/dialects/dialect.d.ts.map +1 -0
  267. package/dist/esm/shape/transforms/sql/dialects/dialect.js +3 -0
  268. package/dist/esm/shape/transforms/sql/dialects/dialect.js.map +1 -0
  269. package/dist/esm/shape/transforms/sql/dialects/index.d.ts +7 -0
  270. package/dist/esm/shape/transforms/sql/dialects/index.d.ts.map +1 -0
  271. package/dist/esm/shape/transforms/sql/dialects/index.js +12 -0
  272. package/dist/esm/shape/transforms/sql/dialects/index.js.map +1 -0
  273. package/dist/esm/shape/transforms/sql/dialects/postgres.d.ts +8 -0
  274. package/dist/esm/shape/transforms/sql/dialects/postgres.d.ts.map +1 -0
  275. package/dist/esm/shape/transforms/sql/dialects/postgres.js +370 -0
  276. package/dist/esm/shape/transforms/sql/dialects/postgres.js.map +1 -0
  277. package/dist/esm/shape/transforms/sql/dialects/sqlite.d.ts +3 -0
  278. package/dist/esm/shape/transforms/sql/dialects/sqlite.d.ts.map +1 -0
  279. package/dist/esm/shape/transforms/sql/dialects/sqlite.js +315 -0
  280. package/dist/esm/shape/transforms/sql/dialects/sqlite.js.map +1 -0
  281. package/dist/esm/shape/transforms/sql/from-sql.d.ts +5 -0
  282. package/dist/esm/shape/transforms/sql/from-sql.d.ts.map +1 -0
  283. package/dist/esm/shape/transforms/sql/from-sql.js +197 -0
  284. package/dist/esm/shape/transforms/sql/from-sql.js.map +1 -0
  285. package/dist/esm/shape/transforms/sql/index.d.ts +4 -0
  286. package/dist/esm/shape/transforms/sql/index.d.ts.map +1 -0
  287. package/dist/esm/shape/transforms/sql/index.js +4 -0
  288. package/dist/esm/shape/transforms/sql/index.js.map +1 -0
  289. package/dist/esm/shape/transforms/sql/options.d.ts +6 -0
  290. package/dist/esm/shape/transforms/sql/options.d.ts.map +1 -0
  291. package/dist/esm/shape/transforms/sql/options.js +2 -0
  292. package/dist/esm/shape/transforms/sql/options.js.map +1 -0
  293. package/dist/esm/shape/transforms/sql/parser/check-decoder.d.ts +25 -0
  294. package/dist/esm/shape/transforms/sql/parser/check-decoder.d.ts.map +1 -0
  295. package/dist/esm/shape/transforms/sql/parser/check-decoder.js +412 -0
  296. package/dist/esm/shape/transforms/sql/parser/check-decoder.js.map +1 -0
  297. package/dist/esm/shape/transforms/sql/parser/create-domain.d.ts +11 -0
  298. package/dist/esm/shape/transforms/sql/parser/create-domain.d.ts.map +1 -0
  299. package/dist/esm/shape/transforms/sql/parser/create-domain.js +91 -0
  300. package/dist/esm/shape/transforms/sql/parser/create-domain.js.map +1 -0
  301. package/dist/esm/shape/transforms/sql/parser/create-table.d.ts +28 -0
  302. package/dist/esm/shape/transforms/sql/parser/create-table.d.ts.map +1 -0
  303. package/dist/esm/shape/transforms/sql/parser/create-table.js +740 -0
  304. package/dist/esm/shape/transforms/sql/parser/create-table.js.map +1 -0
  305. package/dist/esm/shape/transforms/sql/parser/create-type.d.ts +23 -0
  306. package/dist/esm/shape/transforms/sql/parser/create-type.d.ts.map +1 -0
  307. package/dist/esm/shape/transforms/sql/parser/create-type.js +72 -0
  308. package/dist/esm/shape/transforms/sql/parser/create-type.js.map +1 -0
  309. package/dist/esm/shape/transforms/sql/parser/cursor.d.ts +32 -0
  310. package/dist/esm/shape/transforms/sql/parser/cursor.d.ts.map +1 -0
  311. package/dist/esm/shape/transforms/sql/parser/cursor.js +163 -0
  312. package/dist/esm/shape/transforms/sql/parser/cursor.js.map +1 -0
  313. package/dist/esm/shape/transforms/sql/parser/default-decoder.d.ts +10 -0
  314. package/dist/esm/shape/transforms/sql/parser/default-decoder.d.ts.map +1 -0
  315. package/dist/esm/shape/transforms/sql/parser/default-decoder.js +121 -0
  316. package/dist/esm/shape/transforms/sql/parser/default-decoder.js.map +1 -0
  317. package/dist/esm/shape/transforms/sql/parser/lexer.d.ts +14 -0
  318. package/dist/esm/shape/transforms/sql/parser/lexer.d.ts.map +1 -0
  319. package/dist/esm/shape/transforms/sql/parser/lexer.js +253 -0
  320. package/dist/esm/shape/transforms/sql/parser/lexer.js.map +1 -0
  321. package/dist/esm/shape/transforms/sql/parser/pg-types.d.ts +21 -0
  322. package/dist/esm/shape/transforms/sql/parser/pg-types.d.ts.map +1 -0
  323. package/dist/esm/shape/transforms/sql/parser/pg-types.js +208 -0
  324. package/dist/esm/shape/transforms/sql/parser/pg-types.js.map +1 -0
  325. package/dist/esm/shape/transforms/sql/parser/sqlite-types.d.ts +8 -0
  326. package/dist/esm/shape/transforms/sql/parser/sqlite-types.d.ts.map +1 -0
  327. package/dist/esm/shape/transforms/sql/parser/sqlite-types.js +89 -0
  328. package/dist/esm/shape/transforms/sql/parser/sqlite-types.js.map +1 -0
  329. package/dist/esm/shape/transforms/sql/parser/statements.d.ts +9 -0
  330. package/dist/esm/shape/transforms/sql/parser/statements.d.ts.map +1 -0
  331. package/dist/esm/shape/transforms/sql/parser/statements.js +153 -0
  332. package/dist/esm/shape/transforms/sql/parser/statements.js.map +1 -0
  333. package/dist/esm/shape/transforms/sql/parser/type-spec.d.ts +12 -0
  334. package/dist/esm/shape/transforms/sql/parser/type-spec.d.ts.map +1 -0
  335. package/dist/esm/shape/transforms/sql/parser/type-spec.js +130 -0
  336. package/dist/esm/shape/transforms/sql/parser/type-spec.js.map +1 -0
  337. package/dist/esm/shape/transforms/sql/transform.d.ts +5 -0
  338. package/dist/esm/shape/transforms/sql/transform.d.ts.map +1 -0
  339. package/dist/esm/shape/transforms/sql/transform.js +258 -0
  340. package/dist/esm/shape/transforms/sql/transform.js.map +1 -0
  341. package/dist/esm/shape/transforms/typescript/index.d.ts +2 -0
  342. package/dist/esm/shape/transforms/typescript/index.d.ts.map +1 -0
  343. package/dist/esm/shape/transforms/typescript/index.js +2 -0
  344. package/dist/esm/shape/transforms/typescript/index.js.map +1 -0
  345. package/dist/esm/shape/transforms/typescript/transform.d.ts +3 -0
  346. package/dist/esm/shape/transforms/typescript/transform.d.ts.map +1 -0
  347. package/dist/esm/shape/transforms/typescript/transform.js +186 -0
  348. package/dist/esm/shape/transforms/typescript/transform.js.map +1 -0
  349. package/dist/esm/shape/validate.d.ts +7 -0
  350. package/dist/esm/shape/validate.d.ts.map +1 -0
  351. package/dist/esm/shape/validate.js +355 -0
  352. package/dist/esm/shape/validate.js.map +1 -0
  353. package/dist/esm/src/common/array.d.ts +3 -0
  354. package/dist/esm/src/common/array.d.ts.map +1 -0
  355. package/dist/esm/src/common/array.js +12 -0
  356. package/dist/esm/src/common/array.js.map +1 -0
  357. package/dist/esm/src/common/index.d.ts +6 -0
  358. package/dist/esm/src/common/index.d.ts.map +1 -0
  359. package/dist/esm/src/common/index.js +6 -0
  360. package/dist/esm/src/common/index.js.map +1 -0
  361. package/dist/esm/src/common/is.d.ts +5 -0
  362. package/dist/esm/src/common/is.d.ts.map +1 -0
  363. package/dist/esm/src/common/is.js +29 -0
  364. package/dist/esm/src/common/is.js.map +1 -0
  365. package/dist/esm/src/common/object.d.ts +4 -0
  366. package/dist/esm/src/common/object.d.ts.map +1 -0
  367. package/dist/esm/src/common/object.js +15 -0
  368. package/dist/esm/src/common/object.js.map +1 -0
  369. package/dist/esm/src/common/phantom.d.ts +2 -0
  370. package/dist/esm/src/common/phantom.d.ts.map +1 -0
  371. package/dist/esm/src/common/phantom.js +2 -0
  372. package/dist/esm/src/common/phantom.js.map +1 -0
  373. package/dist/esm/src/common/result.d.ts +22 -0
  374. package/dist/esm/src/common/result.d.ts.map +1 -0
  375. package/dist/esm/src/common/result.js +25 -0
  376. package/dist/esm/src/common/result.js.map +1 -0
  377. package/dist/esm/src/common/string.d.ts +3 -0
  378. package/dist/esm/src/common/string.d.ts.map +1 -0
  379. package/dist/esm/src/common/string.js +28 -0
  380. package/dist/esm/src/common/string.js.map +1 -0
  381. package/dist/esm/src/common/types.d.ts +14 -0
  382. package/dist/esm/src/common/types.d.ts.map +1 -0
  383. package/dist/esm/src/common/types.js +2 -0
  384. package/dist/esm/src/common/types.js.map +1 -0
  385. package/dist/esm/src/index.d.ts +2 -0
  386. package/dist/esm/src/index.d.ts.map +1 -0
  387. package/dist/esm/src/index.js +2 -0
  388. package/dist/esm/src/index.js.map +1 -0
  389. package/dist/esm/src/shape/annotate.d.ts +57 -0
  390. package/dist/esm/src/shape/annotate.d.ts.map +1 -0
  391. package/dist/esm/src/shape/annotate.js +34 -0
  392. package/dist/esm/src/shape/annotate.js.map +1 -0
  393. package/dist/esm/src/shape/annotation.d.ts +39 -0
  394. package/dist/esm/src/shape/annotation.d.ts.map +1 -0
  395. package/dist/esm/src/shape/annotation.js +2 -0
  396. package/dist/esm/src/shape/annotation.js.map +1 -0
  397. package/dist/esm/src/shape/base.d.ts +33 -0
  398. package/dist/esm/src/shape/base.d.ts.map +1 -0
  399. package/dist/esm/src/shape/base.js +20 -0
  400. package/dist/esm/src/shape/base.js.map +1 -0
  401. package/dist/esm/src/shape/builder.d.ts +168 -0
  402. package/dist/esm/src/shape/builder.d.ts.map +1 -0
  403. package/dist/esm/src/shape/builder.js +146 -0
  404. package/dist/esm/src/shape/builder.js.map +1 -0
  405. package/dist/esm/src/shape/error.d.ts +5 -0
  406. package/dist/esm/src/shape/error.d.ts.map +1 -0
  407. package/dist/esm/src/shape/error.js +2 -0
  408. package/dist/esm/src/shape/error.js.map +1 -0
  409. package/dist/esm/src/shape/index.d.ts +4 -0
  410. package/dist/esm/src/shape/index.d.ts.map +1 -0
  411. package/dist/esm/src/shape/index.js +4 -0
  412. package/dist/esm/src/shape/index.js.map +1 -0
  413. package/dist/esm/src/shape/number.d.ts +8 -0
  414. package/dist/esm/src/shape/number.d.ts.map +1 -0
  415. package/dist/esm/src/shape/number.js +114 -0
  416. package/dist/esm/src/shape/number.js.map +1 -0
  417. package/dist/esm/src/shape/shape.d.ts +150 -0
  418. package/dist/esm/src/shape/shape.d.ts.map +1 -0
  419. package/dist/esm/src/shape/shape.js +121 -0
  420. package/dist/esm/src/shape/shape.js.map +1 -0
  421. package/dist/esm/src/shape/tags.d.ts +4 -0
  422. package/dist/esm/src/shape/tags.d.ts.map +1 -0
  423. package/dist/esm/src/shape/tags.js +2 -0
  424. package/dist/esm/src/shape/tags.js.map +1 -0
  425. package/dist/esm/src/shape/transforms/index.d.ts +4 -0
  426. package/dist/esm/src/shape/transforms/index.d.ts.map +1 -0
  427. package/dist/esm/src/shape/transforms/index.js +4 -0
  428. package/dist/esm/src/shape/transforms/index.js.map +1 -0
  429. package/dist/esm/src/shape/transforms/json-schema/index.d.ts +3 -0
  430. package/dist/esm/src/shape/transforms/json-schema/index.d.ts.map +1 -0
  431. package/dist/esm/src/shape/transforms/json-schema/index.js +3 -0
  432. package/dist/esm/src/shape/transforms/json-schema/index.js.map +1 -0
  433. package/dist/esm/src/shape/transforms/json-schema/transform.d.ts +5 -0
  434. package/dist/esm/src/shape/transforms/json-schema/transform.d.ts.map +1 -0
  435. package/dist/esm/src/shape/transforms/json-schema/transform.js +734 -0
  436. package/dist/esm/src/shape/transforms/json-schema/transform.js.map +1 -0
  437. package/dist/esm/src/shape/transforms/json-schema/types.d.ts +87 -0
  438. package/dist/esm/src/shape/transforms/json-schema/types.d.ts.map +1 -0
  439. package/dist/esm/src/shape/transforms/json-schema/types.js +2 -0
  440. package/dist/esm/src/shape/transforms/json-schema/types.js.map +1 -0
  441. package/dist/esm/src/shape/transforms/sql/dialects/dialect.d.ts +51 -0
  442. package/dist/esm/src/shape/transforms/sql/dialects/dialect.d.ts.map +1 -0
  443. package/dist/esm/src/shape/transforms/sql/dialects/dialect.js +3 -0
  444. package/dist/esm/src/shape/transforms/sql/dialects/dialect.js.map +1 -0
  445. package/dist/esm/src/shape/transforms/sql/dialects/index.d.ts +7 -0
  446. package/dist/esm/src/shape/transforms/sql/dialects/index.d.ts.map +1 -0
  447. package/dist/esm/src/shape/transforms/sql/dialects/index.js +12 -0
  448. package/dist/esm/src/shape/transforms/sql/dialects/index.js.map +1 -0
  449. package/dist/esm/src/shape/transforms/sql/dialects/postgres.d.ts +8 -0
  450. package/dist/esm/src/shape/transforms/sql/dialects/postgres.d.ts.map +1 -0
  451. package/dist/esm/src/shape/transforms/sql/dialects/postgres.js +370 -0
  452. package/dist/esm/src/shape/transforms/sql/dialects/postgres.js.map +1 -0
  453. package/dist/esm/src/shape/transforms/sql/dialects/sqlite.d.ts +3 -0
  454. package/dist/esm/src/shape/transforms/sql/dialects/sqlite.d.ts.map +1 -0
  455. package/dist/esm/src/shape/transforms/sql/dialects/sqlite.js +315 -0
  456. package/dist/esm/src/shape/transforms/sql/dialects/sqlite.js.map +1 -0
  457. package/dist/esm/src/shape/transforms/sql/from-sql.d.ts +5 -0
  458. package/dist/esm/src/shape/transforms/sql/from-sql.d.ts.map +1 -0
  459. package/dist/esm/src/shape/transforms/sql/from-sql.js +197 -0
  460. package/dist/esm/src/shape/transforms/sql/from-sql.js.map +1 -0
  461. package/dist/esm/src/shape/transforms/sql/index.d.ts +4 -0
  462. package/dist/esm/src/shape/transforms/sql/index.d.ts.map +1 -0
  463. package/dist/esm/src/shape/transforms/sql/index.js +4 -0
  464. package/dist/esm/src/shape/transforms/sql/index.js.map +1 -0
  465. package/dist/esm/src/shape/transforms/sql/options.d.ts +6 -0
  466. package/dist/esm/src/shape/transforms/sql/options.d.ts.map +1 -0
  467. package/dist/esm/src/shape/transforms/sql/options.js +2 -0
  468. package/dist/esm/src/shape/transforms/sql/options.js.map +1 -0
  469. package/dist/esm/src/shape/transforms/sql/parser/check-decoder.d.ts +25 -0
  470. package/dist/esm/src/shape/transforms/sql/parser/check-decoder.d.ts.map +1 -0
  471. package/dist/esm/src/shape/transforms/sql/parser/check-decoder.js +412 -0
  472. package/dist/esm/src/shape/transforms/sql/parser/check-decoder.js.map +1 -0
  473. package/dist/esm/src/shape/transforms/sql/parser/create-domain.d.ts +11 -0
  474. package/dist/esm/src/shape/transforms/sql/parser/create-domain.d.ts.map +1 -0
  475. package/dist/esm/src/shape/transforms/sql/parser/create-domain.js +91 -0
  476. package/dist/esm/src/shape/transforms/sql/parser/create-domain.js.map +1 -0
  477. package/dist/esm/src/shape/transforms/sql/parser/create-table.d.ts +28 -0
  478. package/dist/esm/src/shape/transforms/sql/parser/create-table.d.ts.map +1 -0
  479. package/dist/esm/src/shape/transforms/sql/parser/create-table.js +740 -0
  480. package/dist/esm/src/shape/transforms/sql/parser/create-table.js.map +1 -0
  481. package/dist/esm/src/shape/transforms/sql/parser/create-type.d.ts +23 -0
  482. package/dist/esm/src/shape/transforms/sql/parser/create-type.d.ts.map +1 -0
  483. package/dist/esm/src/shape/transforms/sql/parser/create-type.js +72 -0
  484. package/dist/esm/src/shape/transforms/sql/parser/create-type.js.map +1 -0
  485. package/dist/esm/src/shape/transforms/sql/parser/cursor.d.ts +32 -0
  486. package/dist/esm/src/shape/transforms/sql/parser/cursor.d.ts.map +1 -0
  487. package/dist/esm/src/shape/transforms/sql/parser/cursor.js +163 -0
  488. package/dist/esm/src/shape/transforms/sql/parser/cursor.js.map +1 -0
  489. package/dist/esm/src/shape/transforms/sql/parser/default-decoder.d.ts +10 -0
  490. package/dist/esm/src/shape/transforms/sql/parser/default-decoder.d.ts.map +1 -0
  491. package/dist/esm/src/shape/transforms/sql/parser/default-decoder.js +121 -0
  492. package/dist/esm/src/shape/transforms/sql/parser/default-decoder.js.map +1 -0
  493. package/dist/esm/src/shape/transforms/sql/parser/lexer.d.ts +14 -0
  494. package/dist/esm/src/shape/transforms/sql/parser/lexer.d.ts.map +1 -0
  495. package/dist/esm/src/shape/transforms/sql/parser/lexer.js +253 -0
  496. package/dist/esm/src/shape/transforms/sql/parser/lexer.js.map +1 -0
  497. package/dist/esm/src/shape/transforms/sql/parser/pg-types.d.ts +21 -0
  498. package/dist/esm/src/shape/transforms/sql/parser/pg-types.d.ts.map +1 -0
  499. package/dist/esm/src/shape/transforms/sql/parser/pg-types.js +208 -0
  500. package/dist/esm/src/shape/transforms/sql/parser/pg-types.js.map +1 -0
  501. package/dist/esm/src/shape/transforms/sql/parser/sqlite-types.d.ts +8 -0
  502. package/dist/esm/src/shape/transforms/sql/parser/sqlite-types.d.ts.map +1 -0
  503. package/dist/esm/src/shape/transforms/sql/parser/sqlite-types.js +89 -0
  504. package/dist/esm/src/shape/transforms/sql/parser/sqlite-types.js.map +1 -0
  505. package/dist/esm/src/shape/transforms/sql/parser/statements.d.ts +9 -0
  506. package/dist/esm/src/shape/transforms/sql/parser/statements.d.ts.map +1 -0
  507. package/dist/esm/src/shape/transforms/sql/parser/statements.js +153 -0
  508. package/dist/esm/src/shape/transforms/sql/parser/statements.js.map +1 -0
  509. package/dist/esm/src/shape/transforms/sql/parser/type-spec.d.ts +12 -0
  510. package/dist/esm/src/shape/transforms/sql/parser/type-spec.d.ts.map +1 -0
  511. package/dist/esm/src/shape/transforms/sql/parser/type-spec.js +130 -0
  512. package/dist/esm/src/shape/transforms/sql/parser/type-spec.js.map +1 -0
  513. package/dist/esm/src/shape/transforms/sql/transform.d.ts +5 -0
  514. package/dist/esm/src/shape/transforms/sql/transform.d.ts.map +1 -0
  515. package/dist/esm/src/shape/transforms/sql/transform.js +258 -0
  516. package/dist/esm/src/shape/transforms/sql/transform.js.map +1 -0
  517. package/dist/esm/src/shape/transforms/typescript/index.d.ts +2 -0
  518. package/dist/esm/src/shape/transforms/typescript/index.d.ts.map +1 -0
  519. package/dist/esm/src/shape/transforms/typescript/index.js +2 -0
  520. package/dist/esm/src/shape/transforms/typescript/index.js.map +1 -0
  521. package/dist/esm/src/shape/transforms/typescript/transform.d.ts +3 -0
  522. package/dist/esm/src/shape/transforms/typescript/transform.d.ts.map +1 -0
  523. package/dist/esm/src/shape/transforms/typescript/transform.js +186 -0
  524. package/dist/esm/src/shape/transforms/typescript/transform.js.map +1 -0
  525. package/dist/esm/src/shape/validate.d.ts +7 -0
  526. package/dist/esm/src/shape/validate.d.ts.map +1 -0
  527. package/dist/esm/src/shape/validate.js +355 -0
  528. package/dist/esm/src/shape/validate.js.map +1 -0
  529. package/package.json +40 -0
  530. package/src/common/array.test.ts +19 -0
  531. package/src/common/array.ts +15 -0
  532. package/src/common/index.ts +5 -0
  533. package/src/common/is.ts +23 -0
  534. package/src/common/object.ts +35 -0
  535. package/src/common/phantom.ts +1 -0
  536. package/src/common/result.ts +43 -0
  537. package/src/common/string.ts +28 -0
  538. package/src/common/types.ts +34 -0
  539. package/src/index.ts +1 -0
  540. package/src/shape/annotate.ts +139 -0
  541. package/src/shape/annotation.ts +47 -0
  542. package/src/shape/base.ts +71 -0
  543. package/src/shape/builder.test.ts +728 -0
  544. package/src/shape/builder.ts +475 -0
  545. package/src/shape/error.ts +4 -0
  546. package/src/shape/index.ts +3 -0
  547. package/src/shape/number.ts +118 -0
  548. package/src/shape/shape.test.ts +792 -0
  549. package/src/shape/shape.ts +377 -0
  550. package/src/shape/tags.ts +14 -0
  551. package/src/shape/transforms/index.ts +3 -0
  552. package/src/shape/transforms/json-schema/index.ts +2 -0
  553. package/src/shape/transforms/json-schema/transform.test.ts +850 -0
  554. package/src/shape/transforms/json-schema/transform.ts +882 -0
  555. package/src/shape/transforms/json-schema/types.ts +132 -0
  556. package/src/shape/transforms/sql/dialects/dialect.ts +89 -0
  557. package/src/shape/transforms/sql/dialects/index.ts +14 -0
  558. package/src/shape/transforms/sql/dialects/postgres.ts +392 -0
  559. package/src/shape/transforms/sql/dialects/sqlite.ts +333 -0
  560. package/src/shape/transforms/sql/from-sql.test.ts +704 -0
  561. package/src/shape/transforms/sql/from-sql.ts +210 -0
  562. package/src/shape/transforms/sql/index.ts +3 -0
  563. package/src/shape/transforms/sql/options.ts +6 -0
  564. package/src/shape/transforms/sql/parser/check-decoder.ts +457 -0
  565. package/src/shape/transforms/sql/parser/create-domain.ts +105 -0
  566. package/src/shape/transforms/sql/parser/create-table.ts +809 -0
  567. package/src/shape/transforms/sql/parser/create-type.ts +91 -0
  568. package/src/shape/transforms/sql/parser/cursor.ts +179 -0
  569. package/src/shape/transforms/sql/parser/default-decoder.ts +129 -0
  570. package/src/shape/transforms/sql/parser/lexer.ts +289 -0
  571. package/src/shape/transforms/sql/parser/pg-types.ts +247 -0
  572. package/src/shape/transforms/sql/parser/sqlite-types.ts +103 -0
  573. package/src/shape/transforms/sql/parser/statements.ts +127 -0
  574. package/src/shape/transforms/sql/parser/type-spec.ts +159 -0
  575. package/src/shape/transforms/sql/transform.sqlite.test.ts +448 -0
  576. package/src/shape/transforms/sql/transform.test.ts +880 -0
  577. package/src/shape/transforms/sql/transform.ts +295 -0
  578. package/src/shape/transforms/typescript/index.ts +1 -0
  579. package/src/shape/transforms/typescript/transform.ts +211 -0
  580. package/src/shape/tuple.test.ts +171 -0
  581. package/src/shape/validate.ts +413 -0
  582. package/tsconfig.cjs.json +11 -0
  583. package/tsconfig.esm.json +10 -0
  584. package/tsconfig.json +23 -0
@@ -0,0 +1,880 @@
1
+ import { describe, expect } from "vitest";
2
+ import { R } from "../../../common/result";
3
+ import { annotate, shapes } from "../../shape";
4
+ import { toSQL } from "./transform";
5
+
6
+ const ok = (res: R.Result<string, string[]>): string => {
7
+ if (!R.isOk(res))
8
+ throw new Error(`expected ok, got err: ${res.error.join("; ")}`);
9
+ return res.value;
10
+ };
11
+
12
+ const err = (res: R.Result<string, string[]>): string[] => {
13
+ if (R.isOk(res)) throw new Error(`expected err, got ok: ${res.value}`);
14
+ return res.error;
15
+ };
16
+
17
+ describe("toSQL — top-level errors", (it) => {
18
+ it("non-mapping top-level → err", () => {
19
+ expect(err(toSQL(shapes.string()))).toEqual([
20
+ "toSQL expects a top-level mapping (table) or module",
21
+ ]);
22
+ });
23
+
24
+ it("mapping without title → err", () => {
25
+ const m = shapes.mapping({ id: shapes.int() });
26
+ expect(err(toSQL(m))).toEqual([
27
+ "mapping is missing required title (table name)",
28
+ ]);
29
+ });
30
+
31
+ it("nil/undefined field → err", () => {
32
+ const m = annotate.titled(
33
+ shapes.mapping({ a: shapes.nil(), b: shapes.notDefined() }),
34
+ "t",
35
+ );
36
+ expect(err(toSQL(m))).toEqual([
37
+ "field 'a': nil cannot be a column",
38
+ "field 'b': undefined cannot be a column",
39
+ ]);
40
+ });
41
+
42
+ it("autoIncrement on non-integer → err", () => {
43
+ const m = annotate.titled(
44
+ shapes.mapping({ s: annotate.autoIncremented(shapes.string()) }),
45
+ "t",
46
+ );
47
+ expect(err(toSQL(m))).toEqual([
48
+ "field 's': autoIncrement requires an integer type",
49
+ ]);
50
+ });
51
+
52
+ it("autoIncrement on uint64 → err", () => {
53
+ const m = annotate.titled(
54
+ shapes.mapping({ x: annotate.autoIncremented(shapes.uint64()) }),
55
+ "t",
56
+ );
57
+ expect(err(toSQL(m))).toEqual([
58
+ "field 'x': autoIncrement is not supported on uint64 (numeric column)",
59
+ ]);
60
+ });
61
+
62
+ it("multiple errors collected", () => {
63
+ const m = annotate.titled(
64
+ shapes.mapping({
65
+ a: shapes.nil(),
66
+ b: annotate.autoIncremented(shapes.string()),
67
+ }),
68
+ "t",
69
+ );
70
+ expect(err(toSQL(m))).toEqual([
71
+ "field 'a': nil cannot be a column",
72
+ "field 'b': autoIncrement requires an integer type",
73
+ ]);
74
+ });
75
+ });
76
+
77
+ describe("toSQL — scalars", (it) => {
78
+ it("string → text NOT NULL", () => {
79
+ const m = annotate.titled(shapes.mapping({ name: shapes.string() }), "t");
80
+ expect(ok(toSQL(m))).toBe(
81
+ `CREATE TABLE IF NOT EXISTS "t" (\n "name" text NOT NULL\n);`,
82
+ );
83
+ });
84
+
85
+ it("optional string → no NOT NULL", () => {
86
+ const m = annotate.titled(
87
+ shapes.mapping({ name: annotate.optional(shapes.string()) }),
88
+ "t",
89
+ );
90
+ expect(ok(toSQL(m))).toBe(
91
+ `CREATE TABLE IF NOT EXISTS "t" (\n "name" text\n);`,
92
+ );
93
+ });
94
+
95
+ it("boolean → boolean NOT NULL", () => {
96
+ const m = annotate.titled(shapes.mapping({ flag: shapes.boolean() }), "t");
97
+ expect(ok(toSQL(m))).toBe(
98
+ `CREATE TABLE IF NOT EXISTS "t" (\n "flag" boolean NOT NULL\n);`,
99
+ );
100
+ });
101
+
102
+ it("date → timestamptz", () => {
103
+ const m = annotate.titled(shapes.mapping({ when: shapes.date() }), "t");
104
+ expect(ok(toSQL(m))).toBe(
105
+ `CREATE TABLE IF NOT EXISTS "t" (\n "when" timestamptz NOT NULL\n);`,
106
+ );
107
+ });
108
+
109
+ it("binary → bytea", () => {
110
+ const m = annotate.titled(shapes.mapping({ blob: shapes.binary() }), "t");
111
+ expect(ok(toSQL(m))).toBe(
112
+ `CREATE TABLE IF NOT EXISTS "t" (\n "blob" bytea NOT NULL\n);`,
113
+ );
114
+ });
115
+
116
+ it("unknown → jsonb", () => {
117
+ const m = annotate.titled(
118
+ shapes.mapping({ payload: shapes.unknown() }),
119
+ "t",
120
+ );
121
+ expect(ok(toSQL(m))).toBe(
122
+ `CREATE TABLE IF NOT EXISTS "t" (\n "payload" jsonb NOT NULL\n);`,
123
+ );
124
+ });
125
+ });
126
+
127
+ describe("toSQL — strings", (it) => {
128
+ it("varchar(N) when max set; no min check", () => {
129
+ const m = annotate.titled(
130
+ shapes.mapping({ n: annotate.max(shapes.string(), 32) }),
131
+ "t",
132
+ );
133
+ expect(ok(toSQL(m))).toBe(
134
+ `CREATE TABLE IF NOT EXISTS "t" (\n "n" varchar(32) NOT NULL\n);`,
135
+ );
136
+ });
137
+
138
+ it("text + CHECK char_length when only min set", () => {
139
+ const m = annotate.titled(
140
+ shapes.mapping({ n: annotate.min(shapes.string(), 3) }),
141
+ "t",
142
+ );
143
+ expect(ok(toSQL(m))).toBe(
144
+ `CREATE TABLE IF NOT EXISTS "t" (\n "n" text NOT NULL CHECK (char_length("n") >= 3)\n);`,
145
+ );
146
+ });
147
+
148
+ it("RegExp pattern → ~ check using .source", () => {
149
+ const m = annotate.titled(
150
+ shapes.mapping({ n: annotate.pattern(shapes.string(), /^[a-z]+$/) }),
151
+ "t",
152
+ );
153
+ expect(ok(toSQL(m))).toBe(
154
+ `CREATE TABLE IF NOT EXISTS "t" (\n "n" text NOT NULL CHECK ("n" ~ '^[a-z]+$')\n);`,
155
+ );
156
+ });
157
+
158
+ it("string pattern stored verbatim; single quotes escaped", () => {
159
+ const m = annotate.titled(
160
+ shapes.mapping({ n: annotate.pattern(shapes.string(), "it's") }),
161
+ "t",
162
+ );
163
+ expect(ok(toSQL(m))).toBe(
164
+ `CREATE TABLE IF NOT EXISTS "t" (\n "n" text NOT NULL CHECK ("n" ~ 'it''s')\n);`,
165
+ );
166
+ });
167
+ });
168
+
169
+ describe("toSQL — numbers", (it) => {
170
+ it("number() → double precision (no checks)", () => {
171
+ const m = annotate.titled(shapes.mapping({ x: shapes.number() }), "t");
172
+ expect(ok(toSQL(m))).toBe(
173
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" double precision NOT NULL\n);`,
174
+ );
175
+ });
176
+
177
+ it("int() → bigint (no checks)", () => {
178
+ const m = annotate.titled(shapes.mapping({ x: shapes.int() }), "t");
179
+ expect(ok(toSQL(m))).toBe(
180
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" bigint NOT NULL\n);`,
181
+ );
182
+ });
183
+
184
+ it("int8 → smallint + tag-range CHECKs", () => {
185
+ const m = annotate.titled(shapes.mapping({ x: shapes.int8() }), "t");
186
+ expect(ok(toSQL(m))).toBe(
187
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" smallint NOT NULL CHECK ("x" >= -128) CHECK ("x" <= 127)\n);`,
188
+ );
189
+ });
190
+
191
+ it("uint8 → smallint + [0, 255] CHECKs", () => {
192
+ const m = annotate.titled(shapes.mapping({ x: shapes.uint8() }), "t");
193
+ expect(ok(toSQL(m))).toBe(
194
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" smallint NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 255)\n);`,
195
+ );
196
+ });
197
+
198
+ it("int16 → smallint, no CHECKs (range matches pg type)", () => {
199
+ const m = annotate.titled(shapes.mapping({ x: shapes.int16() }), "t");
200
+ expect(ok(toSQL(m))).toBe(
201
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" smallint NOT NULL\n);`,
202
+ );
203
+ });
204
+
205
+ it("uint16 → integer + [0, 65535] CHECKs", () => {
206
+ const m = annotate.titled(shapes.mapping({ x: shapes.uint16() }), "t");
207
+ expect(ok(toSQL(m))).toBe(
208
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" integer NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 65535)\n);`,
209
+ );
210
+ });
211
+
212
+ it("int32 → integer, no CHECKs", () => {
213
+ const m = annotate.titled(shapes.mapping({ x: shapes.int32() }), "t");
214
+ expect(ok(toSQL(m))).toBe(
215
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" integer NOT NULL\n);`,
216
+ );
217
+ });
218
+
219
+ it("uint32 → bigint + [0, 4294967295] CHECKs", () => {
220
+ const m = annotate.titled(shapes.mapping({ x: shapes.uint32() }), "t");
221
+ expect(ok(toSQL(m))).toBe(
222
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" bigint NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 4294967295)\n);`,
223
+ );
224
+ });
225
+
226
+ it("int64 → bigint, no CHECKs", () => {
227
+ const m = annotate.titled(shapes.mapping({ x: shapes.int64() }), "t");
228
+ expect(ok(toSQL(m))).toBe(
229
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" bigint NOT NULL\n);`,
230
+ );
231
+ });
232
+
233
+ it("uint64 → numeric(20,0) + >= 0 CHECK", () => {
234
+ const m = annotate.titled(shapes.mapping({ x: shapes.uint64() }), "t");
235
+ expect(ok(toSQL(m))).toBe(
236
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" numeric(20,0) NOT NULL CHECK ("x" >= 0)\n);`,
237
+ );
238
+ });
239
+
240
+ it("float / float32 → real", () => {
241
+ const m = annotate.titled(
242
+ shapes.mapping({ a: shapes.float(), b: shapes.float32() }),
243
+ "t",
244
+ );
245
+ expect(ok(toSQL(m))).toBe(
246
+ `CREATE TABLE IF NOT EXISTS "t" (\n "a" real NOT NULL,\n "b" real NOT NULL\n);`,
247
+ );
248
+ });
249
+
250
+ it("float64 → double precision", () => {
251
+ const m = annotate.titled(shapes.mapping({ x: shapes.float64() }), "t");
252
+ expect(ok(toSQL(m))).toBe(
253
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" double precision NOT NULL\n);`,
254
+ );
255
+ });
256
+
257
+ it("tighter anno.min/max replaces tag bound", () => {
258
+ const m = annotate.titled(
259
+ shapes.mapping({ x: annotate.as(shapes.int16(), { min: 0, max: 100 }) }),
260
+ "t",
261
+ );
262
+ expect(ok(toSQL(m))).toBe(
263
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" smallint NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 100)\n);`,
264
+ );
265
+ });
266
+
267
+ it("wider anno.min/max keeps the tag bound (intersection)", () => {
268
+ const m = annotate.titled(
269
+ shapes.mapping({
270
+ x: annotate.as(shapes.uint8(), { min: -1000, max: 1000 }),
271
+ }),
272
+ "t",
273
+ );
274
+ expect(ok(toSQL(m))).toBe(
275
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" smallint NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 255)\n);`,
276
+ );
277
+ });
278
+
279
+ it("anno.min/max on plain number → CHECKs", () => {
280
+ const m = annotate.titled(
281
+ shapes.mapping({ x: annotate.as(shapes.number(), { min: 0, max: 1 }) }),
282
+ "t",
283
+ );
284
+ expect(ok(toSQL(m))).toBe(
285
+ `CREATE TABLE IF NOT EXISTS "t" (\n "x" double precision NOT NULL CHECK ("x" >= 0) CHECK ("x" <= 1)\n);`,
286
+ );
287
+ });
288
+ });
289
+
290
+ describe("toSQL — arrays", (it) => {
291
+ it("array(string) → text[]", () => {
292
+ const m = annotate.titled(
293
+ shapes.mapping({ tags: shapes.array(shapes.string()) }),
294
+ "t",
295
+ );
296
+ expect(ok(toSQL(m))).toBe(
297
+ `CREATE TABLE IF NOT EXISTS "t" (\n "tags" text[] NOT NULL\n);`,
298
+ );
299
+ });
300
+
301
+ it("array(int32) → integer[]", () => {
302
+ const m = annotate.titled(
303
+ shapes.mapping({ ns: shapes.array(shapes.int32()) }),
304
+ "t",
305
+ );
306
+ expect(ok(toSQL(m))).toBe(
307
+ `CREATE TABLE IF NOT EXISTS "t" (\n "ns" integer[] NOT NULL\n);`,
308
+ );
309
+ });
310
+
311
+ it("array(varchar) element honors element max", () => {
312
+ const m = annotate.titled(
313
+ shapes.mapping({ ns: shapes.array(annotate.max(shapes.string(), 8)) }),
314
+ "t",
315
+ );
316
+ expect(ok(toSQL(m))).toBe(
317
+ `CREATE TABLE IF NOT EXISTS "t" (\n "ns" varchar(8)[] NOT NULL\n);`,
318
+ );
319
+ });
320
+
321
+ it("array(mapping) → jsonb", () => {
322
+ const m = annotate.titled(
323
+ shapes.mapping({
324
+ rows: shapes.array(shapes.mapping({ a: shapes.string() })),
325
+ }),
326
+ "t",
327
+ );
328
+ expect(ok(toSQL(m))).toBe(
329
+ `CREATE TABLE IF NOT EXISTS "t" (\n "rows" jsonb NOT NULL\n);`,
330
+ );
331
+ });
332
+
333
+ it("array min/max → cardinality CHECKs", () => {
334
+ const m = annotate.titled(
335
+ shapes.mapping({
336
+ tags: annotate.as(shapes.array(shapes.string()), { min: 1, max: 3 }),
337
+ }),
338
+ "t",
339
+ );
340
+ expect(ok(toSQL(m))).toBe(
341
+ `CREATE TABLE IF NOT EXISTS "t" (\n "tags" text[] NOT NULL CHECK (cardinality("tags") >= 1) CHECK (cardinality("tags") <= 3)\n);`,
342
+ );
343
+ });
344
+
345
+ it("anno.unique on array → column-level UNIQUE (cross-row)", () => {
346
+ const m = annotate.titled(
347
+ shapes.mapping({ tags: annotate.unique(shapes.array(shapes.string())) }),
348
+ "t",
349
+ );
350
+ expect(ok(toSQL(m))).toBe(
351
+ `CREATE TABLE IF NOT EXISTS "t" (\n "tags" text[] NOT NULL UNIQUE\n);`,
352
+ );
353
+ });
354
+
355
+ it("anno.uniqueItems on array → CHECK via helper + prelude function", () => {
356
+ const m = annotate.titled(
357
+ shapes.mapping({
358
+ tags: annotate.uniqueItems(shapes.array(shapes.string())),
359
+ }),
360
+ "t",
361
+ );
362
+ expect(ok(toSQL(m))).toBe(
363
+ [
364
+ `CREATE OR REPLACE FUNCTION shapecraft_array_is_unique(anyarray) RETURNS boolean`,
365
+ ` LANGUAGE sql IMMUTABLE PARALLEL SAFE`,
366
+ ` AS $$ SELECT cardinality($1) = cardinality(ARRAY(SELECT DISTINCT unnest($1))) $$;`,
367
+ ``,
368
+ `CREATE TABLE IF NOT EXISTS "t" (`,
369
+ ` "tags" text[] NOT NULL CHECK (shapecraft_array_is_unique("tags"))`,
370
+ `);`,
371
+ ].join("\n"),
372
+ );
373
+ });
374
+
375
+ it("anno.unique + anno.uniqueItems → both UNIQUE and element CHECK", () => {
376
+ const m = annotate.titled(
377
+ shapes.mapping({
378
+ tags: annotate.unique(
379
+ annotate.uniqueItems(shapes.array(shapes.string())),
380
+ ),
381
+ }),
382
+ "t",
383
+ );
384
+ expect(ok(toSQL(m))).toBe(
385
+ [
386
+ `CREATE OR REPLACE FUNCTION shapecraft_array_is_unique(anyarray) RETURNS boolean`,
387
+ ` LANGUAGE sql IMMUTABLE PARALLEL SAFE`,
388
+ ` AS $$ SELECT cardinality($1) = cardinality(ARRAY(SELECT DISTINCT unnest($1))) $$;`,
389
+ ``,
390
+ `CREATE TABLE IF NOT EXISTS "t" (`,
391
+ ` "tags" text[] NOT NULL UNIQUE CHECK (shapecraft_array_is_unique("tags"))`,
392
+ `);`,
393
+ ].join("\n"),
394
+ );
395
+ });
396
+
397
+ it("anno.uniqueItems on two arrays → only one prelude function emitted", () => {
398
+ const m = annotate.titled(
399
+ shapes.mapping({
400
+ a: annotate.uniqueItems(shapes.array(shapes.string())),
401
+ b: annotate.uniqueItems(shapes.array(shapes.int32())),
402
+ }),
403
+ "t",
404
+ );
405
+ const sql = ok(toSQL(m));
406
+ const fnOccurrences =
407
+ sql.split("CREATE OR REPLACE FUNCTION shapecraft_array_is_unique")
408
+ .length - 1;
409
+ expect(fnOccurrences).toBe(1);
410
+ });
411
+
412
+ it("anno.uniqueItems on jsonb-backed array → err", () => {
413
+ const m = annotate.titled(
414
+ shapes.mapping({
415
+ rows: annotate.uniqueItems(
416
+ shapes.array(shapes.mapping({ a: shapes.string() })),
417
+ ),
418
+ }),
419
+ "t",
420
+ );
421
+ expect(err(toSQL(m))).toEqual([
422
+ "field 'rows': uniqueItems is not supported on jsonb array columns",
423
+ ]);
424
+ });
425
+ });
426
+
427
+ describe("toSQL — vector", (it) => {
428
+ it("vector(3, float32) → vector(3)", () => {
429
+ const m = annotate.titled(
430
+ shapes.mapping({ emb: shapes.vector(3, shapes.float32()) }),
431
+ "t",
432
+ );
433
+ expect(ok(toSQL(m))).toBe(
434
+ `CREATE TABLE IF NOT EXISTS "t" (\n "emb" vector(3) NOT NULL\n);`,
435
+ );
436
+ });
437
+
438
+ it("vector(384) → vector(384) (default format ignored)", () => {
439
+ const m = annotate.titled(shapes.mapping({ emb: shapes.vector(384) }), "t");
440
+ expect(ok(toSQL(m))).toBe(
441
+ `CREATE TABLE IF NOT EXISTS "t" (\n "emb" vector(384) NOT NULL\n);`,
442
+ );
443
+ });
444
+ });
445
+
446
+ describe("toSQL — mappings / records as fields → jsonb", (it) => {
447
+ it("nested mapping → jsonb", () => {
448
+ const m = annotate.titled(
449
+ shapes.mapping({
450
+ profile: shapes.mapping({ a: shapes.string() }),
451
+ }),
452
+ "t",
453
+ );
454
+ expect(ok(toSQL(m))).toBe(
455
+ `CREATE TABLE IF NOT EXISTS "t" (\n "profile" jsonb NOT NULL\n);`,
456
+ );
457
+ });
458
+
459
+ it("record → jsonb", () => {
460
+ const m = annotate.titled(
461
+ shapes.mapping({
462
+ counters: shapes.record(shapes.string(), shapes.int32()),
463
+ }),
464
+ "t",
465
+ );
466
+ expect(ok(toSQL(m))).toBe(
467
+ `CREATE TABLE IF NOT EXISTS "t" (\n "counters" jsonb NOT NULL\n);`,
468
+ );
469
+ });
470
+ });
471
+
472
+ describe("toSQL — unions", (it) => {
473
+ it("string-literal union with title → CREATE TYPE enum + column type ref", () => {
474
+ const role = annotate.titled(
475
+ shapes.union(shapes.literal("admin"), shapes.literal("user")),
476
+ "user_role",
477
+ );
478
+ const m = annotate.titled(shapes.mapping({ role }), "users");
479
+ expect(ok(toSQL(m))).toBe(
480
+ `CREATE TYPE "user_role" AS ENUM ('admin', 'user');\n\nCREATE TABLE IF NOT EXISTS "users" (\n "role" "user_role" NOT NULL\n);`,
481
+ );
482
+ });
483
+
484
+ it("string-literal union without title → text + IN check", () => {
485
+ const u = shapes.union(shapes.literal("a"), shapes.literal("b"));
486
+ const m = annotate.titled(shapes.mapping({ k: u }), "t");
487
+ expect(ok(toSQL(m))).toBe(
488
+ `CREATE TABLE IF NOT EXISTS "t" (\n "k" text NOT NULL CHECK ("k" IN ('a', 'b'))\n);`,
489
+ );
490
+ });
491
+
492
+ it("mixed / non-literal union → jsonb", () => {
493
+ const u = shapes.union(shapes.literal("a"), shapes.number());
494
+ const m = annotate.titled(shapes.mapping({ k: u }), "t");
495
+ expect(ok(toSQL(m))).toBe(
496
+ `CREATE TABLE IF NOT EXISTS "t" (\n "k" jsonb NOT NULL\n);`,
497
+ );
498
+ });
499
+
500
+ it("same titled enum used twice → only one CREATE TYPE", () => {
501
+ const role = annotate.titled(
502
+ shapes.union(shapes.literal("admin"), shapes.literal("user")),
503
+ "user_role",
504
+ );
505
+ const m = annotate.titled(
506
+ shapes.mapping({ role_a: role, role_b: role }),
507
+ "t",
508
+ );
509
+ expect(ok(toSQL(m))).toBe(
510
+ `CREATE TYPE "user_role" AS ENUM ('admin', 'user');\n\nCREATE TABLE IF NOT EXISTS "t" (\n "role_a" "user_role" NOT NULL,\n "role_b" "user_role" NOT NULL\n);`,
511
+ );
512
+ });
513
+ });
514
+
515
+ describe("toSQL — annotations", (it) => {
516
+ it("primary → PRIMARY KEY (always NOT NULL)", () => {
517
+ const m = annotate.titled(
518
+ shapes.mapping({ id: annotate.primary(shapes.int()) }),
519
+ "t",
520
+ );
521
+ expect(ok(toSQL(m))).toBe(
522
+ `CREATE TABLE IF NOT EXISTS "t" (\n "id" bigint NOT NULL PRIMARY KEY\n);`,
523
+ );
524
+ });
525
+
526
+ it("unique on scalar → UNIQUE", () => {
527
+ const m = annotate.titled(
528
+ shapes.mapping({ slug: annotate.unique(shapes.string()) }),
529
+ "t",
530
+ );
531
+ expect(ok(toSQL(m))).toBe(
532
+ `CREATE TABLE IF NOT EXISTS "t" (\n "slug" text NOT NULL UNIQUE\n);`,
533
+ );
534
+ });
535
+
536
+ it("autoIncrement on int → GENERATED ALWAYS AS IDENTITY", () => {
537
+ const m = annotate.titled(
538
+ shapes.mapping({
539
+ id: annotate.primary(annotate.autoIncremented(shapes.int())),
540
+ }),
541
+ "t",
542
+ );
543
+ expect(ok(toSQL(m))).toBe(
544
+ `CREATE TABLE IF NOT EXISTS "t" (\n "id" bigint NOT NULL PRIMARY KEY GENERATED ALWAYS AS IDENTITY\n);`,
545
+ );
546
+ });
547
+
548
+ it("foreign → REFERENCES", () => {
549
+ const m = annotate.titled(
550
+ shapes.mapping({
551
+ user_id: annotate.foreign(shapes.int(), "users", "id"),
552
+ }),
553
+ "t",
554
+ );
555
+ expect(ok(toSQL(m))).toBe(
556
+ `CREATE TABLE IF NOT EXISTS "t" (\n "user_id" bigint NOT NULL REFERENCES "users"("id")\n);`,
557
+ );
558
+ });
559
+
560
+ it("string default", () => {
561
+ const m = annotate.titled(
562
+ shapes.mapping({
563
+ s: annotate.defaulted(shapes.string(), "hi"),
564
+ }),
565
+ "t",
566
+ );
567
+ expect(ok(toSQL(m))).toBe(
568
+ `CREATE TABLE IF NOT EXISTS "t" (\n "s" text NOT NULL DEFAULT 'hi'\n);`,
569
+ );
570
+ });
571
+
572
+ it("number default", () => {
573
+ const m = annotate.titled(
574
+ shapes.mapping({
575
+ n: annotate.defaulted(shapes.int32(), 42),
576
+ }),
577
+ "t",
578
+ );
579
+ expect(ok(toSQL(m))).toBe(
580
+ `CREATE TABLE IF NOT EXISTS "t" (\n "n" integer NOT NULL DEFAULT 42\n);`,
581
+ );
582
+ });
583
+
584
+ it("boolean default", () => {
585
+ const m = annotate.titled(
586
+ shapes.mapping({
587
+ b: annotate.defaulted(shapes.boolean(), true),
588
+ }),
589
+ "t",
590
+ );
591
+ expect(ok(toSQL(m))).toBe(
592
+ `CREATE TABLE IF NOT EXISTS "t" (\n "b" boolean NOT NULL DEFAULT TRUE\n);`,
593
+ );
594
+ });
595
+
596
+ it("null default on optional column", () => {
597
+ const m = annotate.titled(
598
+ shapes.mapping({
599
+ s: annotate.as(annotate.optional(shapes.string()), { default: null }),
600
+ }),
601
+ "t",
602
+ );
603
+ expect(ok(toSQL(m))).toBe(
604
+ `CREATE TABLE IF NOT EXISTS "t" (\n "s" text DEFAULT NULL\n);`,
605
+ );
606
+ });
607
+
608
+ it("object default on jsonb column", () => {
609
+ const m = annotate.titled(
610
+ shapes.mapping({
611
+ meta: annotate.as(shapes.unknown(), { default: { a: 1 } }),
612
+ }),
613
+ "t",
614
+ );
615
+ expect(ok(toSQL(m))).toBe(
616
+ `CREATE TABLE IF NOT EXISTS "t" (\n "meta" jsonb NOT NULL DEFAULT '{"a":1}'::jsonb\n);`,
617
+ );
618
+ });
619
+
620
+ it("unsupported object default on non-jsonb column → err", () => {
621
+ const m = annotate.titled(
622
+ shapes.mapping({
623
+ s: annotate.as(shapes.string(), { default: { a: 1 } }),
624
+ }),
625
+ "t",
626
+ );
627
+ expect(err(toSQL(m))).toEqual([
628
+ "field 's': default value of type object is not supported for column type text",
629
+ ]);
630
+ });
631
+ });
632
+
633
+ describe("toSQL — end-to-end users table", (it) => {
634
+ it("realistic users table with enum, vector, jsonb, default", () => {
635
+ const role = annotate.titled(
636
+ shapes.union(shapes.literal("admin"), shapes.literal("user")),
637
+ "user_role",
638
+ );
639
+ const m = annotate.titled(
640
+ shapes.mapping({
641
+ id: annotate.primary(annotate.autoIncremented(shapes.int())),
642
+ email: annotate.unique(
643
+ annotate.pattern(annotate.max(shapes.string(), 255), /.+@.+/),
644
+ ),
645
+ age: shapes.uint8(),
646
+ role,
647
+ embedding: shapes.vector(384, shapes.float32()),
648
+ profile: shapes.mapping({ bio: shapes.string() }),
649
+ created_at: annotate.as(shapes.date(), { default: null }),
650
+ }),
651
+ "users",
652
+ );
653
+
654
+ const sql = ok(toSQL(m));
655
+ expect(sql).toBe(
656
+ [
657
+ `CREATE TYPE "user_role" AS ENUM ('admin', 'user');`,
658
+ "",
659
+ `CREATE TABLE IF NOT EXISTS "users" (`,
660
+ ` "id" bigint NOT NULL PRIMARY KEY GENERATED ALWAYS AS IDENTITY,`,
661
+ ` "email" varchar(255) NOT NULL UNIQUE CHECK ("email" ~ '.+@.+'),`,
662
+ ` "age" smallint NOT NULL CHECK ("age" >= 0) CHECK ("age" <= 255),`,
663
+ ` "role" "user_role" NOT NULL,`,
664
+ ` "embedding" vector(384) NOT NULL,`,
665
+ ` "profile" jsonb NOT NULL,`,
666
+ ` "created_at" timestamptz NOT NULL DEFAULT NULL`,
667
+ `);`,
668
+ ].join("\n"),
669
+ );
670
+ });
671
+ });
672
+
673
+ describe("toSQL — comments", (it) => {
674
+ it("column with description → single -- comment above it", () => {
675
+ const m = annotate.titled(
676
+ shapes.mapping({
677
+ name: annotate.described(shapes.string(), "the user's name"),
678
+ }),
679
+ "t",
680
+ );
681
+ expect(ok(toSQL(m))).toBe(
682
+ [
683
+ `CREATE TABLE IF NOT EXISTS "t" (`,
684
+ ` -- the user's name`,
685
+ ` "name" text NOT NULL`,
686
+ `);`,
687
+ ].join("\n"),
688
+ );
689
+ });
690
+
691
+ it("column with description + comment → /* */ block above it", () => {
692
+ const m = annotate.titled(
693
+ shapes.mapping({
694
+ name: annotate.commented(
695
+ annotate.described(shapes.string(), "display name"),
696
+ "indexed",
697
+ ),
698
+ }),
699
+ "t",
700
+ );
701
+ expect(ok(toSQL(m))).toBe(
702
+ [
703
+ `CREATE TABLE IF NOT EXISTS "t" (`,
704
+ ` /*`,
705
+ ` * display name`,
706
+ ` * indexed`,
707
+ ` */`,
708
+ ` "name" text NOT NULL`,
709
+ `);`,
710
+ ].join("\n"),
711
+ );
712
+ });
713
+
714
+ it("column with format → -- (format: ...) above it", () => {
715
+ const m = annotate.titled(
716
+ shapes.mapping({ email: annotate.format(shapes.string(), "email") }),
717
+ "t",
718
+ );
719
+ expect(ok(toSQL(m))).toBe(
720
+ [
721
+ `CREATE TABLE IF NOT EXISTS "t" (`,
722
+ ` -- (format: email)`,
723
+ ` "email" text NOT NULL`,
724
+ `);`,
725
+ ].join("\n"),
726
+ );
727
+ });
728
+
729
+ it("table-level description → comment before CREATE TABLE", () => {
730
+ const m = annotate.described(
731
+ annotate.titled(shapes.mapping({ id: shapes.int() }), "t"),
732
+ "rows of stuff",
733
+ );
734
+ expect(ok(toSQL(m))).toBe(
735
+ [
736
+ `-- rows of stuff`,
737
+ `CREATE TABLE IF NOT EXISTS "t" (`,
738
+ ` "id" bigint NOT NULL`,
739
+ `);`,
740
+ ].join("\n"),
741
+ );
742
+ });
743
+
744
+ it("comments coexist with native column constraints", () => {
745
+ const m = annotate.titled(
746
+ shapes.mapping({
747
+ id: annotate.described(
748
+ annotate.primary(shapes.int()),
749
+ "the primary key",
750
+ ),
751
+ }),
752
+ "t",
753
+ );
754
+ expect(ok(toSQL(m))).toBe(
755
+ [
756
+ `CREATE TABLE IF NOT EXISTS "t" (`,
757
+ ` -- the primary key`,
758
+ ` "id" bigint NOT NULL PRIMARY KEY`,
759
+ `);`,
760
+ ].join("\n"),
761
+ );
762
+ });
763
+ });
764
+
765
+ describe("toSQL — module / multi-table", (it) => {
766
+ it("emits CREATE TABLE for each table in module", () => {
767
+ const m = shapes.module({
768
+ product: annotate.titled(
769
+ shapes.mapping({
770
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
771
+ name: shapes.string(),
772
+ }),
773
+ "product",
774
+ ),
775
+ order: annotate.titled(
776
+ shapes.mapping({
777
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
778
+ }),
779
+ "order",
780
+ ),
781
+ });
782
+ const sql = ok(toSQL(m));
783
+ expect(sql).toContain(`CREATE TABLE IF NOT EXISTS "product"`);
784
+ expect(sql).toContain(`CREATE TABLE IF NOT EXISTS "order"`);
785
+ });
786
+
787
+ it("emits tables in topological order (referenced first)", () => {
788
+ const m = shapes.module({
789
+ post: annotate.titled(
790
+ shapes.mapping({
791
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
792
+ author_id: annotate.foreign(shapes.int64(), "user", "id"),
793
+ }),
794
+ "post",
795
+ ),
796
+ user: annotate.titled(
797
+ shapes.mapping({
798
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
799
+ }),
800
+ "user",
801
+ ),
802
+ });
803
+ const sql = ok(toSQL(m));
804
+ const userIdx = sql.indexOf(`CREATE TABLE IF NOT EXISTS "user"`);
805
+ const postIdx = sql.indexOf(`CREATE TABLE IF NOT EXISTS "post"`);
806
+ expect(userIdx).toBeGreaterThanOrEqual(0);
807
+ expect(postIdx).toBeGreaterThan(userIdx);
808
+ });
809
+
810
+ it("composite PK → table-level PRIMARY KEY constraint", () => {
811
+ const m = shapes.module({
812
+ ab: annotate.titled(
813
+ shapes.mapping({
814
+ a_id: annotate.primary(annotate.foreign(shapes.int64(), "a", "id")),
815
+ b_id: annotate.primary(annotate.foreign(shapes.int64(), "b", "id")),
816
+ }),
817
+ "ab",
818
+ ),
819
+ a: annotate.titled(
820
+ shapes.mapping({
821
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
822
+ }),
823
+ "a",
824
+ ),
825
+ b: annotate.titled(
826
+ shapes.mapping({
827
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
828
+ }),
829
+ "b",
830
+ ),
831
+ });
832
+ const sql = ok(toSQL(m));
833
+ expect(sql).toContain(`PRIMARY KEY ("a_id", "b_id")`);
834
+ expect(sql).not.toContain(
835
+ `"a_id" bigint NOT NULL REFERENCES "a"("id") PRIMARY KEY`,
836
+ );
837
+ });
838
+
839
+ it("FK cycle → err", () => {
840
+ const m = shapes.module({
841
+ a: annotate.titled(
842
+ shapes.mapping({
843
+ b_ref: annotate.foreign(shapes.int64(), "b", "id"),
844
+ }),
845
+ "a",
846
+ ),
847
+ b: annotate.titled(
848
+ shapes.mapping({
849
+ a_ref: annotate.foreign(shapes.int64(), "a", "id"),
850
+ }),
851
+ "b",
852
+ ),
853
+ });
854
+ expect(err(toSQL(m))[0]).toMatch(/FK cycle/);
855
+ });
856
+
857
+ it("self-reference allowed (1-node cycle)", () => {
858
+ const m = shapes.module({
859
+ category: annotate.titled(
860
+ shapes.mapping({
861
+ id: annotate.autoIncremented(annotate.primary(shapes.int64())),
862
+ parent: annotate.foreign(shapes.int64(), "category", "id"),
863
+ }),
864
+ "category",
865
+ ),
866
+ });
867
+ const sql = ok(toSQL(m));
868
+ expect(sql).toContain(`CREATE TABLE IF NOT EXISTS "category"`);
869
+ expect(sql).toContain(`REFERENCES "category"("id")`);
870
+ });
871
+
872
+ it("module errors are prefixed by table name", () => {
873
+ const m = shapes.module({
874
+ bad: annotate.titled(shapes.mapping({ x: shapes.nil() }), "bad"),
875
+ });
876
+ expect(err(toSQL(m))).toEqual([
877
+ "table 'bad': field 'x': nil cannot be a column",
878
+ ]);
879
+ });
880
+ });