alpic 0.0.0-dev.f5e699d → 0.0.0-dev.f610c40

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 (191) hide show
  1. package/dist/__tests__/auth.e2e.test.d.ts +1 -0
  2. package/dist/__tests__/auth.e2e.test.js +158 -0
  3. package/dist/__tests__/auth.e2e.test.js.map +1 -0
  4. package/dist/__tests__/deploy-flags.e2e.test.d.ts +1 -0
  5. package/dist/__tests__/deploy-flags.e2e.test.js +111 -0
  6. package/dist/__tests__/deploy-flags.e2e.test.js.map +1 -0
  7. package/dist/__tests__/deploy.e2e.test.d.ts +1 -0
  8. package/dist/__tests__/deploy.e2e.test.js +168 -0
  9. package/dist/__tests__/deploy.e2e.test.js.map +1 -0
  10. package/dist/__tests__/deployment-inspect.e2e.test.d.ts +1 -0
  11. package/dist/__tests__/deployment-inspect.e2e.test.js +113 -0
  12. package/dist/__tests__/deployment-inspect.e2e.test.js.map +1 -0
  13. package/dist/__tests__/deployment-list.e2e.test.d.ts +1 -0
  14. package/dist/__tests__/deployment-list.e2e.test.js +116 -0
  15. package/dist/__tests__/deployment-list.e2e.test.js.map +1 -0
  16. package/dist/__tests__/deployment-logs.e2e.test.d.ts +1 -0
  17. package/dist/__tests__/deployment-logs.e2e.test.js +255 -0
  18. package/dist/__tests__/deployment-logs.e2e.test.js.map +1 -0
  19. package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.d.ts +1 -0
  20. package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.js +260 -0
  21. package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.js.map +1 -0
  22. package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.d.ts +1 -0
  23. package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.js +140 -0
  24. package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.js.map +1 -0
  25. package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.d.ts +1 -0
  26. package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.js +151 -0
  27. package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.js.map +1 -0
  28. package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.d.ts +1 -0
  29. package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.js +343 -0
  30. package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.js.map +1 -0
  31. package/dist/__tests__/environment-variable/environment-variable-validation.test.d.ts +1 -0
  32. package/dist/__tests__/environment-variable/environment-variable-validation.test.js +20 -0
  33. package/dist/__tests__/environment-variable/environment-variable-validation.test.js.map +1 -0
  34. package/dist/__tests__/fixtures/demo-project/index.d.ts +1 -0
  35. package/dist/__tests__/fixtures/demo-project/index.js +4 -0
  36. package/dist/__tests__/fixtures/demo-project/index.js.map +1 -0
  37. package/dist/__tests__/git-flags.e2e.test.d.ts +1 -0
  38. package/dist/__tests__/git-flags.e2e.test.js +124 -0
  39. package/dist/__tests__/git-flags.e2e.test.js.map +1 -0
  40. package/dist/__tests__/git.e2e.test.d.ts +1 -0
  41. package/dist/__tests__/git.e2e.test.js +221 -0
  42. package/dist/__tests__/git.e2e.test.js.map +1 -0
  43. package/dist/__tests__/logs.e2e.test.d.ts +1 -0
  44. package/dist/__tests__/logs.e2e.test.js +227 -0
  45. package/dist/__tests__/logs.e2e.test.js.map +1 -0
  46. package/dist/__tests__/mock-server.d.ts +38 -0
  47. package/dist/__tests__/mock-server.js +728 -0
  48. package/dist/__tests__/mock-server.js.map +1 -0
  49. package/dist/__tests__/publish.e2e.test.d.ts +1 -0
  50. package/dist/__tests__/publish.e2e.test.js +505 -0
  51. package/dist/__tests__/publish.e2e.test.js.map +1 -0
  52. package/dist/__tests__/tunnel.e2e.test.d.ts +1 -0
  53. package/dist/__tests__/tunnel.e2e.test.js +64 -0
  54. package/dist/__tests__/tunnel.e2e.test.js.map +1 -0
  55. package/dist/__tests__/utils.d.ts +72 -0
  56. package/dist/__tests__/utils.js +272 -0
  57. package/dist/__tests__/utils.js.map +1 -0
  58. package/dist/api.d.ts +3 -16
  59. package/dist/api.js +14 -61
  60. package/dist/api.js.map +1 -1
  61. package/dist/commands/deploy.d.ts +7 -4
  62. package/dist/commands/deploy.js +52 -33
  63. package/dist/commands/deploy.js.map +1 -1
  64. package/dist/commands/deployment/inspect.d.ts +11 -0
  65. package/dist/commands/deployment/inspect.js +91 -0
  66. package/dist/commands/deployment/inspect.js.map +1 -0
  67. package/dist/commands/deployment/list.d.ts +11 -0
  68. package/dist/commands/deployment/list.js +97 -0
  69. package/dist/commands/deployment/list.js.map +1 -0
  70. package/dist/commands/deployment/logs.d.ts +12 -0
  71. package/dist/commands/deployment/logs.js +50 -0
  72. package/dist/commands/deployment/logs.js.map +1 -0
  73. package/dist/commands/environment-variable/add.d.ts +14 -0
  74. package/dist/commands/environment-variable/add.js +46 -0
  75. package/dist/commands/environment-variable/add.js.map +1 -0
  76. package/dist/commands/environment-variable/list.d.ts +9 -0
  77. package/dist/commands/environment-variable/list.js +44 -0
  78. package/dist/commands/environment-variable/list.js.map +1 -0
  79. package/dist/commands/environment-variable/remove.d.ts +11 -0
  80. package/dist/commands/environment-variable/remove.js +32 -0
  81. package/dist/commands/environment-variable/remove.js.map +1 -0
  82. package/dist/commands/environment-variable/update.d.ts +13 -0
  83. package/dist/commands/environment-variable/update.js +40 -0
  84. package/dist/commands/environment-variable/update.js.map +1 -0
  85. package/dist/commands/git/connect.d.ts +10 -0
  86. package/dist/commands/git/connect.js +60 -0
  87. package/dist/commands/git/connect.js.map +1 -0
  88. package/dist/commands/git/disconnect.d.ts +9 -0
  89. package/dist/commands/git/disconnect.js +45 -0
  90. package/dist/commands/git/disconnect.js.map +1 -0
  91. package/dist/commands/git.d.ts +6 -0
  92. package/dist/commands/git.js +17 -0
  93. package/dist/commands/git.js.map +1 -0
  94. package/dist/commands/login.d.ts +6 -0
  95. package/dist/commands/login.js +34 -0
  96. package/dist/commands/login.js.map +1 -0
  97. package/dist/commands/logout.d.ts +6 -0
  98. package/dist/commands/logout.js +20 -0
  99. package/dist/commands/logout.js.map +1 -0
  100. package/dist/commands/logs.d.ts +16 -0
  101. package/dist/commands/logs.js +96 -0
  102. package/dist/commands/logs.js.map +1 -0
  103. package/dist/commands/publish.d.ts +15 -0
  104. package/dist/commands/publish.js +51 -0
  105. package/dist/commands/publish.js.map +1 -0
  106. package/dist/commands/telemetry/disable.d.ts +5 -0
  107. package/dist/commands/telemetry/disable.js +14 -0
  108. package/dist/commands/telemetry/disable.js.map +1 -0
  109. package/dist/commands/telemetry/enable.d.ts +5 -0
  110. package/dist/commands/telemetry/enable.js +13 -0
  111. package/dist/commands/telemetry/enable.js.map +1 -0
  112. package/dist/commands/telemetry/status.d.ts +5 -0
  113. package/dist/commands/telemetry/status.js +19 -0
  114. package/dist/commands/telemetry/status.js.map +1 -0
  115. package/dist/commands/tunnel.d.ts +9 -0
  116. package/dist/commands/tunnel.js +53 -0
  117. package/dist/commands/tunnel.js.map +1 -0
  118. package/dist/commands/whoami.d.ts +6 -0
  119. package/dist/commands/whoami.js +13 -0
  120. package/dist/commands/whoami.js.map +1 -0
  121. package/dist/env.d.ts +4 -0
  122. package/dist/env.js +10 -0
  123. package/dist/env.js.map +1 -0
  124. package/dist/lib/alpic-command.d.ts +6 -0
  125. package/dist/lib/alpic-command.js +27 -0
  126. package/dist/lib/alpic-command.js.map +1 -0
  127. package/dist/lib/archive.d.ts +3 -3
  128. package/dist/lib/archive.js +11 -15
  129. package/dist/lib/archive.js.map +1 -1
  130. package/dist/lib/auth/auth.d.ts +2 -0
  131. package/dist/lib/auth/auth.js +21 -0
  132. package/dist/lib/auth/auth.js.map +1 -0
  133. package/dist/lib/auth/oauth/client.d.ts +28 -0
  134. package/dist/lib/auth/oauth/client.js +110 -0
  135. package/dist/lib/auth/oauth/client.js.map +1 -0
  136. package/dist/lib/auth/oauth/constants.d.ts +2 -0
  137. package/dist/lib/auth/oauth/constants.js +3 -0
  138. package/dist/lib/auth/oauth/constants.js.map +1 -0
  139. package/dist/lib/auth/oauth/server/assets/alpic-mountain.png +0 -0
  140. package/dist/lib/auth/oauth/server/assets/authorize.html +195 -0
  141. package/dist/lib/auth/oauth/server/assets/callback.html +88 -0
  142. package/dist/lib/auth/oauth/server/index.d.ts +8 -0
  143. package/dist/lib/auth/oauth/server/index.js +105 -0
  144. package/dist/lib/auth/oauth/server/index.js.map +1 -0
  145. package/dist/lib/auth/whoami.d.ts +1 -0
  146. package/dist/lib/auth/whoami.js +41 -0
  147. package/dist/lib/auth/whoami.js.map +1 -0
  148. package/dist/lib/base-workflow.d.ts +10 -0
  149. package/dist/lib/base-workflow.js +22 -0
  150. package/dist/lib/base-workflow.js.map +1 -0
  151. package/dist/lib/config.d.ts +2 -2
  152. package/dist/lib/config.js +7 -7
  153. package/dist/lib/config.js.map +1 -1
  154. package/dist/lib/deployment.d.ts +83 -6
  155. package/dist/lib/deployment.js +121 -9
  156. package/dist/lib/deployment.js.map +1 -1
  157. package/dist/lib/environment-variable.d.ts +41 -0
  158. package/dist/lib/environment-variable.js +311 -0
  159. package/dist/lib/environment-variable.js.map +1 -0
  160. package/dist/lib/git.d.ts +22 -0
  161. package/dist/lib/git.js +131 -0
  162. package/dist/lib/git.js.map +1 -0
  163. package/dist/lib/global-store.d.ts +28 -0
  164. package/dist/lib/global-store.js +76 -0
  165. package/dist/lib/global-store.js.map +1 -0
  166. package/dist/lib/logs.d.ts +20 -0
  167. package/dist/lib/logs.js +86 -0
  168. package/dist/lib/logs.js.map +1 -0
  169. package/dist/lib/project.d.ts +74 -11
  170. package/dist/lib/project.js +287 -142
  171. package/dist/lib/project.js.map +1 -1
  172. package/dist/lib/publish.d.ts +22 -0
  173. package/dist/lib/publish.js +188 -0
  174. package/dist/lib/publish.js.map +1 -0
  175. package/dist/lib/table.d.ts +8 -0
  176. package/dist/lib/table.js +27 -0
  177. package/dist/lib/table.js.map +1 -0
  178. package/dist/lib/telemetry.d.ts +7 -0
  179. package/dist/lib/telemetry.js +66 -0
  180. package/dist/lib/telemetry.js.map +1 -0
  181. package/dist/lib/utils.d.ts +4 -0
  182. package/dist/lib/utils.js +45 -0
  183. package/dist/lib/utils.js.map +1 -0
  184. package/dist/lib/utils.test.d.ts +1 -0
  185. package/dist/lib/utils.test.js +27 -0
  186. package/dist/lib/utils.test.js.map +1 -0
  187. package/dist/posthog.d.ts +3 -0
  188. package/dist/posthog.js +10 -0
  189. package/dist/posthog.js.map +1 -0
  190. package/dist/types.d.ts +0 -35
  191. package/package.json +41 -13
