@perstack/runtime 0.0.66 → 0.0.67

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
package/README.md CHANGED
@@ -116,7 +116,7 @@ All managers extend `BaseSkillManager` which provides:
116
116
  - `getToolDefinitions()` — Get available tools
117
117
  - `callTool()` — Execute a tool call
118
118
 
119
- **Note:** Interactive skills are only available to the Coordinator Expert. See [Experts documentation](https://docs.perstack.ai/understanding-perstack/experts#why-no-interactive-tools-for-delegates) for details.
119
+ **Note:** Interactive skills are only available to the Coordinator Expert. See [Experts documentation](https://github.com/perstack-ai/perstack/blob/main/docs/understanding-perstack/experts.md#why-no-interactive-tools-for-delegates) for details.
120
120
 
121
121
  ### Initialization Flow
122
122
 
@@ -202,7 +202,7 @@ Job (jobId)
202
202
  | **Run** | Single Expert execution. |
203
203
  | **Checkpoint** | Snapshot at step end. Enables pause/resume. |
204
204
 
205
- For details on step counting, Coordinator vs. Delegated Expert differences, and the full execution model, see [Runtime](https://docs.perstack.ai/understanding-perstack/runtime).
205
+ For details on step counting, Coordinator vs. Delegated Expert differences, and the full execution model, see [Runtime](https://github.com/perstack-ai/perstack/blob/main/docs/understanding-perstack/runtime.md).
206
206
 
207
207
  ### Events, Steps, Checkpoints
208
208
 
@@ -290,18 +290,19 @@ The `status` field in a Checkpoint indicates the current state:
290
290
  - `stoppedByInteractiveTool`, `stoppedByDelegate` — Waiting for external input
291
291
  - `stoppedByExceededMaxSteps`, `stoppedByError` — Run stopped
292
292
 
293
- For stop reasons and error handling, see [Error Handling](https://docs.perstack.ai/using-experts/error-handling).
293
+ For stop reasons and error handling, see [Error Handling](https://github.com/perstack-ai/perstack/blob/main/docs/using-experts/error-handling.md).
294
294
 
295
295
  ## Runtime Adapters
296
296
 
297
297
  The runtime supports multiple execution backends through the adapter pattern. External runtime adapters are provided as separate packages:
298
298
 
299
- | Package | Runtime Name | Description |
300
- | ----------------------- | ------------- | -------------------------- |
301
- | `@perstack/runtime` | `perstack` | Built-in runtime (default) |
302
- | `@perstack/cursor` | `cursor` | Cursor IDE headless mode |
303
- | `@perstack/claude-code` | `claude-code` | Claude Code CLI |
304
- | `@perstack/gemini` | `gemini` | Gemini CLI |
299
+ | Package | Runtime Name | Description |
300
+ | ----------------------- | ------------- | ---------------------------------- |
301
+ | `@perstack/docker` | `docker` | Docker containerized (default) |
302
+ | `@perstack/runtime` | `local` | Built-in runtime without isolation |
303
+ | `@perstack/cursor` | `cursor` | Cursor IDE headless mode |
304
+ | `@perstack/claude-code` | `claude-code` | Claude Code CLI |
305
+ | `@perstack/gemini` | `gemini` | Gemini CLI |
305
306
 
306
307
  ### Registration Pattern
307
308
 
@@ -347,11 +348,11 @@ class MyAdapter extends BaseAdapter {
347
348
  }
348
349
  ```
349
350
 
350
- See [Multi-Runtime Support](https://docs.perstack.ai/using-experts/multi-runtime) for details.
351
+ See [Multi-Runtime Support](https://github.com/perstack-ai/perstack/blob/main/docs/using-experts/multi-runtime.md) for details.
351
352
 
352
353
  ## Related Documentation
353
354
 
354
- - [Runtime](https://docs.perstack.ai/understanding-perstack/runtime) — Full execution model
355
- - [State Management](https://docs.perstack.ai/using-experts/state-management) — Jobs, Runs, and Checkpoints
356
- - [Running Experts](https://docs.perstack.ai/using-experts/running-experts) — CLI usage
357
- - [Multi-Runtime](https://docs.perstack.ai/using-experts/multi-runtime) — Multi-runtime support
355
+ - [Runtime](https://github.com/perstack-ai/perstack/blob/main/docs/understanding-perstack/runtime.md) — Full execution model
356
+ - [State Management](https://github.com/perstack-ai/perstack/blob/main/docs/using-experts/state-management.md) — Jobs, Runs, and Checkpoints
357
+ - [Running Experts](https://github.com/perstack-ai/perstack/blob/main/docs/using-experts/running-experts.md) — CLI usage
358
+ - [Multi-Runtime](https://github.com/perstack-ai/perstack/blob/main/docs/using-experts/multi-runtime.md) — Multi-runtime support
package/dist/bin/cli.js CHANGED
@@ -1,5 +1,5 @@
1
1
  #!/usr/bin/env node
2
- import { package_default, run } from '../chunk-LMD6DWPY.js';
2
+ import { package_default, run } from '../chunk-AQSRQW5R.js';
3
3
  import { parseWithFriendlyError, runCommandInputSchema, perstackConfigSchema } from '@perstack/core';
4
4
  import { Command } from 'commander';
5
5
  import dotenv from 'dotenv';
@@ -14,6 +14,7 @@ function getEnv(envPath) {
14
14
  dotenv.config({ path: envPath, processEnv: env, quiet: true });
15
15
  return env;
16
16
  }
17
+ var ALLOWED_CONFIG_HOSTS = ["raw.githubusercontent.com"];
17
18
  async function getPerstackConfig(configPath) {
18
19
  const configString = await findPerstackConfigString(configPath);
19
20
  if (configString === null) {
@@ -21,8 +22,39 @@ async function getPerstackConfig(configPath) {
21
22
  }
22
23
  return await parsePerstackConfig(configString);
23
24
  }
25
+ function isRemoteUrl(configPath) {
26
+ const lower = configPath.toLowerCase();
27
+ return lower.startsWith("https://") || lower.startsWith("http://");
28
+ }
29
+ async function fetchRemoteConfig(url) {
30
+ let parsed;
31
+ try {
32
+ parsed = new URL(url);
33
+ } catch {
34
+ throw new Error(`Invalid remote config URL: ${url}`);
35
+ }
36
+ if (parsed.protocol !== "https:") {
37
+ throw new Error("Remote config requires HTTPS");
38
+ }
39
+ if (!ALLOWED_CONFIG_HOSTS.includes(parsed.hostname)) {
40
+ throw new Error(`Remote config only allowed from: ${ALLOWED_CONFIG_HOSTS.join(", ")}`);
41
+ }
42
+ try {
43
+ const response = await fetch(url, { redirect: "error" });
44
+ if (!response.ok) {
45
+ throw new Error(`${response.status} ${response.statusText}`);
46
+ }
47
+ return await response.text();
48
+ } catch (error) {
49
+ const message = error instanceof Error ? error.message : String(error);
50
+ throw new Error(`Failed to fetch remote config: ${message}`);
51
+ }
52
+ }
24
53
  async function findPerstackConfigString(configPath) {
25
54
  if (configPath) {
55
+ if (isRemoteUrl(configPath)) {
56
+ return await fetchRemoteConfig(configPath);
57
+ }
26
58
  try {
27
59
  const tomlString = await readFile(path.resolve(process.cwd(), configPath), "utf-8");
28
60
  return tomlString;
@@ -201,7 +233,12 @@ program.command("run").description("Run an Expert with JSON event output").argum
201
233
  ).option("--max-retries <maxRetries>", "Maximum number of generation retries, default is 5").option(
202
234
  "--timeout <timeout>",
203
235
  "Timeout for each generation in milliseconds, default is 60000 (1 minute)"
204
- ).option("--job-id <jobId>", "Job ID for identifying the job").option("--run-id <runId>", "Run ID for identifying the run").option("--env-path <envPath...>", "Path to the environment file, default is .env and .env.local").option("--verbose", "Enable verbose logging").action(async (expertKey, query, options) => {
236
+ ).option("--job-id <jobId>", "Job ID for identifying the job").option("--run-id <runId>", "Run ID for identifying the run").option(
237
+ "--env-path <path>",
238
+ "Path to the environment file (can be specified multiple times), default is .env and .env.local",
239
+ (value, previous) => previous.concat(value),
240
+ []
241
+ ).option("--verbose", "Enable verbose logging").action(async (expertKey, query, options) => {
205
242
  const input = parseWithFriendlyError(runCommandInputSchema, { expertKey, query, options });
206
243
  try {
207
244
  const { perstackConfig, env, providerConfig, model, experts } = await resolveRunContext({
@@ -228,7 +265,8 @@ program.command("run").description("Run an Expert with JSON event output").argum
228
265
  perstackApiKey: env.PERSTACK_API_KEY,
229
266
  perstackBaseSkillCommand: perstackConfig.perstackBaseSkillCommand,
230
267
  env,
231
- proxyUrl: process.env.PERSTACK_PROXY_URL
268
+ proxyUrl: process.env.PERSTACK_PROXY_URL,
269
+ verbose: input.options.verbose
232
270
  }
233
271
  },
234
272
  { eventListener: defaultEventListener, storeCheckpoint, retrieveCheckpoint }
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/cli/get-env.ts","../../src/cli/perstack-toml.ts","../../src/cli/provider-config.ts","../../src/cli/context.ts","../../bin/cli.ts"],"names":["parseWithFriendlyError"],"mappings":"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","file":"cli.js","sourcesContent":["import dotenv from \"dotenv\"\n\nexport function getEnv(envPath: string[]): Record<string, string> {\n const env: Record<string, string> = Object.fromEntries(\n Object.entries(process.env)\n .filter(([_, value]) => !!value)\n .map(([key, value]) => [key, value as string]),\n )\n dotenv.config({ path: envPath, processEnv: env, quiet: true })\n return env\n}\n","import { readFile } from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { type PerstackConfig, parseWithFriendlyError, perstackConfigSchema } from \"@perstack/core\"\nimport TOML from \"smol-toml\"\n\nexport async function getPerstackConfig(configPath?: string): Promise<PerstackConfig> {\n const configString = await findPerstackConfigString(configPath)\n if (configString === null) {\n throw new Error(\"perstack.toml not found. Create one or specify --config path.\")\n }\n return await parsePerstackConfig(configString)\n}\n\nasync function findPerstackConfigString(configPath?: string): Promise<string | null> {\n if (configPath) {\n try {\n const tomlString = await readFile(path.resolve(process.cwd(), configPath), \"utf-8\")\n return tomlString\n } catch {\n throw new Error(`Given config path \"${configPath}\" is not found`)\n }\n }\n return await findPerstackConfigStringRecursively(path.resolve(process.cwd()))\n}\n\nasync function findPerstackConfigStringRecursively(cwd: string): Promise<string | null> {\n try {\n const tomlString = await readFile(path.resolve(cwd, \"perstack.toml\"), \"utf-8\")\n return tomlString\n } catch {\n if (cwd === path.parse(cwd).root) {\n return null\n }\n return await findPerstackConfigStringRecursively(path.dirname(cwd))\n }\n}\n\nasync function parsePerstackConfig(config: string): Promise<PerstackConfig> {\n const toml = TOML.parse(config ?? \"\")\n return parseWithFriendlyError(perstackConfigSchema, toml, \"perstack.toml\")\n}\n","import type { ProviderConfig, ProviderName, ProviderTable } from \"@perstack/core\"\n\ntype SettingRecord = Record<string, unknown>\n\nexport function getProviderConfig(\n provider: ProviderName,\n env: Record<string, string>,\n providerTable?: ProviderTable,\n): ProviderConfig {\n const setting = (providerTable?.setting ?? {}) as SettingRecord\n switch (provider) {\n case \"anthropic\": {\n const apiKey = env.ANTHROPIC_API_KEY\n if (!apiKey) throw new Error(\"ANTHROPIC_API_KEY is not set\")\n return {\n providerName: \"anthropic\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.ANTHROPIC_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"google\": {\n const apiKey = env.GOOGLE_GENERATIVE_AI_API_KEY\n if (!apiKey) throw new Error(\"GOOGLE_GENERATIVE_AI_API_KEY is not set\")\n return {\n providerName: \"google\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.GOOGLE_GENERATIVE_AI_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"openai\": {\n const apiKey = env.OPENAI_API_KEY\n if (!apiKey) throw new Error(\"OPENAI_API_KEY is not set\")\n return {\n providerName: \"openai\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.OPENAI_BASE_URL,\n organization: (setting.organization as string | undefined) ?? env.OPENAI_ORGANIZATION,\n project: (setting.project as string | undefined) ?? env.OPENAI_PROJECT,\n name: setting.name as string | undefined,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"ollama\": {\n return {\n providerName: \"ollama\",\n baseUrl: (setting.baseUrl as string | undefined) ?? env.OLLAMA_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"azure-openai\": {\n const apiKey = env.AZURE_API_KEY\n if (!apiKey) throw new Error(\"AZURE_API_KEY is not set\")\n const resourceName = (setting.resourceName as string | undefined) ?? env.AZURE_RESOURCE_NAME\n const baseUrl = (setting.baseUrl as string | undefined) ?? env.AZURE_BASE_URL\n if (!resourceName && !baseUrl) throw new Error(\"AZURE_RESOURCE_NAME or baseUrl is not set\")\n return {\n providerName: \"azure-openai\",\n apiKey,\n resourceName,\n apiVersion: (setting.apiVersion as string | undefined) ?? env.AZURE_API_VERSION,\n baseUrl,\n headers: setting.headers as Record<string, string> | undefined,\n useDeploymentBasedUrls: setting.useDeploymentBasedUrls as boolean | undefined,\n }\n }\n case \"amazon-bedrock\": {\n const accessKeyId = env.AWS_ACCESS_KEY_ID\n const secretAccessKey = env.AWS_SECRET_ACCESS_KEY\n const sessionToken = env.AWS_SESSION_TOKEN\n if (!accessKeyId) throw new Error(\"AWS_ACCESS_KEY_ID is not set\")\n if (!secretAccessKey) throw new Error(\"AWS_SECRET_ACCESS_KEY is not set\")\n const region = (setting.region as string | undefined) ?? env.AWS_REGION\n if (!region) throw new Error(\"AWS_REGION is not set\")\n return {\n providerName: \"amazon-bedrock\",\n accessKeyId,\n secretAccessKey,\n region,\n sessionToken,\n }\n }\n case \"google-vertex\": {\n return {\n providerName: \"google-vertex\",\n project: (setting.project as string | undefined) ?? env.GOOGLE_VERTEX_PROJECT,\n location: (setting.location as string | undefined) ?? env.GOOGLE_VERTEX_LOCATION,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.GOOGLE_VERTEX_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"deepseek\": {\n const apiKey = env.DEEPSEEK_API_KEY\n if (!apiKey) throw new Error(\"DEEPSEEK_API_KEY is not set\")\n return {\n providerName: \"deepseek\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.DEEPSEEK_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n }\n}\n","import type { PerstackConfig, ProviderConfig, ProviderName } from \"@perstack/core\"\nimport { getEnv } from \"./get-env.js\"\nimport { getPerstackConfig } from \"./perstack-toml.js\"\nimport { getProviderConfig } from \"./provider-config.js\"\n\nconst defaultProvider: ProviderName = \"anthropic\"\nconst defaultModel = \"claude-sonnet-4-5\"\n\nexport type ExpertConfig = NonNullable<PerstackConfig[\"experts\"]>[string]\n\nexport type RunContext = {\n perstackConfig: PerstackConfig\n env: Record<string, string>\n providerConfig: ProviderConfig\n model: string\n experts: Record<string, ExpertConfig & { key: string; name: string; version: string }>\n}\n\nexport type ResolveRunContextInput = {\n configPath?: string\n provider?: string\n model?: string\n envPath?: string[]\n}\n\nexport async function resolveRunContext(input: ResolveRunContextInput): Promise<RunContext> {\n const perstackConfig = await getPerstackConfig(input.configPath)\n const env = getEnv(input.envPath ?? perstackConfig.envPath ?? [\".env\", \".env.local\"])\n const provider = (input.provider ??\n perstackConfig.provider?.providerName ??\n defaultProvider) as ProviderName\n const model = input.model ?? perstackConfig.model ?? defaultModel\n const providerConfig = getProviderConfig(provider, env, perstackConfig.provider)\n const experts = Object.fromEntries(\n Object.entries(perstackConfig.experts ?? {}).map(([name, expert]) => {\n return [\n name,\n {\n key: name,\n name,\n version: expert.version ?? \"1.0.0\",\n description: expert.description,\n instruction: expert.instruction,\n skills: expert.skills ?? {},\n delegates: expert.delegates ?? [],\n tags: expert.tags ?? [],\n },\n ]\n }),\n )\n return {\n perstackConfig,\n env,\n providerConfig,\n model,\n experts: experts as Record<\n string,\n ExpertConfig & { key: string; name: string; version: string }\n >,\n }\n}\n","#!/usr/bin/env node\n\nimport type { Checkpoint, RunEvent, RuntimeEvent } from \"@perstack/core\"\nimport { parseWithFriendlyError, runCommandInputSchema } from \"@perstack/core\"\nimport { Command } from \"commander\"\nimport pkg from \"../package.json\" with { type: \"json\" }\nimport { resolveRunContext } from \"../src/cli/context.js\"\nimport { run } from \"../src/run.js\"\n\nconst defaultEventListener = (event: RunEvent | RuntimeEvent) => console.log(JSON.stringify(event))\n\nconst checkpointStore = new Map<string, Checkpoint>()\nconst storeCheckpoint = async (checkpoint: Checkpoint) => {\n checkpointStore.set(checkpoint.id, checkpoint)\n}\nconst retrieveCheckpoint = async (_jobId: string, checkpointId: string) => {\n const checkpoint = checkpointStore.get(checkpointId)\n if (!checkpoint) {\n throw new Error(`Checkpoint not found: ${checkpointId}`)\n }\n return checkpoint\n}\n\nconst program = new Command()\n .name(\"perstack-runtime\")\n .description(\"Perstack Runtime CLI - Execute Experts directly\")\n .version(pkg.version)\n\nprogram\n .command(\"run\")\n .description(\"Run an Expert with JSON event output\")\n .argument(\"<expertKey>\", \"Expert key to run\")\n .argument(\"<query>\", \"Query to run\")\n .option(\"--config <configPath>\", \"Path to perstack.toml config file\")\n .option(\"--provider <provider>\", \"Provider to use\")\n .option(\"--model <model>\", \"Model to use\")\n .option(\"--temperature <temperature>\", \"Temperature for the model, default is 0.3\")\n .option(\n \"--max-steps <maxSteps>\",\n \"Maximum number of steps to run, default is undefined (no limit)\",\n )\n .option(\"--max-retries <maxRetries>\", \"Maximum number of generation retries, default is 5\")\n .option(\n \"--timeout <timeout>\",\n \"Timeout for each generation in milliseconds, default is 60000 (1 minute)\",\n )\n .option(\"--job-id <jobId>\", \"Job ID for identifying the job\")\n .option(\"--run-id <runId>\", \"Run ID for identifying the run\")\n .option(\"--env-path <envPath...>\", \"Path to the environment file, default is .env and .env.local\")\n .option(\"--verbose\", \"Enable verbose logging\")\n .action(async (expertKey, query, options) => {\n const input = parseWithFriendlyError(runCommandInputSchema, { expertKey, query, options })\n try {\n const { perstackConfig, env, providerConfig, model, experts } = await resolveRunContext({\n configPath: input.options.config,\n provider: input.options.provider,\n model: input.options.model,\n envPath: input.options.envPath,\n })\n await run(\n {\n setting: {\n jobId: input.options.jobId,\n runId: input.options.runId,\n expertKey: input.expertKey,\n input: { text: input.query },\n experts,\n model,\n providerConfig,\n temperature: input.options.temperature ?? perstackConfig.temperature,\n maxSteps: input.options.maxSteps ?? perstackConfig.maxSteps,\n maxRetries: input.options.maxRetries ?? perstackConfig.maxRetries,\n timeout: input.options.timeout ?? perstackConfig.timeout,\n perstackApiBaseUrl: perstackConfig.perstackApiBaseUrl,\n perstackApiKey: env.PERSTACK_API_KEY,\n perstackBaseSkillCommand: perstackConfig.perstackBaseSkillCommand,\n env,\n proxyUrl: process.env.PERSTACK_PROXY_URL,\n },\n },\n { eventListener: defaultEventListener, storeCheckpoint, retrieveCheckpoint },\n )\n } catch (error) {\n if (error instanceof Error) {\n console.error(error.message)\n } else {\n console.error(error)\n }\n process.exit(1)\n }\n })\n\nprogram.parse()\n"]}
1
+ {"version":3,"sources":["../../src/cli/get-env.ts","../../src/cli/perstack-toml.ts","../../src/cli/provider-config.ts","../../src/cli/context.ts","../../bin/cli.ts"],"names":["parseWithFriendlyError"],"mappings":"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","file":"cli.js","sourcesContent":["import dotenv from \"dotenv\"\n\nexport function getEnv(envPath: string[]): Record<string, string> {\n const env: Record<string, string> = Object.fromEntries(\n Object.entries(process.env)\n .filter(([_, value]) => !!value)\n .map(([key, value]) => [key, value as string]),\n )\n dotenv.config({ path: envPath, processEnv: env, quiet: true })\n return env\n}\n","import { readFile } from \"node:fs/promises\"\nimport path from \"node:path\"\nimport { type PerstackConfig, parseWithFriendlyError, perstackConfigSchema } from \"@perstack/core\"\nimport TOML from \"smol-toml\"\n\nconst ALLOWED_CONFIG_HOSTS = [\"raw.githubusercontent.com\"]\n\nexport async function getPerstackConfig(configPath?: string): Promise<PerstackConfig> {\n const configString = await findPerstackConfigString(configPath)\n if (configString === null) {\n throw new Error(\"perstack.toml not found. Create one or specify --config path.\")\n }\n return await parsePerstackConfig(configString)\n}\n\nfunction isRemoteUrl(configPath: string): boolean {\n const lower = configPath.toLowerCase()\n return lower.startsWith(\"https://\") || lower.startsWith(\"http://\")\n}\n\nasync function fetchRemoteConfig(url: string): Promise<string> {\n let parsed: URL\n try {\n parsed = new URL(url)\n } catch {\n throw new Error(`Invalid remote config URL: ${url}`)\n }\n if (parsed.protocol !== \"https:\") {\n throw new Error(\"Remote config requires HTTPS\")\n }\n if (!ALLOWED_CONFIG_HOSTS.includes(parsed.hostname)) {\n throw new Error(`Remote config only allowed from: ${ALLOWED_CONFIG_HOSTS.join(\", \")}`)\n }\n try {\n const response = await fetch(url, { redirect: \"error\" })\n if (!response.ok) {\n throw new Error(`${response.status} ${response.statusText}`)\n }\n return await response.text()\n } catch (error) {\n const message = error instanceof Error ? error.message : String(error)\n throw new Error(`Failed to fetch remote config: ${message}`)\n }\n}\n\nasync function findPerstackConfigString(configPath?: string): Promise<string | null> {\n if (configPath) {\n if (isRemoteUrl(configPath)) {\n return await fetchRemoteConfig(configPath)\n }\n try {\n const tomlString = await readFile(path.resolve(process.cwd(), configPath), \"utf-8\")\n return tomlString\n } catch {\n throw new Error(`Given config path \"${configPath}\" is not found`)\n }\n }\n return await findPerstackConfigStringRecursively(path.resolve(process.cwd()))\n}\n\nasync function findPerstackConfigStringRecursively(cwd: string): Promise<string | null> {\n try {\n const tomlString = await readFile(path.resolve(cwd, \"perstack.toml\"), \"utf-8\")\n return tomlString\n } catch {\n if (cwd === path.parse(cwd).root) {\n return null\n }\n return await findPerstackConfigStringRecursively(path.dirname(cwd))\n }\n}\n\nasync function parsePerstackConfig(config: string): Promise<PerstackConfig> {\n const toml = TOML.parse(config ?? \"\")\n return parseWithFriendlyError(perstackConfigSchema, toml, \"perstack.toml\")\n}\n","import type { ProviderConfig, ProviderName, ProviderTable } from \"@perstack/core\"\n\ntype SettingRecord = Record<string, unknown>\n\nexport function getProviderConfig(\n provider: ProviderName,\n env: Record<string, string>,\n providerTable?: ProviderTable,\n): ProviderConfig {\n const setting = (providerTable?.setting ?? {}) as SettingRecord\n switch (provider) {\n case \"anthropic\": {\n const apiKey = env.ANTHROPIC_API_KEY\n if (!apiKey) throw new Error(\"ANTHROPIC_API_KEY is not set\")\n return {\n providerName: \"anthropic\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.ANTHROPIC_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"google\": {\n const apiKey = env.GOOGLE_GENERATIVE_AI_API_KEY\n if (!apiKey) throw new Error(\"GOOGLE_GENERATIVE_AI_API_KEY is not set\")\n return {\n providerName: \"google\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.GOOGLE_GENERATIVE_AI_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"openai\": {\n const apiKey = env.OPENAI_API_KEY\n if (!apiKey) throw new Error(\"OPENAI_API_KEY is not set\")\n return {\n providerName: \"openai\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.OPENAI_BASE_URL,\n organization: (setting.organization as string | undefined) ?? env.OPENAI_ORGANIZATION,\n project: (setting.project as string | undefined) ?? env.OPENAI_PROJECT,\n name: setting.name as string | undefined,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"ollama\": {\n return {\n providerName: \"ollama\",\n baseUrl: (setting.baseUrl as string | undefined) ?? env.OLLAMA_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"azure-openai\": {\n const apiKey = env.AZURE_API_KEY\n if (!apiKey) throw new Error(\"AZURE_API_KEY is not set\")\n const resourceName = (setting.resourceName as string | undefined) ?? env.AZURE_RESOURCE_NAME\n const baseUrl = (setting.baseUrl as string | undefined) ?? env.AZURE_BASE_URL\n if (!resourceName && !baseUrl) throw new Error(\"AZURE_RESOURCE_NAME or baseUrl is not set\")\n return {\n providerName: \"azure-openai\",\n apiKey,\n resourceName,\n apiVersion: (setting.apiVersion as string | undefined) ?? env.AZURE_API_VERSION,\n baseUrl,\n headers: setting.headers as Record<string, string> | undefined,\n useDeploymentBasedUrls: setting.useDeploymentBasedUrls as boolean | undefined,\n }\n }\n case \"amazon-bedrock\": {\n const accessKeyId = env.AWS_ACCESS_KEY_ID\n const secretAccessKey = env.AWS_SECRET_ACCESS_KEY\n const sessionToken = env.AWS_SESSION_TOKEN\n if (!accessKeyId) throw new Error(\"AWS_ACCESS_KEY_ID is not set\")\n if (!secretAccessKey) throw new Error(\"AWS_SECRET_ACCESS_KEY is not set\")\n const region = (setting.region as string | undefined) ?? env.AWS_REGION\n if (!region) throw new Error(\"AWS_REGION is not set\")\n return {\n providerName: \"amazon-bedrock\",\n accessKeyId,\n secretAccessKey,\n region,\n sessionToken,\n }\n }\n case \"google-vertex\": {\n return {\n providerName: \"google-vertex\",\n project: (setting.project as string | undefined) ?? env.GOOGLE_VERTEX_PROJECT,\n location: (setting.location as string | undefined) ?? env.GOOGLE_VERTEX_LOCATION,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.GOOGLE_VERTEX_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n case \"deepseek\": {\n const apiKey = env.DEEPSEEK_API_KEY\n if (!apiKey) throw new Error(\"DEEPSEEK_API_KEY is not set\")\n return {\n providerName: \"deepseek\",\n apiKey,\n baseUrl: (setting.baseUrl as string | undefined) ?? env.DEEPSEEK_BASE_URL,\n headers: setting.headers as Record<string, string> | undefined,\n }\n }\n }\n}\n","import type { PerstackConfig, ProviderConfig, ProviderName } from \"@perstack/core\"\nimport { getEnv } from \"./get-env.js\"\nimport { getPerstackConfig } from \"./perstack-toml.js\"\nimport { getProviderConfig } from \"./provider-config.js\"\n\nconst defaultProvider: ProviderName = \"anthropic\"\nconst defaultModel = \"claude-sonnet-4-5\"\n\nexport type ExpertConfig = NonNullable<PerstackConfig[\"experts\"]>[string]\n\nexport type RunContext = {\n perstackConfig: PerstackConfig\n env: Record<string, string>\n providerConfig: ProviderConfig\n model: string\n experts: Record<string, ExpertConfig & { key: string; name: string; version: string }>\n}\n\nexport type ResolveRunContextInput = {\n configPath?: string\n provider?: string\n model?: string\n envPath?: string[]\n}\n\nexport async function resolveRunContext(input: ResolveRunContextInput): Promise<RunContext> {\n const perstackConfig = await getPerstackConfig(input.configPath)\n const env = getEnv(input.envPath ?? perstackConfig.envPath ?? [\".env\", \".env.local\"])\n const provider = (input.provider ??\n perstackConfig.provider?.providerName ??\n defaultProvider) as ProviderName\n const model = input.model ?? perstackConfig.model ?? defaultModel\n const providerConfig = getProviderConfig(provider, env, perstackConfig.provider)\n const experts = Object.fromEntries(\n Object.entries(perstackConfig.experts ?? {}).map(([name, expert]) => {\n return [\n name,\n {\n key: name,\n name,\n version: expert.version ?? \"1.0.0\",\n description: expert.description,\n instruction: expert.instruction,\n skills: expert.skills ?? {},\n delegates: expert.delegates ?? [],\n tags: expert.tags ?? [],\n },\n ]\n }),\n )\n return {\n perstackConfig,\n env,\n providerConfig,\n model,\n experts: experts as Record<\n string,\n ExpertConfig & { key: string; name: string; version: string }\n >,\n }\n}\n","#!/usr/bin/env node\n\nimport type { Checkpoint, RunEvent, RuntimeEvent } from \"@perstack/core\"\nimport { parseWithFriendlyError, runCommandInputSchema } from \"@perstack/core\"\nimport { Command } from \"commander\"\nimport pkg from \"../package.json\" with { type: \"json\" }\nimport { resolveRunContext } from \"../src/cli/context.js\"\nimport { run } from \"../src/run.js\"\n\nconst defaultEventListener = (event: RunEvent | RuntimeEvent) => console.log(JSON.stringify(event))\n\nconst checkpointStore = new Map<string, Checkpoint>()\nconst storeCheckpoint = async (checkpoint: Checkpoint) => {\n checkpointStore.set(checkpoint.id, checkpoint)\n}\nconst retrieveCheckpoint = async (_jobId: string, checkpointId: string) => {\n const checkpoint = checkpointStore.get(checkpointId)\n if (!checkpoint) {\n throw new Error(`Checkpoint not found: ${checkpointId}`)\n }\n return checkpoint\n}\n\nconst program = new Command()\n .name(\"perstack-runtime\")\n .description(\"Perstack Runtime CLI - Execute Experts directly\")\n .version(pkg.version)\n\nprogram\n .command(\"run\")\n .description(\"Run an Expert with JSON event output\")\n .argument(\"<expertKey>\", \"Expert key to run\")\n .argument(\"<query>\", \"Query to run\")\n .option(\"--config <configPath>\", \"Path to perstack.toml config file\")\n .option(\"--provider <provider>\", \"Provider to use\")\n .option(\"--model <model>\", \"Model to use\")\n .option(\"--temperature <temperature>\", \"Temperature for the model, default is 0.3\")\n .option(\n \"--max-steps <maxSteps>\",\n \"Maximum number of steps to run, default is undefined (no limit)\",\n )\n .option(\"--max-retries <maxRetries>\", \"Maximum number of generation retries, default is 5\")\n .option(\n \"--timeout <timeout>\",\n \"Timeout for each generation in milliseconds, default is 60000 (1 minute)\",\n )\n .option(\"--job-id <jobId>\", \"Job ID for identifying the job\")\n .option(\"--run-id <runId>\", \"Run ID for identifying the run\")\n .option(\n \"--env-path <path>\",\n \"Path to the environment file (can be specified multiple times), default is .env and .env.local\",\n (value: string, previous: string[]) => previous.concat(value),\n [] as string[],\n )\n .option(\"--verbose\", \"Enable verbose logging\")\n .action(async (expertKey, query, options) => {\n const input = parseWithFriendlyError(runCommandInputSchema, { expertKey, query, options })\n try {\n const { perstackConfig, env, providerConfig, model, experts } = await resolveRunContext({\n configPath: input.options.config,\n provider: input.options.provider,\n model: input.options.model,\n envPath: input.options.envPath,\n })\n await run(\n {\n setting: {\n jobId: input.options.jobId,\n runId: input.options.runId,\n expertKey: input.expertKey,\n input: { text: input.query },\n experts,\n model,\n providerConfig,\n temperature: input.options.temperature ?? perstackConfig.temperature,\n maxSteps: input.options.maxSteps ?? perstackConfig.maxSteps,\n maxRetries: input.options.maxRetries ?? perstackConfig.maxRetries,\n timeout: input.options.timeout ?? perstackConfig.timeout,\n perstackApiBaseUrl: perstackConfig.perstackApiBaseUrl,\n perstackApiKey: env.PERSTACK_API_KEY,\n perstackBaseSkillCommand: perstackConfig.perstackBaseSkillCommand,\n env,\n proxyUrl: process.env.PERSTACK_PROXY_URL,\n verbose: input.options.verbose,\n },\n },\n { eventListener: defaultEventListener, storeCheckpoint, retrieveCheckpoint },\n )\n } catch (error) {\n if (error instanceof Error) {\n console.error(error.message)\n } else {\n console.error(error)\n }\n process.exit(1)\n }\n })\n\nprogram.parse()\n"]}
@@ -5,7 +5,7 @@ import { createDeepSeek } from '@ai-sdk/deepseek';
5
5
  import { createGoogleGenerativeAI } from '@ai-sdk/google';
6
6
  import { createVertex } from '@ai-sdk/google-vertex';
7
7
  import { createOpenAI } from '@ai-sdk/openai';
8
- import { runParamsSchema, createRuntimeEvent, knownModels, stopRunByExceededMaxSteps, continueToNextStep, stopRunByDelegate, stopRunByInteractiveTool, retry, completeRun, finishToolCall, resolveToolResults, attemptCompletion, callDelegate, callInteractiveTool, callTools, resumeToolCalls, finishAllToolCalls, startGeneration, startRun } from '@perstack/core';
8
+ import { runParamsSchema, createRuntimeEvent, knownModels, getFilteredEnv, stopRunByExceededMaxSteps, continueToNextStep, stopRunByDelegate, stopRunByInteractiveTool, retry, completeRun, finishToolCall, resolveToolResults, attemptCompletion, callDelegate, callInteractiveTool, callTools, resumeToolCalls, finishAllToolCalls, startGeneration, startRun } from '@perstack/core';
9
9
  import { createOllama } from 'ollama-ai-provider-v2';
10
10
  import { ProxyAgent, fetch } from 'undici';
11
11
  import { setup, assign, createActor } from 'xstate';
@@ -22,7 +22,7 @@ import { ApiV1Client } from '@perstack/api-client/v1';
22
22
  // package.json
23
23
  var package_default = {
24
24
  name: "@perstack/runtime",
25
- version: "0.0.66",
25
+ version: "0.0.67",
26
26
  description: "Perstack Runtime",
27
27
  author: "Wintermute Technologies, Inc.",
28
28
  license: "Apache-2.0",
@@ -64,7 +64,7 @@ var package_default = {
64
64
  "@ai-sdk/google": "^2.0.44",
65
65
  "@ai-sdk/google-vertex": "^3.0.24",
66
66
  "@ai-sdk/openai": "^2.0.75",
67
- "@modelcontextprotocol/sdk": "^1.23.0",
67
+ "@modelcontextprotocol/sdk": "^1.24.0",
68
68
  "@paralleldrive/cuid2": "^3.0.4",
69
69
  "@perstack/api-client": "workspace:*",
70
70
  "@perstack/core": "workspace:*",
@@ -371,13 +371,14 @@ var McpSkillManager = class extends BaseSkillManager {
371
371
  if (!skill.command) {
372
372
  throw new Error(`Skill ${skill.name} has no command`);
373
373
  }
374
- const env = { ...process.env };
374
+ const requiredEnv = {};
375
375
  for (const envName of skill.requiredEnv) {
376
376
  if (!this._env[envName]) {
377
377
  throw new Error(`Skill ${skill.name} requires environment variable ${envName}`);
378
378
  }
379
- env[envName] = this._env[envName];
379
+ requiredEnv[envName] = this._env[envName];
380
380
  }
381
+ const env = getFilteredEnv(requiredEnv);
381
382
  const startTime = Date.now();
382
383
  const { command, args } = this._getCommandArgs(skill);
383
384
  if (this._eventListener) {
@@ -418,9 +419,44 @@ var McpSkillManager = class extends BaseSkillManager {
418
419
  if (!skill.endpoint) {
419
420
  throw new Error(`Skill ${skill.name} has no endpoint`);
420
421
  }
421
- const transport = new SSEClientTransport(new URL(skill.endpoint));
422
+ const url = new URL(skill.endpoint);
423
+ if (url.protocol !== "https:") {
424
+ throw new Error(`Skill ${skill.name} SSE endpoint must use HTTPS: ${skill.endpoint}`);
425
+ }
426
+ if (this._isPrivateOrLocalIP(url.hostname)) {
427
+ throw new Error(
428
+ `Skill ${skill.name} SSE endpoint cannot use private/local IP: ${skill.endpoint}`
429
+ );
430
+ }
431
+ const transport = new SSEClientTransport(url);
422
432
  await this._mcpClient.connect(transport);
423
433
  }
434
+ _isPrivateOrLocalIP(hostname) {
435
+ if (hostname === "localhost" || hostname === "127.0.0.1" || hostname === "::1" || hostname === "0.0.0.0") {
436
+ return true;
437
+ }
438
+ const ipv4Match = hostname.match(/^(\d+)\.(\d+)\.(\d+)\.(\d+)$/);
439
+ if (ipv4Match) {
440
+ const [, a, b] = ipv4Match.map(Number);
441
+ if (a === 10) return true;
442
+ if (a === 172 && b >= 16 && b <= 31) return true;
443
+ if (a === 192 && b === 168) return true;
444
+ if (a === 169 && b === 254) return true;
445
+ if (a === 127) return true;
446
+ }
447
+ if (hostname.includes(":")) {
448
+ if (hostname.startsWith("fe80:") || hostname.startsWith("fc") || hostname.startsWith("fd")) {
449
+ return true;
450
+ }
451
+ }
452
+ if (hostname.startsWith("::ffff:")) {
453
+ const ipv4Part = hostname.slice(7);
454
+ if (this._isPrivateOrLocalIP(ipv4Part)) {
455
+ return true;
456
+ }
457
+ }
458
+ return false;
459
+ }
424
460
  _getCommandArgs(skill) {
425
461
  const { name, command, packageName, args } = skill;
426
462
  if (!packageName && (!args || args.length === 0)) {
@@ -2146,7 +2182,7 @@ async function run(runInput, options) {
2146
2182
  if (options?.eventListener) {
2147
2183
  const initEvent = createRuntimeEvent("initializeRuntime", setting.jobId, setting.runId, {
2148
2184
  runtimeVersion: package_default.version,
2149
- runtime: "perstack",
2185
+ runtime: "local",
2150
2186
  expertName: expertToRun.name,
2151
2187
  experts: Object.keys(experts),
2152
2188
  model: setting.model,
@@ -2351,5 +2387,5 @@ async function runDelegate(delegation, parentSetting, parentCheckpoint, parentEx
2351
2387
  }
2352
2388
 
2353
2389
  export { getModel, package_default, run, runtimeStateMachine };
2354
- //# sourceMappingURL=chunk-LMD6DWPY.js.map
2355
- //# sourceMappingURL=chunk-LMD6DWPY.js.map
2390
+ //# sourceMappingURL=chunk-AQSRQW5R.js.map
2391
+ //# sourceMappingURL=chunk-AQSRQW5R.js.map