0pflow 0.1.0-dev.303bb09 → 0.1.0-dev.3666a4d

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 (153) hide show
  1. package/dist/__tests__/factory.test.js +13 -0
  2. package/dist/__tests__/factory.test.js.map +1 -1
  3. package/dist/__tests__/integration.e2e.test.js +2 -1
  4. package/dist/__tests__/integration.e2e.test.js.map +1 -1
  5. package/dist/__tests__/integration.test.js +87 -82
  6. package/dist/__tests__/integration.test.js.map +1 -1
  7. package/dist/agent.d.ts +1 -0
  8. package/dist/agent.d.ts.map +1 -1
  9. package/dist/agent.js +17 -2
  10. package/dist/agent.js.map +1 -1
  11. package/dist/cli/app.d.ts +6 -0
  12. package/dist/cli/app.d.ts.map +1 -1
  13. package/dist/cli/app.js +31 -0
  14. package/dist/cli/app.js.map +1 -1
  15. package/dist/cli/cloud-dev.d.ts +7 -0
  16. package/dist/cli/cloud-dev.d.ts.map +1 -0
  17. package/dist/cli/cloud-dev.js +571 -0
  18. package/dist/cli/cloud-dev.js.map +1 -0
  19. package/dist/cli/deploy.d.ts +27 -0
  20. package/dist/cli/deploy.d.ts.map +1 -0
  21. package/dist/cli/deploy.js +332 -0
  22. package/dist/cli/deploy.js.map +1 -0
  23. package/dist/cli/discovery.d.ts.map +1 -1
  24. package/dist/cli/discovery.js +14 -1
  25. package/dist/cli/discovery.js.map +1 -1
  26. package/dist/cli/env.d.ts +3 -1
  27. package/dist/cli/env.d.ts.map +1 -1
  28. package/dist/cli/env.js +8 -3
  29. package/dist/cli/env.js.map +1 -1
  30. package/dist/cli/index.js +103 -7
  31. package/dist/cli/index.js.map +1 -1
  32. package/dist/cli/install.d.ts +4 -0
  33. package/dist/cli/install.d.ts.map +1 -1
  34. package/dist/cli/install.js +32 -37
  35. package/dist/cli/install.js.map +1 -1
  36. package/dist/cli/mcp/lib/scaffolding.d.ts.map +1 -1
  37. package/dist/cli/mcp/lib/scaffolding.js +38 -20
  38. package/dist/cli/mcp/lib/scaffolding.js.map +1 -1
  39. package/dist/cli/mcp/tools/getConnectionInfo.d.ts.map +1 -1
  40. package/dist/cli/mcp/tools/getConnectionInfo.js +4 -10
  41. package/dist/cli/mcp/tools/getConnectionInfo.js.map +1 -1
  42. package/dist/cli/mcp/tools/index.d.ts +0 -7
  43. package/dist/cli/mcp/tools/index.d.ts.map +1 -1
  44. package/dist/cli/mcp/tools/index.js +0 -2
  45. package/dist/cli/mcp/tools/index.js.map +1 -1
  46. package/dist/cli/run.d.ts +2 -0
  47. package/dist/cli/run.d.ts.map +1 -0
  48. package/dist/cli/run.js +560 -0
  49. package/dist/cli/run.js.map +1 -0
  50. package/dist/connections/cloud-auth.d.ts.map +1 -1
  51. package/dist/connections/cloud-auth.js +5 -1
  52. package/dist/connections/cloud-auth.js.map +1 -1
  53. package/dist/connections/cloud-client.d.ts.map +1 -1
  54. package/dist/connections/cloud-client.js +10 -3
  55. package/dist/connections/cloud-client.js.map +1 -1
  56. package/dist/connections/cloud-integration-provider.d.ts +2 -1
  57. package/dist/connections/cloud-integration-provider.d.ts.map +1 -1
  58. package/dist/connections/cloud-integration-provider.js +18 -2
  59. package/dist/connections/cloud-integration-provider.js.map +1 -1
  60. package/dist/connections/connection-labels.d.ts +14 -0
  61. package/dist/connections/connection-labels.d.ts.map +1 -0
  62. package/dist/connections/connection-labels.js +46 -0
  63. package/dist/connections/connection-labels.js.map +1 -0
  64. package/dist/connections/index.d.ts +1 -0
  65. package/dist/connections/index.d.ts.map +1 -1
  66. package/dist/connections/index.js +1 -0
  67. package/dist/connections/index.js.map +1 -1
  68. package/dist/connections/integration-provider.d.ts +1 -0
  69. package/dist/connections/integration-provider.d.ts.map +1 -1
  70. package/dist/connections/local-integration-provider.d.ts +2 -1
  71. package/dist/connections/local-integration-provider.d.ts.map +1 -1
  72. package/dist/connections/local-integration-provider.js +20 -3
  73. package/dist/connections/local-integration-provider.js.map +1 -1
  74. package/dist/connections/nango-client.d.ts +1 -1
  75. package/dist/connections/nango-client.d.ts.map +1 -1
  76. package/dist/connections/nango-client.js +3 -2
  77. package/dist/connections/nango-client.js.map +1 -1
  78. package/dist/connections/resolver.d.ts +4 -4
  79. package/dist/connections/resolver.d.ts.map +1 -1
  80. package/dist/connections/resolver.js +12 -8
  81. package/dist/connections/resolver.js.map +1 -1
  82. package/dist/connections/schema.d.ts +3 -1
  83. package/dist/connections/schema.d.ts.map +1 -1
  84. package/dist/connections/schema.js +10 -4
  85. package/dist/connections/schema.js.map +1 -1
  86. package/dist/dev-ui/api.d.ts +5 -0
  87. package/dist/dev-ui/api.d.ts.map +1 -1
  88. package/dist/dev-ui/api.js +39 -3
  89. package/dist/dev-ui/api.js.map +1 -1
  90. package/dist/dev-ui/dag/extractor.d.ts.map +1 -1
  91. package/dist/dev-ui/dag/extractor.js +7 -19
  92. package/dist/dev-ui/dag/extractor.js.map +1 -1
  93. package/dist/dev-ui/deploy-api.d.ts +14 -0
  94. package/dist/dev-ui/deploy-api.d.ts.map +1 -0
  95. package/dist/dev-ui/deploy-api.js +112 -0
  96. package/dist/dev-ui/deploy-api.js.map +1 -0
  97. package/dist/dev-ui/dev-server.d.ts +4 -0
  98. package/dist/dev-ui/dev-server.d.ts.map +1 -1
  99. package/dist/dev-ui/dev-server.js +140 -20
  100. package/dist/dev-ui/dev-server.js.map +1 -1
  101. package/dist/dev-ui/pty.d.ts +16 -0
  102. package/dist/dev-ui/pty.d.ts.map +1 -0
  103. package/dist/dev-ui/pty.js +87 -0
  104. package/dist/dev-ui/pty.js.map +1 -0
  105. package/dist/dev-ui/watcher.d.ts.map +1 -1
  106. package/dist/dev-ui/watcher.js +6 -2
  107. package/dist/dev-ui/watcher.js.map +1 -1
  108. package/dist/dev-ui/ws.d.ts +26 -1
  109. package/dist/dev-ui/ws.d.ts.map +1 -1
  110. package/dist/dev-ui/ws.js +14 -1
  111. package/dist/dev-ui/ws.js.map +1 -1
  112. package/dist/dev-ui-client/assets/index-D-r3q63O.js +172 -0
  113. package/dist/dev-ui-client/assets/index-Dj9V5GJF.css +32 -0
  114. package/dist/dev-ui-client/index.html +3 -3
  115. package/dist/discover.d.ts.map +1 -1
  116. package/dist/discover.js +5 -2
  117. package/dist/discover.js.map +1 -1
  118. package/dist/factory.d.ts.map +1 -1
  119. package/dist/factory.js +15 -4
  120. package/dist/factory.js.map +1 -1
  121. package/dist/index.d.ts +4 -2
  122. package/dist/index.d.ts.map +1 -1
  123. package/dist/index.js +5 -2
  124. package/dist/index.js.map +1 -1
  125. package/dist/registry-gen.d.ts +6 -0
  126. package/dist/registry-gen.d.ts.map +1 -0
  127. package/dist/registry-gen.js +146 -0
  128. package/dist/registry-gen.js.map +1 -0
  129. package/dist/types.d.ts +14 -7
  130. package/dist/types.d.ts.map +1 -1
  131. package/dist/types.js +18 -1
  132. package/dist/types.js.map +1 -1
  133. package/dist/workflow.d.ts +1 -1
  134. package/dist/workflow.d.ts.map +1 -1
  135. package/dist/workflow.js +25 -3
  136. package/dist/workflow.js.map +1 -1
  137. package/package.json +7 -1
  138. package/templates/app/.dockerignore +6 -0
  139. package/templates/app/Dockerfile +34 -0
  140. package/templates/app/generated/registry.ts +8 -0
  141. package/templates/app/next.config.js +4 -1
  142. package/templates/app/package.json +4 -0
  143. package/templates/app/src/lib/pflow.ts +5 -6
  144. package/dist/cli/init.d.ts +0 -2
  145. package/dist/cli/init.d.ts.map +0 -1
  146. package/dist/cli/init.js +0 -390
  147. package/dist/cli/init.js.map +0 -1
  148. package/dist/cli/mcp/tools/startDevUi.d.ts +0 -15
  149. package/dist/cli/mcp/tools/startDevUi.d.ts.map +0 -1
  150. package/dist/cli/mcp/tools/startDevUi.js +0 -92
  151. package/dist/cli/mcp/tools/startDevUi.js.map +0 -1
  152. package/dist/dev-ui-client/assets/index-B5wNJx8a.js +0 -77
  153. package/dist/dev-ui-client/assets/index-DpOZNNnF.css +0 -1