@@ -0,0 +1,116 @@
1
+ import { rmSync } from "node:fs";
2
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
3
+ import { MockApiServer } from "./mock-server.js";
4
+ import { buildMockDeployment, buildMockProject, cliSession, expectSuccessfulApiCall, setupTestDirectory, } from "./utils.js";
5
+ describe("alpic deployment list (E2E)", () => {
6
+ let mockServer;
7
+ let serverUrl;
8
+ let cwd;
9
+ let binPath;
10
+ const projectId = "test-project-id";
11
+ const teamId = "mock-team-id";
12
+ const environmentId = "test-env-id";
13
+ const environmentName = "production";
14
+ beforeEach(async () => {
15
+ mockServer = new MockApiServer();
16
+ serverUrl = await mockServer.start();
17
+ const setup = await setupTestDirectory();
18
+ cwd = setup.cwd;
19
+ binPath = setup.binPath;
20
+ mockServer.addProject(buildMockProject({
21
+ projectId,
22
+ teamId,
23
+ projectName: "test-project",
24
+ environmentId,
25
+ environmentName,
26
+ }));
27
+ });
28
+ afterEach(async () => {
29
+ await mockServer.stop();
30
+ if (cwd) {
31
+ rmSync(cwd, { recursive: true, force: true });
32
+ }
33
+ });
34
+ it("renders deployment rows correctly", async () => {
35
+ const sevenDaysAgo = new Date(Date.now() - 7 * 24 * 60 * 60 * 1000);
36
+ const thirteenDaysAgo = new Date(Date.now() - 13 * 24 * 60 * 60 * 1000);
37
+ mockServer.addDeployments([
38
+ buildMockDeployment({
39
+ id: "dpl_test_deployed",
40
+ startedAt: sevenDaysAgo,
41
+ completedAt: new Date(sevenDaysAgo.getTime() + 116_000),
42
+ environmentId,
43
+ environmentName,
44
+ }),
45
+ buildMockDeployment({
46
+ id: "dpl_test_canceled",
47
+ status: "canceled",
48
+ sourceCommitId: "def5678",
49
+ sourceCommitMessage: "Cancel deployment",
50
+ authorUsername: null,
51
+ startedAt: thirteenDaysAgo,
52
+ completedAt: new Date(thirteenDaysAgo.getTime() + 12_000),
53
+ isCurrent: false,
54
+ environmentId,
55
+ environmentName,
56
+ }),
57
+ ]);
58
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "list", "--project-id", projectId], {
59
+ cwd,
60
+ env: {
61
+ ALPIC_API_BASE_URL: serverUrl,
62
+ ALPIC_API_KEY: "test-api-key",
63
+ },
64
+ }, async (cli) => {
65
+ await cli.expect("Found 2 deployments");
66
+ });
67
+ expect(exitCode).toBe(0);
68
+ const normalizedOutput = output.replace(/^.*Fetching deployments\.\.\.\r?\n?/gm, "");
69
+ expect(normalizedOutput).toMatchSnapshot();
70
+ const listCall = mockServer.getLastCall("GET", `/v1/projects/${projectId}/deployments`);
71
+ expectSuccessfulApiCall(listCall);
72
+ });
73
+ it("passes environment-id filter to the API server-side", async () => {
74
+ const otherEnvironmentId = "other-env-id";
75
+ mockServer.addDeployments([
76
+ buildMockDeployment({
77
+ environmentId,
78
+ environmentName,
79
+ id: "dpl_env_match",
80
+ }),
81
+ buildMockDeployment({
82
+ environmentId: otherEnvironmentId,
83
+ environmentName: "staging",
84
+ id: "dpl_env_other",
85
+ }),
86
+ ]);
87
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "list", "--project-id", projectId, "--environment-id", environmentId], {
88
+ cwd,
89
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
90
+ }, async (cli) => {
91
+ await cli.expect("Found 1 deployment");
92
+ });
93
+ expect(exitCode).toBe(0);
94
+ expect(output).toContain("dpl_env_match");
95
+ expect(output).not.toContain("dpl_env_other");
96
+ const listCall = mockServer.getLastCall("GET", `/v1/projects/${projectId}/deployments`);
97
+ expectSuccessfulApiCall(listCall);
98
+ expect(listCall?.input).toMatchObject({ environmentId });
99
+ });
100
+ it("passes status filter to the API", async () => {
101
+ const { exitCode } = await cliSession(process.execPath, [binPath, "deployment", "list", "--project-id", projectId, "--status", "deployed", "--status", "failed"], {
102
+ cwd,
103
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
104
+ }, async (cli) => {
105
+ await cli.expect("Found 0 deployments");
106
+ });
107
+ expect(exitCode).toBe(0);
108
+ const listCall = mockServer.getLastCall("GET", `/v1/projects/${projectId}/deployments`);
109
+ expectSuccessfulApiCall(listCall);
110
+ expect(listCall?.input).toMatchObject({
111
+ "status[0]": "deployed",
112
+ "status[1]": "failed",
113
+ });
114
+ });
115
+ }, 15_000);
116
+ //# sourceMappingURL=deployment-list.e2e.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deployment-list.e2e.test.js","sourceRoot":"","sources":["../../src/__tests__/deployment-list.e2e.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1 @@
1
+ export {};
@@ -0,0 +1,255 @@
1
+ import { rmSync } from "node:fs";
2
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
3
+ import { MockApiServer } from "./mock-server.js";
4
+ import { buildMockDeployment, buildMockProject, cliSession, createConfigFile, expectSuccessfulApiCall, setupTestDirectory, } from "./utils.js";
5
+ const DEPLOYMENT_ID = "dpl_logs_test";
6
+ const PROJECT_ID = "prj-logs-test";
7
+ const TEAM_ID = "mock-team-id";
8
+ const ENVIRONMENT_ID = "env-logs-test";
9
+ describe("alpic deployment logs (E2E)", () => {
10
+ let mockServer;
11
+ let serverUrl;
12
+ let cwd;
13
+ let binPath;
14
+ beforeEach(async () => {
15
+ mockServer = new MockApiServer();
16
+ serverUrl = await mockServer.start();
17
+ const setup = await setupTestDirectory();
18
+ cwd = setup.cwd;
19
+ binPath = setup.binPath;
20
+ mockServer.addProject(buildMockProject({
21
+ projectId: PROJECT_ID,
22
+ teamId: TEAM_ID,
23
+ projectName: "logs-project",
24
+ environmentId: ENVIRONMENT_ID,
25
+ environmentName: "production",
26
+ }));
27
+ mockServer.addDeployments([
28
+ buildMockDeployment({
29
+ id: DEPLOYMENT_ID,
30
+ environmentId: ENVIRONMENT_ID,
31
+ environmentName: "production",
32
+ }),
33
+ ]);
34
+ });
35
+ afterEach(async () => {
36
+ await mockServer.stop();
37
+ if (cwd)
38
+ rmSync(cwd, { recursive: true, force: true });
39
+ });
40
+ it("fetches and prints logs for a given --deployment-id", async () => {
41
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
42
+ {
43
+ logs: [
44
+ {
45
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
46
+ content: "Installing dependencies...",
47
+ },
48
+ {
49
+ timestamp: new Date("2024-06-01T12:00:01.000Z"),
50
+ content: "Build succeeded",
51
+ },
52
+ ],
53
+ hasMoreLogs: false,
54
+ },
55
+ ]);
56
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID], {
57
+ cwd,
58
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
59
+ });
60
+ expect(exitCode).toBe(0);
61
+ expect(output).toContain("Installing dependencies...");
62
+ expect(output).toContain("Build succeeded");
63
+ const call = mockServer.getLastCall("GET", /\/v1\/deployments\/.*\/logs/);
64
+ expectSuccessfulApiCall(call);
65
+ });
66
+ it("polls and streams new logs with --follow, stops when hasMoreLogs is false", async () => {
67
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
68
+ {
69
+ logs: [
70
+ {
71
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
72
+ content: "Step 1",
73
+ },
74
+ ],
75
+ hasMoreLogs: true,
76
+ },
77
+ {
78
+ logs: [
79
+ {
80
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
81
+ content: "Step 1",
82
+ },
83
+ {
84
+ timestamp: new Date("2024-06-01T12:00:01.000Z"),
85
+ content: "Step 2",
86
+ },
87
+ ],
88
+ hasMoreLogs: true,
89
+ },
90
+ {
91
+ logs: [
92
+ {
93
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
94
+ content: "Step 1",
95
+ },
96
+ {
97
+ timestamp: new Date("2024-06-01T12:00:01.000Z"),
98
+ content: "Step 2",
99
+ },
100
+ {
101
+ timestamp: new Date("2024-06-01T12:00:02.000Z"),
102
+ content: "Step 3 - done",
103
+ },
104
+ ],
105
+ hasMoreLogs: false,
106
+ },
107
+ ]);
108
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID, "--follow"], {
109
+ cwd,
110
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
111
+ timeoutMs: 15_000,
112
+ });
113
+ expect(exitCode).toBe(0);
114
+ expect(output).toContain("Step 1");
115
+ expect(output).toContain("Step 2");
116
+ expect(output).toContain("Step 3 - done");
117
+ const calls = mockServer.getCalls("GET", /\/v1\/deployments\/.*\/logs/);
118
+ expect(calls.length).toBe(3);
119
+ });
120
+ it("does not duplicate logs when polling with --follow", async () => {
121
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
122
+ {
123
+ logs: [
124
+ {
125
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
126
+ content: "Only once",
127
+ },
128
+ ],
129
+ hasMoreLogs: true,
130
+ },
131
+ {
132
+ logs: [
133
+ {
134
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
135
+ content: "Only once",
136
+ },
137
+ {
138
+ timestamp: new Date("2024-06-01T12:00:01.000Z"),
139
+ content: "Second entry",
140
+ },
141
+ ],
142
+ hasMoreLogs: false,
143
+ },
144
+ ]);
145
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID, "--follow"], {
146
+ cwd,
147
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
148
+ timeoutMs: 15_000,
149
+ });
150
+ expect(exitCode).toBe(0);
151
+ const occurrences = (output.match(/Only once/g) ?? []).length;
152
+ expect(occurrences).toBe(1);
153
+ });
154
+ it("exits immediately without --follow when hasMoreLogs is true", async () => {
155
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
156
+ {
157
+ logs: [
158
+ {
159
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
160
+ content: "Partial build log",
161
+ },
162
+ ],
163
+ hasMoreLogs: true,
164
+ },
165
+ ]);
166
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID], {
167
+ cwd,
168
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
169
+ });
170
+ expect(exitCode).toBe(0);
171
+ expect(output).toContain("Partial build log");
172
+ const calls = mockServer.getCalls("GET", /\/v1\/deployments\/.*\/logs/);
173
+ expect(calls.length).toBe(1);
174
+ });
175
+ it("strips ANSI codes from log content with --no-color", async () => {
176
+ // Content contains ANSI escape codes as CodeBuild tools (npm, tsc, etc.) often emit
177
+ const redText = "\x1b[31mBuild failed\x1b[0m";
178
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
179
+ {
180
+ logs: [{ timestamp: new Date("2024-06-01T12:00:00.000Z"), content: redText }],
181
+ hasMoreLogs: false,
182
+ },
183
+ ]);
184
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID, "--no-color"], {
185
+ cwd,
186
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
187
+ });
188
+ expect(exitCode).toBe(0);
189
+ expect(output).toContain("2024-06-01 12:00:00.000 Build failed");
190
+ expect(output).not.toContain("\x1b[31m");
191
+ });
192
+ it("exits with error when deployment is not found", async () => {
193
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", "dpl_does_not_exist"], {
194
+ cwd,
195
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
196
+ });
197
+ expect(exitCode).not.toBe(0);
198
+ expect(output).toMatch(/error|not found/i);
199
+ });
200
+ it("exits with error when not authenticated", async () => {
201
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--deployment-id", DEPLOYMENT_ID], { cwd, env: { ALPIC_API_BASE_URL: serverUrl } });
202
+ expect(exitCode).not.toBe(0);
203
+ expect(output).toMatch(/not authenticated|alpic login|ALPIC_API_KEY/i);
204
+ });
205
+ it("resolves deployment via --environment-id", async () => {
206
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
207
+ {
208
+ logs: [
209
+ {
210
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
211
+ content: "Env lookup log",
212
+ },
213
+ ],
214
+ hasMoreLogs: false,
215
+ },
216
+ ]);
217
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs", "--environment-id", ENVIRONMENT_ID], {
218
+ cwd,
219
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
220
+ });
221
+ expect(exitCode).toBe(0);
222
+ expect(output).toContain("Env lookup log");
223
+ const call = mockServer.getLastCall("GET", /\/v1\/deployments\/.*\/logs/);
224
+ expectSuccessfulApiCall(call);
225
+ });
226
+ it("resolves deployment via environment-id from config file", async () => {
227
+ mockServer.setDeploymentLogPages(DEPLOYMENT_ID, [
228
+ {
229
+ logs: [
230
+ {
231
+ timestamp: new Date("2024-06-01T12:00:00.000Z"),
232
+ content: "Config file log",
233
+ },
234
+ ],
235
+ hasMoreLogs: false,
236
+ },
237
+ ]);
238
+ createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENVIRONMENT_ID, "production");
239
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs"], {
240
+ cwd,
241
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
242
+ });
243
+ expect(exitCode).toBe(0);
244
+ expect(output).toContain("Config file log");
245
+ });
246
+ it("exits with error when no deployment-id or environment-id provided", async () => {
247
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "deployment", "logs"], {
248
+ cwd,
249
+ env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
250
+ });
251
+ expect(exitCode).not.toBe(0);
252
+ expect(output).toMatch(/environment|deployment-id/i);
253
+ });
254
+ }, 30_000);
255
+ //# sourceMappingURL=deployment-logs.e2e.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deployment-logs.e2e.test.js","sourceRoot":"","sources":["../../src/__tests__/deployment-logs.e2e.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,260 @@
1
+ import { rmSync, writeFileSync } from "node:fs";
2
+ import { join } from "node:path";
3
+ import { afterEach, beforeEach, describe, expect, it } from "vitest";
4
+ import { MockApiServer } from "../mock-server.js";
5
+ import { buildMockProject, cliSession, createConfigFile, expectSuccessfulApiCall, setupTestDirectory, } from "../utils.js";
6
+ describe("alpic environment-variable add (E2E)", () => {
7
+ let mockServer;
8
+ let serverUrl;
9
+ let cwd;
10
+ let binPath;
11
+ const projectId = "test-project-id";
12
+ const teamId = "mock-team-id";
13
+ const environmentId = "test-env-id";
14
+ const environmentName = "production";
15
+ beforeEach(async () => {
16
+ mockServer = new MockApiServer();
17
+ serverUrl = await mockServer.start();
18
+ const setup = await setupTestDirectory();
19
+ cwd = setup.cwd;
20
+ binPath = setup.binPath;
21
+ mockServer.addProject(buildMockProject({
22
+ projectId,
23
+ teamId,
24
+ projectName: "test-project",
25
+ environmentId,
26
+ environmentName,
27
+ }));
28
+ });
29
+ afterEach(async () => {
30
+ await mockServer.stop();
31
+ if (cwd) {
32
+ rmSync(cwd, { recursive: true, force: true });
33
+ }
34
+ });
35
+ it("adds a single variable in non-interactive mode", async () => {
36
+ const { exitCode, output } = await cliSession(process.execPath, [
37
+ binPath,
38
+ "environment-variable",
39
+ "add",
40
+ "--environment-id",
41
+ environmentId,
42
+ "--key",
43
+ "DATABASE_URL",
44
+ "--value",
45
+ "postgres://localhost/db",
46
+ "--non-interactive",
47
+ ], {
48
+ cwd,
49
+ env: {
50
+ ALPIC_API_BASE_URL: serverUrl,
51
+ ALPIC_API_KEY: "test-api-key",
52
+ },
53
+ }, async (cli) => {
54
+ await cli.expect("Successfully added 1 environment variable.");
55
+ });
56
+ expect(exitCode).toBe(0);
57
+ expect(output).toContain("Successfully added 1 environment variable.");
58
+ const call = mockServer.getLastCall("POST", `/v1/environments/${environmentId}/environment-variables`);
59
+ expectSuccessfulApiCall(call);
60
+ expect(call?.input).toMatchObject({
61
+ environmentVariables: [
62
+ {
63
+ key: "DATABASE_URL",
64
+ value: "postgres://localhost/db",
65
+ isSecret: true,
66
+ },
67
+ ],
68
+ });
69
+ });
70
+ it("adds a non-secret variable with --no-secret", async () => {
71
+ const { exitCode } = await cliSession(process.execPath, [
72
+ binPath,
73
+ "environment-variable",
74
+ "add",
75
+ "--environment-id",
76
+ environmentId,
77
+ "--key",
78
+ "PORT",
79
+ "--value",
80
+ "3000",
81
+ "--no-secret",
82
+ "--non-interactive",
83
+ ], {
84
+ cwd,
85
+ env: {
86
+ ALPIC_API_BASE_URL: serverUrl,
87
+ ALPIC_API_KEY: "test-api-key",
88
+ },
89
+ }, async (cli) => {
90
+ await cli.expect("Successfully added 1 environment variable.");
91
+ });
92
+ expect(exitCode).toBe(0);
93
+ const call = mockServer.getLastCall("POST", `/v1/environments/${environmentId}/environment-variables`);
94
+ expectSuccessfulApiCall(call);
95
+ expect(call?.input).toMatchObject({
96
+ environmentVariables: [{ key: "PORT", value: "3000", isSecret: false }],
97
+ });
98
+ });
99
+ it("adds multiple variables from a .env file", async () => {
100
+ const envFilePath = join(cwd, ".env");
101
+ writeFileSync(envFilePath, "DATABASE_URL=postgres://localhost/db\nAPI_KEY=supersecret\nPORT=3000\n");
102
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "environment-variable", "add", "--environment-id", environmentId, "--env-file", envFilePath], {
103
+ cwd,
104
+ env: {
105
+ ALPIC_API_BASE_URL: serverUrl,
106
+ ALPIC_API_KEY: "test-api-key",
107
+ },
108
+ }, async (cli) => {
109
+ await cli.expect("Successfully added 3 environment variables.");
110
+ });
111
+ expect(exitCode).toBe(0);
112
+ expect(output).toContain("Successfully added 3 environment variables.");
113
+ const call = mockServer.getLastCall("POST", `/v1/environments/${environmentId}/environment-variables`);
114
+ expectSuccessfulApiCall(call);
115
+ expect(call?.input).toMatchObject({
116
+ environmentVariables: [
117
+ {
118
+ key: "DATABASE_URL",
119
+ value: "postgres://localhost/db",
120
+ isSecret: true,
121
+ },
122
+ { key: "API_KEY", value: "supersecret", isSecret: true },
123
+ { key: "PORT", value: "3000", isSecret: true },
124
+ ],
125
+ });
126
+ });
127
+ it("adds variables from a .env file with --no-secret", async () => {
128
+ const envFilePath = join(cwd, ".env");
129
+ writeFileSync(envFilePath, "PORT=3000\n");
130
+ const { exitCode } = await cliSession(process.execPath, [
131
+ binPath,
132
+ "environment-variable",
133
+ "add",
134
+ "--environment-id",
135
+ environmentId,
136
+ "--env-file",
137
+ envFilePath,
138
+ "--no-secret",
139
+ ], {
140
+ cwd,
141
+ env: {
142
+ ALPIC_API_BASE_URL: serverUrl,
143
+ ALPIC_API_KEY: "test-api-key",
144
+ },
145
+ }, async (cli) => {
146
+ await cli.expect("Successfully added 1 environment variable.");
147
+ });
148
+ expect(exitCode).toBe(0);
149
+ const call = mockServer.getLastCall("POST", `/v1/environments/${environmentId}/environment-variables`);
150
+ expectSuccessfulApiCall(call);
151
+ expect(call?.input).toMatchObject({
152
+ environmentVariables: [{ key: "PORT", value: "3000", isSecret: false }],
153
+ });
154
+ });
155
+ it("uses environment-id from config file", async () => {
156
+ createConfigFile(cwd, projectId, teamId, "test-project", environmentId, environmentName);
157
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "environment-variable", "add", "--key", "FOO", "--value", "bar", "--non-interactive"], {
158
+ cwd,
159
+ env: {
160
+ ALPIC_API_BASE_URL: serverUrl,
161
+ ALPIC_API_KEY: "test-api-key",
162
+ },
163
+ }, async (cli) => {
164
+ await cli.expect("Successfully added 1 environment variable.");
165
+ });
166
+ expect(exitCode).toBe(0);
167
+ expect(output).toContain(`Using environment id from .alpic config: ${environmentId}`);
168
+ });
169
+ it("errors when --env-file and --key are both provided", async () => {
170
+ const envFilePath = join(cwd, ".env");
171
+ writeFileSync(envFilePath, "FOO=bar\n");
172
+ const { exitCode, output } = await cliSession(process.execPath, [
173
+ binPath,
174
+ "environment-variable",
175
+ "add",
176
+ "--environment-id",
177
+ environmentId,
178
+ "--env-file",
179
+ envFilePath,
180
+ "--key",
181
+ "FOO",
182
+ "--non-interactive",
183
+ ], {
184
+ cwd,
185
+ env: {
186
+ ALPIC_API_BASE_URL: serverUrl,
187
+ ALPIC_API_KEY: "test-api-key",
188
+ },
189
+ });
190
+ expect(exitCode).toBe(1);
191
+ expect(output).toContain("--env-file cannot be used together with --key or --value.");
192
+ });
193
+ it("errors when --value is empty in non-interactive mode", async () => {
194
+ const { exitCode, output } = await cliSession(process.execPath, [
195
+ binPath,
196
+ "environment-variable",
197
+ "add",
198
+ "--environment-id",
199
+ environmentId,
200
+ "--key",
201
+ "FOO",
202
+ "--value",
203
+ "",
204
+ "--non-interactive",
205
+ ], {
206
+ cwd,
207
+ env: {
208
+ ALPIC_API_BASE_URL: serverUrl,
209
+ ALPIC_API_KEY: "test-api-key",
210
+ },
211
+ });
212
+ expect(exitCode).toBe(1);
213
+ expect(output).toContain("Value cannot be empty.");
214
+ });
215
+ it("errors when --key and --value are missing in non-interactive mode", async () => {
216
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "environment-variable", "add", "--environment-id", environmentId, "--non-interactive"], {
217
+ cwd,
218
+ env: {
219
+ ALPIC_API_BASE_URL: serverUrl,
220
+ ALPIC_API_KEY: "test-api-key",
221
+ },
222
+ });
223
+ expect(exitCode).toBe(1);
224
+ expect(output).toContain("--key and --value are required in non-interactive mode.");
225
+ });
226
+ it("exits with error when not authenticated", async () => {
227
+ const { exitCode, output } = await cliSession(process.execPath, [
228
+ binPath,
229
+ "environment-variable",
230
+ "add",
231
+ "--environment-id",
232
+ environmentId,
233
+ "--key",
234
+ "FOO",
235
+ "--value",
236
+ "bar",
237
+ "--non-interactive",
238
+ ], {
239
+ cwd,
240
+ env: {
241
+ ALPIC_API_BASE_URL: serverUrl,
242
+ },
243
+ });
244
+ expect(exitCode).toBe(1);
245
+ expect(output).toContain("Not authenticated");
246
+ expect(output).toMatch(/alpic login|ALPIC_API_KEY/);
247
+ });
248
+ it("exits with error when no environment-id is provided and no config file exists", async () => {
249
+ const { exitCode, output } = await cliSession(process.execPath, [binPath, "environment-variable", "add", "--key", "FOO", "--value", "bar", "--non-interactive"], {
250
+ cwd,
251
+ env: {
252
+ ALPIC_API_BASE_URL: serverUrl,
253
+ ALPIC_API_KEY: "test-api-key",
254
+ },
255
+ });
256
+ expect(exitCode).toBe(1);
257
+ expect(output).toContain("No environment specified");
258
+ });
259
+ }, 15_000);
260
+ //# sourceMappingURL=environment-variable-add.e2e.test.js.map