spindb 0.37.2 → 0.38.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 (811) hide show
  1. package/dist/cli/bin.js +9 -0
  2. package/dist/cli/bin.js.map +1 -0
  3. package/dist/cli/commands/attach.js +102 -0
  4. package/dist/cli/commands/attach.js.map +1 -0
  5. package/dist/cli/commands/backup.js +197 -0
  6. package/dist/cli/commands/backup.js.map +1 -0
  7. package/dist/cli/commands/backups.js +190 -0
  8. package/dist/cli/commands/backups.js.map +1 -0
  9. package/dist/cli/commands/clone.js +119 -0
  10. package/dist/cli/commands/clone.js.map +1 -0
  11. package/dist/cli/commands/config.js +276 -0
  12. package/dist/cli/commands/config.js.map +1 -0
  13. package/dist/cli/commands/connect.js +559 -0
  14. package/dist/cli/commands/connect.js.map +1 -0
  15. package/dist/cli/commands/create.js +952 -0
  16. package/dist/cli/commands/create.js.map +1 -0
  17. package/dist/cli/commands/databases.js +485 -0
  18. package/dist/cli/commands/databases.js.map +1 -0
  19. package/dist/cli/commands/delete.js +106 -0
  20. package/dist/cli/commands/delete.js.map +1 -0
  21. package/dist/cli/commands/deps.js +238 -0
  22. package/dist/cli/commands/deps.js.map +1 -0
  23. package/dist/cli/commands/detach.js +81 -0
  24. package/dist/cli/commands/detach.js.map +1 -0
  25. package/dist/cli/commands/doctor.js +567 -0
  26. package/dist/cli/commands/doctor.js.map +1 -0
  27. package/dist/cli/commands/duckdb.js +207 -0
  28. package/dist/cli/commands/duckdb.js.map +1 -0
  29. package/dist/cli/commands/edit.js +524 -0
  30. package/dist/cli/commands/edit.js.map +1 -0
  31. package/dist/cli/commands/engines.js +1414 -0
  32. package/dist/cli/commands/engines.js.map +1 -0
  33. package/dist/cli/commands/export.js +383 -0
  34. package/dist/cli/commands/export.js.map +1 -0
  35. package/dist/cli/commands/info.js +270 -0
  36. package/dist/cli/commands/info.js.map +1 -0
  37. package/dist/cli/commands/list.js +215 -0
  38. package/dist/cli/commands/list.js.map +1 -0
  39. package/dist/cli/commands/logs.js +81 -0
  40. package/dist/cli/commands/logs.js.map +1 -0
  41. package/dist/cli/commands/menu/backup-handlers.js +1202 -0
  42. package/dist/cli/commands/menu/backup-handlers.js.map +1 -0
  43. package/dist/cli/commands/menu/container-handlers.js +1788 -0
  44. package/dist/cli/commands/menu/container-handlers.js.map +1 -0
  45. package/dist/cli/commands/menu/engine-handlers.js +235 -0
  46. package/dist/cli/commands/menu/engine-handlers.js.map +1 -0
  47. package/dist/cli/commands/menu/index.js +266 -0
  48. package/dist/cli/commands/menu/index.js.map +1 -0
  49. package/dist/cli/commands/menu/settings-handlers.js +320 -0
  50. package/dist/cli/commands/menu/settings-handlers.js.map +1 -0
  51. package/dist/cli/commands/menu/shared.js +13 -0
  52. package/dist/cli/commands/menu/shared.js.map +1 -0
  53. package/dist/cli/commands/menu/shell-handlers.js +1573 -0
  54. package/dist/cli/commands/menu/shell-handlers.js.map +1 -0
  55. package/dist/cli/commands/menu/sql-handlers.js +185 -0
  56. package/dist/cli/commands/menu/sql-handlers.js.map +1 -0
  57. package/dist/cli/commands/menu/update-handlers.js +322 -0
  58. package/dist/cli/commands/menu/update-handlers.js.map +1 -0
  59. package/dist/cli/commands/menu/validators.js +9 -0
  60. package/dist/cli/commands/menu/validators.js.map +1 -0
  61. package/dist/cli/commands/ports.js +166 -0
  62. package/dist/cli/commands/ports.js.map +1 -0
  63. package/dist/cli/commands/pull.js +166 -0
  64. package/dist/cli/commands/pull.js.map +1 -0
  65. package/dist/cli/commands/query.js +180 -0
  66. package/dist/cli/commands/query.js.map +1 -0
  67. package/dist/cli/commands/restore.js +428 -0
  68. package/dist/cli/commands/restore.js.map +1 -0
  69. package/dist/cli/commands/run.js +115 -0
  70. package/dist/cli/commands/run.js.map +1 -0
  71. package/dist/cli/commands/self-update.js +99 -0
  72. package/dist/cli/commands/self-update.js.map +1 -0
  73. package/dist/cli/commands/sqlite.js +207 -0
  74. package/dist/cli/commands/sqlite.js.map +1 -0
  75. package/dist/cli/commands/start.js +196 -0
  76. package/dist/cli/commands/start.js.map +1 -0
  77. package/dist/cli/commands/stop.js +182 -0
  78. package/dist/cli/commands/stop.js.map +1 -0
  79. package/dist/cli/commands/url.js +88 -0
  80. package/dist/cli/commands/url.js.map +1 -0
  81. package/dist/cli/commands/users.js +189 -0
  82. package/dist/cli/commands/users.js.map +1 -0
  83. package/dist/cli/commands/version.js +52 -0
  84. package/dist/cli/commands/version.js.map +1 -0
  85. package/dist/cli/commands/which.js +258 -0
  86. package/dist/cli/commands/which.js.map +1 -0
  87. package/dist/cli/constants.js +212 -0
  88. package/dist/cli/constants.js.map +1 -0
  89. package/dist/cli/helpers.js +1120 -0
  90. package/dist/cli/helpers.js.map +1 -0
  91. package/dist/cli/index.js +146 -0
  92. package/dist/cli/index.js.map +1 -0
  93. package/dist/cli/ui/prompts.js +1002 -0
  94. package/dist/cli/ui/prompts.js.map +1 -0
  95. package/dist/cli/ui/spinner.js +74 -0
  96. package/dist/cli/ui/spinner.js.map +1 -0
  97. package/dist/cli/ui/theme.js +99 -0
  98. package/dist/cli/ui/theme.js.map +1 -0
  99. package/dist/cli/utils/file-follower.js +79 -0
  100. package/dist/cli/utils/file-follower.js.map +1 -0
  101. package/dist/config/backup-formats.js +363 -0
  102. package/dist/config/backup-formats.js.map +1 -0
  103. package/dist/config/defaults.js +25 -0
  104. package/dist/config/defaults.js.map +1 -0
  105. package/dist/config/engine-defaults.js +303 -0
  106. package/dist/config/engine-defaults.js.map +1 -0
  107. package/dist/config/engines-registry.js +103 -0
  108. package/dist/config/engines-registry.js.map +1 -0
  109. package/dist/config/os-dependencies.js +767 -0
  110. package/dist/config/os-dependencies.js.map +1 -0
  111. package/dist/config/paths.js +156 -0
  112. package/dist/config/paths.js.map +1 -0
  113. package/dist/config/version.js +3 -0
  114. package/dist/config/version.js.map +1 -0
  115. package/dist/core/backup-restore.js +219 -0
  116. package/dist/core/backup-restore.js.map +1 -0
  117. package/dist/core/base-binary-manager.js +403 -0
  118. package/dist/core/base-binary-manager.js.map +1 -0
  119. package/dist/core/base-document-binary-manager.js +364 -0
  120. package/dist/core/base-document-binary-manager.js.map +1 -0
  121. package/dist/core/base-embedded-binary-manager.js +364 -0
  122. package/dist/core/base-embedded-binary-manager.js.map +1 -0
  123. package/dist/core/base-server-binary-manager.js +368 -0
  124. package/dist/core/base-server-binary-manager.js.map +1 -0
  125. package/dist/core/config-manager.js +495 -0
  126. package/dist/core/config-manager.js.map +1 -0
  127. package/dist/core/container-manager.js +609 -0
  128. package/dist/core/container-manager.js.map +1 -0
  129. package/dist/core/credential-generator.js +67 -0
  130. package/dist/core/credential-generator.js.map +1 -0
  131. package/dist/core/credential-manager.js +211 -0
  132. package/dist/core/credential-manager.js.map +1 -0
  133. package/dist/core/dblab-utils.js +105 -0
  134. package/dist/core/dblab-utils.js.map +1 -0
  135. package/dist/core/dependency-manager.js +359 -0
  136. package/dist/core/dependency-manager.js.map +1 -0
  137. package/dist/core/docker-exporter.js +1077 -0
  138. package/dist/core/docker-exporter.js.map +1 -0
  139. package/dist/core/error-handler.js +295 -0
  140. package/dist/core/error-handler.js.map +1 -0
  141. package/dist/core/fs-error-utils.js +74 -0
  142. package/dist/core/fs-error-utils.js.map +1 -0
  143. package/dist/core/homebrew-version-manager.js +280 -0
  144. package/dist/core/homebrew-version-manager.js.map +1 -0
  145. package/dist/core/hostdb-client.js +252 -0
  146. package/dist/core/hostdb-client.js.map +1 -0
  147. package/dist/core/hostdb-metadata.js +243 -0
  148. package/dist/core/hostdb-metadata.js.map +1 -0
  149. package/dist/core/hostdb-releases-factory.js +161 -0
  150. package/dist/core/hostdb-releases-factory.js.map +1 -0
  151. package/dist/core/library-env.js +88 -0
  152. package/dist/core/library-env.js.map +1 -0
  153. package/dist/core/pgweb-utils.js +53 -0
  154. package/dist/core/pgweb-utils.js.map +1 -0
  155. package/dist/core/platform-service.js +632 -0
  156. package/dist/core/platform-service.js.map +1 -0
  157. package/dist/core/port-manager.js +136 -0
  158. package/dist/core/port-manager.js.map +1 -0
  159. package/dist/core/process-manager.js +445 -0
  160. package/dist/core/process-manager.js.map +1 -0
  161. package/dist/core/pull-manager.js +418 -0
  162. package/dist/core/pull-manager.js.map +1 -0
  163. package/dist/core/query-parser.js +449 -0
  164. package/dist/core/query-parser.js.map +1 -0
  165. package/dist/core/spawn-utils.js +90 -0
  166. package/dist/core/spawn-utils.js.map +1 -0
  167. package/dist/core/start-with-retry.js +90 -0
  168. package/dist/core/start-with-retry.js.map +1 -0
  169. package/dist/core/test-cleanup.js +85 -0
  170. package/dist/core/test-cleanup.js.map +1 -0
  171. package/dist/core/tls-generator.js +84 -0
  172. package/dist/core/tls-generator.js.map +1 -0
  173. package/dist/core/transaction-manager.js +139 -0
  174. package/dist/core/transaction-manager.js.map +1 -0
  175. package/dist/core/update-manager.js +241 -0
  176. package/dist/core/update-manager.js.map +1 -0
  177. package/dist/core/version-migration.js +260 -0
  178. package/dist/core/version-migration.js.map +1 -0
  179. package/dist/core/version-utils.js +91 -0
  180. package/dist/core/version-utils.js.map +1 -0
  181. package/dist/engines/base-engine.js +179 -0
  182. package/dist/engines/base-engine.js.map +1 -0
  183. package/dist/engines/clickhouse/backup.js +289 -0
  184. package/dist/engines/clickhouse/backup.js.map +1 -0
  185. package/dist/engines/clickhouse/binary-manager.js +145 -0
  186. package/dist/engines/clickhouse/binary-manager.js.map +1 -0
  187. package/dist/engines/clickhouse/binary-urls.js +100 -0
  188. package/dist/engines/clickhouse/binary-urls.js.map +1 -0
  189. package/dist/engines/clickhouse/cli-utils.js +143 -0
  190. package/dist/engines/clickhouse/cli-utils.js.map +1 -0
  191. package/dist/engines/clickhouse/hostdb-releases.js +24 -0
  192. package/dist/engines/clickhouse/hostdb-releases.js.map +1 -0
  193. package/dist/engines/clickhouse/index.js +1077 -0
  194. package/dist/engines/clickhouse/index.js.map +1 -0
  195. package/dist/engines/clickhouse/restore.js +335 -0
  196. package/dist/engines/clickhouse/restore.js.map +1 -0
  197. package/dist/engines/clickhouse/version-maps.js +83 -0
  198. package/dist/engines/clickhouse/version-maps.js.map +1 -0
  199. package/dist/engines/clickhouse/version-validator.js +133 -0
  200. package/dist/engines/clickhouse/version-validator.js.map +1 -0
  201. package/dist/engines/cockroachdb/backup.js +261 -0
  202. package/dist/engines/cockroachdb/backup.js.map +1 -0
  203. package/dist/engines/cockroachdb/binary-manager.js +33 -0
  204. package/dist/engines/cockroachdb/binary-manager.js.map +1 -0
  205. package/dist/engines/cockroachdb/binary-urls.js +33 -0
  206. package/dist/engines/cockroachdb/binary-urls.js.map +1 -0
  207. package/dist/engines/cockroachdb/cli-utils.js +338 -0
  208. package/dist/engines/cockroachdb/cli-utils.js.map +1 -0
  209. package/dist/engines/cockroachdb/hostdb-releases.js +21 -0
  210. package/dist/engines/cockroachdb/hostdb-releases.js.map +1 -0
  211. package/dist/engines/cockroachdb/index.js +1016 -0
  212. package/dist/engines/cockroachdb/index.js.map +1 -0
  213. package/dist/engines/cockroachdb/restore.js +323 -0
  214. package/dist/engines/cockroachdb/restore.js.map +1 -0
  215. package/dist/engines/cockroachdb/version-maps.js +37 -0
  216. package/dist/engines/cockroachdb/version-maps.js.map +1 -0
  217. package/dist/engines/couchdb/api-client.js +64 -0
  218. package/dist/engines/couchdb/api-client.js.map +1 -0
  219. package/dist/engines/couchdb/backup.js +90 -0
  220. package/dist/engines/couchdb/backup.js.map +1 -0
  221. package/dist/engines/couchdb/binary-manager.js +62 -0
  222. package/dist/engines/couchdb/binary-manager.js.map +1 -0
  223. package/dist/engines/couchdb/binary-urls.js +92 -0
  224. package/dist/engines/couchdb/binary-urls.js.map +1 -0
  225. package/dist/engines/couchdb/hostdb-releases.js +21 -0
  226. package/dist/engines/couchdb/hostdb-releases.js.map +1 -0
  227. package/dist/engines/couchdb/index.js +1043 -0
  228. package/dist/engines/couchdb/index.js.map +1 -0
  229. package/dist/engines/couchdb/restore.js +198 -0
  230. package/dist/engines/couchdb/restore.js.map +1 -0
  231. package/dist/engines/couchdb/version-maps.js +67 -0
  232. package/dist/engines/couchdb/version-maps.js.map +1 -0
  233. package/dist/engines/couchdb/version-validator.js +88 -0
  234. package/dist/engines/couchdb/version-validator.js.map +1 -0
  235. package/dist/engines/duckdb/binary-manager.js +33 -0
  236. package/dist/engines/duckdb/binary-manager.js.map +1 -0
  237. package/{engines/duckdb/binary-urls.ts → dist/engines/duckdb/binary-urls.js} +11 -16
  238. package/dist/engines/duckdb/binary-urls.js.map +1 -0
  239. package/dist/engines/duckdb/hostdb-releases.js +21 -0
  240. package/dist/engines/duckdb/hostdb-releases.js.map +1 -0
  241. package/dist/engines/duckdb/index.js +594 -0
  242. package/dist/engines/duckdb/index.js.map +1 -0
  243. package/dist/engines/duckdb/registry.js +265 -0
  244. package/dist/engines/duckdb/registry.js.map +1 -0
  245. package/dist/engines/duckdb/scanner.js +12 -0
  246. package/dist/engines/duckdb/scanner.js.map +1 -0
  247. package/dist/engines/duckdb/version-maps.js +67 -0
  248. package/dist/engines/duckdb/version-maps.js.map +1 -0
  249. package/dist/engines/duckdb/version-validator.js +62 -0
  250. package/dist/engines/duckdb/version-validator.js.map +1 -0
  251. package/dist/engines/ferretdb/backup.js +170 -0
  252. package/dist/engines/ferretdb/backup.js.map +1 -0
  253. package/dist/engines/ferretdb/binary-manager.js +765 -0
  254. package/dist/engines/ferretdb/binary-manager.js.map +1 -0
  255. package/dist/engines/ferretdb/binary-urls.js +135 -0
  256. package/dist/engines/ferretdb/binary-urls.js.map +1 -0
  257. package/dist/engines/ferretdb/index.js +1517 -0
  258. package/dist/engines/ferretdb/index.js.map +1 -0
  259. package/dist/engines/ferretdb/restore.js +310 -0
  260. package/dist/engines/ferretdb/restore.js.map +1 -0
  261. package/{engines/ferretdb/version-maps.ts → dist/engines/ferretdb/version-maps.js} +62 -79
  262. package/dist/engines/ferretdb/version-maps.js.map +1 -0
  263. package/dist/engines/file-based-utils.js +184 -0
  264. package/dist/engines/file-based-utils.js.map +1 -0
  265. package/dist/engines/index.js +124 -0
  266. package/dist/engines/index.js.map +1 -0
  267. package/dist/engines/influxdb/api-client.js +54 -0
  268. package/dist/engines/influxdb/api-client.js.map +1 -0
  269. package/dist/engines/influxdb/backup.js +119 -0
  270. package/dist/engines/influxdb/backup.js.map +1 -0
  271. package/dist/engines/influxdb/binary-manager.js +87 -0
  272. package/dist/engines/influxdb/binary-manager.js.map +1 -0
  273. package/dist/engines/influxdb/binary-urls.js +56 -0
  274. package/dist/engines/influxdb/binary-urls.js.map +1 -0
  275. package/dist/engines/influxdb/hostdb-releases.js +21 -0
  276. package/dist/engines/influxdb/hostdb-releases.js.map +1 -0
  277. package/dist/engines/influxdb/index.js +962 -0
  278. package/dist/engines/influxdb/index.js.map +1 -0
  279. package/dist/engines/influxdb/restore.js +329 -0
  280. package/dist/engines/influxdb/restore.js.map +1 -0
  281. package/dist/engines/influxdb/version-maps.js +64 -0
  282. package/dist/engines/influxdb/version-maps.js.map +1 -0
  283. package/dist/engines/influxdb/version-validator.js +109 -0
  284. package/dist/engines/influxdb/version-validator.js.map +1 -0
  285. package/dist/engines/mariadb/backup.js +178 -0
  286. package/dist/engines/mariadb/backup.js.map +1 -0
  287. package/dist/engines/mariadb/binary-manager.js +33 -0
  288. package/dist/engines/mariadb/binary-manager.js.map +1 -0
  289. package/{engines/mariadb/binary-urls.ts → dist/engines/mariadb/binary-urls.js} +38 -55
  290. package/dist/engines/mariadb/binary-urls.js.map +1 -0
  291. package/dist/engines/mariadb/hostdb-releases.js +21 -0
  292. package/dist/engines/mariadb/hostdb-releases.js.map +1 -0
  293. package/dist/engines/mariadb/index.js +1011 -0
  294. package/dist/engines/mariadb/index.js.map +1 -0
  295. package/dist/engines/mariadb/restore.js +322 -0
  296. package/dist/engines/mariadb/restore.js.map +1 -0
  297. package/dist/engines/mariadb/version-maps.js +63 -0
  298. package/dist/engines/mariadb/version-maps.js.map +1 -0
  299. package/dist/engines/mariadb/version-validator.js +143 -0
  300. package/dist/engines/mariadb/version-validator.js.map +1 -0
  301. package/dist/engines/meilisearch/api-client.js +50 -0
  302. package/dist/engines/meilisearch/api-client.js.map +1 -0
  303. package/dist/engines/meilisearch/backup.js +167 -0
  304. package/dist/engines/meilisearch/backup.js.map +1 -0
  305. package/dist/engines/meilisearch/binary-manager.js +31 -0
  306. package/dist/engines/meilisearch/binary-manager.js.map +1 -0
  307. package/dist/engines/meilisearch/binary-urls.js +56 -0
  308. package/dist/engines/meilisearch/binary-urls.js.map +1 -0
  309. package/dist/engines/meilisearch/hostdb-releases.js +21 -0
  310. package/dist/engines/meilisearch/hostdb-releases.js.map +1 -0
  311. package/dist/engines/meilisearch/index.js +992 -0
  312. package/dist/engines/meilisearch/index.js.map +1 -0
  313. package/dist/engines/meilisearch/restore.js +167 -0
  314. package/dist/engines/meilisearch/restore.js.map +1 -0
  315. package/dist/engines/meilisearch/version-maps.js +67 -0
  316. package/dist/engines/meilisearch/version-maps.js.map +1 -0
  317. package/dist/engines/meilisearch/version-validator.js +109 -0
  318. package/dist/engines/meilisearch/version-validator.js.map +1 -0
  319. package/dist/engines/mongodb/backup.js +109 -0
  320. package/dist/engines/mongodb/backup.js.map +1 -0
  321. package/dist/engines/mongodb/binary-manager.js +36 -0
  322. package/dist/engines/mongodb/binary-manager.js.map +1 -0
  323. package/dist/engines/mongodb/binary-urls.js +46 -0
  324. package/dist/engines/mongodb/binary-urls.js.map +1 -0
  325. package/dist/engines/mongodb/cli-utils.js +131 -0
  326. package/dist/engines/mongodb/cli-utils.js.map +1 -0
  327. package/dist/engines/mongodb/hostdb-releases.js +77 -0
  328. package/dist/engines/mongodb/hostdb-releases.js.map +1 -0
  329. package/dist/engines/mongodb/index.js +873 -0
  330. package/dist/engines/mongodb/index.js.map +1 -0
  331. package/dist/engines/mongodb/restore.js +276 -0
  332. package/dist/engines/mongodb/restore.js.map +1 -0
  333. package/dist/engines/mongodb/version-maps.js +79 -0
  334. package/dist/engines/mongodb/version-maps.js.map +1 -0
  335. package/dist/engines/mongodb/version-validator.js +133 -0
  336. package/dist/engines/mongodb/version-validator.js.map +1 -0
  337. package/dist/engines/mysql/backup.js +210 -0
  338. package/dist/engines/mysql/backup.js.map +1 -0
  339. package/dist/engines/mysql/binary-detection.js +325 -0
  340. package/dist/engines/mysql/binary-detection.js.map +1 -0
  341. package/dist/engines/mysql/binary-manager.js +30 -0
  342. package/dist/engines/mysql/binary-manager.js.map +1 -0
  343. package/dist/engines/mysql/binary-urls.js +87 -0
  344. package/dist/engines/mysql/binary-urls.js.map +1 -0
  345. package/{engines/mysql/hostdb-releases.ts → dist/engines/mysql/hostdb-releases.js} +20 -23
  346. package/dist/engines/mysql/hostdb-releases.js.map +1 -0
  347. package/dist/engines/mysql/index.js +1066 -0
  348. package/dist/engines/mysql/index.js.map +1 -0
  349. package/dist/engines/mysql/restore.js +361 -0
  350. package/dist/engines/mysql/restore.js.map +1 -0
  351. package/dist/engines/mysql/version-maps.js +79 -0
  352. package/dist/engines/mysql/version-maps.js.map +1 -0
  353. package/dist/engines/mysql/version-validator.js +266 -0
  354. package/dist/engines/mysql/version-validator.js.map +1 -0
  355. package/dist/engines/postgresql/backup.js +118 -0
  356. package/dist/engines/postgresql/backup.js.map +1 -0
  357. package/dist/engines/postgresql/binary-manager.js +85 -0
  358. package/dist/engines/postgresql/binary-manager.js.map +1 -0
  359. package/dist/engines/postgresql/binary-urls.js +80 -0
  360. package/dist/engines/postgresql/binary-urls.js.map +1 -0
  361. package/dist/engines/postgresql/hostdb-releases.js +21 -0
  362. package/dist/engines/postgresql/hostdb-releases.js.map +1 -0
  363. package/dist/engines/postgresql/index.js +852 -0
  364. package/dist/engines/postgresql/index.js.map +1 -0
  365. package/dist/engines/postgresql/remote-version.js +109 -0
  366. package/dist/engines/postgresql/remote-version.js.map +1 -0
  367. package/dist/engines/postgresql/restore.js +254 -0
  368. package/dist/engines/postgresql/restore.js.map +1 -0
  369. package/dist/engines/postgresql/version-maps.js +73 -0
  370. package/dist/engines/postgresql/version-maps.js.map +1 -0
  371. package/dist/engines/postgresql/version-validator.js +286 -0
  372. package/dist/engines/postgresql/version-validator.js.map +1 -0
  373. package/dist/engines/qdrant/api-client.js +50 -0
  374. package/dist/engines/qdrant/api-client.js.map +1 -0
  375. package/dist/engines/qdrant/backup.js +115 -0
  376. package/dist/engines/qdrant/backup.js.map +1 -0
  377. package/dist/engines/qdrant/binary-manager.js +31 -0
  378. package/dist/engines/qdrant/binary-manager.js.map +1 -0
  379. package/dist/engines/qdrant/binary-urls.js +92 -0
  380. package/dist/engines/qdrant/binary-urls.js.map +1 -0
  381. package/dist/engines/qdrant/cli-utils.js +39 -0
  382. package/dist/engines/qdrant/cli-utils.js.map +1 -0
  383. package/dist/engines/qdrant/hostdb-releases.js +21 -0
  384. package/dist/engines/qdrant/hostdb-releases.js.map +1 -0
  385. package/dist/engines/qdrant/index.js +1002 -0
  386. package/dist/engines/qdrant/index.js.map +1 -0
  387. package/dist/engines/qdrant/restore.js +154 -0
  388. package/dist/engines/qdrant/restore.js.map +1 -0
  389. package/dist/engines/qdrant/version-maps.js +67 -0
  390. package/dist/engines/qdrant/version-maps.js.map +1 -0
  391. package/dist/engines/qdrant/version-validator.js +109 -0
  392. package/dist/engines/qdrant/version-validator.js.map +1 -0
  393. package/dist/engines/questdb/backup.js +191 -0
  394. package/dist/engines/questdb/backup.js.map +1 -0
  395. package/dist/engines/questdb/binary-manager.js +247 -0
  396. package/dist/engines/questdb/binary-manager.js.map +1 -0
  397. package/dist/engines/questdb/binary-urls.js +27 -0
  398. package/dist/engines/questdb/binary-urls.js.map +1 -0
  399. package/dist/engines/questdb/hostdb-releases.js +21 -0
  400. package/dist/engines/questdb/hostdb-releases.js.map +1 -0
  401. package/dist/engines/questdb/index.js +814 -0
  402. package/dist/engines/questdb/index.js.map +1 -0
  403. package/dist/engines/questdb/restore.js +202 -0
  404. package/dist/engines/questdb/restore.js.map +1 -0
  405. package/dist/engines/questdb/version-maps.js +33 -0
  406. package/dist/engines/questdb/version-maps.js.map +1 -0
  407. package/dist/engines/questdb/version-validator.js +99 -0
  408. package/dist/engines/questdb/version-validator.js.map +1 -0
  409. package/dist/engines/redis/backup.js +292 -0
  410. package/dist/engines/redis/backup.js.map +1 -0
  411. package/dist/engines/redis/binary-manager.js +32 -0
  412. package/dist/engines/redis/binary-manager.js.map +1 -0
  413. package/dist/engines/redis/binary-urls.js +96 -0
  414. package/dist/engines/redis/binary-urls.js.map +1 -0
  415. package/dist/engines/redis/cli-utils.js +38 -0
  416. package/dist/engines/redis/cli-utils.js.map +1 -0
  417. package/dist/engines/redis/hostdb-releases.js +21 -0
  418. package/dist/engines/redis/hostdb-releases.js.map +1 -0
  419. package/dist/engines/redis/index.js +1263 -0
  420. package/dist/engines/redis/index.js.map +1 -0
  421. package/dist/engines/redis/restore.js +338 -0
  422. package/dist/engines/redis/restore.js.map +1 -0
  423. package/dist/engines/redis/version-maps.js +70 -0
  424. package/dist/engines/redis/version-maps.js.map +1 -0
  425. package/dist/engines/redis/version-validator.js +109 -0
  426. package/dist/engines/redis/version-validator.js.map +1 -0
  427. package/dist/engines/sqlite/binary-manager.js +39 -0
  428. package/dist/engines/sqlite/binary-manager.js.map +1 -0
  429. package/{engines/sqlite/binary-urls.ts → dist/engines/sqlite/binary-urls.js} +11 -16
  430. package/dist/engines/sqlite/binary-urls.js.map +1 -0
  431. package/dist/engines/sqlite/hostdb-releases.js +21 -0
  432. package/dist/engines/sqlite/hostdb-releases.js.map +1 -0
  433. package/dist/engines/sqlite/index.js +493 -0
  434. package/dist/engines/sqlite/index.js.map +1 -0
  435. package/dist/engines/sqlite/registry.js +163 -0
  436. package/dist/engines/sqlite/registry.js.map +1 -0
  437. package/dist/engines/sqlite/scanner.js +12 -0
  438. package/dist/engines/sqlite/scanner.js.map +1 -0
  439. package/dist/engines/sqlite/version-maps.js +57 -0
  440. package/dist/engines/sqlite/version-maps.js.map +1 -0
  441. package/dist/engines/surrealdb/backup.js +97 -0
  442. package/dist/engines/surrealdb/backup.js.map +1 -0
  443. package/dist/engines/surrealdb/binary-manager.js +33 -0
  444. package/dist/engines/surrealdb/binary-manager.js.map +1 -0
  445. package/dist/engines/surrealdb/binary-urls.js +33 -0
  446. package/dist/engines/surrealdb/binary-urls.js.map +1 -0
  447. package/dist/engines/surrealdb/cli-utils.js +147 -0
  448. package/dist/engines/surrealdb/cli-utils.js.map +1 -0
  449. package/dist/engines/surrealdb/hostdb-releases.js +21 -0
  450. package/dist/engines/surrealdb/hostdb-releases.js.map +1 -0
  451. package/dist/engines/surrealdb/index.js +1022 -0
  452. package/dist/engines/surrealdb/index.js.map +1 -0
  453. package/dist/engines/surrealdb/restore.js +224 -0
  454. package/dist/engines/surrealdb/restore.js.map +1 -0
  455. package/dist/engines/surrealdb/version-maps.js +36 -0
  456. package/dist/engines/surrealdb/version-maps.js.map +1 -0
  457. package/dist/engines/tigerbeetle/backup.js +36 -0
  458. package/dist/engines/tigerbeetle/backup.js.map +1 -0
  459. package/dist/engines/tigerbeetle/binary-manager.js +72 -0
  460. package/dist/engines/tigerbeetle/binary-manager.js.map +1 -0
  461. package/dist/engines/tigerbeetle/binary-urls.js +49 -0
  462. package/dist/engines/tigerbeetle/binary-urls.js.map +1 -0
  463. package/dist/engines/tigerbeetle/hostdb-releases.js +21 -0
  464. package/dist/engines/tigerbeetle/hostdb-releases.js.map +1 -0
  465. package/dist/engines/tigerbeetle/index.js +559 -0
  466. package/dist/engines/tigerbeetle/index.js.map +1 -0
  467. package/dist/engines/tigerbeetle/restore.js +91 -0
  468. package/dist/engines/tigerbeetle/restore.js.map +1 -0
  469. package/{engines/tigerbeetle/version-maps.ts → dist/engines/tigerbeetle/version-maps.js} +22 -31
  470. package/dist/engines/tigerbeetle/version-maps.js.map +1 -0
  471. package/dist/engines/tigerbeetle/version-validator.js +108 -0
  472. package/dist/engines/tigerbeetle/version-validator.js.map +1 -0
  473. package/dist/engines/typedb/backup.js +129 -0
  474. package/dist/engines/typedb/backup.js.map +1 -0
  475. package/dist/engines/typedb/binary-manager.js +151 -0
  476. package/dist/engines/typedb/binary-manager.js.map +1 -0
  477. package/dist/engines/typedb/binary-urls.js +33 -0
  478. package/dist/engines/typedb/binary-urls.js.map +1 -0
  479. package/dist/engines/typedb/cli-utils.js +163 -0
  480. package/dist/engines/typedb/cli-utils.js.map +1 -0
  481. package/dist/engines/typedb/hostdb-releases.js +21 -0
  482. package/dist/engines/typedb/hostdb-releases.js.map +1 -0
  483. package/dist/engines/typedb/index.js +1003 -0
  484. package/dist/engines/typedb/index.js.map +1 -0
  485. package/dist/engines/typedb/restore.js +279 -0
  486. package/dist/engines/typedb/restore.js.map +1 -0
  487. package/dist/engines/typedb/version-maps.js +40 -0
  488. package/dist/engines/typedb/version-maps.js.map +1 -0
  489. package/dist/engines/typedb/version-validator.js +103 -0
  490. package/dist/engines/typedb/version-validator.js.map +1 -0
  491. package/dist/engines/valkey/backup.js +292 -0
  492. package/dist/engines/valkey/backup.js.map +1 -0
  493. package/dist/engines/valkey/binary-manager.js +33 -0
  494. package/dist/engines/valkey/binary-manager.js.map +1 -0
  495. package/dist/engines/valkey/binary-urls.js +98 -0
  496. package/dist/engines/valkey/binary-urls.js.map +1 -0
  497. package/dist/engines/valkey/cli-utils.js +38 -0
  498. package/dist/engines/valkey/cli-utils.js.map +1 -0
  499. package/dist/engines/valkey/hostdb-releases.js +21 -0
  500. package/dist/engines/valkey/hostdb-releases.js.map +1 -0
  501. package/dist/engines/valkey/index.js +1257 -0
  502. package/dist/engines/valkey/index.js.map +1 -0
  503. package/dist/engines/valkey/restore.js +340 -0
  504. package/dist/engines/valkey/restore.js.map +1 -0
  505. package/dist/engines/valkey/version-maps.js +70 -0
  506. package/dist/engines/valkey/version-maps.js.map +1 -0
  507. package/dist/engines/valkey/version-validator.js +112 -0
  508. package/dist/engines/valkey/version-validator.js.map +1 -0
  509. package/dist/engines/weaviate/api-client.js +50 -0
  510. package/dist/engines/weaviate/api-client.js.map +1 -0
  511. package/dist/engines/weaviate/backup.js +95 -0
  512. package/dist/engines/weaviate/backup.js.map +1 -0
  513. package/dist/engines/weaviate/binary-manager.js +58 -0
  514. package/dist/engines/weaviate/binary-manager.js.map +1 -0
  515. package/dist/engines/weaviate/binary-urls.js +92 -0
  516. package/dist/engines/weaviate/binary-urls.js.map +1 -0
  517. package/dist/engines/weaviate/cli-utils.js +39 -0
  518. package/dist/engines/weaviate/cli-utils.js.map +1 -0
  519. package/dist/engines/weaviate/hostdb-releases.js +21 -0
  520. package/dist/engines/weaviate/hostdb-releases.js.map +1 -0
  521. package/dist/engines/weaviate/index.js +871 -0
  522. package/dist/engines/weaviate/index.js.map +1 -0
  523. package/dist/engines/weaviate/restore.js +185 -0
  524. package/dist/engines/weaviate/restore.js.map +1 -0
  525. package/dist/engines/weaviate/version-maps.js +67 -0
  526. package/dist/engines/weaviate/version-maps.js.map +1 -0
  527. package/dist/engines/weaviate/version-validator.js +109 -0
  528. package/dist/engines/weaviate/version-validator.js.map +1 -0
  529. package/dist/types/index.js +102 -0
  530. package/dist/types/index.js.map +1 -0
  531. package/package.json +12 -9
  532. package/bin/cli.js +0 -68
  533. package/cli/bin.ts +0 -10
  534. package/cli/commands/attach.ts +0 -139
  535. package/cli/commands/backup.ts +0 -290
  536. package/cli/commands/backups.ts +0 -247
  537. package/cli/commands/clone.ts +0 -159
  538. package/cli/commands/config.ts +0 -367
  539. package/cli/commands/connect.ts +0 -684
  540. package/cli/commands/create.ts +0 -1201
  541. package/cli/commands/databases.ts +0 -630
  542. package/cli/commands/delete.ts +0 -133
  543. package/cli/commands/deps.ts +0 -342
  544. package/cli/commands/detach.ts +0 -107
  545. package/cli/commands/doctor.ts +0 -689
  546. package/cli/commands/duckdb.ts +0 -273
  547. package/cli/commands/edit.ts +0 -683
  548. package/cli/commands/engines.ts +0 -1914
  549. package/cli/commands/export.ts +0 -544
  550. package/cli/commands/info.ts +0 -340
  551. package/cli/commands/list.ts +0 -284
  552. package/cli/commands/logs.ts +0 -102
  553. package/cli/commands/menu/backup-handlers.ts +0 -1571
  554. package/cli/commands/menu/container-handlers.ts +0 -2288
  555. package/cli/commands/menu/engine-handlers.ts +0 -355
  556. package/cli/commands/menu/index.ts +0 -342
  557. package/cli/commands/menu/settings-handlers.ts +0 -365
  558. package/cli/commands/menu/shared.ts +0 -23
  559. package/cli/commands/menu/shell-handlers.ts +0 -1811
  560. package/cli/commands/menu/sql-handlers.ts +0 -231
  561. package/cli/commands/menu/update-handlers.ts +0 -378
  562. package/cli/commands/menu/validators.ts +0 -8
  563. package/cli/commands/ports.ts +0 -211
  564. package/cli/commands/pull.ts +0 -223
  565. package/cli/commands/query.ts +0 -241
  566. package/cli/commands/restore.ts +0 -587
  567. package/cli/commands/run.ts +0 -178
  568. package/cli/commands/self-update.ts +0 -121
  569. package/cli/commands/sqlite.ts +0 -273
  570. package/cli/commands/start.ts +0 -218
  571. package/cli/commands/stop.ts +0 -241
  572. package/cli/commands/url.ts +0 -104
  573. package/cli/commands/users.ts +0 -264
  574. package/cli/commands/version.ts +0 -55
  575. package/cli/commands/which.ts +0 -290
  576. package/cli/constants.ts +0 -233
  577. package/cli/helpers.ts +0 -1593
  578. package/cli/index.ts +0 -162
  579. package/cli/ui/prompts.ts +0 -1525
  580. package/cli/ui/spinner.ts +0 -88
  581. package/cli/ui/theme.ts +0 -128
  582. package/cli/utils/file-follower.ts +0 -93
  583. package/config/backup-formats.ts +0 -446
  584. package/config/defaults.ts +0 -56
  585. package/config/engine-defaults.ts +0 -336
  586. package/config/engines-registry.ts +0 -150
  587. package/config/engines.schema.json +0 -135
  588. package/config/os-dependencies.ts +0 -888
  589. package/config/paths.ts +0 -200
  590. package/core/backup-restore.ts +0 -330
  591. package/core/base-binary-manager.ts +0 -562
  592. package/core/base-document-binary-manager.ts +0 -523
  593. package/core/base-embedded-binary-manager.ts +0 -547
  594. package/core/base-server-binary-manager.ts +0 -523
  595. package/core/config-manager.ts +0 -652
  596. package/core/container-manager.ts +0 -787
  597. package/core/credential-generator.ts +0 -93
  598. package/core/credential-manager.ts +0 -259
  599. package/core/dblab-utils.ts +0 -113
  600. package/core/dependency-manager.ts +0 -512
  601. package/core/docker-exporter.ts +0 -1345
  602. package/core/error-handler.ts +0 -419
  603. package/core/fs-error-utils.ts +0 -82
  604. package/core/homebrew-version-manager.ts +0 -352
  605. package/core/hostdb-client.ts +0 -344
  606. package/core/hostdb-metadata.ts +0 -350
  607. package/core/hostdb-releases-factory.ts +0 -237
  608. package/core/library-env.ts +0 -118
  609. package/core/pgweb-utils.ts +0 -62
  610. package/core/platform-service.ts +0 -829
  611. package/core/port-manager.ts +0 -165
  612. package/core/process-manager.ts +0 -576
  613. package/core/pull-manager.ts +0 -511
  614. package/core/query-parser.ts +0 -514
  615. package/core/spawn-utils.ts +0 -122
  616. package/core/start-with-retry.ts +0 -130
  617. package/core/test-cleanup.ts +0 -108
  618. package/core/tls-generator.ts +0 -116
  619. package/core/transaction-manager.ts +0 -158
  620. package/core/update-manager.ts +0 -308
  621. package/core/version-migration.ts +0 -346
  622. package/core/version-utils.ts +0 -104
  623. package/engines/base-engine.ts +0 -340
  624. package/engines/clickhouse/README.md +0 -231
  625. package/engines/clickhouse/backup.ts +0 -398
  626. package/engines/clickhouse/binary-manager.ts +0 -201
  627. package/engines/clickhouse/binary-urls.ts +0 -125
  628. package/engines/clickhouse/cli-utils.ts +0 -176
  629. package/engines/clickhouse/hostdb-releases.ts +0 -30
  630. package/engines/clickhouse/index.ts +0 -1345
  631. package/engines/clickhouse/restore.ts +0 -466
  632. package/engines/clickhouse/version-maps.ts +0 -95
  633. package/engines/clickhouse/version-validator.ts +0 -154
  634. package/engines/cockroachdb/README.md +0 -170
  635. package/engines/cockroachdb/backup.ts +0 -376
  636. package/engines/cockroachdb/binary-manager.ts +0 -45
  637. package/engines/cockroachdb/binary-urls.ts +0 -40
  638. package/engines/cockroachdb/cli-utils.ts +0 -384
  639. package/engines/cockroachdb/hostdb-releases.ts +0 -26
  640. package/engines/cockroachdb/index.ts +0 -1276
  641. package/engines/cockroachdb/restore.ts +0 -455
  642. package/engines/cockroachdb/version-maps.ts +0 -42
  643. package/engines/couchdb/README.md +0 -257
  644. package/engines/couchdb/api-client.ts +0 -81
  645. package/engines/couchdb/backup.ts +0 -137
  646. package/engines/couchdb/binary-manager.ts +0 -86
  647. package/engines/couchdb/binary-urls.ts +0 -115
  648. package/engines/couchdb/hostdb-releases.ts +0 -23
  649. package/engines/couchdb/index.ts +0 -1429
  650. package/engines/couchdb/restore.ts +0 -290
  651. package/engines/couchdb/version-maps.ts +0 -78
  652. package/engines/couchdb/version-validator.ts +0 -111
  653. package/engines/duckdb/README.md +0 -154
  654. package/engines/duckdb/binary-manager.ts +0 -45
  655. package/engines/duckdb/hostdb-releases.ts +0 -23
  656. package/engines/duckdb/index.ts +0 -749
  657. package/engines/duckdb/registry.ts +0 -303
  658. package/engines/duckdb/scanner.ts +0 -22
  659. package/engines/duckdb/version-maps.ts +0 -78
  660. package/engines/duckdb/version-validator.ts +0 -78
  661. package/engines/ferretdb/README.md +0 -262
  662. package/engines/ferretdb/backup.ts +0 -173
  663. package/engines/ferretdb/binary-manager.ts +0 -1095
  664. package/engines/ferretdb/binary-urls.ts +0 -183
  665. package/engines/ferretdb/index.ts +0 -1907
  666. package/engines/ferretdb/restore.ts +0 -357
  667. package/engines/file-based-utils.ts +0 -262
  668. package/engines/index.ts +0 -131
  669. package/engines/influxdb/README.md +0 -180
  670. package/engines/influxdb/api-client.ts +0 -64
  671. package/engines/influxdb/backup.ts +0 -160
  672. package/engines/influxdb/binary-manager.ts +0 -110
  673. package/engines/influxdb/binary-urls.ts +0 -69
  674. package/engines/influxdb/hostdb-releases.ts +0 -23
  675. package/engines/influxdb/index.ts +0 -1272
  676. package/engines/influxdb/restore.ts +0 -417
  677. package/engines/influxdb/version-maps.ts +0 -75
  678. package/engines/influxdb/version-validator.ts +0 -128
  679. package/engines/mariadb/README.md +0 -141
  680. package/engines/mariadb/backup.ts +0 -233
  681. package/engines/mariadb/binary-manager.ts +0 -45
  682. package/engines/mariadb/hostdb-releases.ts +0 -23
  683. package/engines/mariadb/index.ts +0 -1300
  684. package/engines/mariadb/restore.ts +0 -447
  685. package/engines/mariadb/version-maps.ts +0 -72
  686. package/engines/mariadb/version-validator.ts +0 -181
  687. package/engines/meilisearch/README.md +0 -255
  688. package/engines/meilisearch/api-client.ts +0 -61
  689. package/engines/meilisearch/backup.ts +0 -233
  690. package/engines/meilisearch/binary-manager.ts +0 -43
  691. package/engines/meilisearch/binary-urls.ts +0 -69
  692. package/engines/meilisearch/hostdb-releases.ts +0 -26
  693. package/engines/meilisearch/index.ts +0 -1292
  694. package/engines/meilisearch/restore.ts +0 -219
  695. package/engines/meilisearch/version-maps.ts +0 -78
  696. package/engines/meilisearch/version-validator.ts +0 -128
  697. package/engines/mongodb/README.md +0 -162
  698. package/engines/mongodb/backup.ts +0 -127
  699. package/engines/mongodb/binary-manager.ts +0 -48
  700. package/engines/mongodb/binary-urls.ts +0 -63
  701. package/engines/mongodb/cli-utils.ts +0 -171
  702. package/engines/mongodb/hostdb-releases.ts +0 -91
  703. package/engines/mongodb/index.ts +0 -1118
  704. package/engines/mongodb/restore.ts +0 -361
  705. package/engines/mongodb/version-maps.ts +0 -91
  706. package/engines/mongodb/version-validator.ts +0 -160
  707. package/engines/mysql/README.md +0 -142
  708. package/engines/mysql/backup.ts +0 -270
  709. package/engines/mysql/binary-detection.ts +0 -408
  710. package/engines/mysql/binary-manager.ts +0 -42
  711. package/engines/mysql/binary-urls.ts +0 -104
  712. package/engines/mysql/index.ts +0 -1361
  713. package/engines/mysql/restore.ts +0 -500
  714. package/engines/mysql/version-maps.ts +0 -91
  715. package/engines/mysql/version-validator.ts +0 -369
  716. package/engines/postgresql/README.md +0 -158
  717. package/engines/postgresql/backup.ts +0 -151
  718. package/engines/postgresql/binary-manager.ts +0 -114
  719. package/engines/postgresql/binary-urls.ts +0 -99
  720. package/engines/postgresql/hostdb-releases.ts +0 -26
  721. package/engines/postgresql/index.ts +0 -1143
  722. package/engines/postgresql/remote-version.ts +0 -161
  723. package/engines/postgresql/restore.ts +0 -342
  724. package/engines/postgresql/version-maps.ts +0 -83
  725. package/engines/postgresql/version-validator.ts +0 -413
  726. package/engines/qdrant/README.md +0 -222
  727. package/engines/qdrant/api-client.ts +0 -61
  728. package/engines/qdrant/backup.ts +0 -165
  729. package/engines/qdrant/binary-manager.ts +0 -43
  730. package/engines/qdrant/binary-urls.ts +0 -115
  731. package/engines/qdrant/cli-utils.ts +0 -43
  732. package/engines/qdrant/hostdb-releases.ts +0 -23
  733. package/engines/qdrant/index.ts +0 -1312
  734. package/engines/qdrant/restore.ts +0 -203
  735. package/engines/qdrant/version-maps.ts +0 -78
  736. package/engines/qdrant/version-validator.ts +0 -128
  737. package/engines/questdb/README.md +0 -334
  738. package/engines/questdb/backup.ts +0 -220
  739. package/engines/questdb/binary-manager.ts +0 -310
  740. package/engines/questdb/binary-urls.ts +0 -34
  741. package/engines/questdb/hostdb-releases.ts +0 -23
  742. package/engines/questdb/index.ts +0 -1023
  743. package/engines/questdb/restore.ts +0 -260
  744. package/engines/questdb/version-maps.ts +0 -37
  745. package/engines/questdb/version-validator.ts +0 -121
  746. package/engines/redis/README.md +0 -173
  747. package/engines/redis/backup.ts +0 -389
  748. package/engines/redis/binary-manager.ts +0 -44
  749. package/engines/redis/binary-urls.ts +0 -117
  750. package/engines/redis/cli-utils.ts +0 -42
  751. package/engines/redis/hostdb-releases.ts +0 -23
  752. package/engines/redis/index.ts +0 -1583
  753. package/engines/redis/restore.ts +0 -443
  754. package/engines/redis/version-maps.ts +0 -81
  755. package/engines/redis/version-validator.ts +0 -131
  756. package/engines/sqlite/README.md +0 -162
  757. package/engines/sqlite/binary-manager.ts +0 -52
  758. package/engines/sqlite/hostdb-releases.ts +0 -23
  759. package/engines/sqlite/index.ts +0 -641
  760. package/engines/sqlite/registry.ts +0 -198
  761. package/engines/sqlite/scanner.ts +0 -22
  762. package/engines/sqlite/version-maps.ts +0 -64
  763. package/engines/surrealdb/README.md +0 -218
  764. package/engines/surrealdb/backup.ts +0 -131
  765. package/engines/surrealdb/binary-manager.ts +0 -45
  766. package/engines/surrealdb/binary-urls.ts +0 -40
  767. package/engines/surrealdb/cli-utils.ts +0 -173
  768. package/engines/surrealdb/hostdb-releases.ts +0 -23
  769. package/engines/surrealdb/index.ts +0 -1246
  770. package/engines/surrealdb/restore.ts +0 -302
  771. package/engines/surrealdb/version-maps.ts +0 -41
  772. package/engines/tigerbeetle/README.md +0 -61
  773. package/engines/tigerbeetle/backup.ts +0 -49
  774. package/engines/tigerbeetle/binary-manager.ts +0 -95
  775. package/engines/tigerbeetle/binary-urls.ts +0 -62
  776. package/engines/tigerbeetle/hostdb-releases.ts +0 -26
  777. package/engines/tigerbeetle/index.ts +0 -746
  778. package/engines/tigerbeetle/restore.ts +0 -130
  779. package/engines/tigerbeetle/version-validator.ts +0 -126
  780. package/engines/typedb/backup.ts +0 -167
  781. package/engines/typedb/binary-manager.ts +0 -200
  782. package/engines/typedb/binary-urls.ts +0 -40
  783. package/engines/typedb/cli-utils.ts +0 -210
  784. package/engines/typedb/hostdb-releases.ts +0 -23
  785. package/engines/typedb/index.ts +0 -1275
  786. package/engines/typedb/restore.ts +0 -377
  787. package/engines/typedb/version-maps.ts +0 -48
  788. package/engines/typedb/version-validator.ts +0 -127
  789. package/engines/valkey/README.md +0 -219
  790. package/engines/valkey/backup.ts +0 -389
  791. package/engines/valkey/binary-manager.ts +0 -45
  792. package/engines/valkey/binary-urls.ts +0 -122
  793. package/engines/valkey/cli-utils.ts +0 -42
  794. package/engines/valkey/hostdb-releases.ts +0 -23
  795. package/engines/valkey/index.ts +0 -1585
  796. package/engines/valkey/restore.ts +0 -446
  797. package/engines/valkey/version-maps.ts +0 -81
  798. package/engines/valkey/version-validator.ts +0 -131
  799. package/engines/weaviate/README.md +0 -302
  800. package/engines/weaviate/api-client.ts +0 -61
  801. package/engines/weaviate/backup.ts +0 -145
  802. package/engines/weaviate/binary-manager.ts +0 -80
  803. package/engines/weaviate/binary-urls.ts +0 -115
  804. package/engines/weaviate/cli-utils.ts +0 -43
  805. package/engines/weaviate/hostdb-releases.ts +0 -23
  806. package/engines/weaviate/index.ts +0 -1139
  807. package/engines/weaviate/restore.ts +0 -235
  808. package/engines/weaviate/version-maps.ts +0 -78
  809. package/engines/weaviate/version-validator.ts +0 -128
  810. package/types/index.ts +0 -624
  811. /package/{config → dist/config}/engines.json +0 -0
