@lizard-build/cli 0.1.0 → 0.3.29

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 (178) hide show
  1. package/.github/workflows/release.yml +90 -0
  2. package/README.md +41 -0
  3. package/dist/commands/add.js +318 -45
  4. package/dist/commands/add.js.map +1 -1
  5. package/dist/commands/config.d.ts +2 -0
  6. package/dist/commands/config.js +68 -0
  7. package/dist/commands/config.js.map +1 -0
  8. package/dist/commands/docs.d.ts +2 -0
  9. package/dist/commands/docs.js +13 -0
  10. package/dist/commands/docs.js.map +1 -0
  11. package/dist/commands/domain.d.ts +9 -0
  12. package/dist/commands/domain.js +195 -0
  13. package/dist/commands/domain.js.map +1 -0
  14. package/dist/commands/git.js +175 -36
  15. package/dist/commands/git.js.map +1 -1
  16. package/dist/commands/init.d.ts +24 -0
  17. package/dist/commands/init.js +128 -86
  18. package/dist/commands/init.js.map +1 -1
  19. package/dist/commands/link.d.ts +7 -0
  20. package/dist/commands/link.js +104 -33
  21. package/dist/commands/link.js.map +1 -1
  22. package/dist/commands/login.js +4 -3
  23. package/dist/commands/login.js.map +1 -1
  24. package/dist/commands/logs.js +223 -30
  25. package/dist/commands/logs.js.map +1 -1
  26. package/dist/commands/open.js +3 -2
  27. package/dist/commands/open.js.map +1 -1
  28. package/dist/commands/port.d.ts +7 -0
  29. package/dist/commands/port.js +49 -0
  30. package/dist/commands/port.js.map +1 -0
  31. package/dist/commands/projects.js +36 -6
  32. package/dist/commands/projects.js.map +1 -1
  33. package/dist/commands/ps.js +32 -39
  34. package/dist/commands/ps.js.map +1 -1
  35. package/dist/commands/redeploy.js +48 -8
  36. package/dist/commands/redeploy.js.map +1 -1
  37. package/dist/commands/regions.js +2 -5
  38. package/dist/commands/regions.js.map +1 -1
  39. package/dist/commands/restart.js +84 -10
  40. package/dist/commands/restart.js.map +1 -1
  41. package/dist/commands/run.d.ts +9 -0
  42. package/dist/commands/run.js +61 -22
  43. package/dist/commands/run.js.map +1 -1
  44. package/dist/commands/scale.d.ts +10 -0
  45. package/dist/commands/scale.js +166 -0
  46. package/dist/commands/scale.js.map +1 -0
  47. package/dist/commands/secrets.js +200 -89
  48. package/dist/commands/secrets.js.map +1 -1
  49. package/dist/commands/service-set.d.ts +49 -0
  50. package/dist/commands/service-set.js +552 -0
  51. package/dist/commands/service-set.js.map +1 -0
  52. package/dist/commands/service-show.d.ts +11 -0
  53. package/dist/commands/service-show.js +44 -0
  54. package/dist/commands/service-show.js.map +1 -0
  55. package/dist/commands/service.d.ts +8 -0
  56. package/dist/commands/service.js +262 -0
  57. package/dist/commands/service.js.map +1 -0
  58. package/dist/commands/ssh.d.ts +2 -0
  59. package/dist/commands/ssh.js +161 -0
  60. package/dist/commands/ssh.js.map +1 -0
  61. package/dist/commands/status.d.ts +7 -0
  62. package/dist/commands/status.js +49 -38
  63. package/dist/commands/status.js.map +1 -1
  64. package/dist/commands/unlink.d.ts +5 -0
  65. package/dist/commands/unlink.js +18 -0
  66. package/dist/commands/unlink.js.map +1 -0
  67. package/dist/commands/up.d.ts +9 -0
  68. package/dist/commands/up.js +417 -0
  69. package/dist/commands/up.js.map +1 -0
  70. package/dist/commands/upgrade.d.ts +2 -0
  71. package/dist/commands/upgrade.js +79 -0
  72. package/dist/commands/upgrade.js.map +1 -0
  73. package/dist/commands/whoami.js +26 -6
  74. package/dist/commands/whoami.js.map +1 -1
  75. package/dist/commands/workspace.d.ts +8 -0
  76. package/dist/commands/workspace.js +36 -0
  77. package/dist/commands/workspace.js.map +1 -0
  78. package/dist/index.js +204 -82
  79. package/dist/index.js.map +1 -1
  80. package/dist/lib/api.d.ts +17 -2
  81. package/dist/lib/api.js +85 -51
  82. package/dist/lib/api.js.map +1 -1
  83. package/dist/lib/auth.d.ts +3 -11
  84. package/dist/lib/auth.js +16 -36
  85. package/dist/lib/auth.js.map +1 -1
  86. package/dist/lib/config.d.ts +36 -15
  87. package/dist/lib/config.js +71 -58
  88. package/dist/lib/config.js.map +1 -1
  89. package/dist/lib/format.d.ts +1 -0
  90. package/dist/lib/format.js +17 -4
  91. package/dist/lib/format.js.map +1 -1
  92. package/dist/lib/name.d.ts +11 -0
  93. package/dist/lib/name.js +26 -0
  94. package/dist/lib/name.js.map +1 -0
  95. package/dist/lib/picker.d.ts +32 -0
  96. package/dist/lib/picker.js +91 -0
  97. package/dist/lib/picker.js.map +1 -0
  98. package/dist/lib/resolve.d.ts +85 -0
  99. package/dist/lib/resolve.js +203 -0
  100. package/dist/lib/resolve.js.map +1 -0
  101. package/dist/lib/updater.d.ts +16 -0
  102. package/dist/lib/updater.js +102 -0
  103. package/dist/lib/updater.js.map +1 -0
  104. package/lizard-wrapper.sh +2 -0
  105. package/package.json +11 -3
  106. package/src/commands/add.ts +388 -56
  107. package/src/commands/config.ts +80 -0
  108. package/src/commands/docs.ts +15 -0
  109. package/src/commands/domain.ts +248 -0
  110. package/src/commands/git.ts +201 -40
  111. package/src/commands/init.ts +149 -100
  112. package/src/commands/link.ts +127 -35
  113. package/src/commands/login.ts +4 -3
  114. package/src/commands/logs.ts +283 -27
  115. package/src/commands/open.ts +3 -2
  116. package/src/commands/port.ts +57 -0
  117. package/src/commands/projects.ts +43 -6
  118. package/src/commands/ps.ts +39 -60
  119. package/src/commands/redeploy.ts +51 -10
  120. package/src/commands/regions.ts +2 -6
  121. package/src/commands/restart.ts +84 -10
  122. package/src/commands/run.ts +68 -24
  123. package/src/commands/scale.ts +216 -0
  124. package/src/commands/secrets.ts +277 -100
  125. package/src/commands/service-set.ts +669 -0
  126. package/src/commands/service-show.ts +52 -0
  127. package/src/commands/service.ts +298 -0
  128. package/src/commands/ssh.ts +176 -0
  129. package/src/commands/status.ts +51 -46
  130. package/src/commands/unlink.ts +17 -0
  131. package/src/commands/up.ts +461 -0
  132. package/src/commands/upgrade.ts +87 -0
  133. package/src/commands/whoami.ts +34 -6
  134. package/src/commands/workspace.ts +44 -0
  135. package/src/index.ts +214 -85
  136. package/src/lib/api.ts +114 -51
  137. package/src/lib/auth.ts +22 -46
  138. package/src/lib/config.ts +100 -65
  139. package/src/lib/format.ts +18 -4
  140. package/src/lib/name.ts +27 -0
  141. package/src/lib/picker.ts +133 -0
  142. package/src/lib/resolve.ts +285 -0
  143. package/src/lib/updater.ts +106 -0
  144. package/test/cli.test.ts +491 -0
  145. package/test/fixtures/hello-app/Dockerfile +5 -0
  146. package/test/fixtures/hello-app/index.js +5 -0
  147. package/test/unit/api.test.ts +66 -0
  148. package/test/unit/config.test.ts +94 -0
  149. package/test/unit/init.test.ts +211 -0
  150. package/test/unit/json.test.ts +208 -0
  151. package/test/unit/picker.test.ts +161 -0
  152. package/test/unit/resolve.test.ts +124 -0
  153. package/test/unit/service-set.test.ts +355 -0
  154. package/vitest.config.ts +10 -0
  155. package/dist/commands/connect.d.ts +0 -2
  156. package/dist/commands/connect.js +0 -117
  157. package/dist/commands/connect.js.map +0 -1
  158. package/dist/commands/context.d.ts +0 -2
  159. package/dist/commands/context.js +0 -71
  160. package/dist/commands/context.js.map +0 -1
  161. package/dist/commands/deploy.d.ts +0 -2
  162. package/dist/commands/deploy.js +0 -120
  163. package/dist/commands/deploy.js.map +0 -1
  164. package/dist/commands/destroy.d.ts +0 -2
  165. package/dist/commands/destroy.js +0 -51
  166. package/dist/commands/destroy.js.map +0 -1
  167. package/dist/commands/update.d.ts +0 -2
  168. package/dist/commands/update.js +0 -41
  169. package/dist/commands/update.js.map +0 -1
  170. package/dist/commands/version.d.ts +0 -2
  171. package/dist/commands/version.js +0 -37
  172. package/dist/commands/version.js.map +0 -1
  173. package/src/commands/connect.ts +0 -145
  174. package/src/commands/context.ts +0 -93
  175. package/src/commands/deploy.ts +0 -153
  176. package/src/commands/destroy.ts +0 -51
  177. package/src/commands/update.ts +0 -44
  178. package/src/commands/version.ts +0 -37
