alpic 0.0.0-dev.f38a699 → 0.0.0-dev.f52c42d

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 (41) hide show
  1. package/dist/__tests__/deploy.e2e.test.d.ts +1 -0
  2. package/dist/__tests__/deploy.e2e.test.js +185 -0
  3. package/dist/__tests__/deploy.e2e.test.js.map +1 -0
  4. package/dist/__tests__/fixtures/demo-project/index.d.ts +1 -0
  5. package/dist/__tests__/fixtures/demo-project/index.js +4 -0
  6. package/dist/__tests__/fixtures/demo-project/index.js.map +1 -0
  7. package/dist/__tests__/git.e2e.test.d.ts +1 -0
  8. package/dist/__tests__/git.e2e.test.js +182 -0
  9. package/dist/__tests__/git.e2e.test.js.map +1 -0
  10. package/dist/__tests__/mock-server.d.ts +22 -0
  11. package/dist/__tests__/mock-server.js +366 -0
  12. package/dist/__tests__/mock-server.js.map +1 -0
  13. package/dist/__tests__/utils.d.ts +55 -0
  14. package/dist/__tests__/utils.js +174 -0
  15. package/dist/__tests__/utils.js.map +1 -0
  16. package/dist/commands/deploy.d.ts +2 -2
  17. package/dist/commands/deploy.js +4 -4
  18. package/dist/commands/deploy.js.map +1 -1
  19. package/dist/commands/git/connect.d.ts +9 -0
  20. package/dist/commands/git/connect.js +65 -0
  21. package/dist/commands/git/connect.js.map +1 -0
  22. package/dist/commands/git/disconnect.d.ts +9 -0
  23. package/dist/commands/git/disconnect.js +55 -0
  24. package/dist/commands/git/disconnect.js.map +1 -0
  25. package/dist/commands/git.d.ts +6 -0
  26. package/dist/commands/git.js +17 -0
  27. package/dist/commands/git.js.map +1 -0
  28. package/dist/lib/alpic-command.d.ts +4 -0
  29. package/dist/lib/alpic-command.js +20 -0
  30. package/dist/lib/alpic-command.js.map +1 -0
  31. package/dist/lib/auth.d.ts +1 -0
  32. package/dist/lib/auth.js +10 -0
  33. package/dist/lib/auth.js.map +1 -0
  34. package/dist/lib/deployment.js +3 -3
  35. package/dist/lib/deployment.js.map +1 -1
  36. package/dist/lib/git.d.ts +14 -0
  37. package/dist/lib/git.js +106 -0
  38. package/dist/lib/git.js.map +1 -0
  39. package/dist/lib/project.js +6 -2
  40. package/dist/lib/project.js.map +1 -1
  41. package/package.json +9 -12
