alpic 0.0.0-dev.fc2d21b → 0.0.0-dev.fc7e119
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.
- package/dist/__tests__/auth.e2e.test.d.ts +1 -0
- package/dist/__tests__/auth.e2e.test.js +158 -0
- package/dist/__tests__/auth.e2e.test.js.map +1 -0
- package/dist/__tests__/deploy-flags.e2e.test.d.ts +1 -0
- package/dist/__tests__/deploy-flags.e2e.test.js +111 -0
- package/dist/__tests__/deploy-flags.e2e.test.js.map +1 -0
- package/dist/__tests__/deploy.e2e.test.d.ts +1 -0
- package/dist/__tests__/deploy.e2e.test.js +168 -0
- package/dist/__tests__/deploy.e2e.test.js.map +1 -0
- package/dist/__tests__/deployment-inspect.e2e.test.d.ts +1 -0
- package/dist/__tests__/deployment-inspect.e2e.test.js +111 -0
- package/dist/__tests__/deployment-inspect.e2e.test.js.map +1 -0
- package/dist/__tests__/deployment-list.e2e.test.d.ts +1 -0
- package/dist/__tests__/deployment-list.e2e.test.js +108 -0
- package/dist/__tests__/deployment-list.e2e.test.js.map +1 -0
- package/dist/__tests__/deployment-logs.e2e.test.d.ts +1 -0
- package/dist/__tests__/deployment-logs.e2e.test.js +180 -0
- package/dist/__tests__/deployment-logs.e2e.test.js.map +1 -0
- package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.d.ts +1 -0
- package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.js +250 -0
- package/dist/__tests__/environment-variable/environment-variable-add.e2e.test.js.map +1 -0
- package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.d.ts +1 -0
- package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.js +122 -0
- package/dist/__tests__/environment-variable/environment-variable-list.e2e.test.js.map +1 -0
- package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.d.ts +1 -0
- package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.js +139 -0
- package/dist/__tests__/environment-variable/environment-variable-remove.e2e.test.js.map +1 -0
- package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.d.ts +1 -0
- package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.js +319 -0
- package/dist/__tests__/environment-variable/environment-variable-update.e2e.test.js.map +1 -0
- package/dist/__tests__/environment-variable/environment-variable-validation.test.d.ts +1 -0
- package/dist/__tests__/environment-variable/environment-variable-validation.test.js +20 -0
- package/dist/__tests__/environment-variable/environment-variable-validation.test.js.map +1 -0
- package/dist/__tests__/fixtures/demo-project/index.d.ts +1 -0
- package/dist/__tests__/fixtures/demo-project/index.js +4 -0
- package/dist/__tests__/fixtures/demo-project/index.js.map +1 -0
- package/dist/__tests__/git-flags.e2e.test.d.ts +1 -0
- package/dist/__tests__/git-flags.e2e.test.js +124 -0
- package/dist/__tests__/git-flags.e2e.test.js.map +1 -0
- package/dist/__tests__/git.e2e.test.d.ts +1 -0
- package/dist/__tests__/git.e2e.test.js +221 -0
- package/dist/__tests__/git.e2e.test.js.map +1 -0
- package/dist/__tests__/logs.e2e.test.d.ts +1 -0
- package/dist/__tests__/logs.e2e.test.js +197 -0
- package/dist/__tests__/logs.e2e.test.js.map +1 -0
- package/dist/__tests__/mock-server.d.ts +35 -0
- package/dist/__tests__/mock-server.js +646 -0
- package/dist/__tests__/mock-server.js.map +1 -0
- package/dist/__tests__/tunnel.e2e.test.d.ts +1 -0
- package/dist/__tests__/tunnel.e2e.test.js +64 -0
- package/dist/__tests__/tunnel.e2e.test.js.map +1 -0
- package/dist/__tests__/utils.d.ts +70 -0
- package/dist/__tests__/utils.js +275 -0
- package/dist/__tests__/utils.js.map +1 -0
- package/dist/api.d.ts +3 -0
- package/dist/api.js +15 -0
- package/dist/api.js.map +1 -0
- package/dist/commands/deploy.d.ts +12 -0
- package/dist/commands/deploy.js +100 -0
- package/dist/commands/deploy.js.map +1 -0
- package/dist/commands/deployment/inspect.d.ts +11 -0
- package/dist/commands/deployment/inspect.js +68 -0
- package/dist/commands/deployment/inspect.js.map +1 -0
- package/dist/commands/deployment/list.d.ts +11 -0
- package/dist/commands/deployment/list.js +85 -0
- package/dist/commands/deployment/list.js.map +1 -0
- package/dist/commands/deployment/logs.d.ts +12 -0
- package/dist/commands/deployment/logs.js +48 -0
- package/dist/commands/deployment/logs.js.map +1 -0
- package/dist/commands/environment-variable/add.d.ts +14 -0
- package/dist/commands/environment-variable/add.js +46 -0
- package/dist/commands/environment-variable/add.js.map +1 -0
- package/dist/commands/environment-variable/list.d.ts +9 -0
- package/dist/commands/environment-variable/list.js +44 -0
- package/dist/commands/environment-variable/list.js.map +1 -0
- package/dist/commands/environment-variable/remove.d.ts +11 -0
- package/dist/commands/environment-variable/remove.js +32 -0
- package/dist/commands/environment-variable/remove.js.map +1 -0
- package/dist/commands/environment-variable/update.d.ts +13 -0
- package/dist/commands/environment-variable/update.js +40 -0
- package/dist/commands/environment-variable/update.js.map +1 -0
- package/dist/commands/git/connect.d.ts +10 -0
- package/dist/commands/git/connect.js +58 -0
- package/dist/commands/git/connect.js.map +1 -0
- package/dist/commands/git/disconnect.d.ts +9 -0
- package/dist/commands/git/disconnect.js +41 -0
- package/dist/commands/git/disconnect.js.map +1 -0
- package/dist/commands/{hello.d.ts → git.d.ts} +1 -1
- package/dist/commands/git.js +17 -0
- package/dist/commands/git.js.map +1 -0
- package/dist/commands/login.d.ts +6 -0
- package/dist/commands/login.js +34 -0
- package/dist/commands/login.js.map +1 -0
- package/dist/commands/logout.d.ts +6 -0
- package/dist/commands/logout.js +20 -0
- package/dist/commands/logout.js.map +1 -0
- package/dist/commands/logs.d.ts +16 -0
- package/dist/commands/logs.js +96 -0
- package/dist/commands/logs.js.map +1 -0
- package/dist/commands/telemetry/disable.d.ts +5 -0
- package/dist/commands/telemetry/disable.js +14 -0
- package/dist/commands/telemetry/disable.js.map +1 -0
- package/dist/commands/telemetry/enable.d.ts +5 -0
- package/dist/commands/telemetry/enable.js +13 -0
- package/dist/commands/telemetry/enable.js.map +1 -0
- package/dist/commands/telemetry/status.d.ts +5 -0
- package/dist/commands/telemetry/status.js +19 -0
- package/dist/commands/telemetry/status.js.map +1 -0
- package/dist/commands/tunnel.d.ts +9 -0
- package/dist/commands/tunnel.js +45 -0
- package/dist/commands/tunnel.js.map +1 -0
- package/dist/commands/whoami.d.ts +6 -0
- package/dist/commands/whoami.js +13 -0
- package/dist/commands/whoami.js.map +1 -0
- package/dist/env.d.ts +4 -0
- package/dist/env.js +10 -0
- package/dist/env.js.map +1 -0
- package/dist/lib/alpic-command.d.ts +6 -0
- package/dist/lib/alpic-command.js +27 -0
- package/dist/lib/alpic-command.js.map +1 -0
- package/dist/lib/archive.d.ts +7 -0
- package/dist/lib/archive.js +51 -0
- package/dist/lib/archive.js.map +1 -0
- package/dist/lib/auth/auth.d.ts +2 -0
- package/dist/lib/auth/auth.js +21 -0
- package/dist/lib/auth/auth.js.map +1 -0
- package/dist/lib/auth/oauth/client.d.ts +28 -0
- package/dist/lib/auth/oauth/client.js +110 -0
- package/dist/lib/auth/oauth/client.js.map +1 -0
- package/dist/lib/auth/oauth/constants.d.ts +2 -0
- package/dist/lib/auth/oauth/constants.js +3 -0
- package/dist/lib/auth/oauth/constants.js.map +1 -0
- package/dist/lib/auth/oauth/server/assets/alpic-mountain.png +0 -0
- package/dist/lib/auth/oauth/server/assets/authorize.html +195 -0
- package/dist/lib/auth/oauth/server/assets/callback.html +88 -0
- package/dist/lib/auth/oauth/server/index.d.ts +8 -0
- package/dist/lib/auth/oauth/server/index.js +102 -0
- package/dist/lib/auth/oauth/server/index.js.map +1 -0
- package/dist/lib/auth/whoami.d.ts +1 -0
- package/dist/lib/auth/whoami.js +41 -0
- package/dist/lib/auth/whoami.js.map +1 -0
- package/dist/lib/base-workflow.d.ts +10 -0
- package/dist/lib/base-workflow.js +22 -0
- package/dist/lib/base-workflow.js.map +1 -0
- package/dist/lib/config.d.ts +11 -0
- package/dist/lib/config.js +31 -0
- package/dist/lib/config.js.map +1 -0
- package/dist/lib/deployment.d.ts +88 -0
- package/dist/lib/deployment.js +143 -0
- package/dist/lib/deployment.js.map +1 -0
- package/dist/lib/environment-variable.d.ts +41 -0
- package/dist/lib/environment-variable.js +304 -0
- package/dist/lib/environment-variable.js.map +1 -0
- package/dist/lib/git.d.ts +22 -0
- package/dist/lib/git.js +131 -0
- package/dist/lib/git.js.map +1 -0
- package/dist/lib/global-store.d.ts +28 -0
- package/dist/lib/global-store.js +76 -0
- package/dist/lib/global-store.js.map +1 -0
- package/dist/lib/logs.d.ts +20 -0
- package/dist/lib/logs.js +86 -0
- package/dist/lib/logs.js.map +1 -0
- package/dist/lib/project.d.ts +74 -0
- package/dist/lib/project.js +306 -0
- package/dist/lib/project.js.map +1 -0
- package/dist/lib/table.d.ts +8 -0
- package/dist/lib/table.js +27 -0
- package/dist/lib/table.js.map +1 -0
- package/dist/lib/telemetry.d.ts +7 -0
- package/dist/lib/telemetry.js +66 -0
- package/dist/lib/telemetry.js.map +1 -0
- package/dist/lib/upload.d.ts +1 -0
- package/dist/lib/upload.js +14 -0
- package/dist/lib/upload.js.map +1 -0
- package/dist/lib/utils.d.ts +4 -0
- package/dist/lib/utils.js +21 -0
- package/dist/lib/utils.js.map +1 -0
- package/dist/lib/utils.test.d.ts +1 -0
- package/dist/lib/utils.test.js +21 -0
- package/dist/lib/utils.test.js.map +1 -0
- package/dist/posthog.d.ts +3 -0
- package/dist/posthog.js +10 -0
- package/dist/posthog.js.map +1 -0
- package/dist/types.d.ts +7 -0
- package/dist/types.js +2 -0
- package/dist/types.js.map +1 -0
- package/package.json +40 -9
- package/dist/commands/hello.js +0 -10
- package/dist/commands/hello.js.map +0 -1
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.js","sourceRoot":"","sources":["../../../../src/__tests__/fixtures/demo-project/index.js"],"names":[],"mappings":";AAAA,OAAO,CAAC,GAAG,CAAC,mBAAmB,CAAC,CAAC;AACjC,MAAM,CAAC,OAAO,GAAG,EAAE,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
import { rmSync } from "node:fs";
|
|
2
|
+
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
3
|
+
import { MockApiServer } from "./mock-server.js";
|
|
4
|
+
import { buildMockProject, cliSession, createConfigFile, expectSuccessfulApiCall, runGit, setupTestDirectory, } from "./utils.js";
|
|
5
|
+
describe("alpic git connect (E2E, flags only)", () => {
|
|
6
|
+
let mockServer;
|
|
7
|
+
let serverUrl;
|
|
8
|
+
let cwd;
|
|
9
|
+
let binPath;
|
|
10
|
+
const projectId = "project-id";
|
|
11
|
+
const teamId = "team-id";
|
|
12
|
+
const environmentId = "env-id";
|
|
13
|
+
function setupProject(overrides) {
|
|
14
|
+
mockServer.addProject(buildMockProject({
|
|
15
|
+
projectId,
|
|
16
|
+
teamId,
|
|
17
|
+
projectName: "demo",
|
|
18
|
+
environmentId,
|
|
19
|
+
environmentName: "production",
|
|
20
|
+
sourceRepository: null,
|
|
21
|
+
sourceBranch: "main",
|
|
22
|
+
...overrides,
|
|
23
|
+
}));
|
|
24
|
+
createConfigFile(cwd, projectId, teamId, "demo", environmentId, "production");
|
|
25
|
+
}
|
|
26
|
+
beforeEach(async () => {
|
|
27
|
+
mockServer = new MockApiServer();
|
|
28
|
+
serverUrl = await mockServer.start();
|
|
29
|
+
const setup = await setupTestDirectory({ initGit: true });
|
|
30
|
+
cwd = setup.cwd;
|
|
31
|
+
binPath = setup.binPath;
|
|
32
|
+
});
|
|
33
|
+
afterEach(async () => {
|
|
34
|
+
await mockServer.stop();
|
|
35
|
+
if (cwd) {
|
|
36
|
+
rmSync(cwd, { recursive: true, force: true });
|
|
37
|
+
}
|
|
38
|
+
});
|
|
39
|
+
it("--non-interactive auto-selects the only available remote without prompting", async () => {
|
|
40
|
+
setupProject();
|
|
41
|
+
runGit(["remote", "remove", "upstream"], cwd);
|
|
42
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect", "--non-interactive"], {
|
|
43
|
+
cwd,
|
|
44
|
+
env: {
|
|
45
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
46
|
+
ALPIC_API_KEY: "test-api-key",
|
|
47
|
+
},
|
|
48
|
+
}, async (cli) => {
|
|
49
|
+
await cli.expect("✅ Connected");
|
|
50
|
+
});
|
|
51
|
+
expect(exitCode).toBe(0);
|
|
52
|
+
expect(output).toMatch(/Connected .*acme\/frontend.* to project .*demo.*\./);
|
|
53
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
54
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
55
|
+
expect(call.input).toMatchObject({ sourceRepository: "acme/frontend" });
|
|
56
|
+
});
|
|
57
|
+
});
|
|
58
|
+
it("--non-interactive fails when multiple remotes are available and no remote name is specified", async () => {
|
|
59
|
+
setupProject();
|
|
60
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect", "--non-interactive"], {
|
|
61
|
+
cwd,
|
|
62
|
+
env: {
|
|
63
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
64
|
+
ALPIC_API_KEY: "test-api-key",
|
|
65
|
+
},
|
|
66
|
+
}, async (cli) => {
|
|
67
|
+
await cli.expect("Multiple remotes found. Use --remote-name to select a specific remote.");
|
|
68
|
+
});
|
|
69
|
+
expect(exitCode).toBe(1);
|
|
70
|
+
expect(output).toContain("Multiple remotes found. Use --remote-name to select a specific remote.");
|
|
71
|
+
expect(mockServer.getCalls("PATCH", `/v1/projects/${projectId}`)).toHaveLength(0);
|
|
72
|
+
});
|
|
73
|
+
it("--remote-name selects the matching remote without prompting", async () => {
|
|
74
|
+
setupProject();
|
|
75
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect", "--remote-name", "upstream"], {
|
|
76
|
+
cwd,
|
|
77
|
+
env: {
|
|
78
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
79
|
+
ALPIC_API_KEY: "test-api-key",
|
|
80
|
+
},
|
|
81
|
+
});
|
|
82
|
+
expect(exitCode).toBe(0);
|
|
83
|
+
expect(output).toMatch(/Connected .*acme\/backend.* to project .*demo.*\./);
|
|
84
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
85
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
86
|
+
expect(call.input).toMatchObject({ sourceRepository: "acme/backend" });
|
|
87
|
+
});
|
|
88
|
+
});
|
|
89
|
+
it("--remote-name fails with a clear message when the remote does not exist", async () => {
|
|
90
|
+
setupProject();
|
|
91
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect", "--remote-name", "nonexistent"], {
|
|
92
|
+
cwd,
|
|
93
|
+
env: {
|
|
94
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
95
|
+
ALPIC_API_KEY: "test-api-key",
|
|
96
|
+
},
|
|
97
|
+
}, async (cli) => {
|
|
98
|
+
await cli.expect('Remote "nonexistent" not found');
|
|
99
|
+
});
|
|
100
|
+
expect(exitCode).toBe(1);
|
|
101
|
+
expect(output).toContain('Remote "nonexistent" not found');
|
|
102
|
+
expect(mockServer.getCalls("PATCH", `/v1/projects/${projectId}`)).toHaveLength(0);
|
|
103
|
+
});
|
|
104
|
+
it("disconnect --non-interactive skips the confirmation prompt", async () => {
|
|
105
|
+
setupProject({ sourceRepository: "acme/frontend" });
|
|
106
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "disconnect", "--non-interactive"], {
|
|
107
|
+
cwd,
|
|
108
|
+
env: {
|
|
109
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
110
|
+
ALPIC_API_KEY: "test-api-key",
|
|
111
|
+
},
|
|
112
|
+
}, async (cli) => {
|
|
113
|
+
await cli.expect("Disconnected");
|
|
114
|
+
});
|
|
115
|
+
expect(exitCode).toBe(0);
|
|
116
|
+
expect(output).toMatch(/Disconnected .*acme\/frontend.* from project .*demo.*\./);
|
|
117
|
+
expect(output).not.toContain("Are you sure");
|
|
118
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
119
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
120
|
+
expect(call.input).toMatchObject({ sourceRepository: null });
|
|
121
|
+
});
|
|
122
|
+
});
|
|
123
|
+
}, 15_000);
|
|
124
|
+
//# sourceMappingURL=git-flags.e2e.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"git-flags.e2e.test.js","sourceRoot":"","sources":["../../src/__tests__/git-flags.e2e.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,221 @@
|
|
|
1
|
+
import { rmSync } 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, createOAuthEnv, expectSuccessfulApiCall, runGit, setupTestDirectory, } from "./utils.js";
|
|
6
|
+
describe("alpic git connect/disconnect (E2E)", () => {
|
|
7
|
+
let mockServer;
|
|
8
|
+
let serverUrl;
|
|
9
|
+
let cwd;
|
|
10
|
+
let binPath;
|
|
11
|
+
const projectId = "project-id";
|
|
12
|
+
const teamId = "team-id";
|
|
13
|
+
const environmentId = "env-id";
|
|
14
|
+
function setupProject(overrides) {
|
|
15
|
+
mockServer.addProject(buildMockProject({
|
|
16
|
+
projectId,
|
|
17
|
+
teamId,
|
|
18
|
+
projectName: "demo",
|
|
19
|
+
environmentId,
|
|
20
|
+
environmentName: "production",
|
|
21
|
+
sourceRepository: null,
|
|
22
|
+
sourceBranch: "main",
|
|
23
|
+
...overrides,
|
|
24
|
+
}));
|
|
25
|
+
createConfigFile(cwd, projectId, teamId, "demo", environmentId, "production");
|
|
26
|
+
}
|
|
27
|
+
beforeEach(async () => {
|
|
28
|
+
mockServer = new MockApiServer();
|
|
29
|
+
serverUrl = await mockServer.start();
|
|
30
|
+
const setup = await setupTestDirectory({ initGit: true });
|
|
31
|
+
cwd = setup.cwd;
|
|
32
|
+
binPath = setup.binPath;
|
|
33
|
+
});
|
|
34
|
+
afterEach(async () => {
|
|
35
|
+
await mockServer.stop();
|
|
36
|
+
if (cwd) {
|
|
37
|
+
rmSync(cwd, { recursive: true, force: true });
|
|
38
|
+
}
|
|
39
|
+
});
|
|
40
|
+
it("connects using selected remote source when multiple remotes are available", async () => {
|
|
41
|
+
setupProject();
|
|
42
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
43
|
+
cwd,
|
|
44
|
+
env: {
|
|
45
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
46
|
+
ALPIC_API_KEY: "test-api-key",
|
|
47
|
+
},
|
|
48
|
+
}, async (cli) => {
|
|
49
|
+
await cli.expect("Choose the remote source to connect");
|
|
50
|
+
await cli.send("\u001B[B"); // Move down one option (select acme/backend)
|
|
51
|
+
await cli.send("");
|
|
52
|
+
await cli.expect("✅ Connected");
|
|
53
|
+
});
|
|
54
|
+
expect(exitCode).toBe(0);
|
|
55
|
+
expect(output).toMatch(/Connected .*acme\/backend.* to project .*demo.*\./);
|
|
56
|
+
expect(output).toContain("make sure to install the Alpic GitHub App on this repository.");
|
|
57
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
58
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
59
|
+
expect(call.input).toMatchObject({
|
|
60
|
+
sourceRepository: "acme/backend",
|
|
61
|
+
});
|
|
62
|
+
});
|
|
63
|
+
});
|
|
64
|
+
it("requires linked Alpic project config", async () => {
|
|
65
|
+
rmSync(join(cwd, ".alpic"), { recursive: true, force: true });
|
|
66
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
67
|
+
cwd,
|
|
68
|
+
env: {
|
|
69
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
70
|
+
ALPIC_API_KEY: "test-api-key",
|
|
71
|
+
},
|
|
72
|
+
}, async (cli) => {
|
|
73
|
+
await cli.expect("This directory is not linked to an Alpic project. Link it first using `alpic deploy`.");
|
|
74
|
+
});
|
|
75
|
+
expect(exitCode).toBe(1);
|
|
76
|
+
expect(output).toContain("This directory is not linked to an Alpic project. Link it first using `alpic deploy`.");
|
|
77
|
+
});
|
|
78
|
+
it("shows git setup instructions when directory is not a git repository", async () => {
|
|
79
|
+
setupProject();
|
|
80
|
+
rmSync(join(cwd, ".git"), { recursive: true, force: true });
|
|
81
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
82
|
+
cwd,
|
|
83
|
+
env: {
|
|
84
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
85
|
+
ALPIC_API_KEY: "test-api-key",
|
|
86
|
+
},
|
|
87
|
+
}, async (cli) => {
|
|
88
|
+
await cli.expect("This directory is not a git repository.");
|
|
89
|
+
await cli.expect("To set up a git repository");
|
|
90
|
+
await cli.expect("git init");
|
|
91
|
+
});
|
|
92
|
+
expect(exitCode).toBe(0);
|
|
93
|
+
expect(output).toContain("This directory is not a git repository.");
|
|
94
|
+
expect(output).toContain("To set up a git repository");
|
|
95
|
+
expect(output).toContain("git init");
|
|
96
|
+
});
|
|
97
|
+
it("shows git setup instructions when repository has no remotes", async () => {
|
|
98
|
+
setupProject();
|
|
99
|
+
runGit(["remote", "remove", "origin"], cwd);
|
|
100
|
+
runGit(["remote", "remove", "upstream"], cwd);
|
|
101
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
102
|
+
cwd,
|
|
103
|
+
env: {
|
|
104
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
105
|
+
ALPIC_API_KEY: "test-api-key",
|
|
106
|
+
},
|
|
107
|
+
}, async (cli) => {
|
|
108
|
+
await cli.expect("This git repository has no remotes configured.");
|
|
109
|
+
await cli.expect("To set up a git repository");
|
|
110
|
+
await cli.expect("git init");
|
|
111
|
+
});
|
|
112
|
+
expect(exitCode).toBe(0);
|
|
113
|
+
expect(output).toContain("This git repository has no remotes configured.");
|
|
114
|
+
expect(output).toContain("To set up a git repository");
|
|
115
|
+
expect(output).toContain("git init");
|
|
116
|
+
});
|
|
117
|
+
it("when project is already connected, prompts to link another and exits if user declines", async () => {
|
|
118
|
+
setupProject({ sourceRepository: "acme/backend" });
|
|
119
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
120
|
+
cwd,
|
|
121
|
+
env: {
|
|
122
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
123
|
+
ALPIC_API_KEY: "test-api-key",
|
|
124
|
+
},
|
|
125
|
+
}, async (cli) => {
|
|
126
|
+
await cli.expect(/Project .*demo.* is already connected to .*acme\/backend.*\./);
|
|
127
|
+
await cli.expect("Do you want to link another repository?");
|
|
128
|
+
await cli.send(""); // accept default (No)
|
|
129
|
+
await cli.expect("Git connect cancelled");
|
|
130
|
+
});
|
|
131
|
+
expect(exitCode).toBe(1);
|
|
132
|
+
expect(output).toMatch(/Project .*demo.* is already connected to .*acme\/backend.*\./);
|
|
133
|
+
expect(output).toContain("Git connect cancelled");
|
|
134
|
+
expect(mockServer.getCalls("PATCH", "/v1/projects/")).toHaveLength(0);
|
|
135
|
+
});
|
|
136
|
+
it("disconnect fails with clear message when project is not connected", async () => {
|
|
137
|
+
setupProject(); // sourceRepository is null by default
|
|
138
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "disconnect"], {
|
|
139
|
+
cwd,
|
|
140
|
+
env: {
|
|
141
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
142
|
+
ALPIC_API_KEY: "test-api-key",
|
|
143
|
+
},
|
|
144
|
+
}, async (cli) => {
|
|
145
|
+
await cli.expect(/Project .*demo.* is not connected to any repository\./);
|
|
146
|
+
});
|
|
147
|
+
expect(exitCode).toBe(1);
|
|
148
|
+
expect(output).toMatch(/Project .*demo.* is not connected to any repository\./);
|
|
149
|
+
expect(mockServer.getCalls("PATCH", "/v1/projects/")).toHaveLength(0);
|
|
150
|
+
});
|
|
151
|
+
it("disconnects an already connected project repository", async () => {
|
|
152
|
+
setupProject({ sourceRepository: "acme/frontend" });
|
|
153
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "disconnect"], {
|
|
154
|
+
cwd,
|
|
155
|
+
env: {
|
|
156
|
+
ALPIC_API_BASE_URL: serverUrl,
|
|
157
|
+
ALPIC_API_KEY: "test-api-key",
|
|
158
|
+
},
|
|
159
|
+
}, async (cli) => {
|
|
160
|
+
await cli.expect(/Are you sure you want to disconnect .*acme\/frontend.* from project .*demo.*\?/);
|
|
161
|
+
await cli.send("\u001B[A"); // arrow up to select Yes
|
|
162
|
+
await cli.send("");
|
|
163
|
+
await cli.expect(/Disconnected .*acme\/frontend.* from project .*demo.*\./);
|
|
164
|
+
});
|
|
165
|
+
expect(exitCode).toBe(0);
|
|
166
|
+
expect(output).toMatch(/Disconnected .*acme\/frontend.* from project .*demo.*\./);
|
|
167
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
168
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
169
|
+
expect(call.input).toMatchObject({
|
|
170
|
+
sourceRepository: null,
|
|
171
|
+
});
|
|
172
|
+
});
|
|
173
|
+
});
|
|
174
|
+
it("git connect works with OAuth auth (no API key)", async () => {
|
|
175
|
+
setupProject();
|
|
176
|
+
const oauthEnv = createOAuthEnv(cwd, serverUrl);
|
|
177
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "connect"], {
|
|
178
|
+
cwd,
|
|
179
|
+
env: {
|
|
180
|
+
...oauthEnv,
|
|
181
|
+
},
|
|
182
|
+
}, async (cli) => {
|
|
183
|
+
await cli.expect("Choose the remote source to connect");
|
|
184
|
+
await cli.send("\u001B[B"); // Move down one option (select acme/backend)
|
|
185
|
+
await cli.send("");
|
|
186
|
+
await cli.expect("✅ Connected");
|
|
187
|
+
});
|
|
188
|
+
expect(exitCode).toBe(0);
|
|
189
|
+
expect(output).toMatch(/Connected .*acme\/backend.* to project .*demo.*\./);
|
|
190
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
191
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
192
|
+
expect(call.input).toMatchObject({
|
|
193
|
+
sourceRepository: "acme/backend",
|
|
194
|
+
});
|
|
195
|
+
});
|
|
196
|
+
});
|
|
197
|
+
it("git disconnect works with OAuth auth (no API key)", async () => {
|
|
198
|
+
setupProject({ sourceRepository: "acme/frontend" });
|
|
199
|
+
const oauthEnv = createOAuthEnv(cwd, serverUrl);
|
|
200
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "git", "disconnect"], {
|
|
201
|
+
cwd,
|
|
202
|
+
env: {
|
|
203
|
+
...oauthEnv,
|
|
204
|
+
},
|
|
205
|
+
}, async (cli) => {
|
|
206
|
+
await cli.expect(/Are you sure you want to disconnect .*acme\/frontend.* from project .*demo.*\?/);
|
|
207
|
+
await cli.send("\u001B[A"); // arrow up to select Yes
|
|
208
|
+
await cli.send("");
|
|
209
|
+
await cli.expect(/Disconnected .*acme\/frontend.* from project .*demo.*\./);
|
|
210
|
+
});
|
|
211
|
+
expect(exitCode).toBe(0);
|
|
212
|
+
expect(output).toMatch(/Disconnected .*acme\/frontend.* from project .*demo.*\./);
|
|
213
|
+
const updateCall = mockServer.getLastCall("PATCH", `/v1/projects/${projectId}`);
|
|
214
|
+
expectSuccessfulApiCall(updateCall, (call) => {
|
|
215
|
+
expect(call.input).toMatchObject({
|
|
216
|
+
sourceRepository: null,
|
|
217
|
+
});
|
|
218
|
+
});
|
|
219
|
+
});
|
|
220
|
+
}, 20_000);
|
|
221
|
+
//# sourceMappingURL=git.e2e.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"git.e2e.test.js","sourceRoot":"","sources":["../../src/__tests__/git.e2e.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
export {};
|
|
@@ -0,0 +1,197 @@
|
|
|
1
|
+
import { rmSync } from "node:fs";
|
|
2
|
+
import { afterEach, beforeEach, describe, expect, it } from "vitest";
|
|
3
|
+
import { MockApiServer } from "./mock-server.js";
|
|
4
|
+
import { buildMockProject, cliSession, createConfigFile, expectSuccessfulApiCall, setupTestDirectory, } from "./utils.js";
|
|
5
|
+
const ENV_ID = "env-logs-test";
|
|
6
|
+
const PROJECT_ID = "prj-logs-test";
|
|
7
|
+
const TEAM_ID = "mock-team-id";
|
|
8
|
+
function sleep(ms) {
|
|
9
|
+
return new Promise((resolve) => setTimeout(resolve, ms));
|
|
10
|
+
}
|
|
11
|
+
function makeLogs() {
|
|
12
|
+
const base = new Date("2024-06-01T12:00:00.000Z");
|
|
13
|
+
return [
|
|
14
|
+
{ timestamp: new Date(base.getTime() + 0), type: "INFO", requestId: "req-1", content: "server started" },
|
|
15
|
+
{ timestamp: new Date(base.getTime() + 1), type: "WARNING", requestId: "req-1", content: "cache miss" },
|
|
16
|
+
{ timestamp: new Date(base.getTime() + 2), type: "ERROR", requestId: "req-1", content: "tool failed" },
|
|
17
|
+
{ timestamp: new Date(base.getTime() + 3), type: "DEBUG", requestId: "req-2", content: "trace info" },
|
|
18
|
+
];
|
|
19
|
+
}
|
|
20
|
+
describe("alpic logs (E2E)", () => {
|
|
21
|
+
let mockServer;
|
|
22
|
+
let serverUrl;
|
|
23
|
+
let cwd;
|
|
24
|
+
let binPath;
|
|
25
|
+
beforeEach(async () => {
|
|
26
|
+
mockServer = new MockApiServer();
|
|
27
|
+
serverUrl = await mockServer.start();
|
|
28
|
+
const setup = await setupTestDirectory();
|
|
29
|
+
cwd = setup.cwd;
|
|
30
|
+
binPath = setup.binPath;
|
|
31
|
+
mockServer.addProject(buildMockProject({
|
|
32
|
+
projectId: PROJECT_ID,
|
|
33
|
+
teamId: TEAM_ID,
|
|
34
|
+
projectName: "logs-project",
|
|
35
|
+
environmentId: ENV_ID,
|
|
36
|
+
environmentName: "production",
|
|
37
|
+
}));
|
|
38
|
+
});
|
|
39
|
+
afterEach(async () => {
|
|
40
|
+
await mockServer.stop();
|
|
41
|
+
if (cwd)
|
|
42
|
+
rmSync(cwd, { recursive: true, force: true });
|
|
43
|
+
});
|
|
44
|
+
it("fetches and prints logs using environment-id from config", async () => {
|
|
45
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
46
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
47
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs"], {
|
|
48
|
+
cwd,
|
|
49
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
50
|
+
});
|
|
51
|
+
expect(exitCode).toBe(0);
|
|
52
|
+
expect(output).toContain("server started");
|
|
53
|
+
expect(output).toContain("cache miss");
|
|
54
|
+
expect(output).toContain("tool failed");
|
|
55
|
+
expect(output).toContain("trace info");
|
|
56
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
57
|
+
expectSuccessfulApiCall(call);
|
|
58
|
+
});
|
|
59
|
+
it("fetches logs using --environment-id flag", async () => {
|
|
60
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
61
|
+
const { exitCode } = await cliSession(process.execPath, [binPath, "logs", "--environment-id", ENV_ID], {
|
|
62
|
+
cwd,
|
|
63
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
64
|
+
});
|
|
65
|
+
expect(exitCode).toBe(0);
|
|
66
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
67
|
+
expectSuccessfulApiCall(call);
|
|
68
|
+
});
|
|
69
|
+
it("passes --since to the API", async () => {
|
|
70
|
+
mockServer.setRuntimeLogs(ENV_ID, []);
|
|
71
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
72
|
+
const { exitCode } = await cliSession(process.execPath, [binPath, "logs", "--since", "2h"], {
|
|
73
|
+
cwd,
|
|
74
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
75
|
+
});
|
|
76
|
+
expect(exitCode).toBe(0);
|
|
77
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
78
|
+
expectSuccessfulApiCall(call);
|
|
79
|
+
expect(call.input.since).toBe("2h");
|
|
80
|
+
});
|
|
81
|
+
it("passes --until to the API", async () => {
|
|
82
|
+
mockServer.setRuntimeLogs(ENV_ID, []);
|
|
83
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
84
|
+
const { exitCode } = await cliSession(process.execPath, [binPath, "logs", "--until", "2024-06-01T12:00:00Z"], {
|
|
85
|
+
cwd,
|
|
86
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
87
|
+
});
|
|
88
|
+
expect(exitCode).toBe(0);
|
|
89
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
90
|
+
expectSuccessfulApiCall(call);
|
|
91
|
+
expect(call.input.until).toBe("2024-06-01T12:00:00Z");
|
|
92
|
+
});
|
|
93
|
+
it("passes --search to the API", async () => {
|
|
94
|
+
mockServer.setRuntimeLogs(ENV_ID, []);
|
|
95
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
96
|
+
const { exitCode } = await cliSession(process.execPath, [binPath, "logs", "--search", "timeout"], {
|
|
97
|
+
cwd,
|
|
98
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
99
|
+
});
|
|
100
|
+
expect(exitCode).toBe(0);
|
|
101
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
102
|
+
expectSuccessfulApiCall(call);
|
|
103
|
+
expect(call.input.search).toBe("timeout");
|
|
104
|
+
});
|
|
105
|
+
it("filters by --level ERROR — only error logs are returned", async () => {
|
|
106
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
107
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
108
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--level", "ERROR"], {
|
|
109
|
+
cwd,
|
|
110
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
111
|
+
});
|
|
112
|
+
expect(exitCode).toBe(0);
|
|
113
|
+
expect(output).toContain("tool failed");
|
|
114
|
+
expect(output).not.toContain("server started");
|
|
115
|
+
expect(output).not.toContain("cache miss");
|
|
116
|
+
expect(output).not.toContain("trace info");
|
|
117
|
+
const call = mockServer.getLastCall("GET", /\/v1\/environments\/.*\/logs/);
|
|
118
|
+
expectSuccessfulApiCall(call);
|
|
119
|
+
});
|
|
120
|
+
it("accepts multiple --level values", async () => {
|
|
121
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
122
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
123
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--level", "ERROR", "--level", "WARNING"], { cwd, env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" } });
|
|
124
|
+
expect(exitCode).toBe(0);
|
|
125
|
+
expect(output).toContain("tool failed");
|
|
126
|
+
expect(output).toContain("cache miss");
|
|
127
|
+
expect(output).not.toContain("server started");
|
|
128
|
+
expect(output).not.toContain("trace info");
|
|
129
|
+
});
|
|
130
|
+
it("normalises lowercase --level values", async () => {
|
|
131
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
132
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
133
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--level", "error"], {
|
|
134
|
+
cwd,
|
|
135
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
136
|
+
});
|
|
137
|
+
expect(exitCode).toBe(0);
|
|
138
|
+
expect(output).toContain("tool failed");
|
|
139
|
+
expect(output).not.toContain("server started");
|
|
140
|
+
});
|
|
141
|
+
it("shows log level text with --no-color", async () => {
|
|
142
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
143
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
144
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--no-color"], {
|
|
145
|
+
cwd,
|
|
146
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
147
|
+
});
|
|
148
|
+
expect(exitCode).toBe(0);
|
|
149
|
+
expect(output).toContain("INFO server started");
|
|
150
|
+
expect(output).toContain("WARN cache miss");
|
|
151
|
+
expect(output).toContain("ERROR tool failed");
|
|
152
|
+
expect(output).toContain("DEBUG trace info");
|
|
153
|
+
});
|
|
154
|
+
it("exits with error when --limit is used with --follow", async () => {
|
|
155
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
156
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
157
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--follow", "--limit", "10"], {
|
|
158
|
+
cwd,
|
|
159
|
+
env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" },
|
|
160
|
+
});
|
|
161
|
+
expect(exitCode).toBe(1);
|
|
162
|
+
expect(output).toMatch(/--limit cannot be used with --follow/i);
|
|
163
|
+
});
|
|
164
|
+
it("polls repeatedly with --follow", async () => {
|
|
165
|
+
mockServer.setRuntimeLogs(ENV_ID, makeLogs());
|
|
166
|
+
createConfigFile(cwd, PROJECT_ID, TEAM_ID, "logs-project", ENV_ID, "production");
|
|
167
|
+
const { output } = await cliSession(process.execPath, [binPath, "logs", "--follow"], { cwd, env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" }, timeoutMs: 12_000 }, async (cli) => {
|
|
168
|
+
await cli.expect("server started");
|
|
169
|
+
await sleep(3_300);
|
|
170
|
+
cli.stop();
|
|
171
|
+
});
|
|
172
|
+
// We stop the process manually after verifying follow mode had time to poll.
|
|
173
|
+
expect(output).toContain("server started");
|
|
174
|
+
const calls = mockServer.getCalls("GET", /\/v1\/environments\/.*\/logs/);
|
|
175
|
+
expect(calls.length).toBeGreaterThanOrEqual(2);
|
|
176
|
+
const firstCall = calls[0];
|
|
177
|
+
const secondCall = calls[1];
|
|
178
|
+
expect(firstCall).toBeDefined();
|
|
179
|
+
expect(secondCall).toBeDefined();
|
|
180
|
+
expect((firstCall?.input).since).toBe("10m");
|
|
181
|
+
expect((secondCall?.input).since).toMatch(/^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}\.\d{3}Z$/);
|
|
182
|
+
});
|
|
183
|
+
it("exits with error when environment not found", async () => {
|
|
184
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--environment-id", "nonexistent-env"], { cwd, env: { ALPIC_API_BASE_URL: serverUrl, ALPIC_API_KEY: "test-api-key" } });
|
|
185
|
+
expect(exitCode).toBe(1);
|
|
186
|
+
expect(output).toMatch(/error|not found/i);
|
|
187
|
+
});
|
|
188
|
+
it("exits with error when not authenticated", async () => {
|
|
189
|
+
const { exitCode, output } = await cliSession(process.execPath, [binPath, "logs", "--environment-id", ENV_ID], {
|
|
190
|
+
cwd,
|
|
191
|
+
env: { ALPIC_API_BASE_URL: serverUrl },
|
|
192
|
+
});
|
|
193
|
+
expect(exitCode).toBe(1);
|
|
194
|
+
expect(output).toMatch(/not authenticated|alpic login|ALPIC_API_KEY/i);
|
|
195
|
+
});
|
|
196
|
+
}, 30_000);
|
|
197
|
+
//# sourceMappingURL=logs.e2e.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"logs.e2e.test.js","sourceRoot":"","sources":["../../src/__tests__/logs.e2e.test.ts"],"names":[],"mappings":"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"}
|