package/src/lib/auth.ts CHANGED
@@ -1,55 +1,32 @@
1
- import fs from "node:fs";
2
- import path from "node:path";
3
- import os from "node:os";
4
1
  import open from "open";
2
+ import {
3
+ loadConfig,
4
+ saveConfig,
5
+ type Credentials,
6
+ } from "./config.js";
5
7
 
6
- const LIZARD_DIR = path.join(os.homedir(), ".lizard");
7
- const CREDENTIALS_FILE = path.join(LIZARD_DIR, "credentials.json");
8
+ export type { Credentials } from "./config.js";
8
9
 
9
- export interface Credentials {
10
- accessToken: string;
11
- refreshToken?: string;
12
- expiresAt?: string;
13
- userId: string;
14
- username: string;
15
- email?: string;
16
- avatarUrl?: string;
17
- }
18
-
19
- let tokenOverride: string | null = null;
20
-
21
- export function setTokenOverride(token: string) {
22
- tokenOverride = token;
23
- }
24
-
25
- /** Get the active token in priority order: override → env → file */
10
+ /** Get the active token in priority order: env → file */
26
11
  export function getToken(): string | null {
27
- if (tokenOverride) return tokenOverride;
28
12
  if (process.env.LIZARD_TOKEN) return process.env.LIZARD_TOKEN;
29
- const creds = loadCredentials();
30
- return creds?.accessToken ?? null;
13
+ return loadCredentials()?.accessToken ?? null;
31
14
  }