@@ -0,0 +1,366 @@
1
+ import { OpenAPIHandler } from "@orpc/openapi/node";
2
+ import { ORPCError, implement } from "@orpc/server";
3
+ import express, {} from "express";
4
+ import { randomUUID } from "node:crypto";
5
+ import { contract } from "@alpic-ai/api";
6
+ export class MockApiServer {
7
+ server = null;
8
+ app;
9
+ port = null;
10
+ mockData;
11
+ callHistory = [];
12
+ constructor() {
13
+ this.app = express();
14
+ this.mockData = {
15
+ projects: new Map(),
16
+ environments: new Map(),
17
+ deployments: new Map(),
18
+ };
19
+ }
20
+ async start() {
21
+ if (this.server) {
22
+ throw new Error("Server is already running");
23
+ }
24
+ this.configureServer();
25
+ return new Promise((resolve) => {
26
+ this.server = this.app.listen(0, "127.0.0.1", () => {
27
+ const address = this.server.address();
28
+ this.port = address.port;
29
+ const url = `http://127.0.0.1:${this.port}`;
30
+ resolve(url);
31
+ });
32
+ });
33
+ }
34
+ configureServer() {
35
+ this.app.put("/__mock__upload/:token", express.raw({ type: "*/*", limit: "50mb" }), async (req, res) => {
36
+ const apiCall = {
37
+ method: req.method,
38
+ path: req.path,
39
+ timestamp: new Date(),
40
+ input: { token: req.params.token, size: req.body.length },
41
+ responseStatus: 200,
42
+ };
43
+ this.callHistory.push(apiCall);
44
+ res.status(200).end();
45
+ });
46
+ this.app.use(express.json());
47
+ const router = implement(contract).router({
48
+ projects: {
49
+ list: {
50
+ v1: implement(contract.projects.list.v1).handler(async () => {
51
+ return Array.from(this.mockData.projects.values()).map((project) => ({
52
+ id: project.id,
53
+ name: project.name,
54
+ teamId: project.teamId,
55
+ environments: project.environments,
56
+ productionEnvironment: project.productionEnvironment,
57
+ sourceRepository: project.sourceRepository,
58
+ runtime: project.runtime,
59
+ transport: project.transport,
60
+ rootDirectory: project.rootDirectory,
61
+ buildCommand: project.buildCommand,
62
+ buildOutputDir: project.buildOutputDir,
63
+ installCommand: project.installCommand,
64
+ startCommand: project.startCommand,
65
+ createdAt: project.createdAt,
66
+ }));
67
+ }),
68
+ },
69
+ get: {
70
+ v1: implement(contract.projects.get.v1).handler(async ({ input }) => {
71
+ const project = this.mockData.projects.get(input.projectId);
72
+ if (!project) {
73
+ throw new ORPCError("NOT_FOUND", { message: "Project not found" });
74
+ }
75
+ return project;
76
+ }),
77
+ },
78
+ create: {
79
+ v1: implement(contract.projects.create.v1).handler(async ({ input }) => {
80
+ const projectId = randomUUID();
81
+ const teamId = "mock-team-id";
82
+ const productionEnvId = randomUUID();
83
+ const productionEnvName = "production";
84
+ const project = {
85
+ id: projectId,
86
+ name: input.name,
87
+ teamId,
88
+ environments: [],
89
+ productionEnvironment: {
90
+ id: productionEnvId,
91
+ name: productionEnvName,
92
+ },
93
+ sourceRepository: input.sourceRepository ?? null,
94
+ runtime: input.runtime,
95
+ transport: input.transport ?? null,
96
+ rootDirectory: input.rootDirectory ?? null,
97
+ buildCommand: input.settings?.buildCommand ?? null,
98
+ buildOutputDir: input.settings?.buildOutputDir ?? null,
99
+ installCommand: input.settings?.installCommand ?? null,
100
+ startCommand: input.settings?.startCommand ?? null,
101
+ createdAt: new Date(),
102
+ };
103
+ const environment = {
104
+ id: productionEnvId,
105
+ name: productionEnvName,
106
+ sourceBranch: null,
107
+ mcpServerUrl: `https://mcp-${projectId}.alpic.ai`,
108
+ domains: [`https://mcp-${projectId}.alpic.ai`],
109
+ createdAt: new Date(),
110
+ projectId,
111
+ };
112
+ this.mockData.environments.set(productionEnvId, environment);
113
+ const fullProject = {
114
+ ...project,
115
+ productionEnvironment: {
116
+ id: productionEnvId,
117
+ name: productionEnvName,
118
+ mcpServerUrl: `https://mcp-${projectId}.alpic.ai`,
119
+ latestDeployment: null,
120
+ },
121
+ environments: [
122
+ {
123
+ id: productionEnvId,
124
+ name: productionEnvName,
125
+ sourceBranch: null,
126
+ mcpServerUrl: `https://mcp-${projectId}.alpic.ai`,
127
+ createdAt: new Date(),
128
+ projectId,
129
+ latestDeployment: null,
130
+ },
131
+ ],
132
+ };
133
+ this.mockData.projects.set(projectId, fullProject);
134
+ return project;
135
+ }),
136
+ },
137
+ update: {
138
+ v1: implement(contract.projects.update.v1).handler(async ({ input }) => {
139
+ const project = this.mockData.projects.get(input.projectId);
140
+ if (!project) {
141
+ throw new ORPCError("NOT_FOUND", { message: "Project not found" });
142
+ }
143
+ if (input.sourceRepository !== undefined) {
144
+ project.sourceRepository = input.sourceRepository;
145
+ }
146
+ this.mockData.projects.set(input.projectId, project);
147
+ return project;
148
+ }),
149
+ },
150
+ },
151
+ environments: {
152
+ create: {
153
+ v1: implement(contract.environments.create.v1).handler(async ({ input }) => {
154
+ const project = this.mockData.projects.get(input.projectId);
155
+ if (!project) {
156
+ throw new ORPCError("NOT_FOUND", { message: "Project not found" });
157
+ }
158
+ const environmentId = randomUUID();
159
+ const mcpServerUrl = `https://mcp-${environmentId}.alpic.ai`;
160
+ const createdAt = new Date();
161
+ const environment = {
162
+ id: environmentId,
163
+ name: input.name,
164
+ sourceBranch: input.sourceBranch,
165
+ mcpServerUrl,
166
+ domains: [mcpServerUrl],
167
+ createdAt,
168
+ projectId: input.projectId,
169
+ };
170
+ this.mockData.environments.set(environmentId, environment);
171
+ this.mockData.projects.set(input.projectId, {
172
+ ...project,
173
+ environments: [
174
+ ...project.environments,
175
+ {
176
+ id: environmentId,
177
+ name: input.name,
178
+ sourceBranch: input.sourceBranch,
179
+ mcpServerUrl,
180
+ createdAt,
181
+ projectId: input.projectId,
182
+ latestDeployment: null,
183
+ },
184
+ ],
185
+ });
186
+ return {
187
+ id: environmentId,
188
+ name: input.name,
189
+ sourceBranch: input.sourceBranch,
190
+ createdAt,
191
+ projectId: input.projectId,
192
+ urls: [mcpServerUrl],
193
+ };
194
+ }),
195
+ },
196
+ get: {
197
+ v1: implement(contract.environments.get.v1).handler(async ({ input }) => {
198
+ const environment = this.mockData.environments.get(input.environmentId);
199
+ if (!environment) {
200
+ throw new ORPCError("NOT_FOUND", { message: "Environment not found" });
201
+ }
202
+ return environment;
203
+ }),
204
+ },
205
+ deploy: {
206
+ v1: implement(contract.environments.deploy.v1).handler(async ({ input }) => {
207
+ const environment = this.mockData.environments.get(input.environmentId);
208
+ if (!environment) {
209
+ throw new ORPCError("NOT_FOUND", { message: "Environment not found" });
210
+ }
211
+ const deploymentId = randomUUID();
212
+ const deployment = {
213
+ id: deploymentId,
214
+ status: "ongoing",
215
+ sourceRef: environment.sourceBranch,
216
+ sourceCommitId: randomUUID().slice(0, 7),
217
+ sourceCommitMessage: "Mock deployment",
218
+ authorUsername: "test-user",
219
+ authorAvatarUrl: null,
220
+ startedAt: new Date(),
221
+ completedAt: null,
222
+ };
223
+ this.mockData.deployments.set(deploymentId, deployment);
224
+ return deployment;
225
+ }),
226
+ },
227
+ },
228
+ deployments: {
229
+ get: {
230
+ v1: implement(contract.deployments.get.v1).handler(async ({ input }) => {
231
+ const deployment = this.mockData.deployments.get(input.deploymentId);
232
+ if (!deployment) {
233
+ throw new ORPCError("NOT_FOUND", { message: "Deployment not found" });
234
+ }
235
+ if (deployment.status === "ongoing") {
236
+ const updatedDeployment = {
237
+ ...deployment,
238
+ status: "deployed",
239
+ completedAt: new Date(),
240
+ };
241
+ this.mockData.deployments.set(input.deploymentId, updatedDeployment);
242
+ return updatedDeployment;
243
+ }
244
+ return deployment;
245
+ }),
246
+ },
247
+ uploadArtifact: {
248
+ v1: implement(contract.deployments.uploadArtifact.v1).handler(async () => {
249
+ const token = randomUUID();
250
+ const baseUrl = `http://127.0.0.1:${this.port}`;
251
+ const uploadUrl = `${baseUrl}/__mock__upload/${token}`;
252
+ return {
253
+ uploadUrl,
254
+ token,
255
+ expiresAt: new Date(Date.now() + 3600000),
256
+ };
257
+ }),
258
+ },
259
+ },
260
+ });
261
+ const handler = new OpenAPIHandler(router);
262
+ this.app.use(async (req, res) => {
263
+ const apiCall = {
264
+ method: req.method,
265
+ path: req.path,
266
+ timestamp: new Date(),
267
+ };
268
+ if (req.body) {
269
+ apiCall.input = req.body;
270
+ }
271
+ const result = await handler.handle(req, res, {
272
+ context: {},
273
+ });
274
+ if (!result.matched) {
275
+ apiCall.responseStatus = 404;
276
+ this.callHistory.push(apiCall);
277
+ res.status(404).json({ error: "Not found" });
278
+ return;
279
+ }
280
+ apiCall.responseStatus = res.statusCode || 200;
281
+ this.callHistory.push(apiCall);
282
+ });
283
+ }
284
+ async stop() {
285
+ return new Promise((resolve, reject) => {
286
+ if (!this.server) {
287
+ resolve();
288
+ return;
289
+ }
290
+ this.server.close((error) => {
291
+ if (error) {
292
+ console.error("[MOCK] Error stopping server:", error);
293
+ reject(error);
294
+ }
295
+ else {
296
+ this.server = null;
297
+ resolve();
298
+ }
299
+ });
300
+ });
301
+ }
302
+ getCalls(method, pathPattern) {
303
+ return this.callHistory.filter((call) => {
304
+ if (method && call.method !== method)
305
+ return false;
306
+ if (pathPattern) {
307
+ if (typeof pathPattern === "string") {
308
+ if (!call.path.includes(pathPattern))
309
+ return false;
310
+ }
311
+ else {
312
+ if (!pathPattern.test(call.path))
313
+ return false;
314
+ }
315
+ }
316
+ return true;
317
+ });
318
+ }
319
+ getLastCall(method, pathPattern) {
320
+ for (let i = this.callHistory.length - 1; i >= 0; i--) {
321
+ const call = this.callHistory[i];
322
+ if (method && call.method !== method)
323
+ continue;
324
+ if (pathPattern) {
325
+ if (typeof pathPattern === "string") {
326
+ if (!call.path.includes(pathPattern))
327
+ continue;
328
+ }
329
+ else {
330
+ if (!pathPattern.test(call.path))
331
+ continue;
332
+ }
333
+ }
334
+ return call;
335
+ }
336
+ return undefined;
337
+ }
338
+ addProject(project) {
339
+ this.mockData.projects.set(project.id, project);
340
+ if (project.productionEnvironment) {
341
+ const env = {
342
+ id: project.productionEnvironment.id,
343
+ name: project.productionEnvironment.name,
344
+ sourceBranch: null,
345
+ mcpServerUrl: project.productionEnvironment.mcpServerUrl,
346
+ domains: [project.productionEnvironment.mcpServerUrl],
347
+ createdAt: new Date(),
348
+ projectId: project.id,
349
+ };
350
+ this.mockData.environments.set(env.id, env);
351
+ }
352
+ for (const env of project.environments) {
353
+ const fullEnv = {
354
+ id: env.id,
355
+ name: env.name,
356
+ sourceBranch: env.sourceBranch,
357
+ mcpServerUrl: env.mcpServerUrl,
358
+ domains: [env.mcpServerUrl],
359
+ createdAt: env.createdAt,
360
+ projectId: project.id,
361
+ };
362
+ this.mockData.environments.set(env.id, fullEnv);
363
+ }
364
+ }
365
+ }
366
+ //# sourceMappingURL=mock-server.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"mock-server.js","sourceRoot":"","sources":["../../src/__tests__/mock-server.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,55 @@
1
+ import type { RouterOutput } from "@alpic-ai/api";
2
+ import { type ApiCall } from "./mock-server";
3
+ type Expectable = string | RegExp;
4
+ type ProjectGetV1 = RouterOutput["projects"]["get"]["v1"];
5
+ export type CliSessionOptions = {
6
+ cwd?: string;
7
+ env?: Record<string, string | undefined>;
8
+ cols?: number;
9
+ rows?: number;
10
+ timeoutMs?: number;
11
+ };
12
+ export type CliController = {
13
+ /** Wait until stdout/stderr contains a string or matches a regex */
14
+ expect: (pattern: Expectable, opts?: {
15
+ timeoutMs?: number;
16
+ }) => Promise<void>;
17
+ /** Send a line (adds newline) */
18
+ send: (line: string) => Promise<void>;
19
+ /** Send raw data (no newline) */
20
+ write: (data: string) => void;
21
+ /** Current accumulated output */
22
+ output: () => string;
23
+ /** Clear accumulated output (handy between phases) */
24
+ clear: () => void;
25
+ /** End session (SIGTERM) */
26
+ stop: () => void;
27
+ };
28
+ export declare function cliSession(cmd: string, args: string[], options: CliSessionOptions, fn: (cli: CliController) => Promise<void>): Promise<{
29
+ exitCode: number | null;
30
+ output: string;
31
+ }>;
32
+ export type BuildMockProjectParams = {
33
+ projectId: string;
34
+ teamId: string;
35
+ projectName: string;
36
+ environmentId: string;
37
+ environmentName: string;
38
+ sourceRepository?: string | null;
39
+ sourceBranch?: string | null;
40
+ };
41
+ export declare function buildMockProject(params: BuildMockProjectParams): ProjectGetV1;
42
+ export type SetupTestDirectoryOptions = {
43
+ /** Temp directory prefix (default: "alpic-cli-e2e") */
44
+ prefix?: string;
45
+ /** If true, run git init and add origin/upstream remotes */
46
+ initGit?: boolean;
47
+ };
48
+ export declare function setupTestDirectory(options?: SetupTestDirectoryOptions): Promise<{
49
+ cwd: string;
50
+ binPath: string;
51
+ }>;
52
+ export declare function createConfigFile(cwd: string, projectId: string, teamId: string, projectName: string, environmentId: string, environmentName: string): void;
53
+ export declare function runGit(args: string[], cwd: string): void;
54
+ export declare function expectSuccessfulApiCall(call: ApiCall | undefined, customAssertions?: (call: ApiCall) => void): asserts call is ApiCall;
55
+ export {};
@@ -0,0 +1,174 @@
1
+ import pty from "node-pty";
2
+ import { execFileSync } from "node:child_process";
3
+ import { randomUUID } from "node:crypto";
4
+ import { mkdirSync, writeFileSync } from "node:fs";
5
+ import { cp, mkdtemp } from "node:fs/promises";
6
+ import { tmpdir } from "node:os";
7
+ import { dirname, join } from "node:path";
8
+ import { fileURLToPath } from "node:url";
9
+ import { expect } from "vitest";
10
+ import {} from "./mock-server";
11
+ const __testDirname = dirname(fileURLToPath(import.meta.url));
12
+ export async function cliSession(cmd, args, options, fn) {
13
+ const { cwd, env, cols = 100, rows = 30, timeoutMs = 8000 } = options;
14
+ const mergedEnv = {
15
+ ...process.env,
16
+ NO_COLOR: "1",
17
+ // Resolve @alpic-ai/api to TypeScript source via the "development" export condition,
18
+ // so e2e tests run without building the api package first.
19
+ NODE_OPTIONS: [process.env.NODE_OPTIONS, "--conditions=development"].filter(Boolean).join(" "),
20
+ ...Object.fromEntries(Object.entries(env ?? {}).filter(([, v]) => v !== undefined)),
21
+ };
22
+ for (const [k, v] of Object.entries(mergedEnv)) {
23
+ if (typeof v !== "string") {
24
+ console.error("[pty] bad env entry:", k, v, typeof v);
25
+ throw new Error(`env value for ${k} is not a string`);
26
+ }
27
+ }
28
+ const term = pty.spawn(cmd, args, {
29
+ cwd,
30
+ env: mergedEnv,
31
+ name: "xterm-256color",
32
+ cols,
33
+ rows,
34
+ });
35
+ let buf = "";
36
+ const append = (data) => {
37
+ buf += data;
38
+ };
39
+ term.onData(append);
40
+ let exitCode = null;
41
+ const exitPromise = new Promise((resolve) => {
42
+ term.onExit((e) => {
43
+ exitCode = e.exitCode ?? null;
44
+ resolve();
45
+ });
46
+ });
47
+ const controller = {
48
+ expect: (pattern, opts) => waitFor(() => matches(buf, pattern), opts?.timeoutMs ?? timeoutMs, () => buf, pattern),
49
+ send: async (line) => {
50
+ term.write(line + "\r");
51
+ },
52
+ write: (data) => term.write(data),
53
+ output: () => buf,
54
+ clear: () => {
55
+ buf = "";
56
+ },
57
+ stop: () => {
58
+ try {
59
+ term.kill();
60
+ }
61
+ catch {
62
+ // ignore
63
+ }
64
+ },
65
+ };
66
+ try {
67
+ await fn(controller);
68
+ }
69
+ finally {
70
+ await Promise.race([exitPromise, sleep(1500)]);
71
+ controller.stop();
72
+ await Promise.race([exitPromise, sleep(200)]);
73
+ }
74
+ return { exitCode, output: buf };
75
+ }
76
+ function matches(text, pattern) {
77
+ if (typeof pattern === "string")
78
+ return text.includes(pattern);
79
+ return pattern.test(text);
80
+ }
81
+ async function waitFor(predicate, timeoutMs, getOutput, pattern) {
82
+ const start = Date.now();
83
+ while (true) {
84
+ if (predicate())
85
+ return;
86
+ if (Date.now() - start > timeoutMs) {
87
+ const out = getOutput();
88
+ throw new Error(`Timed out after ${timeoutMs}ms waiting for: ${patternToString(pattern)}\n\n` +
89
+ `--- current output ---\n${out}\n--- end output ---\n`);
90
+ }
91
+ await sleep(20);
92
+ }
93
+ }
94
+ function patternToString(p) {
95
+ return typeof p === "string" ? JSON.stringify(p) : `/${p.source}/${p.flags}`;
96
+ }
97
+ function sleep(ms) {
98
+ return new Promise((r) => setTimeout(r, ms));
99
+ }
100
+ export function buildMockProject(params) {
101
+ const { projectId, teamId, projectName, environmentId, environmentName, sourceRepository = null, sourceBranch = null, } = params;
102
+ const mcpServerUrl = `https://mcp-${projectId}.alpic.ai`;
103
+ return {
104
+ id: projectId,
105
+ name: projectName,
106
+ teamId,
107
+ environments: [
108
+ {
109
+ id: environmentId,
110
+ name: environmentName,
111
+ sourceBranch,
112
+ mcpServerUrl,
113
+ createdAt: new Date(),
114
+ projectId,
115
+ latestDeployment: null,
116
+ },
117
+ ],
118
+ productionEnvironment: {
119
+ id: environmentId,
120
+ name: environmentName,
121
+ mcpServerUrl,
122
+ latestDeployment: null,
123
+ },
124
+ sourceRepository,
125
+ runtime: "node24",
126
+ transport: null,
127
+ rootDirectory: null,
128
+ buildCommand: null,
129
+ buildOutputDir: null,
130
+ installCommand: null,
131
+ startCommand: null,
132
+ createdAt: new Date(),
133
+ };
134
+ }
135
+ export async function setupTestDirectory(options = {}) {
136
+ const { prefix = "alpic-cli-e2e", initGit = false } = options;
137
+ const uniqueId = randomUUID();
138
+ const cwd = await mkdtemp(join(tmpdir(), `${prefix}-${uniqueId}-`));
139
+ const fixturePath = join(__testDirname, "fixtures", "demo-project");
140
+ await cp(fixturePath, cwd, { recursive: true });
141
+ if (initGit) {
142
+ runGit(["init"], cwd);
143
+ runGit(["remote", "add", "origin", "git@github.com:acme/frontend.git"], cwd);
144
+ runGit(["remote", "add", "upstream", "https://github.com/acme/backend.git"], cwd);
145
+ }
146
+ const binPath = join(process.cwd(), "bin", "dev.js");
147
+ return { cwd, binPath };
148
+ }
149
+ export function createConfigFile(cwd, projectId, teamId, projectName, environmentId, environmentName) {
150
+ const configDir = join(cwd, ".alpic");
151
+ mkdirSync(configDir, { recursive: true });
152
+ const configPath = join(configDir, "project.json");
153
+ writeFileSync(configPath, JSON.stringify({
154
+ projectId,
155
+ teamId,
156
+ projectName,
157
+ environmentId,
158
+ environmentName,
159
+ }));
160
+ }
161
+ export function runGit(args, cwd) {
162
+ execFileSync("git", args, {
163
+ cwd,
164
+ stdio: ["ignore", "pipe", "pipe"],
165
+ });
166
+ }
167
+ export function expectSuccessfulApiCall(call, customAssertions) {
168
+ expect(call).toBeDefined();
169
+ expect(call?.responseStatus).toBe(200);
170
+ if (customAssertions && call) {
171
+ customAssertions(call);
172
+ }
173
+ }
174
+ //# sourceMappingURL=utils.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"utils.js","sourceRoot":"","sources":["../../src/__tests__/utils.ts"],"names":[],"mappings":"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"}
@@ -1,5 +1,5 @@
1
- import { Command } from "@oclif/core";
2
- export declare class Deploy extends Command {
1
+ import { AlpicCommand } from "../lib/alpic-command.js";
2
+ export declare class Deploy extends AlpicCommand {
3
3
  static description: string;
4
4
  static examples: string[];
5
5
  static args: {
@@ -3,11 +3,13 @@ import { Args, Command } from "@oclif/core";
3
3
  import chalk from "chalk";
4
4
  import { readFileSync, rmSync } from "node:fs";
5
5
  import { api } from "../api.js";
6
+ import { AlpicCommand } from "../lib/alpic-command.js";
6
7
  import { createTarArchive, getFilesToPack } from "../lib/archive.js";
8
+ import { ensureApiKey } from "../lib/auth.js";
7
9
  import { deployAndWait, formatElapsed } from "../lib/deployment.js";
8
10
  import { resolveDeployDir, resolveProjectForDeploy } from "../lib/project.js";
9
11
  import { uploadToPresignedUrl } from "../lib/upload.js";
10
- export class Deploy extends Command {
12
+ export class Deploy extends AlpicCommand {
11
13
  static description = "Deploy a project to Alpic";
12
14
  static examples = ["<%= config.bin %> deploy", "<%= config.bin %> deploy ./my-app"];
13
15
  static args = {
@@ -19,9 +21,7 @@ export class Deploy extends Command {
19
21
  async run() {
20
22
  const { args } = await this.parse(Deploy);
21
23
  p.intro("Deploying to Alpic");
22
- const apiKey = process.env.ALPIC_API_KEY;
23
- if (!apiKey) {
24
- p.cancel("ALPIC_API_KEY environment variable is required. Get your API key from Team settings in the Alpic dashboard.");
24
+ if (!ensureApiKey()) {
25
25
  this.exit(1);
26
26
  return;
27
27
  }