@tinybirdco/sdk 0.0.1

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 (258) hide show
  1. package/README.md +518 -0
  2. package/bin/tinybird.js +7 -0
  3. package/dist/api/branches.d.ts +98 -0
  4. package/dist/api/branches.d.ts.map +1 -0
  5. package/dist/api/branches.js +203 -0
  6. package/dist/api/branches.js.map +1 -0
  7. package/dist/api/branches.test.d.ts +2 -0
  8. package/dist/api/branches.test.d.ts.map +1 -0
  9. package/dist/api/branches.test.js +286 -0
  10. package/dist/api/branches.test.js.map +1 -0
  11. package/dist/api/build.d.ts +130 -0
  12. package/dist/api/build.d.ts.map +1 -0
  13. package/dist/api/build.js +143 -0
  14. package/dist/api/build.js.map +1 -0
  15. package/dist/api/build.test.d.ts +2 -0
  16. package/dist/api/build.test.d.ts.map +1 -0
  17. package/dist/api/build.test.js +138 -0
  18. package/dist/api/build.test.js.map +1 -0
  19. package/dist/api/deploy.d.ts +39 -0
  20. package/dist/api/deploy.d.ts.map +1 -0
  21. package/dist/api/deploy.js +135 -0
  22. package/dist/api/deploy.js.map +1 -0
  23. package/dist/api/deploy.test.d.ts +2 -0
  24. package/dist/api/deploy.test.d.ts.map +1 -0
  25. package/dist/api/deploy.test.js +118 -0
  26. package/dist/api/deploy.test.js.map +1 -0
  27. package/dist/api/workspaces.d.ts +46 -0
  28. package/dist/api/workspaces.d.ts.map +1 -0
  29. package/dist/api/workspaces.js +39 -0
  30. package/dist/api/workspaces.js.map +1 -0
  31. package/dist/api/workspaces.test.d.ts +2 -0
  32. package/dist/api/workspaces.test.d.ts.map +1 -0
  33. package/dist/api/workspaces.test.js +65 -0
  34. package/dist/api/workspaces.test.js.map +1 -0
  35. package/dist/cli/auth.d.ts +86 -0
  36. package/dist/cli/auth.d.ts.map +1 -0
  37. package/dist/cli/auth.js +284 -0
  38. package/dist/cli/auth.js.map +1 -0
  39. package/dist/cli/branch-store.d.ts +53 -0
  40. package/dist/cli/branch-store.d.ts.map +1 -0
  41. package/dist/cli/branch-store.js +91 -0
  42. package/dist/cli/branch-store.js.map +1 -0
  43. package/dist/cli/branch-store.test.d.ts +2 -0
  44. package/dist/cli/branch-store.test.d.ts.map +1 -0
  45. package/dist/cli/branch-store.test.js +115 -0
  46. package/dist/cli/branch-store.test.js.map +1 -0
  47. package/dist/cli/commands/branch.d.ts +82 -0
  48. package/dist/cli/commands/branch.d.ts.map +1 -0
  49. package/dist/cli/commands/branch.js +215 -0
  50. package/dist/cli/commands/branch.js.map +1 -0
  51. package/dist/cli/commands/build.d.ts +43 -0
  52. package/dist/cli/commands/build.d.ts.map +1 -0
  53. package/dist/cli/commands/build.js +138 -0
  54. package/dist/cli/commands/build.js.map +1 -0
  55. package/dist/cli/commands/dev.d.ts +78 -0
  56. package/dist/cli/commands/dev.d.ts.map +1 -0
  57. package/dist/cli/commands/dev.js +226 -0
  58. package/dist/cli/commands/dev.js.map +1 -0
  59. package/dist/cli/commands/init.d.ts +45 -0
  60. package/dist/cli/commands/init.d.ts.map +1 -0
  61. package/dist/cli/commands/init.js +277 -0
  62. package/dist/cli/commands/init.js.map +1 -0
  63. package/dist/cli/commands/init.test.d.ts +2 -0
  64. package/dist/cli/commands/init.test.d.ts.map +1 -0
  65. package/dist/cli/commands/init.test.js +158 -0
  66. package/dist/cli/commands/init.test.js.map +1 -0
  67. package/dist/cli/commands/login.d.ts +37 -0
  68. package/dist/cli/commands/login.d.ts.map +1 -0
  69. package/dist/cli/commands/login.js +64 -0
  70. package/dist/cli/commands/login.js.map +1 -0
  71. package/dist/cli/config.d.ts +114 -0
  72. package/dist/cli/config.d.ts.map +1 -0
  73. package/dist/cli/config.js +258 -0
  74. package/dist/cli/config.js.map +1 -0
  75. package/dist/cli/config.test.d.ts +2 -0
  76. package/dist/cli/config.test.d.ts.map +1 -0
  77. package/dist/cli/config.test.js +243 -0
  78. package/dist/cli/config.test.js.map +1 -0
  79. package/dist/cli/env.d.ts +29 -0
  80. package/dist/cli/env.d.ts.map +1 -0
  81. package/dist/cli/env.js +66 -0
  82. package/dist/cli/env.js.map +1 -0
  83. package/dist/cli/git.d.ts +29 -0
  84. package/dist/cli/git.d.ts.map +1 -0
  85. package/dist/cli/git.js +114 -0
  86. package/dist/cli/git.js.map +1 -0
  87. package/dist/cli/git.test.d.ts +2 -0
  88. package/dist/cli/git.test.d.ts.map +1 -0
  89. package/dist/cli/git.test.js +125 -0
  90. package/dist/cli/git.test.js.map +1 -0
  91. package/dist/cli/index.d.ts +7 -0
  92. package/dist/cli/index.d.ts.map +1 -0
  93. package/dist/cli/index.js +337 -0
  94. package/dist/cli/index.js.map +1 -0
  95. package/dist/cli/utils/schema-validation.d.ts +95 -0
  96. package/dist/cli/utils/schema-validation.d.ts.map +1 -0
  97. package/dist/cli/utils/schema-validation.js +175 -0
  98. package/dist/cli/utils/schema-validation.js.map +1 -0
  99. package/dist/cli/utils/schema-validation.test.d.ts +5 -0
  100. package/dist/cli/utils/schema-validation.test.d.ts.map +1 -0
  101. package/dist/cli/utils/schema-validation.test.js +173 -0
  102. package/dist/cli/utils/schema-validation.test.js.map +1 -0
  103. package/dist/client/base.d.ts +116 -0
  104. package/dist/client/base.d.ts.map +1 -0
  105. package/dist/client/base.js +328 -0
  106. package/dist/client/base.js.map +1 -0
  107. package/dist/client/types.d.ts +137 -0
  108. package/dist/client/types.d.ts.map +1 -0
  109. package/dist/client/types.js +43 -0
  110. package/dist/client/types.js.map +1 -0
  111. package/dist/generator/client.d.ts +44 -0
  112. package/dist/generator/client.d.ts.map +1 -0
  113. package/dist/generator/client.js +144 -0
  114. package/dist/generator/client.js.map +1 -0
  115. package/dist/generator/datasource.d.ts +57 -0
  116. package/dist/generator/datasource.d.ts.map +1 -0
  117. package/dist/generator/datasource.js +169 -0
  118. package/dist/generator/datasource.js.map +1 -0
  119. package/dist/generator/datasource.test.d.ts +2 -0
  120. package/dist/generator/datasource.test.d.ts.map +1 -0
  121. package/dist/generator/datasource.test.js +254 -0
  122. package/dist/generator/datasource.test.js.map +1 -0
  123. package/dist/generator/index.d.ts +131 -0
  124. package/dist/generator/index.d.ts.map +1 -0
  125. package/dist/generator/index.js +121 -0
  126. package/dist/generator/index.js.map +1 -0
  127. package/dist/generator/index.test.d.ts +2 -0
  128. package/dist/generator/index.test.d.ts.map +1 -0
  129. package/dist/generator/index.test.js +175 -0
  130. package/dist/generator/index.test.js.map +1 -0
  131. package/dist/generator/loader.d.ts +156 -0
  132. package/dist/generator/loader.d.ts.map +1 -0
  133. package/dist/generator/loader.js +295 -0
  134. package/dist/generator/loader.js.map +1 -0
  135. package/dist/generator/pipe.d.ts +72 -0
  136. package/dist/generator/pipe.d.ts.map +1 -0
  137. package/dist/generator/pipe.js +174 -0
  138. package/dist/generator/pipe.js.map +1 -0
  139. package/dist/generator/pipe.test.d.ts +2 -0
  140. package/dist/generator/pipe.test.d.ts.map +1 -0
  141. package/dist/generator/pipe.test.js +393 -0
  142. package/dist/generator/pipe.test.js.map +1 -0
  143. package/dist/index.d.ts +74 -0
  144. package/dist/index.d.ts.map +1 -0
  145. package/dist/index.js +73 -0
  146. package/dist/index.js.map +1 -0
  147. package/dist/infer/index.d.ts +202 -0
  148. package/dist/infer/index.d.ts.map +1 -0
  149. package/dist/infer/index.js +5 -0
  150. package/dist/infer/index.js.map +1 -0
  151. package/dist/schema/datasource.d.ts +135 -0
  152. package/dist/schema/datasource.d.ts.map +1 -0
  153. package/dist/schema/datasource.js +105 -0
  154. package/dist/schema/datasource.js.map +1 -0
  155. package/dist/schema/datasource.test.d.ts +2 -0
  156. package/dist/schema/datasource.test.d.ts.map +1 -0
  157. package/dist/schema/datasource.test.js +142 -0
  158. package/dist/schema/datasource.test.js.map +1 -0
  159. package/dist/schema/engines.d.ts +157 -0
  160. package/dist/schema/engines.d.ts.map +1 -0
  161. package/dist/schema/engines.js +155 -0
  162. package/dist/schema/engines.js.map +1 -0
  163. package/dist/schema/engines.test.d.ts +2 -0
  164. package/dist/schema/engines.test.d.ts.map +1 -0
  165. package/dist/schema/engines.test.js +221 -0
  166. package/dist/schema/engines.test.js.map +1 -0
  167. package/dist/schema/params.d.ts +106 -0
  168. package/dist/schema/params.d.ts.map +1 -0
  169. package/dist/schema/params.js +138 -0
  170. package/dist/schema/params.js.map +1 -0
  171. package/dist/schema/params.test.d.ts +2 -0
  172. package/dist/schema/params.test.d.ts.map +1 -0
  173. package/dist/schema/params.test.js +175 -0
  174. package/dist/schema/params.test.js.map +1 -0
  175. package/dist/schema/pipe.d.ts +436 -0
  176. package/dist/schema/pipe.d.ts.map +1 -0
  177. package/dist/schema/pipe.js +484 -0
  178. package/dist/schema/pipe.js.map +1 -0
  179. package/dist/schema/pipe.test.d.ts +2 -0
  180. package/dist/schema/pipe.test.d.ts.map +1 -0
  181. package/dist/schema/pipe.test.js +488 -0
  182. package/dist/schema/pipe.test.js.map +1 -0
  183. package/dist/schema/project.d.ts +202 -0
  184. package/dist/schema/project.d.ts.map +1 -0
  185. package/dist/schema/project.js +188 -0
  186. package/dist/schema/project.js.map +1 -0
  187. package/dist/schema/project.test.d.ts +2 -0
  188. package/dist/schema/project.test.d.ts.map +1 -0
  189. package/dist/schema/project.test.js +180 -0
  190. package/dist/schema/project.test.js.map +1 -0
  191. package/dist/schema/types.d.ts +140 -0
  192. package/dist/schema/types.d.ts.map +1 -0
  193. package/dist/schema/types.js +174 -0
  194. package/dist/schema/types.js.map +1 -0
  195. package/dist/schema/types.test.d.ts +2 -0
  196. package/dist/schema/types.test.d.ts.map +1 -0
  197. package/dist/schema/types.test.js +176 -0
  198. package/dist/schema/types.test.js.map +1 -0
  199. package/dist/test/handlers.d.ts +58 -0
  200. package/dist/test/handlers.d.ts.map +1 -0
  201. package/dist/test/handlers.js +62 -0
  202. package/dist/test/handlers.js.map +1 -0
  203. package/dist/test/setup.d.ts +5 -0
  204. package/dist/test/setup.d.ts.map +1 -0
  205. package/dist/test/setup.js +11 -0
  206. package/dist/test/setup.js.map +1 -0
  207. package/package.json +57 -0
  208. package/src/api/branches.test.ts +377 -0
  209. package/src/api/branches.ts +334 -0
  210. package/src/api/build.test.ts +216 -0
  211. package/src/api/build.ts +266 -0
  212. package/src/api/deploy.test.ts +193 -0
  213. package/src/api/deploy.ts +163 -0
  214. package/src/api/workspaces.test.ts +81 -0
  215. package/src/api/workspaces.ts +77 -0
  216. package/src/cli/auth.ts +358 -0
  217. package/src/cli/branch-store.test.ts +139 -0
  218. package/src/cli/branch-store.ts +137 -0
  219. package/src/cli/commands/branch.ts +306 -0
  220. package/src/cli/commands/build.ts +183 -0
  221. package/src/cli/commands/dev.ts +334 -0
  222. package/src/cli/commands/init.test.ts +249 -0
  223. package/src/cli/commands/init.ts +323 -0
  224. package/src/cli/commands/login.ts +98 -0
  225. package/src/cli/config.test.ts +359 -0
  226. package/src/cli/config.ts +335 -0
  227. package/src/cli/env.ts +86 -0
  228. package/src/cli/git.test.ts +147 -0
  229. package/src/cli/git.ts +125 -0
  230. package/src/cli/index.ts +382 -0
  231. package/src/cli/utils/schema-validation.test.ts +222 -0
  232. package/src/cli/utils/schema-validation.ts +272 -0
  233. package/src/client/base.ts +414 -0
  234. package/src/client/types.ts +165 -0
  235. package/src/generator/client.ts +194 -0
  236. package/src/generator/datasource.test.ts +297 -0
  237. package/src/generator/datasource.ts +217 -0
  238. package/src/generator/index.test.ts +209 -0
  239. package/src/generator/index.ts +203 -0
  240. package/src/generator/loader.ts +406 -0
  241. package/src/generator/pipe.test.ts +441 -0
  242. package/src/generator/pipe.ts +220 -0
  243. package/src/index.ts +191 -0
  244. package/src/infer/index.ts +247 -0
  245. package/src/schema/datasource.test.ts +187 -0
  246. package/src/schema/datasource.ts +195 -0
  247. package/src/schema/engines.test.ts +247 -0
  248. package/src/schema/engines.ts +271 -0
  249. package/src/schema/params.test.ts +208 -0
  250. package/src/schema/params.ts +249 -0
  251. package/src/schema/pipe.test.ts +588 -0
  252. package/src/schema/pipe.ts +832 -0
  253. package/src/schema/project.test.ts +236 -0
  254. package/src/schema/project.ts +394 -0
  255. package/src/schema/types.test.ts +212 -0
  256. package/src/schema/types.ts +366 -0
  257. package/src/test/handlers.ts +79 -0
  258. package/src/test/setup.ts +13 -0