32
15
 
33
16
  export function loadCredentials(): Credentials | null {
34
- try {
35
- const data = fs.readFileSync(CREDENTIALS_FILE, "utf-8");
36
- return JSON.parse(data) as Credentials;
37
- } catch {
38
- return null;
39
- }
17
+ return loadConfig().credentials ?? null;
40
18
  }
41
19
 
42
20
  export function saveCredentials(creds: Credentials) {
43
- fs.mkdirSync(LIZARD_DIR, { recursive: true });
44
- fs.writeFileSync(CREDENTIALS_FILE, JSON.stringify(creds, null, 2), {
45
- mode: 0o600,
46
- });
21
+ const config = loadConfig();
22
+ config.credentials = creds;
23
+ saveConfig(config);
47
24
  }
48
25
 
49
26
  export function clearCredentials() {
50
- try {
51
- fs.unlinkSync(CREDENTIALS_FILE);
52
- } catch {}
27
+ const config = loadConfig();
28
+ delete config.credentials;
29
+ saveConfig(config);
53
30
  }
54
31
 
55
32
  export function isLoggedIn(): boolean {
@@ -65,10 +42,9 @@ function isTTY(): boolean {
65
42
  * Returns credentials or throws.
66
43
  */
67
44
  export async function requireAuth(): Promise<Credentials> {
68
- // Token override or env var — we don't have full Credentials, fake it
69
- if (tokenOverride || process.env.LIZARD_TOKEN) {
45
+ if (process.env.LIZARD_TOKEN) {
70
46
  return {
71
- accessToken: (tokenOverride || process.env.LIZARD_TOKEN)!,
47
+ accessToken: process.env.LIZARD_TOKEN,
72
48
  userId: "",
73
49
  username: "",
74
50
  };
@@ -77,14 +53,14 @@ export async function requireAuth(): Promise<Credentials> {
77
53
  const creds = loadCredentials();
78
54
  if (creds) return creds;
79
55
 
80
- // Not logged in
81
56
  if (!isTTY()) {
82
- throw new Error(
57
+ const err = new Error(
83
58
  "Not authenticated. Set LIZARD_TOKEN or run `lizard login` first.",
84
- );
59
+ ) as Error & { code: string };
60
+ err.code = "NOT_AUTHENTICATED";
61
+ throw err;
85
62
  }
86
63
 
87
- // Auto-login
88
64
  const { performLogin } = await import("../commands/login.js");
89
65
  return performLogin();
90
66
  }
@@ -101,7 +77,7 @@ export async function openURL(url: string) {
101
77
  !process.env.WAYLAND_DISPLAY;
102
78
 
103
79
  if (isSSH || isCI || noDisplay) {
104
- return false; // caller should show URL manually
80
+ return false;
105
81
  }
106
82
 
107
83
  try {
package/src/lib/config.ts CHANGED
@@ -2,92 +2,127 @@ import fs from "node:fs";
2
2
  import path from "node:path";
3
3
  import os from "node:os";
4
4
 
5
- const CONFIG_DIR = ".lizard";
6
- const CONFIG_FILE = "config.json";
7
- const GLOBAL_SETTINGS_FILE = path.join(os.homedir(), ".lizard", "settings.json");
5
+ export const DEFAULT_REGION = "us-east-1";
8
6
 
9
- export interface ProjectConfig {
10
- workspaceId?: string;
11
- projectId: string;
12
- projectName?: string;
13
- environment?: string;
7
+ // Resolve at every call so tests (and exotic shells that mutate HOME) get a
8
+ // fresh path. Cost is negligible — we hit disk anyway. LIZARD_HOME lets tests
9
+ // (and power users) point the config dir somewhere other than ~/.lizard.
10
+ function configDir(): string {
11
+ return process.env.LIZARD_HOME
12
+ ? path.join(process.env.LIZARD_HOME, ".lizard")
13
+ : path.join(os.homedir(), ".lizard");
14
14
  }
15
-
16
- export interface GlobalSettings {
17
- defaultWorkspace?: string;
15
+ function configFile(): string {
16
+ return path.join(configDir(), "config.json");
18
17
  }
19
18
 
20
- /** Find project config by walking up from cwd */
21
- export function findProjectConfig(): ProjectConfig | null {
22
- let dir = process.cwd();
23
- while (true) {
24
- const configPath = path.join(dir, CONFIG_DIR, CONFIG_FILE);
25
- if (fs.existsSync(configPath)) {
26
- try {
27
- return JSON.parse(fs.readFileSync(configPath, "utf-8"));
28
- } catch {
29
- return null;
30
- }
31
- }
32
- const parent = path.dirname(dir);
33
- if (parent === dir) break;
34
- dir = parent;
35
- }
36
- return null;
19
+ export interface Credentials {
20
+ accessToken: string;
21
+ refreshToken?: string;
22
+ expiresAt?: string;
23
+ userId: string;
24
+ username: string;
25
+ email?: string;
26
+ avatarUrl?: string;
37
27
  }
38
28
 
39
- /** Save project config in cwd */
40
- export function saveProjectConfig(config: ProjectConfig) {
41
- const dir = path.join(process.cwd(), CONFIG_DIR);
42
- fs.mkdirSync(dir, { recursive: true });
43
- fs.writeFileSync(path.join(dir, CONFIG_FILE), JSON.stringify(config, null, 2));
29
+ export interface ProjectLink {
30
+ projectId: string;
31
+ projectName?: string;
32
+ /** Workspace the project belongs to. Lazy-filled for legacy links. */
33
+ workspaceId?: string;
34
+ workspaceName?: string;
35
+ /** Active service for this cwd. `appId/appName` are kept as aliases for backwards compat. */
36
+ serviceId?: string;
37
+ serviceName?: string;
38
+ /** @deprecated use serviceId */
39
+ appId?: string;
40
+ /** @deprecated use serviceName */
41
+ appName?: string;
42
+ }
44
43
 
45
- // Add .lizard/ to .gitignore if not already there
46
- const gitignorePath = path.join(process.cwd(), ".gitignore");
47
- try {
48
- const existing = fs.existsSync(gitignorePath)
49
- ? fs.readFileSync(gitignorePath, "utf-8")
50
- : "";
51
- if (!existing.includes(".lizard/")) {
52
- fs.appendFileSync(
53
- gitignorePath,
54
- (existing.endsWith("\n") ? "" : "\n") + ".lizard/\n",
55
- );
56
- }
57
- } catch {}
44
+ export interface Config {
45
+ credentials?: Credentials;
46
+ projects?: Record<string, ProjectLink>;
58
47
  }
59
48
 
60
- export function loadGlobalSettings(): GlobalSettings {
49
+ export function loadConfig(): Config {
61
50
  try {
62
- return JSON.parse(fs.readFileSync(GLOBAL_SETTINGS_FILE, "utf-8"));
51
+ return JSON.parse(fs.readFileSync(configFile(), "utf-8")) as Config;
63
52
  } catch {
64
53
  return {};
65
54
  }
66
55
  }
67
56
 
68
- export function saveGlobalSettings(settings: GlobalSettings) {
69
- const dir = path.dirname(GLOBAL_SETTINGS_FILE);
70
- fs.mkdirSync(dir, { recursive: true });
71
- fs.writeFileSync(GLOBAL_SETTINGS_FILE, JSON.stringify(settings, null, 2));
57
+ export function saveConfig(config: Config) {
58
+ fs.mkdirSync(configDir(), { recursive: true });
59
+ fs.writeFileSync(configFile(), JSON.stringify(config, null, 2), {
60
+ mode: 0o600,
61
+ });
72
62
  }
73
63
 
74
64
  /**
75
- * Resolve projectId from: --project flag .lizard/config.json → error
65
+ * Read the link for a directory. Normalises legacy `appId/appName` into
66
+ * `serviceId/serviceName` so callers only have to look at one pair.
76
67
  */
77
- export function resolveProjectId(flagValue?: string): string {
78
- if (flagValue) return flagValue;
79
- const config = findProjectConfig();
80
- if (config?.projectId) return config.projectId;
81
- throw new Error(
82
- "No project linked. Run `lizard init` or `lizard link` first, or use --project <id>.",
83
- );
68
+ export function getProjectLink(cwd: string = process.cwd()): ProjectLink | null {
69
+ const raw = loadConfig().projects?.[cwd];
70
+ if (!raw) return null;
71
+ return {
72
+ ...raw,
73
+ serviceId: raw.serviceId ?? raw.appId,
74
+ serviceName: raw.serviceName ?? raw.appName,
75
+ };
76
+ }
77
+
78
+ export function setProjectLink(link: ProjectLink, cwd: string = process.cwd()) {
79
+ const config = loadConfig();
80
+ config.projects ??= {};
81
+ // Mirror service↔app for older readers.
82
+ const normalised: ProjectLink = {
83
+ ...link,
84
+ appId: link.serviceId ?? link.appId,
85
+ appName: link.serviceName ?? link.appName,
86
+ serviceId: link.serviceId ?? link.appId,
87
+ serviceName: link.serviceName ?? link.appName,
88
+ };
89
+ config.projects[cwd] = normalised;
90
+ saveConfig(config);
91
+ }
92
+
93
+ export function updateProjectLink(
94
+ patch: Partial<ProjectLink>,
95
+ cwd: string = process.cwd(),
96
+ ) {
97
+ const existing = getProjectLink(cwd);
98
+ if (!existing) return;
99
+ setProjectLink({ ...existing, ...patch }, cwd);
100
+ }
101
+
102
+ export function clearProjectLink(cwd: string = process.cwd()) {
103
+ const config = loadConfig();
104
+ if (config.projects) {
105
+ delete config.projects[cwd];
106
+ saveConfig(config);
107
+ }
84
108
  }
85
109
 
86
110
  /**
87
- * Resolve environment from: --environment flag .lizard/config.json "production"
111
+ * Resolve a project flag (name, slug, or ID) to an actual project ID.
112
+ * When no flag is given, falls back to the linked cwd. Hits the API only
113
+ * when a flag is provided so name/slug lookups work as advertised.
88
114
  */
89
- export function resolveEnvironment(flagValue?: string): string {
90
- if (flagValue) return flagValue;
91
- const config = findProjectConfig();
92
- return config?.environment ?? "production";
115
+ export async function resolveProjectId(flagValue?: string): Promise<string> {
116
+ if (!flagValue) {
117
+ const link = getProjectLink();
118
+ if (link?.projectId) return link.projectId;
119
+ throw new Error("No project linked. Run `lizard init` or pass --project <id>.");
120
+ }
121
+ const { api } = await import("./api.js");
122
+ const projects = await api.get<Array<{ id: string; name: string; slug: string }>>("/api/projects");
123
+ const match = projects.find(
124
+ (p) => p.id === flagValue || p.slug === flagValue || p.name === flagValue,
125
+ );
126
+ if (!match) throw new Error(`Project "${flagValue}" not found.`);
127
+ return match.id;
93
128
  }
package/src/lib/format.ts CHANGED
@@ -34,29 +34,43 @@ export function info(msg: string) {
34
34
  process.stderr.write(msg + "\n");
35
35
  }
36
36
 
37
+ export function link(url: string, text?: string): string {
38
+ const label = text ?? url;
39
+ if (!isTTY()) return label;
40
+ return `\x1b]8;;${url}\x1b\\${label}\x1b]8;;\x1b\\`;
41
+ }
42
+
43
+ const ANSI_RE = /\x1b\[[0-9;]*m/g;
44
+ function visibleLength(s: string): number {
45
+ return s.replace(ANSI_RE, "").length;
46
+ }
47
+ function padVisible(s: string, width: number): string {
48
+ return s + " ".repeat(Math.max(0, width - visibleLength(s)));
49
+ }
50
+
37
51
  export function table(
38
52
  headers: string[],
39
53
  rows: string[][],
40
54
  ) {
41
55
  if (rows.length === 0) return;
42
56
 
43
- const widths = headers.map((h) => h.length);
57
+ const widths = headers.map((h) => visibleLength(h));
44
58
  for (const row of rows) {
45
59
  for (let i = 0; i < row.length; i++) {
46
60
  if (i < widths.length) {
47
- widths[i] = Math.max(widths[i], (row[i] || "").length);
61
+ widths[i] = Math.max(widths[i], visibleLength(row[i] || ""));
48
62
  }
49
63
  }
50
64
  }
51
65
 
52
66
  const header = headers
53
- .map((h, i) => h.toUpperCase().padEnd(widths[i]))
67
+ .map((h, i) => padVisible(h.toUpperCase(), widths[i]))
54
68
  .join(" ");
55
69
  console.log(chalk.dim(header));
56
70
 
57
71
  for (const row of rows) {
58
72
  const line = headers
59
- .map((_, i) => (row[i] || "").padEnd(widths[i]))
73
+ .map((_, i) => padVisible(row[i] || "", widths[i]))
60
74
  .join(" ");
61
75
  console.log(line);
62
76
  }
@@ -0,0 +1,27 @@
1
+ // Shared name validator. Mirrors server/src/services/var-transform.ts.
2
+ // LIZARD-55: 1–40 chars, [a-z0-9-], must start and end with [a-z0-9].
3
+
4
+ export const NAME_REGEX = /^[a-z0-9]([a-z0-9-]{0,38}[a-z0-9])?$/;
5
+ export const NAME_VALIDATION_HINT =
6
+ "1–40 chars, lowercase a–z, digits, hyphens; can't start or end with a hyphen";
7
+
8
+ export function validateName(name: string): string | null {
9
+ if (!name) return "name is required";
10
+ if (name.length > 40) return "name must be 40 characters or fewer";
11
+ if (!NAME_REGEX.test(name)) return `invalid name (${NAME_VALIDATION_HINT})`;
12
+ return null;
13
+ }
14
+
15
+ /** Mirrors `slugifyName` in lizard-client/src/lib/api.ts. */
16
+ export function slugifyName(name: string): string {
17
+ return name
18
+ .toLowerCase()
19
+ .replace(/[^a-z0-9_-]+/g, "-")
20
+ .replace(/^-+|-+$/g, "");
21
+ }
22
+
23
+ /** Mirrors `addonRefName` in lizard-client — what users type inside ${{...}}. */
24
+ export function addonRefName(addon: { name?: string | null; type?: string; addonType?: string }): string {
25
+ const display = addon.name || addon.type || addon.addonType || "";
26
+ return slugifyName(display) || display;
27
+ }
@@ -0,0 +1,133 @@
1
+ import * as p from "@clack/prompts";
2
+ import { api, withQuery, type Workspace } from "./api.js";
3
+ import { isTTY } from "./format.js";
4
+
5
+ export async function fetchWorkspaces(): Promise<Workspace[]> {
6
+ return api.get<Workspace[]>("/api/workspaces");
7
+ }
8
+
9
+ export function matchWorkspace(
10
+ workspaces: Workspace[],
11
+ idOrSlugOrName: string,
12
+ ): Workspace | undefined {
13
+ const lower = idOrSlugOrName.toLowerCase();
14
+ return workspaces.find(
15
+ (w) =>
16
+ w.id.toLowerCase() === lower ||
17
+ w.slug.toLowerCase() === lower ||
18
+ w.name.toLowerCase() === lower,
19
+ );
20
+ }
21
+
22
+ /**
23
+ * Resolve a workspace by flag value. Fetches the list once; throws with the
24
+ * available names when the value doesn't match anything the user belongs to.
25
+ */
26
+ export async function resolveWorkspace(flagValue: string): Promise<Workspace> {
27
+ const workspaces = await fetchWorkspaces();
28
+ const match = matchWorkspace(workspaces, flagValue);
29
+ if (!match) {
30
+ throw new Error(
31
+ `Workspace "${flagValue}" not found. Available: ${
32
+ workspaces.map((w) => w.name).join(", ") || "(none)"
33
+ }`,
34
+ );
35
+ }
36
+ return match;
37
+ }
38
+
39
+ /**
40
+ * Pick a workspace for project creation / linking.
41
+ *
42
+ * 1. --workspace flag wins.
43
+ * 2. If only one workspace exists, auto-select it.
44
+ * 3. Non-TTY → prefer personal, else first.
45
+ * 4. TTY → prompt.
46
+ *
47
+ * `projectNameHint` lets the caller short-circuit when --name X uniquely
48
+ * identifies a workspace (see init flow case 5).
49
+ */
50
+ export async function pickWorkspace(opts: {
51
+ flag?: string;
52
+ projectNameHint?: string;
53
+ workspaces?: Workspace[];
54
+ }): Promise<Workspace> {
55
+ if (opts.flag) return resolveWorkspace(opts.flag);
56
+
57
+ const list = opts.workspaces ?? (await fetchWorkspaces());
58
+ if (list.length === 0) {
59
+ throw new Error(
60
+ "No workspaces available. The backend should always return a personal workspace — please report this.",
61
+ );
62
+ }
63
+ if (list.length === 1) return list[0];
64
+
65
+ // Hint-based auto-pick: a --name match that is unique across workspaces.
66
+ if (opts.projectNameHint) {
67
+ const matches = await findWorkspacesContainingProject(
68
+ list,
69
+ opts.projectNameHint,
70
+ );
71
+ if (matches.length === 1) return matches[0].workspace;
72
+ if (matches.length > 1) {
73
+ const detail = matches
74
+ .map((m) => ` • ${m.project.name} in ${m.workspace.name} (--workspace ${m.workspace.slug})`)
75
+ .join("\n");
76
+ throw new Error(
77
+ `Multiple projects named "${opts.projectNameHint}" found:\n${detail}\nPass --workspace, or run \`lizard init\` without --name to choose interactively.`,
78
+ );
79
+ }
80
+ }
81
+
82
+ if (!isTTY()) {
83
+ // Non-TTY fallback: personal first, else first ws.
84
+ const personal = list.find((w) => w.isPersonal);
85
+ return personal ?? list[0];
86
+ }
87
+
88
+ const sel = await p.select({
89
+ message: "Workspace",
90
+ options: list.map((w) => ({
91
+ value: w.id,
92
+ label: w.name,
93
+ hint: `${w.role}${w.isPersonal ? " · personal" : ""} · ${w.projectCount ?? 0} project${(w.projectCount ?? 0) === 1 ? "" : "s"}`,
94
+ })),
95
+ });
96
+ if (p.isCancel(sel)) process.exit(5);
97
+ return list.find((w) => w.id === sel)!;
98
+ }
99
+
100
+ interface ProjectLite {
101
+ id: string;
102
+ name: string;
103
+ slug: string;
104
+ workspaceId?: string | null;
105
+ }
106
+
107
+ async function findWorkspacesContainingProject(
108
+ workspaces: Workspace[],
109
+ nameOrSlugOrId: string,
110
+ ): Promise<Array<{ workspace: Workspace; project: ProjectLite }>> {
111
+ const all = await api.get<ProjectLite[]>("/api/projects");
112
+ const lower = nameOrSlugOrId.toLowerCase();
113
+ const matches: Array<{ workspace: Workspace; project: ProjectLite }> = [];
114
+ for (const w of workspaces) {
115
+ const inWs = all.find(
116
+ (pr) =>
117
+ pr.workspaceId === w.id &&
118
+ (pr.id.toLowerCase() === lower ||
119
+ pr.slug.toLowerCase() === lower ||
120
+ pr.name.toLowerCase() === lower),
121
+ );
122
+ if (inWs) matches.push({ workspace: w, project: inWs });
123
+ }
124
+ return matches;
125
+ }
126
+
127
+ export async function listProjectsInWorkspace(
128
+ workspaceId: string,
129
+ ): Promise<ProjectLite[]> {
130
+ return api.get<ProjectLite[]>(
131
+ withQuery("/api/projects", { workspaceId }),
132
+ );
133
+ }