@lizard-build/cli 0.1.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (115) hide show
  1. package/dist/commands/add.d.ts +2 -0
  2. package/dist/commands/add.js +72 -0
  3. package/dist/commands/add.js.map +1 -0
  4. package/dist/commands/connect.d.ts +2 -0
  5. package/dist/commands/connect.js +117 -0
  6. package/dist/commands/connect.js.map +1 -0
  7. package/dist/commands/context.d.ts +2 -0
  8. package/dist/commands/context.js +71 -0
  9. package/dist/commands/context.js.map +1 -0
  10. package/dist/commands/deploy.d.ts +2 -0
  11. package/dist/commands/deploy.js +120 -0
  12. package/dist/commands/deploy.js.map +1 -0
  13. package/dist/commands/destroy.d.ts +2 -0
  14. package/dist/commands/destroy.js +51 -0
  15. package/dist/commands/destroy.js.map +1 -0
  16. package/dist/commands/git.d.ts +2 -0
  17. package/dist/commands/git.js +67 -0
  18. package/dist/commands/git.js.map +1 -0
  19. package/dist/commands/init.d.ts +2 -0
  20. package/dist/commands/init.js +107 -0
  21. package/dist/commands/init.js.map +1 -0
  22. package/dist/commands/link.d.ts +2 -0
  23. package/dist/commands/link.js +50 -0
  24. package/dist/commands/link.js.map +1 -0
  25. package/dist/commands/login.d.ts +7 -0
  26. package/dist/commands/login.js +123 -0
  27. package/dist/commands/login.js.map +1 -0
  28. package/dist/commands/logout.d.ts +2 -0
  29. package/dist/commands/logout.js +17 -0
  30. package/dist/commands/logout.js.map +1 -0
  31. package/dist/commands/logs.d.ts +2 -0
  32. package/dist/commands/logs.js +92 -0
  33. package/dist/commands/logs.js.map +1 -0
  34. package/dist/commands/open.d.ts +2 -0
  35. package/dist/commands/open.js +16 -0
  36. package/dist/commands/open.js.map +1 -0
  37. package/dist/commands/projects.d.ts +2 -0
  38. package/dist/commands/projects.js +28 -0
  39. package/dist/commands/projects.js.map +1 -0
  40. package/dist/commands/ps.d.ts +2 -0
  41. package/dist/commands/ps.js +52 -0
  42. package/dist/commands/ps.js.map +1 -0
  43. package/dist/commands/redeploy.d.ts +2 -0
  44. package/dist/commands/redeploy.js +69 -0
  45. package/dist/commands/redeploy.js.map +1 -0
  46. package/dist/commands/regions.d.ts +2 -0
  47. package/dist/commands/regions.js +23 -0
  48. package/dist/commands/regions.js.map +1 -0
  49. package/dist/commands/restart.d.ts +2 -0
  50. package/dist/commands/restart.js +21 -0
  51. package/dist/commands/restart.js.map +1 -0
  52. package/dist/commands/run.d.ts +2 -0
  53. package/dist/commands/run.js +33 -0
  54. package/dist/commands/run.js.map +1 -0
  55. package/dist/commands/secrets.d.ts +2 -0
  56. package/dist/commands/secrets.js +138 -0
  57. package/dist/commands/secrets.js.map +1 -0
  58. package/dist/commands/status.d.ts +2 -0
  59. package/dist/commands/status.js +51 -0
  60. package/dist/commands/status.js.map +1 -0
  61. package/dist/commands/update.d.ts +2 -0
  62. package/dist/commands/update.js +41 -0
  63. package/dist/commands/update.js.map +1 -0
  64. package/dist/commands/version.d.ts +2 -0
  65. package/dist/commands/version.js +37 -0
  66. package/dist/commands/version.js.map +1 -0
  67. package/dist/commands/whoami.d.ts +2 -0
  68. package/dist/commands/whoami.js +21 -0
  69. package/dist/commands/whoami.js.map +1 -0
  70. package/dist/index.d.ts +2 -0
  71. package/dist/index.js +151 -0
  72. package/dist/index.js.map +1 -0
  73. package/dist/lib/api.d.ts +19 -0
  74. package/dist/lib/api.js +114 -0
  75. package/dist/lib/api.js.map +1 -0
  76. package/dist/lib/auth.d.ts +23 -0
  77. package/dist/lib/auth.js +89 -0
  78. package/dist/lib/auth.js.map +1 -0
  79. package/dist/lib/config.d.ts +23 -0
  80. package/dist/lib/config.js +77 -0
  81. package/dist/lib/config.js.map +1 -0
  82. package/dist/lib/format.d.ts +11 -0
  83. package/dist/lib/format.js +86 -0
  84. package/dist/lib/format.js.map +1 -0
  85. package/install.sh +59 -0
  86. package/package.json +35 -0
  87. package/src/commands/add.ts +100 -0
  88. package/src/commands/connect.ts +145 -0
  89. package/src/commands/context.ts +93 -0
  90. package/src/commands/deploy.ts +153 -0
  91. package/src/commands/destroy.ts +51 -0
  92. package/src/commands/git.ts +80 -0
  93. package/src/commands/init.ts +139 -0
  94. package/src/commands/link.ts +63 -0
  95. package/src/commands/login.ts +158 -0
  96. package/src/commands/logout.ts +17 -0
  97. package/src/commands/logs.ts +104 -0
  98. package/src/commands/open.ts +17 -0
  99. package/src/commands/projects.ts +45 -0
  100. package/src/commands/ps.ts +80 -0
  101. package/src/commands/redeploy.ts +74 -0
  102. package/src/commands/regions.ts +38 -0
  103. package/src/commands/restart.ts +24 -0
  104. package/src/commands/run.ts +43 -0
  105. package/src/commands/secrets.ts +175 -0
  106. package/src/commands/status.ts +65 -0
  107. package/src/commands/update.ts +44 -0
  108. package/src/commands/version.ts +37 -0
  109. package/src/commands/whoami.ts +27 -0
  110. package/src/index.ts +168 -0
  111. package/src/lib/api.ts +134 -0
  112. package/src/lib/auth.ts +113 -0
  113. package/src/lib/config.ts +93 -0
  114. package/src/lib/format.ts +95 -0
  115. package/tsconfig.json +17 -0
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerAdd(program: Command): void;
@@ -0,0 +1,72 @@
1
+ import chalk from "chalk";
2
+ import * as p from "@clack/prompts";
3
+ import { api } from "../lib/api.js";
4
+ import { resolveProjectId } from "../lib/config.js";
5
+ import { success, info, isJSONMode, printJSON, isTTY, table, } from "../lib/format.js";
6
+ const CATALOG = [
7
+ { name: "postgres", label: "PostgreSQL", description: "Relational database" },
8
+ { name: "redis", label: "Redis", description: "In-memory key-value store" },
9
+ { name: "mysql", label: "MySQL", description: "Relational database" },
10
+ { name: "mongodb", label: "MongoDB", description: "Document database" },
11
+ ];
12
+ export function registerAdd(program) {
13
+ program
14
+ .command("add")
15
+ .argument("[name]", "Service name from catalog (postgres, redis, mysql, mongodb)")
16
+ .description("Add a service to the project")
17
+ .option("--list", "Show available services")
18
+ .option("--region <region>", "Region for the service")
19
+ .action(async (name, opts) => {
20
+ // Show catalog
21
+ if (opts.list || (!name && !isTTY())) {
22
+ if (isJSONMode()) {
23
+ printJSON(CATALOG);
24
+ }
25
+ else {
26
+ table(["Name", "Description"], CATALOG.map((c) => [c.name, c.description]));
27
+ }
28
+ return;
29
+ }
30
+ // Interactive selection
31
+ if (!name) {
32
+ const selected = await p.select({
33
+ message: "Select service to add",
34
+ options: CATALOG.map((c) => ({
35
+ value: c.name,
36
+ label: c.label,
37
+ hint: c.description,
38
+ })),
39
+ });
40
+ if (p.isCancel(selected))
41
+ process.exit(5);
42
+ name = selected;
43
+ }
44
+ // Validate name is in catalog
45
+ const catalogEntry = CATALOG.find((c) => c.name === name);
46
+ if (!catalogEntry) {
47
+ throw new Error(`Unknown service "${name}". Available: ${CATALOG.map((c) => c.name).join(", ")}`);
48
+ }
49
+ const projectId = resolveProjectId(program.opts().project);
50
+ const region = opts.region || program.opts().region;
51
+ info(`Adding ${chalk.cyan(catalogEntry.label)}...`);
52
+ const addon = await api.post(`/api/projects/${projectId}/addons`, {
53
+ addonType: name,
54
+ region,
55
+ });
56
+ if (isJSONMode()) {
57
+ printJSON(addon);
58
+ return;
59
+ }
60
+ success(`${catalogEntry.label} added`);
61
+ if (addon.hostname) {
62
+ info(` Host: ${chalk.cyan(addon.hostname)}`);
63
+ }
64
+ if (addon.envVars) {
65
+ info(chalk.dim("\n Environment variables added to project:"));
66
+ for (const [key, val] of Object.entries(addon.envVars)) {
67
+ info(` ${chalk.bold(key)}=${chalk.dim(val)}`);
68
+ }
69
+ }
70
+ });
71
+ }
72
+ //# sourceMappingURL=add.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"add.js","sourceRoot":"","sources":["../../src/commands/add.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerConnect(program: Command): void;
@@ -0,0 +1,117 @@
1
+ import { execSync } from "node:child_process";
2
+ import chalk from "chalk";
3
+ import * as p from "@clack/prompts";
4
+ import { api } from "../lib/api.js";
5
+ import { resolveProjectId } from "../lib/config.js";
6
+ import { info, isJSONMode, printJSON, isTTY } from "../lib/format.js";
7
+ const CLIENT_COMMANDS = {
8
+ postgres: "psql",
9
+ mysql: "mysql",
10
+ mongodb: "mongosh",
11
+ redis: "redis-cli",
12
+ };
13
+ export function registerConnect(program) {
14
+ program
15
+ .command("connect")
16
+ .argument("[service]", "Service type or ID (postgres, redis, etc.)")
17
+ .description("Connect to a managed service")
18
+ .option("--url", "Print connection string without connecting")
19
+ .action(async (service, opts) => {
20
+ const projectId = resolveProjectId(program.opts().project);
21
+ // Get addons
22
+ const addons = await api.get(`/api/projects/${projectId}/addons`);
23
+ if (addons.length === 0) {
24
+ throw new Error("No managed services in this project. Use `lizard add`.");
25
+ }
26
+ let addon;
27
+ if (service) {
28
+ // Match by type or ID
29
+ addon =
30
+ addons.find((a) => a.addonType === service) ||
31
+ addons.find((a) => a.id === service) ||
32
+ addons.find((a) => a.name === service);
33
+ }
34
+ else if (addons.length === 1) {
35
+ addon = addons[0];
36
+ }
37
+ else {
38
+ if (!isTTY()) {
39
+ throw new Error("Multiple services found. Specify one: " +
40
+ addons.map((a) => a.addonType || a.name).join(", "));
41
+ }
42
+ const selected = await p.select({
43
+ message: "Select service to connect to",
44
+ options: addons.map((a) => ({
45
+ value: a.id,
46
+ label: a.name || a.addonType,
47
+ hint: a.addonType,
48
+ })),
49
+ });
50
+ if (p.isCancel(selected))
51
+ process.exit(5);
52
+ addon = addons.find((a) => a.id === selected);
53
+ }
54
+ if (!addon) {
55
+ throw new Error(`Service "${service}" not found`);
56
+ }
57
+ if (addon.status !== "running") {
58
+ throw new Error(`Service is ${addon.status}, not running`);
59
+ }
60
+ // Build connection string from secrets
61
+ const secrets = await api.get(`/api/projects/${projectId}/secrets`);
62
+ const connString = findConnectionString(addon.addonType, secrets);
63
+ if (opts.url || isJSONMode()) {
64
+ if (isJSONMode()) {
65
+ printJSON({ type: addon.addonType, connectionString: connString });
66
+ }
67
+ else {
68
+ console.log(connString || "Connection string not found in secrets");
69
+ }
70
+ return;
71
+ }
72
+ if (!connString) {
73
+ throw new Error("Connection string not found in project secrets. Check `lizard secret list --show`.");
74
+ }
75
+ // Connect using native client
76
+ const clientCmd = CLIENT_COMMANDS[addon.addonType];
77
+ if (!clientCmd) {
78
+ info(`Connection string: ${connString}`);
79
+ return;
80
+ }
81
+ info(chalk.dim(`Connecting via ${clientCmd}...\n`));
82
+ try {
83
+ if (addon.addonType === "postgres") {
84
+ execSync(`${clientCmd} "${connString}"`, { stdio: "inherit" });
85
+ }
86
+ else if (addon.addonType === "redis") {
87
+ // redis-cli -u redis://...
88
+ execSync(`${clientCmd} -u "${connString}"`, { stdio: "inherit" });
89
+ }
90
+ else if (addon.addonType === "mysql") {
91
+ execSync(`${clientCmd} "${connString}"`, { stdio: "inherit" });
92
+ }
93
+ else if (addon.addonType === "mongodb") {
94
+ execSync(`${clientCmd} "${connString}"`, { stdio: "inherit" });
95
+ }
96
+ }
97
+ catch (err) {
98
+ process.exit(err.status || 1);
99
+ }
100
+ });
101
+ }
102
+ function findConnectionString(type, secrets) {
103
+ const envKeys = {
104
+ postgres: ["DATABASE_URL", "POSTGRES_URL", "PG_URL"],
105
+ mysql: ["MYSQL_URL", "DATABASE_URL"],
106
+ mongodb: ["MONGODB_URL", "MONGO_URL"],
107
+ redis: ["REDIS_URL"],
108
+ };
109
+ const keys = envKeys[type] || [];
110
+ for (const key of keys) {
111
+ const s = secrets.find((s) => s.key === key);
112
+ if (s)
113
+ return s.value;
114
+ }
115
+ return null;
116
+ }
117
+ //# sourceMappingURL=connect.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"connect.js","sourceRoot":"","sources":["../../src/commands/connect.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerContext(program: Command): void;
@@ -0,0 +1,71 @@
1
+ import chalk from "chalk";
2
+ import { api } from "../lib/api.js";
3
+ import { resolveProjectId, findProjectConfig } from "../lib/config.js";
4
+ import { isJSONMode, printJSON, statusColor, table } from "../lib/format.js";
5
+ export function registerContext(program) {
6
+ program
7
+ .command("context")
8
+ .description("Show full project context (optimized for AI agents)")
9
+ .action(async () => {
10
+ const projectId = resolveProjectId(program.opts().project);
11
+ const config = findProjectConfig();
12
+ const [project, services, secrets] = await Promise.all([
13
+ api.get(`/api/projects/${projectId}`),
14
+ api.get(`/api/projects/${projectId}/services`),
15
+ api.get(`/api/projects/${projectId}/secrets`).catch(() => []),
16
+ ]);
17
+ const context = {
18
+ project: {
19
+ id: project.id,
20
+ name: project.name,
21
+ slug: project.slug,
22
+ },
23
+ environment: config?.environment || "production",
24
+ apps: (services.apps || []).map((a) => ({
25
+ id: a.id,
26
+ name: a.name,
27
+ status: a.status,
28
+ domain: a.domain,
29
+ repo: a.repo,
30
+ branch: a.branch,
31
+ cpuLimit: a.cpuLimit,
32
+ memoryLimit: a.memoryLimit,
33
+ })),
34
+ addons: (services.addons || []).map((a) => ({
35
+ id: a.id,
36
+ name: a.name,
37
+ type: a.addonType,
38
+ status: a.status,
39
+ hostname: a.hostname,
40
+ })),
41
+ secrets: secrets.map((s) => s.key), // names only, not values
42
+ };
43
+ // Always JSON for pipe, since this is designed for AI agents
44
+ if (isJSONMode() || !process.stdout.isTTY) {
45
+ printJSON(context);
46
+ return;
47
+ }
48
+ // Human-readable
49
+ console.log(chalk.bold(context.project.name) + chalk.dim(` (${context.project.id})`));
50
+ console.log(chalk.dim(`Environment: ${context.environment}`));
51
+ console.log();
52
+ if (context.apps.length > 0) {
53
+ console.log(chalk.bold("Apps:"));
54
+ table(["Name", "Status", "Domain"], context.apps.map((a) => [
55
+ a.name,
56
+ statusColor(a.status),
57
+ a.domain ? `https://${a.domain}` : "—",
58
+ ]));
59
+ console.log();
60
+ }
61
+ if (context.addons.length > 0) {
62
+ console.log(chalk.bold("Addons:"));
63
+ table(["Name", "Type", "Status", "Host"], context.addons.map((a) => [a.name, a.type, statusColor(a.status), a.hostname || "—"]));
64
+ console.log();
65
+ }
66
+ if (context.secrets.length > 0) {
67
+ console.log(chalk.bold("Secrets:") + " " + context.secrets.join(", "));
68
+ }
69
+ });
70
+ }
71
+ //# sourceMappingURL=context.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"context.js","sourceRoot":"","sources":["../../src/commands/context.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerDeploy(program: Command): void;
@@ -0,0 +1,120 @@
1
+ import chalk from "chalk";
2
+ import ora from "ora";
3
+ import { api, streamSSE } from "../lib/api.js";
4
+ import { resolveProjectId } from "../lib/config.js";
5
+ import { success, info, error, isJSONMode, printJSON, statusColor, } from "../lib/format.js";
6
+ export function registerDeploy(program) {
7
+ program
8
+ .command("deploy")
9
+ .description("Deploy the current project")
10
+ .option("--detach", "Run in background without streaming logs")
11
+ .option("--region <region>", "Region for deployment")
12
+ .action(async (opts) => {
13
+ const projectId = resolveProjectId(program.opts().project);
14
+ // Check if there's already an app, if so redeploy
15
+ const services = await api.get(`/api/projects/${projectId}/services`);
16
+ if (services.apps && services.apps.length > 0) {
17
+ // Redeploy existing app
18
+ const app = services.apps[0];
19
+ info(`Redeploying ${chalk.bold(app.name)}...`);
20
+ await api.post(`/api/apps/${app.id}/redeploy`);
21
+ if (opts.detach) {
22
+ if (isJSONMode()) {
23
+ printJSON({ appId: app.id, status: "deploying" });
24
+ }
25
+ else {
26
+ success(`Redeploy started for ${app.name}`);
27
+ info(chalk.dim(` Check status: lizard deploy status ${app.id}`));
28
+ }
29
+ return;
30
+ }
31
+ // Stream build logs
32
+ await streamBuildLogs(app.id);
33
+ return;
34
+ }
35
+ // First deploy — create app
36
+ // TODO: detect repo from git remote, create app from repo
37
+ throw new Error("First deploy requires an existing app. Create one from the dashboard or use `lizard init` + push via git.");
38
+ });
39
+ program
40
+ .command("deploy-status")
41
+ .argument("<id>", "App or deploy ID")
42
+ .description("Show deployment status")
43
+ .action(async (id) => {
44
+ const app = await api.get(`/api/apps/${id}`);
45
+ if (isJSONMode()) {
46
+ printJSON(app);
47
+ return;
48
+ }
49
+ console.log(`${chalk.bold(app.name)} ${statusColor(app.status)}`);
50
+ if (app.domain)
51
+ console.log(` URL: ${chalk.cyan(`https://${app.domain}`)}`);
52
+ if (app.builds?.length) {
53
+ const latest = app.builds[0];
54
+ console.log(` Latest build: ${statusColor(latest.status)}`);
55
+ }
56
+ });
57
+ }
58
+ async function streamBuildLogs(appId) {
59
+ // Wait a moment for the build to start
60
+ const spinner = ora("Waiting for build...").start();
61
+ // Poll until we get a build ID
62
+ let buildId = null;
63
+ for (let i = 0; i < 30; i++) {
64
+ await sleep(2000);
65
+ try {
66
+ const app = await api.get(`/api/apps/${appId}`);
67
+ if (app.builds?.length) {
68
+ const latest = app.builds[0];
69
+ if (["building", "deploying", "running", "failed"].includes(latest.status)) {
70
+ buildId = latest.id;
71
+ break;
72
+ }
73
+ }
74
+ }
75
+ catch { }
76
+ }
77
+ spinner.stop();
78
+ if (!buildId) {
79
+ info(chalk.dim("No build found. Check `lizard deploy status <id>` for status."));
80
+ return;
81
+ }
82
+ info(chalk.dim("Streaming build logs...\n"));
83
+ await streamSSE(`/api/builds/${buildId}/logs`, (event, data) => {
84
+ if (event === "done" || event === "error") {
85
+ if (event === "error") {
86
+ error(`Build failed: ${data}`);
87
+ }
88
+ else {
89
+ success("Build complete");
90
+ }
91
+ return false;
92
+ }
93
+ // Print log line
94
+ try {
95
+ const parsed = JSON.parse(data);
96
+ if (parsed.line) {
97
+ process.stdout.write(parsed.line + "\n");
98
+ }
99
+ else if (typeof parsed === "string") {
100
+ process.stdout.write(parsed + "\n");
101
+ }
102
+ }
103
+ catch {
104
+ process.stdout.write(data + "\n");
105
+ }
106
+ return true;
107
+ });
108
+ // Check final status
109
+ const app = await api.get(`/api/apps/${appId}`);
110
+ if (app.status === "running") {
111
+ success(`Deployed! ${app.domain ? chalk.cyan(`https://${app.domain}`) : ""}`);
112
+ }
113
+ else if (app.status === "failed") {
114
+ error("Deploy failed. Check logs with `lizard logs --build`");
115
+ }
116
+ }
117
+ function sleep(ms) {
118
+ return new Promise((r) => setTimeout(r, ms));
119
+ }
120
+ //# sourceMappingURL=deploy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"deploy.js","sourceRoot":"","sources":["../../src/commands/deploy.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerDestroy(program: Command): void;
@@ -0,0 +1,51 @@
1
+ import chalk from "chalk";
2
+ import * as p from "@clack/prompts";
3
+ import { api } from "../lib/api.js";
4
+ import { resolveProjectId } from "../lib/config.js";
5
+ import { success, isJSONMode, printJSON, isTTY } from "../lib/format.js";
6
+ export function registerDestroy(program) {
7
+ program
8
+ .command("destroy")
9
+ .argument("<id>", "Service ID to destroy")
10
+ .description("Destroy a service (irreversible)")
11
+ .action(async (id) => {
12
+ const projectId = resolveProjectId(program.opts().project);
13
+ const yes = program.opts().yes;
14
+ // Confirm
15
+ if (!yes) {
16
+ if (!isTTY()) {
17
+ throw new Error("Use -y to confirm destruction in non-interactive mode");
18
+ }
19
+ const confirm = await p.confirm({
20
+ message: `Destroy service ${chalk.bold(id)}? This is irreversible.`,
21
+ });
22
+ if (p.isCancel(confirm) || !confirm) {
23
+ process.exit(5);
24
+ }
25
+ }
26
+ // Try as app first, then as addon
27
+ try {
28
+ await api.delete(`/api/apps/${id}`);
29
+ if (isJSONMode()) {
30
+ printJSON({ id, status: "destroyed", type: "app" });
31
+ }
32
+ else {
33
+ success(`Service ${id} destroyed`);
34
+ }
35
+ return;
36
+ }
37
+ catch (err) {
38
+ if (err.status !== 404)
39
+ throw err;
40
+ }
41
+ // Try as addon
42
+ await api.delete(`/api/projects/${projectId}/addons/${id}`);
43
+ if (isJSONMode()) {
44
+ printJSON({ id, status: "destroyed", type: "addon" });
45
+ }
46
+ else {
47
+ success(`Service ${id} destroyed`);
48
+ }
49
+ });
50
+ }
51
+ //# sourceMappingURL=destroy.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"destroy.js","sourceRoot":"","sources":["../../src/commands/destroy.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,MAAM,OAAO,CAAC;AAC1B,OAAO,KAAK,CAAC,MAAM,gBAAgB,CAAC;AAEpC,OAAO,EAAE,GAAG,EAAE,MAAM,eAAe,CAAC;AACpC,OAAO,EAAE,gBAAgB,EAAE,MAAM,kBAAkB,CAAC;AACpD,OAAO,EAAE,OAAO,EAAE,UAAU,EAAE,SAAS,EAAE,KAAK,EAAE,MAAM,kBAAkB,CAAC;AAEzE,MAAM,UAAU,eAAe,CAAC,OAAgB;IAC9C,OAAO;SACJ,OAAO,CAAC,SAAS,CAAC;SAClB,QAAQ,CAAC,MAAM,EAAE,uBAAuB,CAAC;SACzC,WAAW,CAAC,kCAAkC,CAAC;SAC/C,MAAM,CAAC,KAAK,EAAE,EAAU,EAAE,EAAE;QAC3B,MAAM,SAAS,GAAG,gBAAgB,CAAC,OAAO,CAAC,IAAI,EAAE,CAAC,OAAO,CAAC,CAAC;QAC3D,MAAM,GAAG,GAAG,OAAO,CAAC,IAAI,EAAE,CAAC,GAAG,CAAC;QAE/B,UAAU;QACV,IAAI,CAAC,GAAG,EAAE,CAAC;YACT,IAAI,CAAC,KAAK,EAAE,EAAE,CAAC;gBACb,MAAM,IAAI,KAAK,CAAC,uDAAuD,CAAC,CAAC;YAC3E,CAAC;YACD,MAAM,OAAO,GAAG,MAAM,CAAC,CAAC,OAAO,CAAC;gBAC9B,OAAO,EAAE,mBAAmB,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,yBAAyB;aACpE,CAAC,CAAC;YACH,IAAI,CAAC,CAAC,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC;gBACpC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC;YAClB,CAAC;QACH,CAAC;QAED,kCAAkC;QAClC,IAAI,CAAC;YACH,MAAM,GAAG,CAAC,MAAM,CAAC,aAAa,EAAE,EAAE,CAAC,CAAC;YACpC,IAAI,UAAU,EAAE,EAAE,CAAC;gBACjB,SAAS,CAAC,EAAE,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC,CAAC;YACtD,CAAC;iBAAM,CAAC;gBACN,OAAO,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;YACrC,CAAC;YACD,OAAO;QACT,CAAC;QAAC,OAAO,GAAQ,EAAE,CAAC;YAClB,IAAI,GAAG,CAAC,MAAM,KAAK,GAAG;gBAAE,MAAM,GAAG,CAAC;QACpC,CAAC;QAED,eAAe;QACf,MAAM,GAAG,CAAC,MAAM,CAAC,iBAAiB,SAAS,WAAW,EAAE,EAAE,CAAC,CAAC;QAC5D,IAAI,UAAU,EAAE,EAAE,CAAC;YACjB,SAAS,CAAC,EAAE,EAAE,EAAE,MAAM,EAAE,WAAW,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;QACxD,CAAC;aAAM,CAAC;YACN,OAAO,CAAC,WAAW,EAAE,YAAY,CAAC,CAAC;QACrC,CAAC;IACH,CAAC,CAAC,CAAC;AACP,CAAC"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerGit(program: Command): void;
@@ -0,0 +1,67 @@
1
+ import chalk from "chalk";
2
+ import { api } from "../lib/api.js";
3
+ import { resolveProjectId } from "../lib/config.js";
4
+ import { info, isJSONMode, printJSON } from "../lib/format.js";
5
+ export function registerGit(program) {
6
+ const git = program
7
+ .command("git")
8
+ .description("Git integration");
9
+ git
10
+ .command("connect")
11
+ .argument("<repo>", "GitHub repository (user/repo)")
12
+ .description("Connect Git repository for auto-deploy")
13
+ .option("--branch <name>", "Branch for auto-deploy", "main")
14
+ .action(async (repo, opts) => {
15
+ const projectId = resolveProjectId(program.opts().project);
16
+ // This requires a server endpoint for programmatic webhook setup
17
+ // For now, guide the user to use the dashboard
18
+ if (isJSONMode()) {
19
+ printJSON({
20
+ error: "not_implemented",
21
+ message: "Git connect via CLI requires server endpoint. Use the dashboard.",
22
+ });
23
+ }
24
+ else {
25
+ info(`To connect ${chalk.cyan(repo)} for auto-deploy:`);
26
+ info(` 1. Open your project on lizard.build`);
27
+ info(` 2. Go to Settings → Git Integration`);
28
+ info(` 3. Connect ${repo} (branch: ${opts.branch})`);
29
+ info("");
30
+ info(chalk.dim("CLI git connect will be available in a future update."));
31
+ }
32
+ });
33
+ git
34
+ .command("disconnect")
35
+ .description("Disconnect Git auto-deploy")
36
+ .action(async () => {
37
+ info(chalk.dim("Git disconnect via CLI will be available in a future update."));
38
+ });
39
+ git
40
+ .command("status")
41
+ .description("Show Git integration status")
42
+ .action(async () => {
43
+ const projectId = resolveProjectId(program.opts().project);
44
+ // Get apps to check for repo info
45
+ const services = await api.get(`/api/projects/${projectId}/services`);
46
+ const appsWithRepo = (services.apps || []).filter((a) => a.repo);
47
+ if (isJSONMode()) {
48
+ printJSON({
49
+ connected: appsWithRepo.length > 0,
50
+ apps: appsWithRepo.map((a) => ({
51
+ name: a.name,
52
+ repo: a.repo,
53
+ branch: a.branch,
54
+ })),
55
+ });
56
+ return;
57
+ }
58
+ if (appsWithRepo.length === 0) {
59
+ console.log("No Git repositories connected.");
60
+ return;
61
+ }
62
+ for (const app of appsWithRepo) {
63
+ console.log(`${chalk.bold(app.name)}: ${chalk.cyan(app.repo)} (${app.branch || "main"})`);
64
+ }
65
+ });
66
+ }
67
+ //# sourceMappingURL=git.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"git.js","sourceRoot":"","sources":["../../src/commands/git.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ import { Command } from "commander";
2
+ export declare function registerInit(program: Command): void;