@@ -0,0 +1,62 @@
1
+ /**
2
+ * MSW handlers for API mocking in tests
3
+ */
4
+ import { http, HttpResponse } from "msw";
5
+ export const BASE_URL = "https://api.tinybird.co";
6
+ /**
7
+ * Create build success response
8
+ */
9
+ export function createBuildSuccessResponse(options) {
10
+ return {
11
+ result: "success",
12
+ build: {
13
+ id: options?.buildId ?? "build-123",
14
+ changed_pipe_names: options?.changedPipes ?? [],
15
+ new_pipe_names: options?.newPipes ?? [],
16
+ deleted_pipe_names: options?.deletedPipes ?? [],
17
+ changed_datasource_names: options?.changedDatasources ?? [],
18
+ new_datasource_names: options?.newDatasources ?? [],
19
+ deleted_datasource_names: options?.deletedDatasources ?? [],
20
+ },
21
+ };
22
+ }
23
+ /**
24
+ * Create build failure response
25
+ */
26
+ export function createBuildFailureResponse(error) {
27
+ return {
28
+ result: "failed",
29
+ error,
30
+ };
31
+ }
32
+ /**
33
+ * Create build failure response with multiple errors
34
+ */
35
+ export function createBuildMultipleErrorsResponse(errors) {
36
+ return {
37
+ result: "failed",
38
+ errors,
39
+ };
40
+ }
41
+ /**
42
+ * Create no changes response
43
+ */
44
+ export function createNoChangesResponse() {
45
+ return {
46
+ result: "no_changes",
47
+ };
48
+ }
49
+ /**
50
+ * Default handlers for build and deploy endpoints
51
+ */
52
+ export const handlers = [
53
+ // Build endpoint - success by default
54
+ http.post(`${BASE_URL}/v1/build`, () => {
55
+ return HttpResponse.json(createBuildSuccessResponse());
56
+ }),
57
+ // Deploy endpoint - success by default
58
+ http.post(`${BASE_URL}/v1/deploy`, () => {
59
+ return HttpResponse.json(createBuildSuccessResponse());
60
+ }),
61
+ ];
62
+ //# sourceMappingURL=handlers.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"handlers.js","sourceRoot":"","sources":["../../src/test/handlers.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,IAAI,EAAE,YAAY,EAAE,MAAM,KAAK,CAAC;AAEzC,MAAM,CAAC,MAAM,QAAQ,GAAG,yBAAyB,CAAC;AAElD;;GAEG;AACH,MAAM,UAAU,0BAA0B,CAAC,OAQ1C;IACC,OAAO;QACL,MAAM,EAAE,SAAS;QACjB,KAAK,EAAE;YACL,EAAE,EAAE,OAAO,EAAE,OAAO,IAAI,WAAW;YACnC,kBAAkB,EAAE,OAAO,EAAE,YAAY,IAAI,EAAE;YAC/C,cAAc,EAAE,OAAO,EAAE,QAAQ,IAAI,EAAE;YACvC,kBAAkB,EAAE,OAAO,EAAE,YAAY,IAAI,EAAE;YAC/C,wBAAwB,EAAE,OAAO,EAAE,kBAAkB,IAAI,EAAE;YAC3D,oBAAoB,EAAE,OAAO,EAAE,cAAc,IAAI,EAAE;YACnD,wBAAwB,EAAE,OAAO,EAAE,kBAAkB,IAAI,EAAE;SAC5D;KACF,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,0BAA0B,CAAC,KAAa;IACtD,OAAO;QACL,MAAM,EAAE,QAAQ;QAChB,KAAK;KACN,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,iCAAiC,CAC/C,MAAmD;IAEnD,OAAO;QACL,MAAM,EAAE,QAAQ;QAChB,MAAM;KACP,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,UAAU,uBAAuB;IACrC,OAAO;QACL,MAAM,EAAE,YAAY;KACrB,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,MAAM,CAAC,MAAM,QAAQ,GAAG;IACtB,sCAAsC;IACtC,IAAI,CAAC,IAAI,CAAC,GAAG,QAAQ,WAAW,EAAE,GAAG,EAAE;QACrC,OAAO,YAAY,CAAC,IAAI,CAAC,0BAA0B,EAAE,CAAC,CAAC;IACzD,CAAC,CAAC;IAEF,uCAAuC;IACvC,IAAI,CAAC,IAAI,CAAC,GAAG,QAAQ,YAAY,EAAE,GAAG,EAAE;QACtC,OAAO,YAAY,CAAC,IAAI,CAAC,0BAA0B,EAAE,CAAC,CAAC;IACzD,CAAC,CAAC;CACH,CAAC"}
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Test setup file for MSW
3
+ */
4
+ export declare const server: import("msw/node").SetupServerApi;
5
+ //# sourceMappingURL=setup.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.d.ts","sourceRoot":"","sources":["../../src/test/setup.ts"],"names":[],"mappings":"AAAA;;GAEG;AAMH,eAAO,MAAM,MAAM,mCAA2B,CAAC"}
@@ -0,0 +1,11 @@
1
+ /**
2
+ * Test setup file for MSW
3
+ */
4
+ import { beforeAll, afterEach, afterAll } from "vitest";
5
+ import { setupServer } from "msw/node";
6
+ import { handlers } from "./handlers.js";
7
+ export const server = setupServer(...handlers);
8
+ beforeAll(() => server.listen({ onUnhandledRequest: "error" }));
9
+ afterEach(() => server.resetHandlers());
10
+ afterAll(() => server.close());
11
+ //# sourceMappingURL=setup.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"setup.js","sourceRoot":"","sources":["../../src/test/setup.ts"],"names":[],"mappings":"AAAA;;GAEG;AAEH,OAAO,EAAE,SAAS,EAAE,SAAS,EAAE,QAAQ,EAAE,MAAM,QAAQ,CAAC;AACxD,OAAO,EAAE,WAAW,EAAE,MAAM,UAAU,CAAC;AACvC,OAAO,EAAE,QAAQ,EAAE,MAAM,eAAe,CAAC;AAEzC,MAAM,CAAC,MAAM,MAAM,GAAG,WAAW,CAAC,GAAG,QAAQ,CAAC,CAAC;AAE/C,SAAS,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,kBAAkB,EAAE,OAAO,EAAE,CAAC,CAAC,CAAC;AAChE,SAAS,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,aAAa,EAAE,CAAC,CAAC;AACxC,QAAQ,CAAC,GAAG,EAAE,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC"}
package/package.json ADDED
@@ -0,0 +1,57 @@
1
+ {
2
+ "name": "@tinybirdco/sdk",
3
+ "version": "0.0.1",
4
+ "description": "TypeScript SDK for Tinybird Forward - define datasources and pipes as TypeScript",
5
+ "type": "module",
6
+ "main": "./dist/index.js",
7
+ "module": "./dist/index.js",
8
+ "types": "./dist/index.d.ts",
9
+ "bin": {
10
+ "tinybird": "./bin/tinybird.js"
11
+ },
12
+ "exports": {
13
+ ".": {
14
+ "import": "./dist/index.js",
15
+ "types": "./dist/index.d.ts"
16
+ }
17
+ },
18
+ "files": [
19
+ "dist",
20
+ "bin",
21
+ "src"
22
+ ],
23
+ "keywords": [
24
+ "tinybird",
25
+ "analytics",
26
+ "typescript",
27
+ "sdk",
28
+ "clickhouse",
29
+ "cli"
30
+ ],
31
+ "author": "Tinybird",
32
+ "license": "MIT",
33
+ "dependencies": {
34
+ "chokidar": "^4.0.0",
35
+ "commander": "^12.0.0",
36
+ "esbuild": "^0.24.0"
37
+ },
38
+ "devDependencies": {
39
+ "@types/node": "^20.0.0",
40
+ "@vitest/coverage-v8": "^4.0.18",
41
+ "msw": "^2.12.7",
42
+ "typescript": "^5.3.0",
43
+ "vitest": "^4.0.18"
44
+ },
45
+ "engines": {
46
+ "node": ">=20.0.0"
47
+ },
48
+ "scripts": {
49
+ "build": "tsc",
50
+ "dev": "tsc --watch",
51
+ "typecheck": "tsc --noEmit",
52
+ "clean": "rm -rf dist",
53
+ "test": "vitest",
54
+ "test:run": "vitest run",
55
+ "test:coverage": "vitest run --coverage"
56
+ }
57
+ }
@@ -0,0 +1,377 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ import {
3
+ BranchApiError,
4
+ createBranch,
5
+ listBranches,
6
+ getBranch,
7
+ deleteBranch,
8
+ branchExists,
9
+ getOrCreateBranch,
10
+ type BranchApiConfig,
11
+ } from "./branches.js";
12
+
13
+ // Mock fetch globally
14
+ const mockFetch = vi.fn();
15
+ global.fetch = mockFetch;
16
+
17
+ describe("Branch API client", () => {
18
+ const config: BranchApiConfig = {
19
+ baseUrl: "https://api.tinybird.co",
20
+ token: "p.test-token",
21
+ };
22
+
23
+ beforeEach(() => {
24
+ mockFetch.mockReset();
25
+ vi.useFakeTimers();
26
+ });
27
+
28
+ afterEach(() => {
29
+ vi.useRealTimers();
30
+ });
31
+
32
+ describe("createBranch", () => {
33
+ it("creates a branch and returns it", async () => {
34
+ const mockBranch = {
35
+ id: "branch-123",
36
+ name: "my-feature",
37
+ token: "p.branch-token",
38
+ created_at: "2024-01-01T00:00:00Z",
39
+ };
40
+
41
+ // 1. POST to /v1/environments returns a job
42
+ mockFetch.mockResolvedValueOnce({
43
+ ok: true,
44
+ json: () => Promise.resolve({
45
+ job: { id: "job-123", status: "waiting" },
46
+ workspace: { id: "ws-123" },
47
+ }),
48
+ });
49
+
50
+ // 2. Poll job - returns done
51
+ mockFetch.mockResolvedValueOnce({
52
+ ok: true,
53
+ json: () => Promise.resolve({ id: "job-123", status: "done" }),
54
+ });
55
+
56
+ // 3. Get branch with token
57
+ mockFetch.mockResolvedValueOnce({
58
+ ok: true,
59
+ json: () => Promise.resolve(mockBranch),
60
+ });
61
+
62
+ const result = await createBranch(config, "my-feature");
63
+
64
+ expect(mockFetch).toHaveBeenCalledTimes(3);
65
+ expect(mockFetch).toHaveBeenNthCalledWith(
66
+ 1,
67
+ "https://api.tinybird.co/v1/environments?name=my-feature",
68
+ {
69
+ method: "POST",
70
+ headers: {
71
+ Authorization: "Bearer p.test-token",
72
+ },
73
+ }
74
+ );
75
+ expect(mockFetch).toHaveBeenNthCalledWith(
76
+ 2,
77
+ "https://api.tinybird.co/v0/jobs/job-123",
78
+ {
79
+ method: "GET",
80
+ headers: {
81
+ Authorization: "Bearer p.test-token",
82
+ },
83
+ }
84
+ );
85
+ expect(mockFetch).toHaveBeenNthCalledWith(
86
+ 3,
87
+ "https://api.tinybird.co/v0/environments/my-feature?with_token=true",
88
+ {
89
+ method: "GET",
90
+ headers: {
91
+ Authorization: "Bearer p.test-token",
92
+ },
93
+ }
94
+ );
95
+ expect(result).toEqual(mockBranch);
96
+ });
97
+
98
+ it("polls job until done", async () => {
99
+ const mockBranch = {
100
+ id: "branch-123",
101
+ name: "my-feature",
102
+ token: "p.branch-token",
103
+ created_at: "2024-01-01T00:00:00Z",
104
+ };
105
+
106
+ // 1. POST to /v1/environments returns a job
107
+ mockFetch.mockResolvedValueOnce({
108
+ ok: true,
109
+ json: () => Promise.resolve({
110
+ job: { id: "job-123", status: "waiting" },
111
+ workspace: { id: "ws-123" },
112
+ }),
113
+ });
114
+
115
+ // 2. Poll job - waiting
116
+ mockFetch.mockResolvedValueOnce({
117
+ ok: true,
118
+ json: () => Promise.resolve({ id: "job-123", status: "waiting" }),
119
+ });
120
+
121
+ // 3. Poll job - working
122
+ mockFetch.mockResolvedValueOnce({
123
+ ok: true,
124
+ json: () => Promise.resolve({ id: "job-123", status: "working" }),
125
+ });
126
+
127
+ // 4. Poll job - done
128
+ mockFetch.mockResolvedValueOnce({
129
+ ok: true,
130
+ json: () => Promise.resolve({ id: "job-123", status: "done" }),
131
+ });
132
+
133
+ // 5. Get branch with token
134
+ mockFetch.mockResolvedValueOnce({
135
+ ok: true,
136
+ json: () => Promise.resolve(mockBranch),
137
+ });
138
+
139
+ // Start the async operation
140
+ const promise = createBranch(config, "my-feature");
141
+
142
+ // Advance timers and run all pending promises
143
+ await vi.runAllTimersAsync();
144
+
145
+ const result = await promise;
146
+
147
+ expect(mockFetch).toHaveBeenCalledTimes(5);
148
+ expect(result).toEqual(mockBranch);
149
+ });
150
+
151
+ it("throws BranchApiError on job error", async () => {
152
+ // 1. POST to /v1/environments returns a job
153
+ mockFetch.mockResolvedValueOnce({
154
+ ok: true,
155
+ json: () => Promise.resolve({
156
+ job: { id: "job-123", status: "waiting" },
157
+ }),
158
+ });
159
+
160
+ // 2. Poll job - error
161
+ mockFetch.mockResolvedValueOnce({
162
+ ok: true,
163
+ json: () => Promise.resolve({ id: "job-123", status: "error", error: "Something went wrong" }),
164
+ });
165
+
166
+ await expect(createBranch(config, "my-feature")).rejects.toThrow(
167
+ BranchApiError
168
+ );
169
+ });
170
+
171
+ it("throws BranchApiError on failure", async () => {
172
+ mockFetch.mockResolvedValueOnce({
173
+ ok: false,
174
+ status: 400,
175
+ statusText: "Bad Request",
176
+ text: () => Promise.resolve("Branch already exists"),
177
+ });
178
+
179
+ await expect(createBranch(config, "existing")).rejects.toThrow(
180
+ BranchApiError
181
+ );
182
+ });
183
+ });
184
+
185
+ describe("listBranches", () => {
186
+ it("returns array of branches", async () => {
187
+ const mockBranches = [
188
+ { id: "1", name: "feature-a", created_at: "2024-01-01T00:00:00Z" },
189
+ { id: "2", name: "feature-b", created_at: "2024-01-02T00:00:00Z" },
190
+ ];
191
+
192
+ mockFetch.mockResolvedValueOnce({
193
+ ok: true,
194
+ json: () => Promise.resolve({ environments: mockBranches }),
195
+ });
196
+
197
+ const result = await listBranches(config);
198
+
199
+ expect(mockFetch).toHaveBeenCalledWith(
200
+ "https://api.tinybird.co/v1/environments",
201
+ {
202
+ method: "GET",
203
+ headers: {
204
+ Authorization: "Bearer p.test-token",
205
+ },
206
+ }
207
+ );
208
+ expect(result).toEqual(mockBranches);
209
+ });
210
+
211
+ it("returns empty array when no branches", async () => {
212
+ mockFetch.mockResolvedValueOnce({
213
+ ok: true,
214
+ json: () => Promise.resolve({ environments: undefined }),
215
+ });
216
+
217
+ const result = await listBranches(config);
218
+ expect(result).toEqual([]);
219
+ });
220
+ });
221
+
222
+ describe("getBranch", () => {
223
+ it("returns branch with token", async () => {
224
+ const mockBranch = {
225
+ id: "branch-123",
226
+ name: "my-feature",
227
+ token: "p.branch-token",
228
+ created_at: "2024-01-01T00:00:00Z",
229
+ };
230
+
231
+ mockFetch.mockResolvedValueOnce({
232
+ ok: true,
233
+ json: () => Promise.resolve(mockBranch),
234
+ });
235
+
236
+ const result = await getBranch(config, "my-feature");
237
+
238
+ expect(mockFetch).toHaveBeenCalledWith(
239
+ "https://api.tinybird.co/v0/environments/my-feature?with_token=true",
240
+ {
241
+ method: "GET",
242
+ headers: {
243
+ Authorization: "Bearer p.test-token",
244
+ },
245
+ }
246
+ );
247
+ expect(result).toEqual(mockBranch);
248
+ });
249
+
250
+ it("throws BranchApiError when branch not found", async () => {
251
+ mockFetch.mockResolvedValueOnce({
252
+ ok: false,
253
+ status: 404,
254
+ statusText: "Not Found",
255
+ text: () => Promise.resolve("Branch not found"),
256
+ });
257
+
258
+ await expect(getBranch(config, "nonexistent")).rejects.toThrow(
259
+ BranchApiError
260
+ );
261
+ });
262
+ });
263
+
264
+ describe("deleteBranch", () => {
265
+ it("deletes a branch successfully", async () => {
266
+ mockFetch.mockResolvedValueOnce({
267
+ ok: true,
268
+ });
269
+
270
+ await deleteBranch(config, "my-feature");
271
+
272
+ expect(mockFetch).toHaveBeenCalledWith(
273
+ "https://api.tinybird.co/v1/environments/my-feature",
274
+ {
275
+ method: "DELETE",
276
+ headers: {
277
+ Authorization: "Bearer p.test-token",
278
+ },
279
+ }
280
+ );
281
+ });
282
+ });
283
+
284
+ describe("branchExists", () => {
285
+ it("returns true when branch exists", async () => {
286
+ mockFetch.mockResolvedValueOnce({
287
+ ok: true,
288
+ json: () =>
289
+ Promise.resolve({
290
+ environments: [
291
+ { id: "1", name: "my-feature", created_at: "2024-01-01" },
292
+ ],
293
+ }),
294
+ });
295
+
296
+ const result = await branchExists(config, "my-feature");
297
+ expect(result).toBe(true);
298
+ });
299
+
300
+ it("returns false when branch does not exist", async () => {
301
+ mockFetch.mockResolvedValueOnce({
302
+ ok: true,
303
+ json: () => Promise.resolve({ environments: [] }),
304
+ });
305
+
306
+ const result = await branchExists(config, "nonexistent");
307
+ expect(result).toBe(false);
308
+ });
309
+
310
+ it("throws on API error", async () => {
311
+ mockFetch.mockRejectedValueOnce(new Error("Network error"));
312
+
313
+ await expect(branchExists(config, "any")).rejects.toThrow("Network error");
314
+ });
315
+ });
316
+
317
+ describe("getOrCreateBranch", () => {
318
+ it("returns existing branch if found", async () => {
319
+ const mockBranch = {
320
+ id: "branch-123",
321
+ name: "existing-feature",
322
+ token: "p.branch-token",
323
+ created_at: "2024-01-01T00:00:00Z",
324
+ };
325
+
326
+ mockFetch.mockResolvedValueOnce({
327
+ ok: true,
328
+ json: () => Promise.resolve(mockBranch),
329
+ });
330
+
331
+ const result = await getOrCreateBranch(config, "existing-feature");
332
+ expect(result).toEqual({ ...mockBranch, wasCreated: false });
333
+ });
334
+
335
+ it("creates branch if not found", async () => {
336
+ const newBranch = {
337
+ id: "branch-456",
338
+ name: "new-feature",
339
+ token: "p.new-token",
340
+ created_at: "2024-01-02T00:00:00Z",
341
+ };
342
+
343
+ // 1. getBranch returns 404
344
+ mockFetch.mockResolvedValueOnce({
345
+ ok: false,
346
+ status: 404,
347
+ statusText: "Not Found",
348
+ text: () => Promise.resolve("Not found"),
349
+ });
350
+
351
+ // 2. createBranch: POST to /v1/environments returns a job
352
+ mockFetch.mockResolvedValueOnce({
353
+ ok: true,
354
+ json: () => Promise.resolve({
355
+ job: { id: "job-456", status: "waiting" },
356
+ workspace: { id: "ws-456" },
357
+ }),
358
+ });
359
+
360
+ // 3. Poll job - done
361
+ mockFetch.mockResolvedValueOnce({
362
+ ok: true,
363
+ json: () => Promise.resolve({ id: "job-456", status: "done" }),
364
+ });
365
+
366
+ // 4. Get branch with token
367
+ mockFetch.mockResolvedValueOnce({
368
+ ok: true,
369
+ json: () => Promise.resolve(newBranch),
370
+ });
371
+
372
+ const result = await getOrCreateBranch(config, "new-feature");
373
+ expect(result).toEqual({ ...newBranch, wasCreated: true });
374
+ expect(mockFetch).toHaveBeenCalledTimes(4);
375
+ });
376
+ });
377
+ });