@@ -0,0 +1,1202 @@
1
+ import chalk from 'chalk';
2
+ import inquirer from 'inquirer';
3
+ import { existsSync } from 'fs';
4
+ import { rm, mkdir } from 'fs/promises';
5
+ import { join } from 'path';
6
+ import { tmpdir } from 'os';
7
+ import { containerManager } from '../../../core/container-manager.js';
8
+ import { getMissingDependencies } from '../../../core/dependency-manager.js';
9
+ import { platformService } from '../../../core/platform-service.js';
10
+ import { portManager } from '../../../core/port-manager.js';
11
+ import { getEngine } from '../../../engines/index.js';
12
+ import { defaults } from '../../../config/defaults.js';
13
+ import { getBackupExtension, getBackupSpinnerLabel, getDefaultFormat, } from '../../../config/backup-formats.js';
14
+ import { generateBackupTimestamp, estimateBackupSize, checkBackupSize, } from '../../../core/backup-restore.js';
15
+ import { promptCreateOptions, promptContainerName, promptContainerSelect, promptDatabaseName, promptDatabaseSelect, promptBackupFormat, promptBackupFilename, promptBackupDirectory, promptInstallDependencies, promptConfirm, escapeablePrompt, filterableListPrompt, BACK_VALUE, MAIN_MENU_VALUE, ESCAPE_VALUE, } from '../../ui/prompts.js';
16
+ import { createSpinner } from '../../ui/spinner.js';
17
+ import { header, uiSuccess, uiError, uiWarning, connectionBox, formatBytes, } from '../../ui/theme.js';
18
+ import { getEngineIcon, getPageSize } from '../../constants.js';
19
+ import { Engine, assertExhaustive } from '../../../types/index.js';
20
+ import { pressEnterToContinue } from './shared.js';
21
+ import { SpinDBError, ErrorCodes } from '../../../core/error-handler.js';
22
+ import { validateTypedbConnectionString } from './validators.js';
23
+ // Strip surrounding quotes from paths (handles drag-and-drop paths)
24
+ function stripQuotes(path) {
25
+ return path.replace(/^['"]|['"]$/g, '').trim();
26
+ }
27
+ /**
28
+ * Mask the password portion of a connection string for display.
29
+ * Example: postgresql://user:secretpass@host:5432/db → postgresql://user:****@host:5432/db
30
+ * Handles passwords containing '@' characters by using URL parsing.
31
+ */
32
+ function maskConnectionStringPassword(connectionString) {
33
+ if (!connectionString)
34
+ return connectionString;
35
+ try {
36
+ // Use URL constructor for robust parsing (handles @ in passwords)
37
+ const url = new URL(connectionString);
38
+ if (url.password) {
39
+ url.password = '****';
40
+ return url.toString();
41
+ }
42
+ return connectionString;
43
+ }
44
+ catch {
45
+ // Fallback for malformed URLs - use greedy regex that captures everything up to last @
46
+ try {
47
+ return connectionString.replace(/^([a-z+]+:\/\/[^:]*:)(.+)(@[^@]+)$/i, (_, prefix, _password, suffix) => `${prefix}****${suffix}`);
48
+ }
49
+ catch {
50
+ return connectionString;
51
+ }
52
+ }
53
+ }
54
+ /**
55
+ * Validate a connection string for the given engine.
56
+ * Returns true if valid, or an error message string if invalid.
57
+ *
58
+ * Empty input is intentionally allowed (returns true) to support the
59
+ * "press Enter to go back" UX pattern. Callers must check for empty
60
+ * input after the prompt returns (e.g., `if (!connectionString.trim()) return`).
61
+ */
62
+ function validateConnectionString(input, engine) {
63
+ // Allow empty input for "press Enter to go back" UX - callers handle this case
64
+ if (!input)
65
+ return true;
66
+ switch (engine) {
67
+ case Engine.PostgreSQL:
68
+ if (!input.startsWith('postgresql://') &&
69
+ !input.startsWith('postgres://')) {
70
+ return 'Connection string must start with postgresql:// or postgres://';
71
+ }
72
+ break;
73
+ case Engine.MySQL:
74
+ if (!input.startsWith('mysql://')) {
75
+ return 'Connection string must start with mysql://';
76
+ }
77
+ break;
78
+ case Engine.MariaDB:
79
+ if (!input.startsWith('mysql://') && !input.startsWith('mariadb://')) {
80
+ return 'Connection string must start with mysql:// or mariadb://';
81
+ }
82
+ break;
83
+ case Engine.MongoDB:
84
+ case Engine.FerretDB:
85
+ if (!input.startsWith('mongodb://') &&
86
+ !input.startsWith('mongodb+srv://')) {
87
+ return 'Connection string must start with mongodb:// or mongodb+srv://';
88
+ }
89
+ break;
90
+ case Engine.Redis:
91
+ if (!input.startsWith('redis://') && !input.startsWith('rediss://')) {
92
+ return 'Connection string must start with redis:// or rediss://';
93
+ }
94
+ break;
95
+ case Engine.Valkey:
96
+ if (!input.startsWith('redis://') &&
97
+ !input.startsWith('rediss://') &&
98
+ !input.startsWith('valkey://') &&
99
+ !input.startsWith('valkeys://')) {
100
+ return 'Connection string must start with redis://, rediss://, valkey://, or valkeys://';
101
+ }
102
+ break;
103
+ case Engine.ClickHouse:
104
+ if (!input.startsWith('clickhouse://') &&
105
+ !input.startsWith('http://') &&
106
+ !input.startsWith('https://')) {
107
+ return 'Connection string must start with clickhouse://, http://, or https://';
108
+ }
109
+ break;
110
+ case Engine.Qdrant:
111
+ if (!input.startsWith('qdrant://') &&
112
+ !input.startsWith('http://') &&
113
+ !input.startsWith('https://')) {
114
+ return 'Connection string must start with qdrant://, http://, or https://';
115
+ }
116
+ break;
117
+ case Engine.Meilisearch:
118
+ if (!input.startsWith('meilisearch://') &&
119
+ !input.startsWith('http://') &&
120
+ !input.startsWith('https://')) {
121
+ return 'Connection string must start with meilisearch://, http://, or https://';
122
+ }
123
+ break;
124
+ case Engine.CouchDB:
125
+ if (!input.startsWith('couchdb://') &&
126
+ !input.startsWith('http://') &&
127
+ !input.startsWith('https://')) {
128
+ return 'Connection string must start with couchdb://, http://, or https://';
129
+ }
130
+ break;
131
+ case Engine.CockroachDB:
132
+ if (!input.startsWith('postgresql://') &&
133
+ !input.startsWith('postgres://')) {
134
+ return 'Connection string must start with postgresql:// or postgres://';
135
+ }
136
+ break;
137
+ case Engine.SurrealDB:
138
+ if (!input.startsWith('surrealdb://') &&
139
+ !input.startsWith('ws://') &&
140
+ !input.startsWith('wss://') &&
141
+ !input.startsWith('http://') &&
142
+ !input.startsWith('https://')) {
143
+ return 'Connection string must start with surrealdb://, ws://, wss://, http://, or https://';
144
+ }
145
+ break;
146
+ case Engine.QuestDB:
147
+ // QuestDB uses PostgreSQL wire protocol
148
+ if (!input.startsWith('postgresql://') &&
149
+ !input.startsWith('postgres://')) {
150
+ return 'Connection string must start with postgresql:// or postgres://';
151
+ }
152
+ break;
153
+ case Engine.TypeDB:
154
+ {
155
+ const typedbError = validateTypedbConnectionString(input);
156
+ if (typedbError)
157
+ return typedbError;
158
+ }
159
+ break;
160
+ case Engine.InfluxDB:
161
+ if (!input.startsWith('influxdb://') &&
162
+ !input.startsWith('http://') &&
163
+ !input.startsWith('https://')) {
164
+ return 'Connection string must start with influxdb://, http://, or https://';
165
+ }
166
+ break;
167
+ case Engine.Weaviate:
168
+ if (!input.startsWith('http://') && !input.startsWith('https://')) {
169
+ return 'Connection string must start with http:// or https://';
170
+ }
171
+ break;
172
+ case Engine.TigerBeetle:
173
+ return 'TigerBeetle does not support remote dumps (custom binary protocol)';
174
+ case Engine.SQLite:
175
+ case Engine.DuckDB:
176
+ return 'File-based engines do not support remote connection strings';
177
+ default:
178
+ assertExhaustive(engine);
179
+ }
180
+ return true;
181
+ }
182
+ /**
183
+ * Prompt for a connection string with validation and password masking.
184
+ * Shows a hint about pressing Enter to go back and Escape for main menu.
185
+ *
186
+ * @param engine - The database engine for connection string validation
187
+ * @returns The connection string, or null if empty/escaped
188
+ */
189
+ async function promptConnectionString(engine) {
190
+ console.log(chalk.gray(' Enter connection string, or press Enter to go back (esc - main menu)'));
191
+ const { connectionString } = await escapeablePrompt([
192
+ {
193
+ type: 'input',
194
+ name: 'connectionString',
195
+ message: 'Connection string:',
196
+ transformer: (input) => maskConnectionStringPassword(input.trim()),
197
+ validate: (input) => validateConnectionString(input.trim(), engine),
198
+ },
199
+ ]);
200
+ const trimmed = connectionString.trim();
201
+ return trimmed || null;
202
+ }
203
+ export async function handleCreateForRestore() {
204
+ console.log();
205
+ const answers = await promptCreateOptions();
206
+ let { name: containerName } = answers;
207
+ const { engine, version, port, database } = answers;
208
+ console.log();
209
+ console.log(header('Creating Database Container'));
210
+ console.log();
211
+ const dbEngine = getEngine(engine);
212
+ const portAvailable = await portManager.isPortAvailable(port);
213
+ if (!portAvailable) {
214
+ console.log(uiError(`Port ${port} is in use. Please choose a different port.`));
215
+ return null;
216
+ }
217
+ const binarySpinner = createSpinner(`Checking ${dbEngine.displayName} ${version} binaries...`);
218
+ binarySpinner.start();
219
+ const isInstalled = await dbEngine.isBinaryInstalled(version);
220
+ if (isInstalled) {
221
+ binarySpinner.succeed(`${dbEngine.displayName} ${version} binaries ready (cached)`);
222
+ }
223
+ else {
224
+ binarySpinner.text = `Downloading ${dbEngine.displayName} ${version} binaries...`;
225
+ await dbEngine.ensureBinaries(version, ({ message }) => {
226
+ binarySpinner.text = message;
227
+ });
228
+ binarySpinner.succeed(`${dbEngine.displayName} ${version} binaries downloaded`);
229
+ }
230
+ while (await containerManager.exists(containerName)) {
231
+ console.log(chalk.yellow(` Container "${containerName}" already exists.`));
232
+ containerName = await promptContainerName();
233
+ }
234
+ const createSpinnerInstance = createSpinner('Creating container...');
235
+ createSpinnerInstance.start();
236
+ await containerManager.create(containerName, {
237
+ engine: dbEngine.name,
238
+ version,
239
+ port,
240
+ database,
241
+ });
242
+ createSpinnerInstance.succeed('Container created');
243
+ const initSpinner = createSpinner('Initializing database cluster...');
244
+ initSpinner.start();
245
+ await dbEngine.initDataDir(containerName, version, {
246
+ superuser: defaults.superuser,
247
+ });
248
+ initSpinner.succeed('Database cluster initialized');
249
+ const startSpinner = createSpinner(`Starting ${dbEngine.displayName}...`);
250
+ startSpinner.start();
251
+ const config = await containerManager.getConfig(containerName);
252
+ if (!config) {
253
+ startSpinner.fail('Failed to get container config');
254
+ return null;
255
+ }
256
+ await dbEngine.start(config);
257
+ await containerManager.updateConfig(containerName, { status: 'running' });
258
+ startSpinner.succeed(`${dbEngine.displayName} started`);
259
+ if (database !== 'postgres') {
260
+ const dbSpinner = createSpinner(`Creating database "${database}"...`);
261
+ dbSpinner.start();
262
+ await dbEngine.createDatabase(config, database);
263
+ dbSpinner.succeed(`Database "${database}" created`);
264
+ }
265
+ console.log();
266
+ console.log(uiSuccess('Container ready for restore'));
267
+ console.log();
268
+ return { name: containerName, config };
269
+ }
270
+ export async function handleRestore() {
271
+ // Use a loop instead of recursion for "back" navigation
272
+ while (true) {
273
+ const containers = await containerManager.list();
274
+ const running = containers.filter((c) => c.status === 'running');
275
+ // Build filterable container choices
276
+ const containerChoices = running.map((c) => ({
277
+ name: `${c.name} ${chalk.gray(`(${getEngineIcon(c.engine)}${c.engine} ${c.version}, port ${c.port})`)} ${chalk.green('● running')}`,
278
+ value: c.name,
279
+ short: c.name,
280
+ }));
281
+ // Build footer with action and navigation options
282
+ const footerChoices = [
283
+ new inquirer.Separator(),
284
+ {
285
+ name: `${chalk.green('➕')} Create new container`,
286
+ value: '__create_new__',
287
+ short: 'Create new',
288
+ },
289
+ new inquirer.Separator(),
290
+ { name: `${chalk.blue('←')} Back`, value: BACK_VALUE },
291
+ {
292
+ name: `${chalk.blue('⌂')} Back to main menu ${chalk.gray('(esc)')}`,
293
+ value: MAIN_MENU_VALUE,
294
+ },
295
+ new inquirer.Separator(),
296
+ ];
297
+ const allChoices = [...containerChoices, ...footerChoices];
298
+ const selectedContainer = await filterableListPrompt(allChoices, 'Select container to restore to:', {
299
+ filterableCount: containerChoices.length,
300
+ pageSize: getPageSize(),
301
+ emptyText: 'No containers match filter',
302
+ });
303
+ // Handle navigation (including escape)
304
+ if (selectedContainer === ESCAPE_VALUE ||
305
+ selectedContainer === BACK_VALUE ||
306
+ selectedContainer === MAIN_MENU_VALUE) {
307
+ return;
308
+ }
309
+ let containerName;
310
+ let config;
311
+ if (selectedContainer === '__create_new__') {
312
+ const createResult = await handleCreateForRestore();
313
+ if (!createResult)
314
+ return;
315
+ containerName = createResult.name;
316
+ config = createResult.config;
317
+ }
318
+ else {
319
+ containerName = selectedContainer;
320
+ config = await containerManager.getConfig(containerName);
321
+ if (!config) {
322
+ console.error(uiError(`Container "${containerName}" not found`));
323
+ return;
324
+ }
325
+ }
326
+ const depsSpinner = createSpinner('Checking required tools...');
327
+ depsSpinner.start();
328
+ let missingDeps = await getMissingDependencies(config.engine);
329
+ if (missingDeps.length > 0) {
330
+ depsSpinner.warn(`Missing tools: ${missingDeps.map((d) => d.name).join(', ')}`);
331
+ const installed = await promptInstallDependencies(missingDeps[0].binary, config.engine);
332
+ if (!installed) {
333
+ return;
334
+ }
335
+ missingDeps = await getMissingDependencies(config.engine);
336
+ if (missingDeps.length > 0) {
337
+ console.log(uiError(`Still missing tools: ${missingDeps.map((d) => d.name).join(', ')}`));
338
+ return;
339
+ }
340
+ console.log(chalk.green(' ✓ All required tools are now available'));
341
+ console.log();
342
+ }
343
+ else {
344
+ depsSpinner.succeed('Required tools available');
345
+ }
346
+ // All engines now support dumpFromConnectionString
347
+ const restoreChoices = [
348
+ {
349
+ name: `${chalk.magenta('☰')} Dump file (drag and drop or enter path)`,
350
+ value: 'file',
351
+ },
352
+ ];
353
+ restoreChoices.push({
354
+ name: `${chalk.cyan('↗')} Connection string ${chalk.gray('(pull from remote database)')}`,
355
+ value: 'connection',
356
+ });
357
+ restoreChoices.push(new inquirer.Separator(), {
358
+ name: `${chalk.blue('←')} Back`,
359
+ value: '__back__',
360
+ });
361
+ const { restoreSource } = await escapeablePrompt([
362
+ {
363
+ type: 'list',
364
+ name: 'restoreSource',
365
+ message: 'Restore from:',
366
+ choices: restoreChoices,
367
+ },
368
+ ]);
369
+ if (restoreSource === '__back__') {
370
+ continue; // Go back to container selection
371
+ }
372
+ let backupPath = '';
373
+ let isTempFile = false;
374
+ if (restoreSource === 'connection') {
375
+ const connectionString = await promptConnectionString(config.engine);
376
+ if (!connectionString) {
377
+ continue; // Return to container selection
378
+ }
379
+ const engine = getEngine(config.engine);
380
+ const timestamp = Date.now();
381
+ const defaultFormat = getDefaultFormat(config.engine);
382
+ const dumpExtension = getBackupExtension(config.engine, defaultFormat);
383
+ const tempDumpPath = join(tmpdir(), `spindb-dump-${timestamp}${dumpExtension}`);
384
+ let dumpSuccess = false;
385
+ let attempts = 0;
386
+ const maxAttempts = 2;
387
+ while (!dumpSuccess && attempts < maxAttempts) {
388
+ attempts++;
389
+ const dumpSpinner = createSpinner('Creating dump from remote database...');
390
+ dumpSpinner.start();
391
+ try {
392
+ const dumpResult = await engine.dumpFromConnectionString(connectionString, tempDumpPath);
393
+ dumpSpinner.succeed('Dump created from remote database');
394
+ if (dumpResult.warnings?.length) {
395
+ for (const warning of dumpResult.warnings) {
396
+ console.log(chalk.yellow(` ${warning}`));
397
+ }
398
+ }
399
+ backupPath = tempDumpPath;
400
+ isTempFile = true;
401
+ dumpSuccess = true;
402
+ }
403
+ catch (error) {
404
+ const e = error;
405
+ dumpSpinner.fail('Failed to create dump');
406
+ // Handle version mismatch errors with helpful message
407
+ if (e instanceof SpinDBError &&
408
+ e.code === ErrorCodes.VERSION_MISMATCH) {
409
+ console.log();
410
+ console.log(uiError('PostgreSQL version mismatch:'));
411
+ console.log(chalk.gray(` ${e.message}`));
412
+ if (e.suggestion) {
413
+ console.log();
414
+ console.log(uiWarning('To fix this:'));
415
+ console.log(chalk.yellow(` ${e.suggestion}`));
416
+ }
417
+ console.log();
418
+ try {
419
+ await rm(tempDumpPath, { force: true });
420
+ }
421
+ catch {
422
+ // Ignore cleanup errors
423
+ }
424
+ await pressEnterToContinue();
425
+ return;
426
+ }
427
+ // Handle connection errors
428
+ if (e instanceof SpinDBError &&
429
+ e.code === ErrorCodes.CONNECTION_FAILED) {
430
+ console.log();
431
+ console.log(uiError('Connection failed:'));
432
+ console.log(chalk.gray(` ${e.message}`));
433
+ if (e.suggestion) {
434
+ console.log(chalk.yellow(` ${e.suggestion}`));
435
+ }
436
+ console.log();
437
+ await pressEnterToContinue();
438
+ return;
439
+ }
440
+ // Handle missing tool errors
441
+ if (e.message.includes('pg_dump not found') ||
442
+ e.message.includes('mysqldump not found') ||
443
+ e.message.includes('ENOENT')) {
444
+ const missingTool = e.message.includes('mysqldump')
445
+ ? 'mysqldump'
446
+ : 'pg_dump';
447
+ const toolEngine = missingTool === 'mysqldump' ? 'mysql' : 'postgresql';
448
+ const installed = await promptInstallDependencies(missingTool, toolEngine);
449
+ if (installed) {
450
+ // Installation counts toward maxAttempts - retry with newly installed tools
451
+ continue;
452
+ }
453
+ }
454
+ else {
455
+ const dumpTool = config.engine === 'mysql' ? 'mysqldump' : 'pg_dump';
456
+ console.log();
457
+ console.log(uiError(`${dumpTool} error:`));
458
+ console.log(chalk.gray(` ${e.message}`));
459
+ console.log();
460
+ }
461
+ try {
462
+ await rm(tempDumpPath, { force: true });
463
+ }
464
+ catch {
465
+ // Ignore cleanup errors
466
+ }
467
+ await pressEnterToContinue();
468
+ return;
469
+ }
470
+ }
471
+ if (!dumpSuccess) {
472
+ console.log(uiError('Failed to create dump after retries'));
473
+ return;
474
+ }
475
+ }
476
+ else {
477
+ console.log(chalk.gray(' Drag & drop, enter path (abs or rel), or press Enter to go back (esc - main menu)'));
478
+ const { backupPath: rawBackupPath } = await escapeablePrompt([
479
+ {
480
+ type: 'input',
481
+ name: 'backupPath',
482
+ message: 'Backup file path:',
483
+ validate: (input) => {
484
+ if (!input)
485
+ return true;
486
+ const cleanPath = stripQuotes(input);
487
+ if (!existsSync(cleanPath))
488
+ return 'File not found';
489
+ return true;
490
+ },
491
+ },
492
+ ]);
493
+ if (!rawBackupPath.trim()) {
494
+ continue; // Return to container selection
495
+ }
496
+ backupPath = stripQuotes(rawBackupPath);
497
+ }
498
+ const engine = getEngine(config.engine);
499
+ // Get existing databases in this container
500
+ const existingDatabases = config.databases || [config.database];
501
+ // Redis uses numbered databases 0-15, so "create new" doesn't apply
502
+ const isRedis = config.engine === 'redis';
503
+ let restoreMode;
504
+ if (isRedis) {
505
+ // Redis: Always restore to existing database (0-15)
506
+ restoreMode = 'replace';
507
+ }
508
+ else {
509
+ const result = await escapeablePrompt([
510
+ {
511
+ type: 'list',
512
+ name: 'restoreMode',
513
+ message: 'How would you like to restore?',
514
+ choices: [
515
+ {
516
+ name: `${chalk.green('➕')} Create new database ${chalk.gray('(keeps existing databases intact)')}`,
517
+ value: 'new',
518
+ },
519
+ {
520
+ name: `${chalk.yellow('↻')} Replace existing database ${chalk.gray('(overwrites data)')}`,
521
+ value: 'replace',
522
+ disabled: existingDatabases.length === 0
523
+ ? 'No existing databases'
524
+ : false,
525
+ },
526
+ new inquirer.Separator(),
527
+ {
528
+ name: `${chalk.blue('←')} Back`,
529
+ value: '__back__',
530
+ },
531
+ ],
532
+ },
533
+ ]);
534
+ restoreMode = result.restoreMode;
535
+ }
536
+ if (restoreMode === '__back__') {
537
+ continue; // Return to container selection
538
+ }
539
+ let databaseName;
540
+ if (restoreMode === 'new') {
541
+ // Show existing databases for context
542
+ if (existingDatabases.length > 0) {
543
+ console.log();
544
+ console.log(chalk.gray(' Existing databases in this container:'));
545
+ for (const db of existingDatabases) {
546
+ console.log(chalk.gray(` • ${db}`));
547
+ }
548
+ console.log();
549
+ }
550
+ // Prompt for new database name (must not already exist)
551
+ const result = await promptDatabaseName(containerName, config.engine, {
552
+ allowBack: true,
553
+ existingDatabases,
554
+ disallowExisting: true,
555
+ });
556
+ if (result === null) {
557
+ continue; // Return to container selection
558
+ }
559
+ databaseName = result;
560
+ }
561
+ else {
562
+ // Replace existing database - show selection
563
+ if (existingDatabases.length === 1) {
564
+ databaseName = existingDatabases[0];
565
+ }
566
+ else {
567
+ const result = await promptDatabaseSelect(existingDatabases, 'Select database to replace:', { includeBack: true });
568
+ if (result === null) {
569
+ continue; // Return to container selection
570
+ }
571
+ databaseName = result;
572
+ }
573
+ // Confirm overwrite
574
+ const confirmed = await promptConfirm(`This will overwrite all data in "${databaseName}". Continue?`, false);
575
+ if (!confirmed) {
576
+ continue; // Return to container selection
577
+ }
578
+ // Redis doesn't need drop/create - databases 0-15 always exist
579
+ if (!isRedis) {
580
+ // Drop the existing database before restore
581
+ console.log();
582
+ const dropSpinner = createSpinner(`Dropping existing database "${databaseName}"...`);
583
+ dropSpinner.start();
584
+ try {
585
+ await engine.dropDatabase(config, databaseName);
586
+ dropSpinner.succeed(`Dropped database "${databaseName}"`);
587
+ }
588
+ catch (error) {
589
+ dropSpinner.fail(`Failed to drop database "${databaseName}"`);
590
+ console.log(uiError(error.message));
591
+ await pressEnterToContinue();
592
+ return;
593
+ }
594
+ }
595
+ }
596
+ const detectSpinner = createSpinner('Detecting backup format...');
597
+ detectSpinner.start();
598
+ const format = await engine.detectBackupFormat(backupPath);
599
+ detectSpinner.succeed(`Detected: ${format.description}`);
600
+ // For Redis .redis text files, ask about merge vs replace behavior
601
+ let flushBeforeRestore = false;
602
+ if (isRedis && format.format === 'redis') {
603
+ const { restoreBehavior } = await escapeablePrompt([
604
+ {
605
+ type: 'list',
606
+ name: 'restoreBehavior',
607
+ message: 'How should existing data be handled?',
608
+ choices: [
609
+ {
610
+ name: `${chalk.yellow('↻')} Replace all ${chalk.gray('(FLUSHDB - clear database first)')}`,
611
+ value: 'replace',
612
+ },
613
+ {
614
+ name: `${chalk.green('➕')} Merge ${chalk.gray('(add/update keys, keep others)')}`,
615
+ value: 'merge',
616
+ },
617
+ ],
618
+ },
619
+ ]);
620
+ flushBeforeRestore = restoreBehavior === 'replace';
621
+ }
622
+ // Redis doesn't need createDatabase - databases 0-15 always exist
623
+ if (!isRedis) {
624
+ const dbSpinner = createSpinner(`Creating database "${databaseName}"...`);
625
+ dbSpinner.start();
626
+ await engine.createDatabase(config, databaseName);
627
+ dbSpinner.succeed(`Database "${databaseName}" ready`);
628
+ }
629
+ const restoreSpinner = createSpinner('Restoring backup...');
630
+ restoreSpinner.start();
631
+ const result = await engine.restore(config, backupPath, {
632
+ database: databaseName,
633
+ createDatabase: false,
634
+ flush: flushBeforeRestore,
635
+ });
636
+ if (result.code === 0) {
637
+ restoreSpinner.succeed('Backup restored successfully');
638
+ }
639
+ else {
640
+ const stderr = result.stderr || '';
641
+ if (stderr.includes('unsupported version') ||
642
+ stderr.includes('Archive version') ||
643
+ stderr.includes('too old')) {
644
+ restoreSpinner.fail('Version compatibility detected');
645
+ console.log();
646
+ console.log(uiError('PostgreSQL version incompatibility detected:'));
647
+ console.log(uiWarning('Your pg_restore version is too old for this backup file.'));
648
+ console.log(chalk.yellow('Cleaning up failed database...'));
649
+ try {
650
+ await engine.dropDatabase(config, databaseName);
651
+ console.log(chalk.gray(`✓ Removed database "${databaseName}"`));
652
+ }
653
+ catch {
654
+ console.log(chalk.yellow(`Warning: Could not remove database "${databaseName}"`));
655
+ }
656
+ console.log();
657
+ const versionMatch = stderr.match(/PostgreSQL (\d+)/);
658
+ const requiredVersion = versionMatch ? versionMatch[1] : '17';
659
+ console.log(chalk.gray(`This backup was created with PostgreSQL ${requiredVersion}`));
660
+ console.log();
661
+ console.log();
662
+ console.log(uiWarning(`To restore this backup, download PostgreSQL ${requiredVersion} binaries:`));
663
+ console.log(chalk.cyan(` spindb engines download postgresql ${requiredVersion}`));
664
+ console.log();
665
+ console.log(chalk.gray('Then create a new container with that version and try the restore again.'));
666
+ await pressEnterToContinue();
667
+ return;
668
+ }
669
+ else {
670
+ // Other restore errors - show warnings
671
+ restoreSpinner.warn('Restore completed with warnings');
672
+ if (result.stderr) {
673
+ console.log();
674
+ console.log(chalk.yellow(' Warnings/Errors:'));
675
+ const lines = result.stderr.split('\n').filter((l) => l.trim());
676
+ const displayLines = lines.slice(0, 20);
677
+ for (const line of displayLines) {
678
+ console.log(chalk.gray(` ${line}`));
679
+ }
680
+ if (lines.length > 20) {
681
+ console.log(chalk.gray(` ... and ${lines.length - 20} more lines`));
682
+ }
683
+ }
684
+ }
685
+ }
686
+ if (result.code === 0) {
687
+ const connectionString = engine.getConnectionString(config, databaseName);
688
+ console.log();
689
+ console.log(uiSuccess(`Database "${databaseName}" restored`));
690
+ console.log(chalk.gray(' Connection string:'));
691
+ console.log(chalk.cyan(` ${connectionString}`));
692
+ const copied = await platformService.copyToClipboard(connectionString);
693
+ if (copied) {
694
+ console.log(chalk.gray(' ✓ Connection string copied to clipboard'));
695
+ }
696
+ else {
697
+ console.log(chalk.gray(' (Could not copy to clipboard)'));
698
+ }
699
+ console.log();
700
+ }
701
+ if (isTempFile) {
702
+ try {
703
+ await rm(backupPath, { force: true });
704
+ }
705
+ catch {
706
+ // Ignore cleanup errors
707
+ }
708
+ }
709
+ await pressEnterToContinue();
710
+ return; // Exit the wizard loop after successful restore
711
+ }
712
+ }
713
+ /**
714
+ * Shared backup flow for both main menu and container submenu
715
+ * Reduces code duplication between handleBackup and handleBackupForContainer
716
+ *
717
+ * @param containerName - The container to backup
718
+ * @param database - Optional database name (skips database selection if provided)
719
+ */
720
+ async function performBackupFlow(containerName, database) {
721
+ const config = await containerManager.getConfig(containerName);
722
+ if (!config) {
723
+ console.log(uiError(`Container "${containerName}" not found`));
724
+ return;
725
+ }
726
+ const engine = getEngine(config.engine);
727
+ // Check dependencies
728
+ const depsSpinner = createSpinner('Checking required tools...');
729
+ depsSpinner.start();
730
+ let missingDeps = await getMissingDependencies(config.engine);
731
+ if (missingDeps.length > 0) {
732
+ depsSpinner.warn(`Missing tools: ${missingDeps.map((d) => d.name).join(', ')}`);
733
+ const installed = await promptInstallDependencies(missingDeps[0].binary, config.engine);
734
+ if (!installed) {
735
+ return;
736
+ }
737
+ missingDeps = await getMissingDependencies(config.engine);
738
+ if (missingDeps.length > 0) {
739
+ console.log(uiError(`Still missing tools: ${missingDeps.map((d) => d.name).join(', ')}`));
740
+ return;
741
+ }
742
+ console.log(chalk.green(' ✓ All required tools are now available'));
743
+ console.log();
744
+ }
745
+ else {
746
+ depsSpinner.succeed('Required tools available');
747
+ }
748
+ // Use provided database or select from available databases
749
+ const databases = config.databases || [config.database];
750
+ let databaseName;
751
+ if (database) {
752
+ // Use the pre-selected database from the container submenu
753
+ databaseName = database;
754
+ }
755
+ else if (databases.length > 1) {
756
+ databaseName = await promptDatabaseSelect(databases, 'Select database to backup:');
757
+ }
758
+ else {
759
+ databaseName = databases[0];
760
+ }
761
+ // Show estimated size
762
+ const estimatedSize = await estimateBackupSize(config);
763
+ if (estimatedSize !== null) {
764
+ console.log(chalk.gray(` Estimated database size: ${formatBytes(estimatedSize)}`));
765
+ console.log();
766
+ }
767
+ // Select format
768
+ const format = await promptBackupFormat(config.engine);
769
+ // Select output directory
770
+ const outputDir = await promptBackupDirectory();
771
+ if (!outputDir)
772
+ return;
773
+ // Ensure directory exists
774
+ if (!existsSync(outputDir)) {
775
+ await mkdir(outputDir, { recursive: true });
776
+ }
777
+ // Get filename
778
+ const defaultFilename = `${containerName}-${databaseName}-backup-${generateBackupTimestamp()}`;
779
+ const filename = await promptBackupFilename(defaultFilename);
780
+ const extension = getBackupExtension(config.engine, format);
781
+ const outputPath = join(outputDir, `${filename}${extension}`);
782
+ const spinnerLabel = getBackupSpinnerLabel(config.engine, format);
783
+ const backupSpinner = createSpinner(`Creating ${spinnerLabel} backup of "${databaseName}"...`);
784
+ backupSpinner.start();
785
+ try {
786
+ const result = await engine.backup(config, outputPath, {
787
+ database: databaseName,
788
+ format,
789
+ });
790
+ backupSpinner.succeed('Backup created successfully');
791
+ console.log();
792
+ console.log(uiSuccess('Backup complete'));
793
+ console.log();
794
+ console.log(chalk.gray(' Saved to:'), chalk.cyan(result.path));
795
+ console.log(chalk.gray(' Size:'), chalk.white(formatBytes(result.size)));
796
+ console.log(chalk.gray(' Format:'), chalk.white(result.format));
797
+ console.log();
798
+ }
799
+ catch (error) {
800
+ const e = error;
801
+ backupSpinner.fail('Backup failed');
802
+ console.log();
803
+ console.log(uiError(e.message));
804
+ console.log();
805
+ }
806
+ }
807
+ export async function handleBackup() {
808
+ const containers = await containerManager.list();
809
+ const running = containers.filter((c) => c.status === 'running');
810
+ if (running.length === 0) {
811
+ console.log(uiWarning('No running containers. Start a container first.'));
812
+ await pressEnterToContinue();
813
+ return;
814
+ }
815
+ const containerName = await promptContainerSelect(running, 'Select container to backup:', { includeBack: true });
816
+ if (!containerName)
817
+ return;
818
+ await performBackupFlow(containerName);
819
+ await pressEnterToContinue();
820
+ }
821
+ /**
822
+ * Handle backup for a specific container (used from container submenu)
823
+ * Skips container selection since we already know which container
824
+ */
825
+ export async function handleBackupForContainer(containerName, database) {
826
+ await performBackupFlow(containerName, database);
827
+ await pressEnterToContinue();
828
+ }
829
+ /**
830
+ * Handle restore for a specific container (used from container submenu)
831
+ * Skips container selection since we already know which container
832
+ *
833
+ * @param containerName - The container to restore to
834
+ * @param database - Optional database name (pre-selects target database if provided)
835
+ */
836
+ export async function handleRestoreForContainer(containerName, database) {
837
+ const config = await containerManager.getConfig(containerName);
838
+ if (!config) {
839
+ console.log(uiError(`Container "${containerName}" not found`));
840
+ await pressEnterToContinue();
841
+ return;
842
+ }
843
+ const engine = getEngine(config.engine);
844
+ // Check dependencies
845
+ const depsSpinner = createSpinner('Checking required tools...');
846
+ depsSpinner.start();
847
+ let missingDeps = await getMissingDependencies(config.engine);
848
+ if (missingDeps.length > 0) {
849
+ depsSpinner.warn(`Missing tools: ${missingDeps.map((d) => d.name).join(', ')}`);
850
+ const installed = await promptInstallDependencies(missingDeps[0].binary, config.engine);
851
+ if (!installed) {
852
+ return;
853
+ }
854
+ missingDeps = await getMissingDependencies(config.engine);
855
+ if (missingDeps.length > 0) {
856
+ console.log(uiError(`Still missing tools: ${missingDeps.map((d) => d.name).join(', ')}`));
857
+ return;
858
+ }
859
+ console.log(chalk.green(' ✓ All required tools are now available'));
860
+ console.log();
861
+ }
862
+ else {
863
+ depsSpinner.succeed('Required tools available');
864
+ }
865
+ // Restore source selection (file or connection string)
866
+ // All engines now support dumpFromConnectionString
867
+ const restoreChoices = [
868
+ {
869
+ name: `${chalk.magenta('☰')} Dump file (drag and drop or enter path)`,
870
+ value: 'file',
871
+ },
872
+ {
873
+ name: `${chalk.cyan('↗')} Connection string ${chalk.gray('(pull from remote database)')}`,
874
+ value: 'connection',
875
+ },
876
+ ];
877
+ restoreChoices.push(new inquirer.Separator(), {
878
+ name: `${chalk.blue('←')} Back`,
879
+ value: '__back__',
880
+ });
881
+ const { restoreSource } = await escapeablePrompt([
882
+ {
883
+ type: 'list',
884
+ name: 'restoreSource',
885
+ message: 'Restore from:',
886
+ choices: restoreChoices,
887
+ },
888
+ ]);
889
+ if (restoreSource === '__back__') {
890
+ return;
891
+ }
892
+ let backupPath = '';
893
+ let isTempFile = false;
894
+ if (restoreSource === 'connection') {
895
+ const connectionString = await promptConnectionString(config.engine);
896
+ if (!connectionString) {
897
+ return;
898
+ }
899
+ const timestamp = Date.now();
900
+ const defaultFormat = getDefaultFormat(config.engine);
901
+ const dumpExtension = getBackupExtension(config.engine, defaultFormat);
902
+ const tempDumpPath = join(tmpdir(), `spindb-dump-${timestamp}${dumpExtension}`);
903
+ const dumpSpinner = createSpinner('Creating dump from remote database...');
904
+ dumpSpinner.start();
905
+ try {
906
+ const dumpResult = await engine.dumpFromConnectionString(connectionString, tempDumpPath);
907
+ dumpSpinner.succeed('Dump created from remote database');
908
+ if (dumpResult.warnings?.length) {
909
+ for (const warning of dumpResult.warnings) {
910
+ console.log(chalk.yellow(` ${warning}`));
911
+ }
912
+ }
913
+ backupPath = tempDumpPath;
914
+ isTempFile = true;
915
+ }
916
+ catch (error) {
917
+ const e = error;
918
+ dumpSpinner.fail('Failed to create dump');
919
+ console.log(uiError(e.message));
920
+ await pressEnterToContinue();
921
+ return;
922
+ }
923
+ }
924
+ else {
925
+ // Handle file restore
926
+ console.log(chalk.gray(' Drag & drop, enter path (abs or rel), or press Enter to go back (esc - main menu)'));
927
+ const { backupPath: rawBackupPath } = await escapeablePrompt([
928
+ {
929
+ type: 'input',
930
+ name: 'backupPath',
931
+ message: 'Backup file path:',
932
+ validate: (input) => {
933
+ if (!input)
934
+ return true;
935
+ const cleanPath = stripQuotes(input);
936
+ if (!existsSync(cleanPath))
937
+ return 'File not found';
938
+ return true;
939
+ },
940
+ },
941
+ ]);
942
+ if (!rawBackupPath.trim()) {
943
+ return;
944
+ }
945
+ backupPath = stripQuotes(rawBackupPath);
946
+ }
947
+ // Check backup file size and warn if large
948
+ const sizeCheck = checkBackupSize(backupPath);
949
+ if (sizeCheck.level === 'very_large') {
950
+ console.log();
951
+ console.log(chalk.yellow(` ⚠ Large backup file: ${formatBytes(sizeCheck.size)}`));
952
+ console.log(chalk.gray(' This restore may take a while.'));
953
+ console.log();
954
+ const confirmed = await promptConfirm('Continue with restore?', true);
955
+ if (!confirmed) {
956
+ if (isTempFile) {
957
+ await rm(backupPath, { force: true }).catch(() => { });
958
+ }
959
+ return;
960
+ }
961
+ }
962
+ else if (sizeCheck.level === 'large') {
963
+ console.log(chalk.gray(` Backup file size: ${formatBytes(sizeCheck.size)}`));
964
+ }
965
+ // Detect backup format
966
+ const format = await engine.detectBackupFormat(backupPath);
967
+ console.log(chalk.gray(` Detected format: ${format.description}`));
968
+ console.log();
969
+ // Get existing databases in this container
970
+ const existingDatabases = config.databases || [config.database];
971
+ // Redis uses numbered databases 0-15, so "create new" doesn't apply
972
+ const isRedis = config.engine === 'redis';
973
+ let restoreMode;
974
+ if (isRedis) {
975
+ // Redis: Always restore to existing database (0-15)
976
+ restoreMode = 'replace';
977
+ }
978
+ else {
979
+ const result = await escapeablePrompt([
980
+ {
981
+ type: 'list',
982
+ name: 'restoreMode',
983
+ message: 'How would you like to restore?',
984
+ choices: [
985
+ {
986
+ name: `${chalk.green('➕')} Create new database ${chalk.gray('(keeps existing databases intact)')}`,
987
+ value: 'new',
988
+ },
989
+ {
990
+ name: `${chalk.yellow('↻')} Replace existing database ${chalk.gray('(overwrites data)')}`,
991
+ value: 'replace',
992
+ disabled: existingDatabases.length === 0 ? 'No existing databases' : false,
993
+ },
994
+ new inquirer.Separator(),
995
+ {
996
+ name: `${chalk.blue('←')} Back`,
997
+ value: '__back__',
998
+ },
999
+ ],
1000
+ },
1001
+ ]);
1002
+ restoreMode = result.restoreMode;
1003
+ }
1004
+ if (restoreMode === '__back__') {
1005
+ if (isTempFile) {
1006
+ await rm(backupPath, { force: true }).catch(() => { });
1007
+ }
1008
+ return;
1009
+ }
1010
+ let databaseName;
1011
+ if (restoreMode === 'new') {
1012
+ // Show existing databases for context
1013
+ if (existingDatabases.length > 0) {
1014
+ console.log();
1015
+ console.log(chalk.gray(' Existing databases in this container:'));
1016
+ for (const db of existingDatabases) {
1017
+ console.log(chalk.gray(` • ${db}`));
1018
+ }
1019
+ console.log();
1020
+ }
1021
+ // Prompt for new database name
1022
+ const result = await promptDatabaseName(containerName, config.engine, {
1023
+ existingDatabases,
1024
+ });
1025
+ if (!result) {
1026
+ if (isTempFile) {
1027
+ await rm(backupPath, { force: true }).catch(() => { });
1028
+ }
1029
+ return;
1030
+ }
1031
+ databaseName = result;
1032
+ // Create the new database
1033
+ const createDbSpinner = createSpinner(`Creating database "${databaseName}"...`);
1034
+ createDbSpinner.start();
1035
+ try {
1036
+ await engine.createDatabase(config, databaseName);
1037
+ createDbSpinner.succeed(`Database "${databaseName}" created`);
1038
+ // Update container config with new database
1039
+ const updatedDbs = [...existingDatabases, databaseName];
1040
+ await containerManager.updateConfig(containerName, {
1041
+ databases: updatedDbs,
1042
+ });
1043
+ }
1044
+ catch (error) {
1045
+ const e = error;
1046
+ createDbSpinner.fail('Failed to create database');
1047
+ console.log(uiError(e.message));
1048
+ if (isTempFile) {
1049
+ await rm(backupPath, { force: true }).catch(() => { });
1050
+ }
1051
+ await pressEnterToContinue();
1052
+ return;
1053
+ }
1054
+ }
1055
+ else {
1056
+ // Replace existing database - use pre-selected or auto-select if only one
1057
+ if (database) {
1058
+ // Use the pre-selected database from the container submenu
1059
+ databaseName = database;
1060
+ console.log(chalk.gray(` Target database: ${databaseName}`));
1061
+ }
1062
+ else if (existingDatabases.length === 1) {
1063
+ databaseName = existingDatabases[0];
1064
+ console.log(chalk.gray(` Using database: ${databaseName}`));
1065
+ }
1066
+ else {
1067
+ const { database: selectedDb } = await escapeablePrompt([
1068
+ {
1069
+ type: 'list',
1070
+ name: 'database',
1071
+ message: 'Select database to replace:',
1072
+ choices: existingDatabases.map((db) => ({ name: db, value: db })),
1073
+ },
1074
+ ]);
1075
+ databaseName = selectedDb;
1076
+ }
1077
+ }
1078
+ // For Redis .redis text files, ask about merge vs replace behavior
1079
+ let flushBeforeRestore = false;
1080
+ if (isRedis && format.format === 'redis') {
1081
+ const { restoreBehavior } = await escapeablePrompt([
1082
+ {
1083
+ type: 'list',
1084
+ name: 'restoreBehavior',
1085
+ message: 'How should existing data be handled?',
1086
+ choices: [
1087
+ {
1088
+ name: `${chalk.yellow('↻')} Replace all ${chalk.gray('(FLUSHDB - clear database first)')}`,
1089
+ value: 'replace',
1090
+ },
1091
+ {
1092
+ name: `${chalk.green('➕')} Merge ${chalk.gray('(add/update keys, keep others)')}`,
1093
+ value: 'merge',
1094
+ },
1095
+ ],
1096
+ },
1097
+ ]);
1098
+ flushBeforeRestore = restoreBehavior === 'replace';
1099
+ }
1100
+ // Perform restore
1101
+ const restoreSpinner = createSpinner(`Restoring to "${databaseName}" in ${containerName}...`);
1102
+ restoreSpinner.start();
1103
+ try {
1104
+ const result = await engine.restore(config, backupPath, {
1105
+ database: databaseName,
1106
+ flush: flushBeforeRestore,
1107
+ });
1108
+ if (result.code === 0) {
1109
+ restoreSpinner.succeed('Restore completed successfully');
1110
+ const connectionString = engine.getConnectionString(config, databaseName);
1111
+ console.log();
1112
+ console.log(uiSuccess(`Database "${databaseName}" restored`));
1113
+ console.log(chalk.gray(' Connection string:'));
1114
+ console.log(chalk.cyan(` ${connectionString}`));
1115
+ const copied = await platformService.copyToClipboard(connectionString);
1116
+ if (copied) {
1117
+ console.log(chalk.gray(' ✓ Connection string copied to clipboard'));
1118
+ }
1119
+ console.log();
1120
+ }
1121
+ else {
1122
+ restoreSpinner.warn('Restore completed with warnings');
1123
+ if (result.stderr) {
1124
+ console.log();
1125
+ console.log(chalk.yellow(' Warnings/Errors:'));
1126
+ const lines = result.stderr.split('\n').filter((l) => l.trim());
1127
+ const displayLines = lines.slice(0, 10);
1128
+ for (const line of displayLines) {
1129
+ console.log(chalk.gray(` ${line}`));
1130
+ }
1131
+ if (lines.length > 10) {
1132
+ console.log(chalk.gray(` ... and ${lines.length - 10} more lines`));
1133
+ }
1134
+ }
1135
+ }
1136
+ }
1137
+ catch (error) {
1138
+ const e = error;
1139
+ restoreSpinner.fail('Restore failed');
1140
+ console.log();
1141
+ console.log(uiError(e.message));
1142
+ console.log();
1143
+ }
1144
+ await pressEnterToContinue();
1145
+ }
1146
+ export async function handleClone() {
1147
+ const containers = await containerManager.list();
1148
+ const stopped = containers.filter((c) => c.status !== 'running');
1149
+ if (containers.length === 0) {
1150
+ console.log(uiWarning('No containers found'));
1151
+ return;
1152
+ }
1153
+ if (stopped.length === 0) {
1154
+ console.log(uiWarning('All containers are running. Stop a container first to clone it.'));
1155
+ return;
1156
+ }
1157
+ const sourceName = await promptContainerSelect(stopped, 'Select container to clone:', { includeBack: true });
1158
+ if (!sourceName)
1159
+ return;
1160
+ const sourceConfig = await containerManager.getConfig(sourceName);
1161
+ if (!sourceConfig) {
1162
+ console.log(uiError(`Container "${sourceName}" not found`));
1163
+ return;
1164
+ }
1165
+ const { targetName } = await escapeablePrompt([
1166
+ {
1167
+ type: 'input',
1168
+ name: 'targetName',
1169
+ message: 'Name for the cloned container:',
1170
+ default: `${sourceName}-copy`,
1171
+ validate: (input) => {
1172
+ if (!input)
1173
+ return 'Name is required';
1174
+ if (!/^[a-zA-Z][a-zA-Z0-9_-]*$/.test(input)) {
1175
+ return 'Name must start with a letter and contain only letters, numbers, hyphens, and underscores';
1176
+ }
1177
+ return true;
1178
+ },
1179
+ },
1180
+ ]);
1181
+ // Check if target container already exists
1182
+ if (await containerManager.exists(targetName, { engine: sourceConfig.engine })) {
1183
+ console.log(uiError(`Container "${targetName}" already exists`));
1184
+ return;
1185
+ }
1186
+ const spinner = createSpinner(`Cloning ${sourceName} to ${targetName}...`);
1187
+ spinner.start();
1188
+ try {
1189
+ const newConfig = await containerManager.clone(sourceName, targetName);
1190
+ spinner.succeed(`Cloned "${sourceName}" to "${targetName}"`);
1191
+ const engine = getEngine(newConfig.engine);
1192
+ const connectionString = engine.getConnectionString(newConfig);
1193
+ console.log();
1194
+ console.log(connectionBox(targetName, connectionString, newConfig.port));
1195
+ }
1196
+ catch (error) {
1197
+ const e = error;
1198
+ spinner.fail(`Failed to clone "${sourceName}"`);
1199
+ console.log(uiError(e.message));
1200
+ }
1201
+ }
1202
+ //# sourceMappingURL=backup-handlers.js.map