package/dist/workflow.js CHANGED
@@ -33,7 +33,7 @@ function createDurableContext(config) {
33
33
  `Declared integrations: [${declared}]. ` +
34
34
  `Add "${integrationId}" to the node's integrations array.`);
35
35
  }
36
- const connectionId = await resolveConnectionId(config.sql, config.workflowName, _currentNodeName, integrationId);
36
+ const connectionId = await resolveConnectionId(config.sql, config.workflowName, _currentNodeName, integrationId, config.appSchema);
37
37
  if (!connectionId) {
38
38
  throw new Error(`No connection configured for integration "${integrationId}" ` +
39
39
  `(workflow="${config.workflowName}", node="${_currentNodeName}"). ` +
@@ -56,31 +56,50 @@ function createDurableContext(config) {
56
56
  */
57
57
  const POOL_KEY = Symbol.for("opflow.getWorkflowPool()");
58
58
  const PROVIDER_KEY = Symbol.for("opflow.getWorkflowIntegrationProvider()");
59
+ const SCHEMA_KEY = Symbol.for("opflow.getWorkflowAppSchema()");
59
60
  function getWorkflowPool() {
60
61
  return globalThis[POOL_KEY] ?? null;
61
62
  }
62
63
  function getWorkflowIntegrationProvider() {
63
64
  return globalThis[PROVIDER_KEY] ?? null;
64
65
  }
66
+ function getWorkflowAppSchema() {
67
+ return globalThis[SCHEMA_KEY] ?? "public";
68
+ }
65
69
  /**
66
70
  * Configure the workflow runtime (called by factory)
67
71
  * @internal
68
72
  */
69
- export function configureWorkflowRuntime(sql, integrationProvider) {
73
+ export function configureWorkflowRuntime(sql, integrationProvider, appSchema) {
70
74
  globalThis[POOL_KEY] = sql;
71
75
  globalThis[PROVIDER_KEY] = integrationProvider;
76
+ globalThis[SCHEMA_KEY] = appSchema;
77
+ }
78
+ // Global cache for workflow executables to prevent duplicate DBOS registration
79
+ // when bundlers (Turbopack) re-evaluate the same module in multiple chunks.
80
+ const WORKFLOW_CACHE_KEY = Symbol.for("opflow.workflowCache");
81
+ function getWorkflowCache() {
82
+ const g = globalThis;
83
+ if (!g[WORKFLOW_CACHE_KEY])
84
+ g[WORKFLOW_CACHE_KEY] = new Map();
85
+ return g[WORKFLOW_CACHE_KEY];
72
86
  }
73
87
  /**
74
88
  * Factory for creating workflow executables
75
89
  */
76
90
  export const Workflow = {
77
91
  create(definition) {
92
+ // Return cached executable if already registered (bundler re-evaluation)
93
+ const cached = getWorkflowCache().get(definition.name);
94
+ if (cached)
95
+ return cached;
78
96
  // Create the DBOS-registered workflow function
79
97
  async function workflowImpl(inputs) {
80
98
  const ctx = createDurableContext({
81
99
  sql: getWorkflowPool(),
82
100
  integrationProvider: getWorkflowIntegrationProvider(),
83
101
  workflowName: definition.name,
102
+ appSchema: getWorkflowAppSchema(),
84
103
  });
85
104
  return definition.run(ctx, inputs);
86
105
  }
@@ -88,7 +107,7 @@ export const Workflow = {
88
107
  const durableWorkflow = DBOS.registerWorkflow(workflowImpl, {
89
108
  name: definition.name,
90
109
  });
91
- return {
110
+ const executable = {
92
111
  name: definition.name,
93
112
  type: "workflow",
94
113
  description: definition.description,
@@ -98,6 +117,8 @@ export const Workflow = {
98
117
  // execute ignores the ctx param and uses DBOS context instead
99
118
  execute: (_ctx, inputs) => durableWorkflow(inputs),
100
119
  };
120
+ getWorkflowCache().set(definition.name, executable);
121
+ return executable;
101
122
  },
102
123
  /**
103
124
  * Create a wrapper workflow for running a node standalone.
@@ -114,6 +135,7 @@ export const Workflow = {
114
135
  sql: getWorkflowPool(),
115
136
  integrationProvider: getWorkflowIntegrationProvider(),
116
137
  workflowName: _parentWorkflowName ?? wrapperName,
138
+ appSchema: getWorkflowAppSchema(),
117
139
  });
118
140
  _parentWorkflowName = undefined;
119
141
  return ctx.run(node, inputs);
@@ -1 +1 @@
1
- {"version":3,"file":"workflow.js","sourceRoot":"","sources":["../src/workflow.ts"],"names":[],"mappings":"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"}
1
+ {"version":3,"file":"workflow.js","sourceRoot":"","sources":["../src/workflow.ts"],"names":[],"mappings":"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"}
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "0pflow",
3
- "version": "0.1.0-dev.303bb09",
3
+ "version": "0.1.0-dev.3666a4d",
4
4
  "type": "module",
5
5
  "repository": {
6
6
  "type": "git",
@@ -43,8 +43,11 @@
43
43
  "handlebars": "^4.7.8",
44
44
  "jiti": "^2.6.1",
45
45
  "linkedom": "^0.18.12",
46
+ "node-pty": "1.2.0-beta.11",
46
47
  "pg": "^8.17.2",
47
48
  "picocolors": "^1.1.1",
49
+ "react-markdown": "^10.1.0",
50
+ "roughjs": "^4.6.6",
48
51
  "tree-sitter-typescript": "^0.23.2",
49
52
  "web-tree-sitter": "^0.24.7",
50
53
  "ws": "^8.18.0",
@@ -58,6 +61,9 @@
58
61
  "@types/react-dom": "^19.0.0",
59
62
  "@types/ws": "^8.5.0",
60
63
  "@vitejs/plugin-react": "^4.3.0",
64
+ "@xterm/addon-fit": "^0.11.0",
65
+ "@xterm/addon-web-links": "^0.12.0",
66
+ "@xterm/xterm": "^6.0.0",
61
67
  "@xyflow/react": "^12.0.0",
62
68
  "react": "^19.0.0",
63
69
  "react-dom": "^19.0.0",
@@ -0,0 +1,6 @@
1
+ node_modules
2
+ .git
3
+ .next
4
+ dist
5
+ .dbos
6
+ package-lock.json
@@ -0,0 +1,34 @@
1
+ FROM node:22-alpine AS base
2
+
3
+ # --- Dependencies ---
4
+ FROM base AS deps
5
+ WORKDIR /app
6
+ COPY package.json ./
7
+ RUN npm install
8
+
9
+ # --- Builder ---
10
+ FROM base AS builder
11
+ WORKDIR /app
12
+ COPY --from=deps /app/node_modules ./node_modules
13
+ COPY . .
14
+ ENV NODE_ENV=production
15
+ ENV SKIP_ENV_VALIDATION=1
16
+ RUN npm run build
17
+
18
+ # --- Runner ---
19
+ FROM base AS runner
20
+ WORKDIR /app
21
+ ENV NODE_ENV=production
22
+
23
+ RUN addgroup --system --gid 1001 nodejs && adduser --system --uid 1001 nextjs
24
+
25
+ COPY --from=builder /app/public ./public
26
+ COPY --from=builder /app/specs ./specs
27
+ COPY --from=builder --chown=nextjs:nodejs /app/.next/standalone ./
28
+ COPY --from=builder --chown=nextjs:nodejs /app/.next/static ./.next/static
29
+
30
+ USER nextjs
31
+ EXPOSE 3000
32
+ ENV PORT=3000
33
+ ENV HOSTNAME="0.0.0.0"
34
+ CMD ["node", "server.js"]
@@ -0,0 +1,8 @@
1
+ // generated/registry.ts
2
+ // Auto-generated by 0pflow build - do not edit directly
3
+
4
+ export const workflows = {};
5
+
6
+ export const agents = {};
7
+
8
+ export const nodes = {};
@@ -5,6 +5,9 @@
5
5
  import "./src/env.js";
6
6
 
7
7
  /** @type {import("next").NextConfig} */
8
- const config = {};
8
+ const config = {
9
+ output: "standalone",
10
+ serverExternalPackages: ["@dbos-inc/dbos-sdk"],
11
+ };
9
12
 
10
13
  export default config;
@@ -4,6 +4,9 @@
4
4
  "private": true,
5
5
  "type": "module",
6
6
  "scripts": {
7
+ "preinstall": "[ ! -f /tmp/node_modules.lock ] || flock /tmp/node_modules.lock true",
8
+ "postinstall": "rm -rf node_modules/0pflow",
9
+ "prebuild": "0pflow build",
7
10
  "build": "next build",
8
11
  "check": "biome check .",
9
12
  "check:unsafe": "biome check --write --unsafe .",
@@ -12,6 +15,7 @@
12
15
  "db:migrate": "drizzle-kit migrate",
13
16
  "db:push": "drizzle-kit push",
14
17
  "db:studio": "drizzle-kit studio",
18
+ "predev": "0pflow build",
15
19
  "dev": "next dev --turbo",
16
20
  "0pflow": "0pflow",
17
21
  "preview": "next build && next start",
@@ -1,4 +1,5 @@
1
- import { create0pflow, discover } from "0pflow";
1
+ import { create0pflow } from "0pflow";
2
+ import { workflows, agents, nodes } from "../../generated/registry";
2
3
  import "server-only";
3
4
 
4
5
  type PflowInstance = Awaited<ReturnType<typeof create0pflow>>;
@@ -11,14 +12,12 @@ export async function getPflow(): Promise<PflowInstance> {
11
12
  if (initPromise) return initPromise;
12
13
 
13
14
  initPromise = (async () => {
14
- const discovered = await discover(process.cwd());
15
- if (discovered.warnings.length > 0) {
16
- console.warn("0pflow discovery warnings:", discovered.warnings);
17
- }
18
15
  const instance = await create0pflow({
19
16
  databaseUrl: process.env.DATABASE_URL!,
20
17
  appName: "{{app_name}}",
21
- ...discovered,
18
+ workflows,
19
+ agents,
20
+ nodes,
22
21
  });
23
22
  console.log("0pflow initialized");
24
23
  return instance;
@@ -1,2 +0,0 @@
1
- export declare function runInit(): Promise<void>;
2
- //# sourceMappingURL=init.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"init.d.ts","sourceRoot":"","sources":["../../src/cli/init.ts"],"names":[],"mappings":"AAsGA,wBAAsB,OAAO,IAAI,OAAO,CAAC,IAAI,CAAC,CAmU7C"}
package/dist/cli/init.js DELETED
@@ -1,390 +0,0 @@
1
- import { execSync, spawn } from "node:child_process";
2
- import { existsSync, readFileSync, readdirSync } from "node:fs";
3
- import { basename, join, resolve } from "node:path";
4
- import * as p from "@clack/prompts";
5
- import pc from "picocolors";
6
- import { scaffoldApp, createDatabase, setupAppSchema, } from "./mcp/lib/scaffolding.js";
7
- import { packageRoot } from "./mcp/config.js";
8
- // Monorepo root (only valid in dev mode)
9
- const monorepoRoot = join(packageRoot, "..", "..");
10
- function isDevMode() {
11
- const corePath = join(monorepoRoot, "packages", "core");
12
- return existsSync(corePath);
13
- }
14
- function isCwdEmpty() {
15
- try {
16
- const entries = readdirSync(process.cwd());
17
- // Ignore dotfiles like .git
18
- return entries.filter((e) => !e.startsWith(".")).length === 0;
19
- }
20
- catch {
21
- return true;
22
- }
23
- }
24
- function isClaudeAvailable() {
25
- try {
26
- execSync("claude --version", { stdio: "ignore" });
27
- return true;
28
- }
29
- catch {
30
- return false;
31
- }
32
- }
33
- /**
34
- * Poll Tiger Cloud until the service is ready (status != "creating").
35
- * Returns true if ready, false on timeout.
36
- */
37
- async function waitForDatabase(serviceId, timeoutMs = 5 * 60 * 1000, intervalMs = 5000) {
38
- const start = Date.now();
39
- const { execSync } = await import("node:child_process");
40
- while (Date.now() - start < timeoutMs) {
41
- try {
42
- const stdout = execSync(`tiger service get ${serviceId} -o json`, {
43
- encoding: "utf-8",
44
- stdio: ["pipe", "pipe", "pipe"],
45
- });
46
- const info = JSON.parse(stdout);
47
- if (info.status && info.status !== "creating") {
48
- return true;
49
- }
50
- }
51
- catch {
52
- // tiger CLI not available or service not found — keep trying
53
- }
54
- await new Promise((r) => setTimeout(r, intervalMs));
55
- }
56
- return false;
57
- }
58
- function isExisting0pflow() {
59
- try {
60
- const pkgPath = join(process.cwd(), "package.json");
61
- if (!existsSync(pkgPath))
62
- return false;
63
- const pkg = JSON.parse(readFileSync(pkgPath, "utf-8"));
64
- const deps = { ...pkg.dependencies, ...pkg.devDependencies };
65
- return "0pflow" in deps;
66
- }
67
- catch {
68
- return false;
69
- }
70
- }
71
- function launchClaude(cwd, { yolo = false } = {}) {
72
- const prompt = "Welcome to your 0pflow project! What workflow would you like to create? Here are some ideas:\n\n" +
73
- '- "Enrich leads from a CSV file with company data"\n' +
74
- '- "Monitor website uptime and send Slack alerts"\n' +
75
- '- "Sync Salesforce contacts to our database nightly"\n' +
76
- '- "Score and route inbound leads based on firmographics"\n\n' +
77
- "Describe what you'd like to automate and I'll help you build it with /create-workflow.";
78
- const claudeArgs = [];
79
- if (isDevMode())
80
- claudeArgs.push("--plugin-dir", monorepoRoot);
81
- if (yolo)
82
- claudeArgs.push("--dangerously-skip-permissions");
83
- claudeArgs.push(prompt);
84
- const child = spawn("claude", claudeArgs, {
85
- cwd,
86
- stdio: "inherit",
87
- });
88
- child.on("exit", (code) => process.exit(code ?? 0));
89
- }
90
- export async function runInit() {
91
- p.intro(pc.red("0pflow") + pc.dim(" — create a new project"));
92
- // ── Existing project check ──────────────────────────────────────────
93
- if (isExisting0pflow()) {
94
- const action = await p.select({
95
- message: "This directory is already an 0pflow project.",
96
- options: [
97
- { value: "claude", label: "Launch Claude Code" },
98
- { value: "claude-yolo", label: "Launch Claude Code with --dangerously-skip-permissions" },
99
- { value: "new", label: "Create a new project in a subdirectory" },
100
- ],
101
- });
102
- if (p.isCancel(action)) {
103
- p.cancel("Cancelled.");
104
- process.exit(0);
105
- }
106
- if (action === "claude" || action === "claude-yolo") {
107
- if (!isClaudeAvailable()) {
108
- p.log.error("Claude Code CLI not found. Install it from https://claude.ai/code");
109
- process.exit(1);
110
- }
111
- p.outro(pc.green("Launching Claude Code..."));
112
- launchClaude(process.cwd(), { yolo: action === "claude-yolo" });
113
- return;
114
- }
115
- // fall through to normal wizard (cwdEmpty will be false, so directory defaults to ./<name>)
116
- }
117
- const cwdEmpty = isCwdEmpty();
118
- // ── Project name ────────────────────────────────────────────────────
119
- // Default to current directory name if it looks like a valid project name
120
- const dirName = basename(process.cwd()).toLowerCase().replace(/[^a-z0-9-]/g, "-");
121
- const defaultName = cwdEmpty && /^[a-z][a-z0-9-]*$/.test(dirName) ? dirName : undefined;
122
- const projectName = await p.text({
123
- message: "Project name",
124
- ...(defaultName ? { initialValue: defaultName } : { placeholder: "my-app" }),
125
- validate(value) {
126
- if (!value)
127
- return "Project name is required";
128
- if (!/^[a-z][a-z0-9-]*$/.test(value)) {
129
- return "Must be lowercase letters, numbers, and hyphens (start with a letter)";
130
- }
131
- },
132
- });
133
- if (p.isCancel(projectName)) {
134
- p.cancel("Cancelled.");
135
- process.exit(0);
136
- }
137
- // ── Directory ───────────────────────────────────────────────────────
138
- const cwd = process.cwd();
139
- const defaultDir = cwdEmpty ? "." : `./${projectName}`;
140
- const defaultLabel = cwdEmpty
141
- ? `${cwd} (current directory)`
142
- : `./${projectName}`;
143
- const useDirectory = await p.select({
144
- message: "Where should we create it?",
145
- options: cwdEmpty
146
- ? [
147
- { value: "default", label: `Here — ${cwd}` },
148
- { value: "custom", label: "Other directory" },
149
- ]
150
- : [
151
- { value: "default", label: `./${projectName}` },
152
- { value: "custom", label: "Other directory" },
153
- ],
154
- });
155
- if (p.isCancel(useDirectory)) {
156
- p.cancel("Cancelled.");
157
- process.exit(0);
158
- }
159
- let directory;
160
- if (useDirectory === "custom") {
161
- const customDir = await p.text({
162
- message: "Directory path",
163
- placeholder: `./${projectName}`,
164
- validate(value) {
165
- if (!value)
166
- return "Directory is required";
167
- },
168
- });
169
- if (p.isCancel(customDir)) {
170
- p.cancel("Cancelled.");
171
- process.exit(0);
172
- }
173
- directory = customDir;
174
- }
175
- else {
176
- directory = defaultDir;
177
- }
178
- if (p.isCancel(directory)) {
179
- p.cancel("Cancelled.");
180
- process.exit(0);
181
- }
182
- // ── Database ────────────────────────────────────────────────────────
183
- const dbChoice = await p.select({
184
- message: "Database setup",
185
- options: [
186
- {
187
- value: "new",
188
- label: "Create new Tiger Cloud database (free)",
189
- },
190
- {
191
- value: "existing",
192
- label: "Use existing Tiger Cloud database",
193
- },
194
- ],
195
- });
196
- if (p.isCancel(dbChoice)) {
197
- p.cancel("Cancelled.");
198
- process.exit(0);
199
- }
200
- let serviceId;
201
- if (dbChoice === "existing") {
202
- // Fetch available databases from Tiger Cloud
203
- let services = [];
204
- try {
205
- const stdout = execSync("tiger service list -o json", {
206
- encoding: "utf-8",
207
- stdio: ["pipe", "pipe", "pipe"],
208
- });
209
- services = JSON.parse(stdout);
210
- }
211
- catch {
212
- // tiger CLI not available or not logged in
213
- }
214
- if (services.length > 0) {
215
- const selected = await p.select({
216
- message: "Select a database",
217
- options: [
218
- ...services.map((s) => ({
219
- value: s.service_id,
220
- label: `${s.name} (${s.service_id})`,
221
- hint: s.status,
222
- })),
223
- { value: "__manual__", label: "Enter service ID manually" },
224
- ],
225
- });
226
- if (p.isCancel(selected)) {
227
- p.cancel("Cancelled.");
228
- process.exit(0);
229
- }
230
- if (selected === "__manual__") {
231
- const sid = await p.text({
232
- message: "Tiger Cloud service ID",
233
- placeholder: "abc123def4",
234
- validate(value) {
235
- if (!value)
236
- return "Service ID is required";
237
- },
238
- });
239
- if (p.isCancel(sid)) {
240
- p.cancel("Cancelled.");
241
- process.exit(0);
242
- }
243
- serviceId = sid;
244
- }
245
- else {
246
- serviceId = selected;
247
- }
248
- }
249
- else {
250
- p.log.warn("Could not fetch databases. Enter the service ID manually.");
251
- const sid = await p.text({
252
- message: "Tiger Cloud service ID",
253
- placeholder: "abc123def4",
254
- validate(value) {
255
- if (!value)
256
- return "Service ID is required";
257
- },
258
- });
259
- if (p.isCancel(sid)) {
260
- p.cancel("Cancelled.");
261
- process.exit(0);
262
- }
263
- serviceId = sid;
264
- }
265
- }
266
- // ── Execute ─────────────────────────────────────────────────────────
267
- const s = p.spinner();
268
- // Start database creation first (async) if creating new
269
- let dbPromise;
270
- if (dbChoice === "new") {
271
- s.start("Creating Tiger Cloud database...");
272
- dbPromise = createDatabase({ name: projectName }).then((result) => {
273
- if (!result.success) {
274
- throw new Error(result.error || "Failed to create database");
275
- }
276
- return result;
277
- });
278
- }
279
- // Scaffold app (parallel with db provisioning)
280
- if (!dbPromise) {
281
- s.start("Scaffolding project...");
282
- }
283
- const scaffoldResult = await scaffoldApp({
284
- appName: projectName,
285
- directory,
286
- installDeps: false,
287
- });
288
- if (!scaffoldResult.success) {
289
- s.stop(pc.red("Failed to scaffold project"));
290
- p.log.error(scaffoldResult.message);
291
- process.exit(1);
292
- }
293
- s.stop(pc.green("Scaffolded project"));
294
- // Install dependencies
295
- const appPath = scaffoldResult.path;
296
- s.start("Installing dependencies...");
297
- try {
298
- execSync("npm install", { cwd: appPath, stdio: "pipe" });
299
- s.stop(pc.green("Installed dependencies"));
300
- }
301
- catch (err) {
302
- s.stop(pc.yellow("npm install failed (you can retry manually)"));
303
- }
304
- // Wait for database and setup schema
305
- if (dbChoice === "new" && dbPromise) {
306
- s.start("Waiting for database to be ready...");
307
- try {
308
- const dbResult = await dbPromise;
309
- serviceId = dbResult.service_id;
310
- s.stop(pc.green(`Database created (${serviceId})`));
311
- }
312
- catch (err) {
313
- s.stop(pc.yellow("Database creation failed"));
314
- p.log.warn(`You can create one later with: tiger service create --name ${projectName}`);
315
- }
316
- }
317
- if (serviceId) {
318
- s.start("Configuring database schema...");
319
- const schemaName = projectName.replace(/-/g, "_");
320
- const schemaResult = await setupAppSchema({
321
- directory: appPath,
322
- serviceId,
323
- appName: schemaName,
324
- });
325
- if (schemaResult.success) {
326
- s.stop(pc.green("Database schema configured"));
327
- }
328
- else {
329
- // Database might still be provisioning — try waiting
330
- if (dbChoice === "new") {
331
- s.message("Database still provisioning, waiting...");
332
- const ready = await waitForDatabase(serviceId);
333
- if (ready) {
334
- const retry = await setupAppSchema({
335
- directory: appPath,
336
- serviceId,
337
- appName: schemaName,
338
- });
339
- if (retry.success) {
340
- s.stop(pc.green("Database schema configured"));
341
- }
342
- else {
343
- s.stop(pc.yellow("Schema setup failed"));
344
- p.log.warn(retry.message);
345
- }
346
- }
347
- else {
348
- s.stop(pc.yellow("Database not ready yet"));
349
- p.log.warn(`Run later: 0pflow init won't retry. Use the MCP tools or set up manually.`);
350
- }
351
- }
352
- else {
353
- s.stop(pc.yellow("Schema setup failed"));
354
- p.log.warn(schemaResult.message);
355
- }
356
- }
357
- }
358
- // ── Launch Claude? ──────────────────────────────────────────────────
359
- const hasClaude = isClaudeAvailable();
360
- if (hasClaude) {
361
- const launchChoice = await p.select({
362
- message: "Launch Claude Code to design your first workflow?",
363
- options: [
364
- { value: "claude", label: "Yes" },
365
- { value: "claude-yolo", label: "Yes, with --dangerously-skip-permissions" },
366
- { value: "no", label: "No, I'll do it later" },
367
- ],
368
- });
369
- if (!p.isCancel(launchChoice) && launchChoice !== "no") {
370
- p.outro(pc.green("Launching Claude Code..."));
371
- launchClaude(resolve(appPath), { yolo: launchChoice === "claude-yolo" });
372
- return;
373
- }
374
- }
375
- // ── Done ────────────────────────────────────────────────────────────
376
- const cdCmd = directory === "." ? "" : `cd ${directory} && `;
377
- const pluginFlag = isDevMode() ? ` --plugin-dir ${monorepoRoot}` : "";
378
- const claudeCmd = `${cdCmd}claude${pluginFlag}`;
379
- p.outro(pc.green("Project created!"));
380
- console.log();
381
- console.log(pc.bold(" To launch Claude Code later:"));
382
- console.log(pc.cyan(` ${claudeCmd}`));
383
- console.log();
384
- console.log(pc.bold(" Example prompts to get started:"));
385
- console.log(pc.dim(' "Create a workflow that enriches leads from a CSV file"'));
386
- console.log(pc.dim(' "Build a workflow that monitors website uptime and sends Slack alerts"'));
387
- console.log(pc.dim(' "Create a data pipeline that syncs Salesforce contacts to our database"'));
388
- console.log();
389
- }
390
- //# sourceMappingURL=init.js.map