@privateconnect/sdk 0.7.4 → 0.7.7

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
@@ -1,174 +1,192 @@
1
- # Private Connect SDK
2
-
3
- TypeScript SDK for Private Connect - programmatic access to services and agent orchestration.
4
-
5
- ## Installation
6
-
7
- ```bash
8
- npm install @privateconnect/sdk
9
- # or
10
- pnpm add @privateconnect/sdk
11
- ```
12
-
13
- ## Quick Start
14
-
15
- ```typescript
16
- import { PrivateConnect, connect } from '@privateconnect/sdk';
17
-
18
- // Quick connect to a service
19
- const db = await connect('postgres-prod');
20
- console.log(db.connectionString); // postgres://localhost:5432/...
21
-
22
- // Or use the full client
23
- const pc = new PrivateConnect({
24
- apiKey: process.env.PRIVATECONNECT_API_KEY
25
- });
26
-
27
- // List available services
28
- const services = await pc.services.list();
29
-
30
- // Connect to a specific service
31
- const redis = await pc.connect('redis-cache');
32
- console.log(redis.connectionString); // redis://localhost:6379
33
- ```
34
-
35
- ## Agent Orchestration
36
-
37
- The SDK enables multi-agent orchestration - coordinating work across agents running on different machines.
38
-
39
- ### List Agents
40
-
41
- ```typescript
42
- // Get all agents in your workspace
43
- const agents = await pc.agents.list();
44
-
45
- // Get only online agents
46
- const online = await pc.agents.list({ onlineOnly: true });
47
-
48
- // Find agents with specific capabilities
49
- const gpuAgents = await pc.agents.findByCapability('gpu');
50
- ```
51
-
52
- ### Register Capabilities
53
-
54
- Tell other agents what you can do:
55
-
56
- ```typescript
57
- await pc.agents.registerCapabilities([
58
- { name: 'database', metadata: { type: 'postgres', version: '15' } },
59
- { name: 'gpu', metadata: { model: 'A100', memory: '80GB' } },
60
- ]);
61
- ```
62
-
63
- ### Agent Messaging
64
-
65
- Coordinate with other agents:
66
-
67
- ```typescript
68
- // Send a message to a specific agent
69
- await pc.agents.sendMessage(targetAgentId, {
70
- action: 'run-migration',
71
- database: 'users',
72
- });
73
-
74
- // Broadcast to all agents
75
- await pc.agents.broadcast({
76
- type: 'deployment-starting',
77
- service: 'api',
78
- });
79
-
80
- // Get your messages
81
- const messages = await pc.agents.getMessages({ unreadOnly: true });
82
- for (const msg of messages) {
83
- console.log(`From ${msg.from.name}: ${JSON.stringify(msg.payload)}`);
84
- }
85
- ```
86
-
87
- ### Orchestration Sessions
88
-
89
- Create ephemeral sessions for coordinated workflows:
90
-
91
- ```typescript
92
- // Create a session
93
- const session = await pc.sessions.create('deploy-v2.1', {
94
- ttlMinutes: 30,
95
- metadata: { version: '2.1.0' },
96
- });
97
-
98
- // ... coordinate agents ...
99
-
100
- // End the session
101
- await pc.sessions.end(session.id);
102
- ```
103
-
104
- ## Connection Strings
105
-
106
- Get properly formatted connection strings for common services:
107
-
108
- ```typescript
109
- const db = await pc.connect('postgres-prod');
110
- // db.connectionString = 'postgres://localhost:5432/postgres'
111
- // db.envVar = 'DATABASE_URL'
112
-
113
- const cache = await pc.connect('redis-cache');
114
- // cache.connectionString = 'redis://localhost:6379'
115
- // cache.envVar = 'REDIS_URL'
116
-
117
- const api = await pc.connect('internal-api');
118
- // api.connectionString = 'http://localhost:8080'
119
- // api.envVar = 'API_URL'
120
- ```
121
-
122
- ## Environment Variables
123
-
124
- The SDK can read configuration from environment variables:
125
-
126
- ```bash
127
- export PRIVATECONNECT_API_KEY=your-api-key
128
- ```
129
-
130
- ```typescript
131
- // API key automatically read from env
132
- const connection = await connect('my-service');
133
- ```
134
-
135
- ## API Reference
136
-
137
- ### `PrivateConnect`
138
-
139
- Main client class.
140
-
141
- ```typescript
142
- const pc = new PrivateConnect({
143
- apiKey: string, // Required: Your API key
144
- hubUrl?: string, // Optional: Hub URL (default: https://api.privateconnect.co)
145
- agentId?: string, // Optional: Agent ID (auto-detected)
146
- });
147
- ```
148
-
149
- ### `pc.services`
150
-
151
- - `list()` - List all services
152
- - `get(name)` - Get a service by name
153
- - `getConnection(name)` - Get connection details for a service
154
-
155
- ### `pc.agents`
156
-
157
- - `list(options?)` - List all agents
158
- - `findByCapability(capability)` - Find agents by capability
159
- - `registerCapabilities(capabilities)` - Register this agent's capabilities
160
- - `sendMessage(toAgentId, payload, options?)` - Send a message
161
- - `broadcast(payload, options?)` - Broadcast to all agents
162
- - `getMessages(options?)` - Get received messages
163
- - `markRead(messageIds)` - Mark messages as read
164
-
165
- ### `pc.sessions`
166
-
167
- - `create(name, options?)` - Create an orchestration session
168
- - `end(sessionId)` - End a session
169
- - `getActive()` - Get active sessions
170
-
171
- ## License
172
-
173
- [FSL-1.1-MIT](LICENSE)
174
-
1
+ # Private Connect SDK
2
+
3
+ Define your connections in `pconnect.yml`. Access them from anywhere your app, CI, an AI agent.
4
+
5
+ ## Install
6
+
7
+ ```bash
8
+ npm install @privateconnect/sdk
9
+ ```
10
+
11
+ ## Quick Start
12
+
13
+ **1. Create `pconnect.yml` in your project root:**
14
+
15
+ ```yaml
16
+ resources:
17
+ staging-db:
18
+ type: postgres
19
+ host: internal-db
20
+ port: 5432
21
+ access:
22
+ mode: tcp
23
+ redis-cache:
24
+ type: redis
25
+ host: redis.internal
26
+ port: 6379
27
+ access:
28
+ mode: tcp
29
+ ```
30
+
31
+ **2. Use it in your code:**
32
+
33
+ ```typescript
34
+ import { PrivateConnect } from '@privateconnect/sdk';
35
+
36
+ const pc = PrivateConnect.fromManifest();
37
+
38
+ const db = pc.resource('staging-db');
39
+ console.log(db.connectionString); // postgres://internal-db:5432
40
+ console.log(db.envVar); // DATABASE_URL
41
+
42
+ const cache = pc.resource('redis-cache');
43
+ console.log(cache.connectionString); // redis://redis.internal:6379
44
+ ```
45
+
46
+ **3. Run `connect dev` to make it live:**
47
+
48
+ ```bash
49
+ connect dev
50
+ # → Provisions tunnels for every resource in pconnect.yml
51
+ # → Your app's connection strings now resolve to live services
52
+ ```
53
+
54
+ That's it. Your app declares what it connects to. `connect dev` makes it work. An AI agent (or a teammate) can modify `pconnect.yml` to change the topology — without touching application code.
55
+
56
+ ## Why This Matters
57
+
58
+ The `pconnect.yml` file is a diffable, reviewable, mergeable description of your project's connectivity. When an AI modifies it — adding a read replica, changing a TTL, switching an access mode — that's a PR you can review and merge. The SDK reads the manifest; the agent provisions it.
59
+
60
+ ## Manifest API
61
+
62
+ ### `PrivateConnect.fromManifest(path?, config?)`
63
+
64
+ Load a manifest and return a configured client. Auto-discovers `pconnect.yml` in the current directory (or parents) if no path is given.
65
+
66
+ ```typescript
67
+ // Auto-discover
68
+ const pc = PrivateConnect.fromManifest();
69
+
70
+ // Explicit path
71
+ const pc = PrivateConnect.fromManifest('./infra/pconnect.yml');
72
+
73
+ // With hub API access (for grants, orchestration, etc.)
74
+ const pc = PrivateConnect.fromManifest('./pconnect.yml', {
75
+ apiKey: process.env.PRIVATECONNECT_API_KEY,
76
+ });
77
+ ```
78
+
79
+ ### `pc.resource(name)`
80
+
81
+ Get a resource by name. Returns its resolved type, host, port, connection string, and suggested environment variable.
82
+
83
+ ```typescript
84
+ const db = pc.resource('staging-db');
85
+ // {
86
+ // name: 'staging-db',
87
+ // type: 'postgres',
88
+ // host: 'internal-db',
89
+ // port: 5432,
90
+ // connectionString: 'postgres://internal-db:5432',
91
+ // envVar: 'DATABASE_URL',
92
+ // accessMode: 'tcp',
93
+ // via: 'direct'
94
+ // }
95
+ ```
96
+
97
+ ### `pc.resources()`
98
+
99
+ List all resources declared in the manifest.
100
+
101
+ ```typescript
102
+ const all = pc.resources();
103
+ all.forEach(r => console.log(`${r.name}: ${r.connectionString}`));
104
+ ```
105
+
106
+ ### `pc.envBlock()`
107
+
108
+ Generate a `.env`-compatible block for all resources.
109
+
110
+ ```typescript
111
+ console.log(pc.envBlock());
112
+ // DATABASE_URL=postgres://internal-db:5432
113
+ // REDIS_URL=redis://redis.internal:6379
114
+ // API_URL=http://payments.service.internal:3000
115
+ ```
116
+
117
+ ## Hub API
118
+
119
+ When you pass an API key, you also get access to the hub APIs for grants, agents, and services.
120
+
121
+ ### Grants
122
+
123
+ Grant an AI agent temporary, scoped access to a private resource.
124
+
125
+ ```typescript
126
+ const pc = PrivateConnect.fromManifest('./pconnect.yml', {
127
+ apiKey: process.env.PRIVATECONNECT_API_KEY,
128
+ });
129
+
130
+ const grant = await pc.grants.create({
131
+ agentLabel: 'claude',
132
+ resourceType: 'db',
133
+ resourceName: 'postgres',
134
+ ttl: '5m',
135
+ });
136
+ console.log(grant.token); // gnt_...
137
+ ```
138
+
139
+ ### Agent Orchestration
140
+
141
+ Coordinate agents across machines.
142
+
143
+ ```typescript
144
+ const agents = await pc.agents.list({ onlineOnly: true });
145
+ const gpuAgents = await pc.agents.findByCapability('gpu');
146
+
147
+ await pc.agents.sendMessage(gpuAgents[0].id, {
148
+ action: 'run-training',
149
+ dataset: 'v2',
150
+ });
151
+ ```
152
+
153
+ ### Services
154
+
155
+ Query hub-registered services directly.
156
+
157
+ ```typescript
158
+ const services = await pc.services.list();
159
+ const conn = await pc.connect('prod-db');
160
+ console.log(conn.connectionString);
161
+ ```
162
+
163
+ ## Manifest Format
164
+
165
+ The `pconnect.yml` file supports the following resource types:
166
+
167
+ | Type | Default Port | Connection String | Env Var |
168
+ |------|-------------|-------------------|---------|
169
+ | `postgres` | 5432 | `postgres://host:port` | `DATABASE_URL` |
170
+ | `mysql` | 3306 | `mysql://host:port` | `DATABASE_URL` |
171
+ | `redis` | 6379 | `redis://host:port` | `REDIS_URL` |
172
+ | `http` | 80 | `http://host:port` | `API_URL` |
173
+ | `generic-tcp` | — | `tcp://host:port` | `<NAME>_URL` |
174
+
175
+ Access modes:
176
+ - `tcp` — direct TCP tunnel (databases, Redis, generic)
177
+ - `http` — HTTP-level proxying
178
+
179
+ Transport:
180
+ - `direct` (default) — connect directly to the target
181
+ - `hub` — route through the Private Connect hub when the target isn't directly reachable
182
+
183
+ ## Environment Variables
184
+
185
+ | Variable | Purpose |
186
+ |----------|---------|
187
+ | `PRIVATECONNECT_API_KEY` | API key for hub access (grants, orchestration) |
188
+ | `CONNECT_HUB_URL` | Hub URL (default: `https://api.privateconnect.co`) |
189
+
190
+ ## License
191
+
192
+ [FSL-1.1-MIT](LICENSE)
package/dist/index.d.ts CHANGED
@@ -1,34 +1,35 @@
1
1
  /**
2
2
  * Private Connect SDK
3
3
  *
4
- * Programmatic access to Private Connect services, grants, and agent orchestration.
4
+ * Define your connections in pconnect.yml. Access them from anywhere.
5
5
  *
6
6
  * @example
7
7
  * ```typescript
8
8
  * import { PrivateConnect } from '@privateconnect/sdk';
9
9
  *
10
- * const pc = new PrivateConnect({ apiKey: 'your-api-key' });
10
+ * // Load your project's connection manifest
11
+ * const pc = PrivateConnect.fromManifest();
11
12
  *
12
- * // Connect to a service (assumes tunnel is already open)
13
- * const db = await pc.connect('postgres-prod');
14
- * console.log(db.connectionString); // postgres://localhost:5432/...
13
+ * // Get a resource declared in pconnect.yml
14
+ * const db = pc.resource('staging-db');
15
+ * console.log(db.connectionString); // postgres://internal-db:5432
16
+ * console.log(db.envVar); // DATABASE_URL
15
17
  *
16
- * // Grant an AI agent temporary access
17
- * const grant = await pc.grants.create({
18
+ * // With an API key, you also get hub API access
19
+ * const pc2 = PrivateConnect.fromManifest('./pconnect.yml', {
20
+ * apiKey: process.env.PRIVATECONNECT_API_KEY,
21
+ * });
22
+ * const grant = await pc2.grants.create({
18
23
  * agentLabel: 'claude',
19
24
  * resourceType: 'db',
20
25
  * resourceName: 'postgres',
21
26
  * ttl: '5m',
22
27
  * });
23
- * console.log(grant.token); // gnt_...
24
- *
25
- * // List all agents
26
- * const agents = await pc.agents.list();
27
28
  * ```
28
29
  */
29
30
  export interface PrivateConnectConfig {
30
- /** API key for authentication */
31
- apiKey: string;
31
+ /** API key for authentication. Required for hub API calls; optional for manifest-only usage. */
32
+ apiKey?: string;
32
33
  /** Hub URL (default: https://api.privateconnect.co) */
33
34
  hubUrl?: string;
34
35
  /** Agent ID (auto-detected from local config if not provided) */
@@ -100,6 +101,34 @@ export interface GrantCreateOptions {
100
101
  /** Duration string: 60s, 5m, 1h, 1d. Omit for persistent grant. */
101
102
  ttl?: string;
102
103
  }
104
+ export declare const RESOURCE_TYPES: readonly ["postgres", "mysql", "redis", "http", "generic-tcp"];
105
+ export type ResourceType = typeof RESOURCE_TYPES[number];
106
+ export declare const ACCESS_MODES: readonly ["tcp", "http"];
107
+ export type AccessMode = typeof ACCESS_MODES[number];
108
+ export declare const TRANSPORT_MODES: readonly ["direct", "hub"];
109
+ export type TransportVia = typeof TRANSPORT_MODES[number];
110
+ export interface ManifestResourceConfig {
111
+ type: ResourceType;
112
+ host?: string;
113
+ port?: number;
114
+ targetHost?: string;
115
+ targetPort?: number;
116
+ url?: string;
117
+ access: {
118
+ mode: AccessMode;
119
+ via?: TransportVia;
120
+ };
121
+ }
122
+ export interface ManifestResource {
123
+ name: string;
124
+ type: ResourceType;
125
+ host: string;
126
+ port: number;
127
+ connectionString: string;
128
+ envVar: string;
129
+ accessMode: AccessMode;
130
+ via: TransportVia;
131
+ }
103
132
  export declare class AgentsAPI {
104
133
  private client;
105
134
  constructor(client: PrivateConnect);
@@ -170,13 +199,53 @@ export declare class GrantsAPI {
170
199
  }
171
200
  export declare class PrivateConnect {
172
201
  private config;
202
+ private manifest;
203
+ private manifestPath?;
173
204
  /** Agents API for discovery and orchestration */
174
205
  agents: AgentsAPI;
175
206
  /** Services API for connecting to services */
176
207
  services: ServicesAPI;
177
208
  /** Grants API for managing scoped access tokens (time-limited or persistent) */
178
209
  grants: GrantsAPI;
179
- constructor(config: PrivateConnectConfig);
210
+ constructor(config?: PrivateConnectConfig);
211
+ /**
212
+ * Load a pconnect.yml manifest and return a configured client.
213
+ *
214
+ * Auto-discovers pconnect.yml in the current directory (or parents) if no
215
+ * path is given. Pass a config to also enable hub API access.
216
+ *
217
+ * @example
218
+ * ```typescript
219
+ * const pc = PrivateConnect.fromManifest();
220
+ * const db = pc.resource('staging-db');
221
+ * console.log(db.connectionString); // postgres://internal-db:5432
222
+ * ```
223
+ */
224
+ static fromManifest(manifestPath?: string, config?: PrivateConnectConfig): PrivateConnect;
225
+ /**
226
+ * Get a resource declared in pconnect.yml by name.
227
+ *
228
+ * Returns its type, host, port, connection string, and suggested env var.
229
+ * When `connect dev` is running, the connection string points to a live
230
+ * local tunnel.
231
+ */
232
+ resource(name: string): ManifestResource;
233
+ /**
234
+ * List all resources declared in the loaded manifest.
235
+ */
236
+ resources(): ManifestResource[];
237
+ /**
238
+ * Generate a `.env`-compatible block for all manifest resources.
239
+ *
240
+ * @example
241
+ * ```typescript
242
+ * const pc = PrivateConnect.fromManifest();
243
+ * console.log(pc.envBlock());
244
+ * // DATABASE_URL=postgres://internal-db:5432
245
+ * // REDIS_URL=redis://redis.internal:6379
246
+ * ```
247
+ */
248
+ envBlock(): string;
180
249
  /** The resolved agent ID, or undefined if not configured. */
181
250
  get agentId(): string | undefined;
182
251
  /** The hub URL this client is connected to. */
@@ -193,11 +262,18 @@ export declare class PrivateConnect {
193
262
  * Used by APIs that require an authenticated agent identity.
194
263
  */
195
264
  requireAgentId(): string;
265
+ /** Requires an API key or throws. */
266
+ private requireApiKey;
196
267
  /** Internal fetch with API key auth. */
197
- fetch(path: string, options?: RequestInit): Promise<Response>;
268
+ fetch(urlPath: string, options?: RequestInit): Promise<Response>;
198
269
  }
199
270
  export default PrivateConnect;
200
- /** Convenience function for quick one-off connections. */
271
+ /** Convenience function for quick one-off connections via the hub API. */
201
272
  export declare function connect(serviceName: string, config?: PrivateConnectConfig & {
202
273
  grantToken?: string;
203
274
  }): Promise<Connection>;
275
+ /**
276
+ * Load pconnect.yml and get a resource by name.
277
+ * Shorthand for `PrivateConnect.fromManifest().resource(name)`.
278
+ */
279
+ export declare function fromManifest(manifestPath?: string): PrivateConnect;
package/dist/index.js CHANGED
@@ -2,37 +2,169 @@
2
2
  /**
3
3
  * Private Connect SDK
4
4
  *
5
- * Programmatic access to Private Connect services, grants, and agent orchestration.
5
+ * Define your connections in pconnect.yml. Access them from anywhere.
6
6
  *
7
7
  * @example
8
8
  * ```typescript
9
9
  * import { PrivateConnect } from '@privateconnect/sdk';
10
10
  *
11
- * const pc = new PrivateConnect({ apiKey: 'your-api-key' });
11
+ * // Load your project's connection manifest
12
+ * const pc = PrivateConnect.fromManifest();
12
13
  *
13
- * // Connect to a service (assumes tunnel is already open)
14
- * const db = await pc.connect('postgres-prod');
15
- * console.log(db.connectionString); // postgres://localhost:5432/...
14
+ * // Get a resource declared in pconnect.yml
15
+ * const db = pc.resource('staging-db');
16
+ * console.log(db.connectionString); // postgres://internal-db:5432
17
+ * console.log(db.envVar); // DATABASE_URL
16
18
  *
17
- * // Grant an AI agent temporary access
18
- * const grant = await pc.grants.create({
19
+ * // With an API key, you also get hub API access
20
+ * const pc2 = PrivateConnect.fromManifest('./pconnect.yml', {
21
+ * apiKey: process.env.PRIVATECONNECT_API_KEY,
22
+ * });
23
+ * const grant = await pc2.grants.create({
19
24
  * agentLabel: 'claude',
20
25
  * resourceType: 'db',
21
26
  * resourceName: 'postgres',
22
27
  * ttl: '5m',
23
28
  * });
24
- * console.log(grant.token); // gnt_...
25
- *
26
- * // List all agents
27
- * const agents = await pc.agents.list();
28
29
  * ```
29
30
  */
30
31
  Object.defineProperty(exports, "__esModule", { value: true });
31
- exports.PrivateConnect = exports.GrantsAPI = exports.ServicesAPI = exports.AgentsAPI = void 0;
32
+ exports.PrivateConnect = exports.GrantsAPI = exports.ServicesAPI = exports.AgentsAPI = exports.TRANSPORT_MODES = exports.ACCESS_MODES = exports.RESOURCE_TYPES = void 0;
32
33
  exports.connect = connect;
34
+ exports.fromManifest = fromManifest;
33
35
  const fs = require("fs");
34
36
  const path = require("path");
35
37
  const os = require("os");
38
+ const yaml = require("js-yaml");
39
+ // ─────────────────────────────────────────────────────────────────────────────
40
+ // Manifest Types
41
+ // ─────────────────────────────────────────────────────────────────────────────
42
+ exports.RESOURCE_TYPES = ['postgres', 'mysql', 'redis', 'http', 'generic-tcp'];
43
+ exports.ACCESS_MODES = ['tcp', 'http'];
44
+ exports.TRANSPORT_MODES = ['direct', 'hub'];
45
+ // ─────────────────────────────────────────────────────────────────────────────
46
+ // Manifest Parsing
47
+ // ─────────────────────────────────────────────────────────────────────────────
48
+ const MANIFEST_FILENAMES = [
49
+ 'pconnect.yml',
50
+ 'pconnect.yaml',
51
+ 'pconnect.json',
52
+ '.pconnect.yml',
53
+ '.pconnect.yaml',
54
+ '.pconnect.json',
55
+ ];
56
+ const DEFAULT_PORTS = {
57
+ postgres: 5432,
58
+ mysql: 3306,
59
+ redis: 6379,
60
+ http: 80,
61
+ 'generic-tcp': 0,
62
+ };
63
+ const PROTOCOL_SCHEMES = {
64
+ postgres: 'postgres',
65
+ mysql: 'mysql',
66
+ redis: 'redis',
67
+ http: 'http',
68
+ 'generic-tcp': 'tcp',
69
+ };
70
+ const ENV_VAR_MAP = {
71
+ postgres: 'DATABASE_URL',
72
+ mysql: 'DATABASE_URL',
73
+ redis: 'REDIS_URL',
74
+ http: 'API_URL',
75
+ 'generic-tcp': 'TCP_URL',
76
+ };
77
+ function findManifest(startDir) {
78
+ let dir = startDir || process.cwd();
79
+ for (let depth = 0; depth < 4; depth++) {
80
+ for (const filename of MANIFEST_FILENAMES) {
81
+ const filePath = path.join(dir, filename);
82
+ if (fs.existsSync(filePath))
83
+ return filePath;
84
+ }
85
+ const parent = path.dirname(dir);
86
+ if (parent === dir)
87
+ break;
88
+ dir = parent;
89
+ }
90
+ return null;
91
+ }
92
+ function parseManifestFile(filePath) {
93
+ const content = fs.readFileSync(filePath, 'utf-8');
94
+ let raw;
95
+ if (filePath.endsWith('.json')) {
96
+ raw = JSON.parse(content);
97
+ }
98
+ else {
99
+ raw = yaml.load(content);
100
+ }
101
+ if (!raw || typeof raw !== 'object' || !raw.resources || typeof raw.resources !== 'object') {
102
+ return {};
103
+ }
104
+ const resources = {};
105
+ const rawResources = raw.resources;
106
+ for (const [name, value] of Object.entries(rawResources)) {
107
+ if (!value || typeof value !== 'object')
108
+ continue;
109
+ const obj = value;
110
+ const type = obj.type;
111
+ if (!exports.RESOURCE_TYPES.includes(type))
112
+ continue;
113
+ const accessObj = (obj.access && typeof obj.access === 'object')
114
+ ? obj.access
115
+ : { mode: 'tcp' };
116
+ resources[name] = {
117
+ type: type,
118
+ host: typeof obj.host === 'string' ? obj.host : undefined,
119
+ port: typeof obj.port === 'number' ? obj.port : undefined,
120
+ targetHost: typeof obj.targetHost === 'string' ? obj.targetHost : undefined,
121
+ targetPort: typeof obj.targetPort === 'number' ? obj.targetPort : undefined,
122
+ url: typeof obj.url === 'string' ? obj.url : undefined,
123
+ access: {
124
+ mode: (exports.ACCESS_MODES.includes(accessObj.mode)
125
+ ? accessObj.mode
126
+ : 'tcp'),
127
+ via: (exports.TRANSPORT_MODES.includes(accessObj.via)
128
+ ? accessObj.via
129
+ : 'direct'),
130
+ },
131
+ };
132
+ }
133
+ return resources;
134
+ }
135
+ function resolveManifestResource(name, config) {
136
+ let host;
137
+ let port;
138
+ if (config.type === 'http' && config.url) {
139
+ try {
140
+ const parsed = new URL(config.url);
141
+ host = parsed.hostname;
142
+ port = parsed.port ? parseInt(parsed.port, 10) : (parsed.protocol === 'https:' ? 443 : 80);
143
+ }
144
+ catch {
145
+ host = config.host || config.targetHost || 'localhost';
146
+ port = config.port || config.targetPort || DEFAULT_PORTS[config.type];
147
+ }
148
+ }
149
+ else {
150
+ host = config.targetHost || config.host || 'localhost';
151
+ port = config.targetPort || config.port || DEFAULT_PORTS[config.type];
152
+ }
153
+ const scheme = PROTOCOL_SCHEMES[config.type];
154
+ const connectionString = `${scheme}://${host}:${port}`;
155
+ const envVar = ENV_VAR_MAP[config.type]
156
+ || `${name.toUpperCase().replace(/-/g, '_')}_URL`;
157
+ return {
158
+ name,
159
+ type: config.type,
160
+ host,
161
+ port,
162
+ connectionString,
163
+ envVar,
164
+ accessMode: config.access.mode,
165
+ via: config.access.via || 'direct',
166
+ };
167
+ }
36
168
  // ─────────────────────────────────────────────────────────────────────────────
37
169
  // Tracking
38
170
  // ─────────────────────────────────────────────────────────────────────────────
@@ -183,8 +315,8 @@ class ServicesAPI {
183
315
  }
184
316
  const port = service.tunnelPort || service.targetPort;
185
317
  const host = 'localhost';
186
- let connectionString = '';
187
- let envVar = 'SERVICE_URL';
318
+ let connectionString;
319
+ let envVar;
188
320
  if (service.targetPort === 5432 || service.protocol === 'postgres') {
189
321
  connectionString = `postgres://${host}:${port}/postgres`;
190
322
  envVar = 'DATABASE_URL';
@@ -288,7 +420,8 @@ exports.GrantsAPI = GrantsAPI;
288
420
  // Main Client
289
421
  // ─────────────────────────────────────────────────────────────────────────────
290
422
  class PrivateConnect {
291
- constructor(config) {
423
+ constructor(config = {}) {
424
+ this.manifest = new Map();
292
425
  this.config = {
293
426
  apiKey: config.apiKey,
294
427
  hubUrl: config.hubUrl || 'https://api.privateconnect.co',
@@ -301,6 +434,87 @@ class PrivateConnect {
301
434
  trackSdkUsage(this.config.hubUrl);
302
435
  }
303
436
  }
437
+ // ─────────────────────────────────────────────────────────────────────────
438
+ // Manifest API — the primary way to use the SDK
439
+ // ─────────────────────────────────────────────────────────────────────────
440
+ /**
441
+ * Load a pconnect.yml manifest and return a configured client.
442
+ *
443
+ * Auto-discovers pconnect.yml in the current directory (or parents) if no
444
+ * path is given. Pass a config to also enable hub API access.
445
+ *
446
+ * @example
447
+ * ```typescript
448
+ * const pc = PrivateConnect.fromManifest();
449
+ * const db = pc.resource('staging-db');
450
+ * console.log(db.connectionString); // postgres://internal-db:5432
451
+ * ```
452
+ */
453
+ static fromManifest(manifestPath, config) {
454
+ const resolvedPath = manifestPath
455
+ ? path.resolve(manifestPath)
456
+ : findManifest();
457
+ if (!resolvedPath) {
458
+ throw new Error('No pconnect.yml found. Create one in your project root:\n\n' +
459
+ ' resources:\n' +
460
+ ' staging-db:\n' +
461
+ ' type: postgres\n' +
462
+ ' host: internal-db\n' +
463
+ ' port: 5432\n' +
464
+ ' access:\n' +
465
+ ' mode: tcp\n');
466
+ }
467
+ const instance = new PrivateConnect(config || { disableTracking: true });
468
+ instance.manifestPath = resolvedPath;
469
+ const rawResources = parseManifestFile(resolvedPath);
470
+ for (const [name, rawConfig] of Object.entries(rawResources)) {
471
+ instance.manifest.set(name, resolveManifestResource(name, rawConfig));
472
+ }
473
+ return instance;
474
+ }
475
+ /**
476
+ * Get a resource declared in pconnect.yml by name.
477
+ *
478
+ * Returns its type, host, port, connection string, and suggested env var.
479
+ * When `connect dev` is running, the connection string points to a live
480
+ * local tunnel.
481
+ */
482
+ resource(name) {
483
+ const r = this.manifest.get(name);
484
+ if (!r) {
485
+ const available = Array.from(this.manifest.keys());
486
+ const msg = available.length
487
+ ? `Available: ${available.join(', ')}`
488
+ : 'No resources loaded. Did you call PrivateConnect.fromManifest()?';
489
+ throw new Error(`Resource "${name}" not found. ${msg}`);
490
+ }
491
+ return r;
492
+ }
493
+ /**
494
+ * List all resources declared in the loaded manifest.
495
+ */
496
+ resources() {
497
+ return Array.from(this.manifest.values());
498
+ }
499
+ /**
500
+ * Generate a `.env`-compatible block for all manifest resources.
501
+ *
502
+ * @example
503
+ * ```typescript
504
+ * const pc = PrivateConnect.fromManifest();
505
+ * console.log(pc.envBlock());
506
+ * // DATABASE_URL=postgres://internal-db:5432
507
+ * // REDIS_URL=redis://redis.internal:6379
508
+ * ```
509
+ */
510
+ envBlock() {
511
+ return this.resources()
512
+ .map(r => `${r.envVar}=${r.connectionString}`)
513
+ .join('\n');
514
+ }
515
+ // ─────────────────────────────────────────────────────────────────────────
516
+ // Existing APIs
517
+ // ─────────────────────────────────────────────────────────────────────────
304
518
  /** The resolved agent ID, or undefined if not configured. */
305
519
  get agentId() {
306
520
  return this.config.agentId;
@@ -329,13 +543,23 @@ class PrivateConnect {
329
543
  }
330
544
  return this.config.agentId;
331
545
  }
546
+ /** Requires an API key or throws. */
547
+ requireApiKey() {
548
+ if (!this.config.apiKey) {
549
+ throw new Error('API key required for hub API calls. Either:\n' +
550
+ ' 1. Set PRIVATECONNECT_API_KEY environment variable, or\n' +
551
+ ' 2. Pass apiKey in the config.');
552
+ }
553
+ return this.config.apiKey;
554
+ }
332
555
  /** Internal fetch with API key auth. */
333
- async fetch(path, options) {
334
- const url = `${this.config.hubUrl}${path}`;
556
+ async fetch(urlPath, options) {
557
+ const apiKey = this.requireApiKey();
558
+ const url = `${this.config.hubUrl}${urlPath}`;
335
559
  const response = await fetch(url, {
336
560
  ...options,
337
561
  headers: {
338
- 'x-api-key': this.config.apiKey,
562
+ 'x-api-key': apiKey,
339
563
  'Content-Type': 'application/json',
340
564
  ...options?.headers,
341
565
  },
@@ -349,7 +573,7 @@ class PrivateConnect {
349
573
  }
350
574
  exports.PrivateConnect = PrivateConnect;
351
575
  exports.default = PrivateConnect;
352
- /** Convenience function for quick one-off connections. */
576
+ /** Convenience function for quick one-off connections via the hub API. */
353
577
  async function connect(serviceName, config) {
354
578
  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;
355
579
  if (!apiKey) {
@@ -358,4 +582,11 @@ async function connect(serviceName, config) {
358
582
  const client = new PrivateConnect({ ...config, apiKey });
359
583
  return client.connect(serviceName, { grantToken: config?.grantToken });
360
584
  }
361
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Private Connect SDK\r\n *\r\n * Programmatic access to Private Connect services, grants, and agent orchestration.\r\n *\r\n * @example\r\n * ```typescript\r\n * import { PrivateConnect } from '@privateconnect/sdk';\r\n *\r\n * const pc = new PrivateConnect({ apiKey: 'your-api-key' });\r\n *\r\n * // Connect to a service (assumes tunnel is already open)\r\n * const db = await pc.connect('postgres-prod');\r\n * console.log(db.connectionString); // postgres://localhost:5432/...\r\n *\r\n * // Grant an AI agent temporary access\r\n * const grant = await pc.grants.create({\r\n *   agentLabel: 'claude',\r\n *   resourceType: 'db',\r\n *   resourceName: 'postgres',\r\n *   ttl: '5m',\r\n * });\r\n * console.log(grant.token); // gnt_...\r\n *\r\n * // List all agents\r\n * const agents = await pc.agents.list();\r\n * ```\r\n */\r\n\r\nimport * as fs from 'fs';\r\nimport * as path from 'path';\r\nimport * as os from 'os';\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Types\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport interface PrivateConnectConfig {\r\n  /** API key for authentication */\r\n  apiKey: string;\r\n  /** Hub URL (default: https://api.privateconnect.co) */\r\n  hubUrl?: string;\r\n  /** Agent ID (auto-detected from local config if not provided) */\r\n  agentId?: string;\r\n  /** Disable usage tracking (default: false) */\r\n  disableTracking?: boolean;\r\n}\r\n\r\nexport interface Service {\r\n  id: string;\r\n  name: string;\r\n  targetHost: string;\r\n  targetPort: number;\r\n  tunnelPort?: number;\r\n  protocol: string;\r\n  status: string;\r\n  agentLabel?: string;\r\n}\r\n\r\nexport interface Agent {\r\n  id: string;\r\n  name?: string;\r\n  label: string;\r\n  isOnline: boolean;\r\n  lastSeenAt: string;\r\n  capabilities: string[];\r\n  services: string[];\r\n}\r\n\r\nexport interface Connection {\r\n  service: string;\r\n  host: string;\r\n  port: number;\r\n  connectionString: string;\r\n  envVar: string;\r\n  /** Present when using a grant-based connection */\r\n  grantToken?: string;\r\n  /** Present when using a grant-based connection */\r\n  grantEndpoint?: string;\r\n}\r\n\r\nexport interface Message {\r\n  id: string;\r\n  from: { id: string; name?: string; label?: string };\r\n  channel: string;\r\n  type: string;\r\n  payload: Record<string, unknown>;\r\n  createdAt: string;\r\n  isRead: boolean;\r\n}\r\n\r\nexport interface Grant {\r\n  id: string;\r\n  agentLabel: string;\r\n  resourceType: string;\r\n  resourceName: string;\r\n  scope: string;\r\n  tokenPrefix?: string;\r\n  persistent: boolean;\r\n  expiresAt: string | null;\r\n  expiresInMinutes?: number | null;\r\n  token?: string;\r\n  endpoint?: string;\r\n}\r\n\r\nexport interface GrantCreateOptions {\r\n  agentLabel: string;\r\n  resourceType: 'db' | 'api' | 'path';\r\n  resourceName: string;\r\n  scope?: 'read-only' | 'full';\r\n  /** Duration string: 60s, 5m, 1h, 1d. Omit for persistent grant. */\r\n  ttl?: string;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Tracking\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nfunction trackSdkUsage(hubUrl: string): void {\r\n  const data = JSON.stringify({\r\n    os: typeof process !== 'undefined' ? process.platform : 'browser',\r\n    arch: typeof process !== 'undefined' ? (process.arch === 'arm64' ? 'arm64' : 'x64') : 'unknown',\r\n    version: 'sdk',\r\n    source: 'sdk',\r\n  });\r\n\r\n  fetch(`${hubUrl}/v1/events/install`, {\r\n    method: 'POST',\r\n    headers: { 'Content-Type': 'application/json' },\r\n    body: data,\r\n  }).catch(() => {});\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agent ID Detection\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nconst CONFIG_PATH = path.join(os.homedir(), '.private-connect', 'config.json');\r\n\r\n/**\r\n * Detect the agent ID from local config or environment.\r\n * Returns undefined if no agent is configured — callers that need an agent ID\r\n * should surface a clear error rather than using a fake one.\r\n */\r\nfunction detectAgentId(): string | undefined {\r\n  // 1. Environment variables (highest priority — useful in CI)\r\n  const envId = process.env.PRIVATECONNECT_AGENT_ID || process.env.CONNECT_AGENT_ID;\r\n  if (envId) return envId;\r\n\r\n  // 2. Local config file (~/.private-connect/config.json)\r\n  try {\r\n    if (fs.existsSync(CONFIG_PATH)) {\r\n      const raw = fs.readFileSync(CONFIG_PATH, 'utf-8');\r\n      const config = JSON.parse(raw);\r\n      if (config.agentId && typeof config.agentId === 'string') {\r\n        return config.agentId;\r\n      }\r\n    }\r\n  } catch {\r\n    // Config unreadable or malformed — fall through\r\n  }\r\n\r\n  return undefined;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agents API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class AgentsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(options?: { onlineOnly?: boolean }): Promise<Agent[]> {\r\n    const response = await this.client.fetch('/v1/agents/orchestration');\r\n    const data = await response.json();\r\n    let agents = data.agents || [];\r\n\r\n    if (options?.onlineOnly) {\r\n      agents = agents.filter((a: any) => a.isOnline);\r\n    }\r\n\r\n    return agents.map((a: any) => ({\r\n      id: a.id,\r\n      name: a.name,\r\n      label: a.label,\r\n      isOnline: a.isOnline,\r\n      lastSeenAt: a.lastSeenAt,\r\n      capabilities: a.capabilities?.map((c: any) => c.name) || [],\r\n      services: a.services?.map((s: any) => s.name) || [],\r\n    }));\r\n  }\r\n\r\n  async findByCapability(capability: string): Promise<Agent[]> {\r\n    const response = await this.client.fetch(`/v1/agents/by-capability/${encodeURIComponent(capability)}`);\r\n    const data = await response.json();\r\n    return data.agents || [];\r\n  }\r\n\r\n  async registerCapabilities(capabilities: Array<{ name: string; metadata?: Record<string, unknown> }>): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/capabilities`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ capabilities }),\r\n    });\r\n  }\r\n\r\n  async sendMessage(\r\n    toAgentId: string,\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string; type?: 'request' | 'response' | 'event' }\r\n  ): Promise<{ messageId: string }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/send`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ toAgentId, payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async broadcast(\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string }\r\n  ): Promise<{ sent: number }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/broadcast`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async getMessages(options?: { channel?: string; unreadOnly?: boolean; limit?: number }): Promise<Message[]> {\r\n    const agentId = this.client.requireAgentId();\r\n    const params = new URLSearchParams();\r\n    if (options?.channel) params.set('channel', options.channel);\r\n    if (options?.unreadOnly !== undefined) params.set('unreadOnly', String(options.unreadOnly));\r\n    if (options?.limit) params.set('limit', String(options.limit));\r\n\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages?${params}`);\r\n    const data = await response.json();\r\n    return data.messages || [];\r\n  }\r\n\r\n  async markRead(messageIds: string[]): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/messages/read`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ messageIds }),\r\n    });\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Services API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class ServicesAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(): Promise<Service[]> {\r\n    const response = await this.client.fetch('/v1/services');\r\n    return response.json();\r\n  }\r\n\r\n  async get(name: string): Promise<Service | null> {\r\n    const services = await this.list();\r\n    return services.find(s => s.name.toLowerCase() === name.toLowerCase()) || null;\r\n  }\r\n\r\n  /**\r\n   * Get connection details for a service.\r\n   *\r\n   * If `grantToken` is provided, returns a proxied connection via the hub's\r\n   * grant endpoint instead of assuming a local tunnel.\r\n   */\r\n  async getConnection(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    if (options?.grantToken) {\r\n      return this.getGrantConnection(serviceName, options.grantToken);\r\n    }\r\n\r\n    const service = await this.get(serviceName);\r\n    if (!service) {\r\n      throw new Error(`Service \"${serviceName}\" not found`);\r\n    }\r\n\r\n    const port = service.tunnelPort || service.targetPort;\r\n    const host = 'localhost';\r\n\r\n    let connectionString = '';\r\n    let envVar = 'SERVICE_URL';\r\n\r\n    if (service.targetPort === 5432 || service.protocol === 'postgres') {\r\n      connectionString = `postgres://${host}:${port}/postgres`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 3306 || service.protocol === 'mysql') {\r\n      connectionString = `mysql://${host}:${port}`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 6379 || service.protocol === 'redis') {\r\n      connectionString = `redis://${host}:${port}`;\r\n      envVar = 'REDIS_URL';\r\n    } else if (service.targetPort === 27017 || service.protocol === 'mongodb') {\r\n      connectionString = `mongodb://${host}:${port}`;\r\n      envVar = 'MONGODB_URI';\r\n    } else if (service.protocol === 'http' || service.protocol === 'https') {\r\n      connectionString = `http://${host}:${port}`;\r\n      envVar = 'API_URL';\r\n    } else {\r\n      connectionString = `tcp://${host}:${port}`;\r\n      envVar = `${serviceName.toUpperCase().replace(/-/g, '_')}_URL`;\r\n    }\r\n\r\n    return { service: serviceName, host, port, connectionString, envVar };\r\n  }\r\n\r\n  private getGrantConnection(serviceName: string, grantToken: string): Connection {\r\n    const hubUrl = this.client.hubUrl;\r\n    const grantEndpoint = `${hubUrl}/grant/${encodeURIComponent(serviceName)}`;\r\n\r\n    return {\r\n      service: serviceName,\r\n      host: new URL(hubUrl).hostname,\r\n      port: new URL(hubUrl).port ? parseInt(new URL(hubUrl).port, 10) : 443,\r\n      connectionString: grantEndpoint,\r\n      envVar: 'GRANT_URL',\r\n      grantToken,\r\n      grantEndpoint,\r\n    };\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Grants API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class GrantsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  /**\r\n   * Create a time-limited access grant for an AI agent or external consumer.\r\n   */\r\n  async create(options: GrantCreateOptions): Promise<Grant> {\r\n    const response = await this.client.fetch('/v1/grants', {\r\n      method: 'POST',\r\n      body: JSON.stringify(options),\r\n    });\r\n    const data = await response.json();\r\n    return data.grant;\r\n  }\r\n\r\n  /**\r\n   * List active grants in the workspace.\r\n   */\r\n  async list(options?: { includeExpired?: boolean }): Promise<Grant[]> {\r\n    const params = new URLSearchParams();\r\n    if (options?.includeExpired) params.set('includeExpired', 'true');\r\n    const response = await this.client.fetch(`/v1/grants?${params}`);\r\n    const data = await response.json();\r\n    return data.grants || [];\r\n  }\r\n\r\n  /**\r\n   * Revoke an active grant immediately.\r\n   */\r\n  async revoke(grantId: string): Promise<void> {\r\n    await this.client.fetch(`/v1/grants/${grantId}`, { method: 'DELETE' });\r\n  }\r\n\r\n  /**\r\n   * Validate a grant token (public endpoint — no API key required).\r\n   * Returns the grant if valid, null otherwise.\r\n   */\r\n  async validate(token: string): Promise<Grant | null> {\r\n    try {\r\n      const url = `${this.client.hubUrl}/v1/grants/validate`;\r\n      const response = await fetch(url, {\r\n        method: 'POST',\r\n        headers: { 'Content-Type': 'application/json' },\r\n        body: JSON.stringify({ token }),\r\n      });\r\n      if (!response.ok) return null;\r\n      const data = await response.json();\r\n      return data.valid ? data.grant : null;\r\n    } catch {\r\n      return null;\r\n    }\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Main Client\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class PrivateConnect {\r\n  private config: { apiKey: string; hubUrl: string; agentId?: string };\r\n\r\n  /** Agents API for discovery and orchestration */\r\n  public agents: AgentsAPI;\r\n\r\n  /** Services API for connecting to services */\r\n  public services: ServicesAPI;\r\n\r\n  /** Grants API for managing scoped access tokens (time-limited or persistent) */\r\n  public grants: GrantsAPI;\r\n\r\n  constructor(config: PrivateConnectConfig) {\r\n    this.config = {\r\n      apiKey: config.apiKey,\r\n      hubUrl: config.hubUrl || 'https://api.privateconnect.co',\r\n      agentId: config.agentId || detectAgentId(),\r\n    };\r\n\r\n    this.agents = new AgentsAPI(this);\r\n    this.services = new ServicesAPI(this);\r\n    this.grants = new GrantsAPI(this);\r\n\r\n    if (!config.disableTracking) {\r\n      trackSdkUsage(this.config.hubUrl);\r\n    }\r\n  }\r\n\r\n  /** The resolved agent ID, or undefined if not configured. */\r\n  get agentId(): string | undefined {\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** The hub URL this client is connected to. */\r\n  get hubUrl(): string {\r\n    return this.config.hubUrl;\r\n  }\r\n\r\n  /**\r\n   * Shorthand: get connection details for a service.\r\n   * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.\r\n   */\r\n  async connect(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    return this.services.getConnection(serviceName, options);\r\n  }\r\n\r\n  /**\r\n   * Returns the agent ID or throws if not configured.\r\n   * Used by APIs that require an authenticated agent identity.\r\n   */\r\n  requireAgentId(): string {\r\n    if (!this.config.agentId) {\r\n      throw new Error(\r\n        'Agent ID not found. Either:\\n' +\r\n        '  1. Run \"connect up\" to register this machine, or\\n' +\r\n        '  2. Set PRIVATECONNECT_AGENT_ID environment variable, or\\n' +\r\n        '  3. Pass agentId in the PrivateConnect constructor.'\r\n      );\r\n    }\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** Internal fetch with API key auth. */\r\n  async fetch(path: string, options?: RequestInit): Promise<Response> {\r\n    const url = `${this.config.hubUrl}${path}`;\r\n    const response = await fetch(url, {\r\n      ...options,\r\n      headers: {\r\n        'x-api-key': this.config.apiKey,\r\n        'Content-Type': 'application/json',\r\n        ...options?.headers,\r\n      },\r\n    });\r\n\r\n    if (!response.ok) {\r\n      const error = await response.json().catch(() => ({ message: 'Unknown error' }));\r\n      throw new Error(error.message || `Request failed: ${response.status}`);\r\n    }\r\n\r\n    return response;\r\n  }\r\n}\r\n\r\nexport default PrivateConnect;\r\n\r\n/** Convenience function for quick one-off connections. */\r\nexport async function connect(\r\n  serviceName: string,\r\n  config?: PrivateConnectConfig & { grantToken?: string },\r\n): Promise<Connection> {\r\n  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;\r\n  if (!apiKey) {\r\n    throw new Error('API key required. Set PRIVATECONNECT_API_KEY or pass config.apiKey');\r\n  }\r\n\r\n  const client = new PrivateConnect({ ...config, apiKey });\r\n  return client.connect(serviceName, { grantToken: config?.grantToken });\r\n}\r\n"]}
585
+ /**
586
+ * Load pconnect.yml and get a resource by name.
587
+ * Shorthand for `PrivateConnect.fromManifest().resource(name)`.
588
+ */
589
+ function fromManifest(manifestPath) {
590
+ return PrivateConnect.fromManifest(manifestPath);
591
+ }
592
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Private Connect SDK\r\n *\r\n * Define your connections in pconnect.yml. Access them from anywhere.\r\n *\r\n * @example\r\n * ```typescript\r\n * import { PrivateConnect } from '@privateconnect/sdk';\r\n *\r\n * // Load your project's connection manifest\r\n * const pc = PrivateConnect.fromManifest();\r\n *\r\n * // Get a resource declared in pconnect.yml\r\n * const db = pc.resource('staging-db');\r\n * console.log(db.connectionString); // postgres://internal-db:5432\r\n * console.log(db.envVar);           // DATABASE_URL\r\n *\r\n * // With an API key, you also get hub API access\r\n * const pc2 = PrivateConnect.fromManifest('./pconnect.yml', {\r\n *   apiKey: process.env.PRIVATECONNECT_API_KEY,\r\n * });\r\n * const grant = await pc2.grants.create({\r\n *   agentLabel: 'claude',\r\n *   resourceType: 'db',\r\n *   resourceName: 'postgres',\r\n *   ttl: '5m',\r\n * });\r\n * ```\r\n */\r\n\r\nimport * as fs from 'fs';\r\nimport * as path from 'path';\r\nimport * as os from 'os';\r\nimport * as yaml from 'js-yaml';\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Types\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport interface PrivateConnectConfig {\r\n  /** API key for authentication. Required for hub API calls; optional for manifest-only usage. */\r\n  apiKey?: string;\r\n  /** Hub URL (default: https://api.privateconnect.co) */\r\n  hubUrl?: string;\r\n  /** Agent ID (auto-detected from local config if not provided) */\r\n  agentId?: string;\r\n  /** Disable usage tracking (default: false) */\r\n  disableTracking?: boolean;\r\n}\r\n\r\nexport interface Service {\r\n  id: string;\r\n  name: string;\r\n  targetHost: string;\r\n  targetPort: number;\r\n  tunnelPort?: number;\r\n  protocol: string;\r\n  status: string;\r\n  agentLabel?: string;\r\n}\r\n\r\nexport interface Agent {\r\n  id: string;\r\n  name?: string;\r\n  label: string;\r\n  isOnline: boolean;\r\n  lastSeenAt: string;\r\n  capabilities: string[];\r\n  services: string[];\r\n}\r\n\r\nexport interface Connection {\r\n  service: string;\r\n  host: string;\r\n  port: number;\r\n  connectionString: string;\r\n  envVar: string;\r\n  /** Present when using a grant-based connection */\r\n  grantToken?: string;\r\n  /** Present when using a grant-based connection */\r\n  grantEndpoint?: string;\r\n}\r\n\r\nexport interface Message {\r\n  id: string;\r\n  from: { id: string; name?: string; label?: string };\r\n  channel: string;\r\n  type: string;\r\n  payload: Record<string, unknown>;\r\n  createdAt: string;\r\n  isRead: boolean;\r\n}\r\n\r\nexport interface Grant {\r\n  id: string;\r\n  agentLabel: string;\r\n  resourceType: string;\r\n  resourceName: string;\r\n  scope: string;\r\n  tokenPrefix?: string;\r\n  persistent: boolean;\r\n  expiresAt: string | null;\r\n  expiresInMinutes?: number | null;\r\n  token?: string;\r\n  endpoint?: string;\r\n}\r\n\r\nexport interface GrantCreateOptions {\r\n  agentLabel: string;\r\n  resourceType: 'db' | 'api' | 'path';\r\n  resourceName: string;\r\n  scope?: 'read-only' | 'full';\r\n  /** Duration string: 60s, 5m, 1h, 1d. Omit for persistent grant. */\r\n  ttl?: string;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Manifest Types\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport const RESOURCE_TYPES = ['postgres', 'mysql', 'redis', 'http', 'generic-tcp'] as const;\r\nexport type ResourceType = typeof RESOURCE_TYPES[number];\r\n\r\nexport const ACCESS_MODES = ['tcp', 'http'] as const;\r\nexport type AccessMode = typeof ACCESS_MODES[number];\r\n\r\nexport const TRANSPORT_MODES = ['direct', 'hub'] as const;\r\nexport type TransportVia = typeof TRANSPORT_MODES[number];\r\n\r\nexport interface ManifestResourceConfig {\r\n  type: ResourceType;\r\n  host?: string;\r\n  port?: number;\r\n  targetHost?: string;\r\n  targetPort?: number;\r\n  url?: string;\r\n  access: { mode: AccessMode; via?: TransportVia };\r\n}\r\n\r\nexport interface ManifestResource {\r\n  name: string;\r\n  type: ResourceType;\r\n  host: string;\r\n  port: number;\r\n  connectionString: string;\r\n  envVar: string;\r\n  accessMode: AccessMode;\r\n  via: TransportVia;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Manifest Parsing\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nconst MANIFEST_FILENAMES = [\r\n  'pconnect.yml',\r\n  'pconnect.yaml',\r\n  'pconnect.json',\r\n  '.pconnect.yml',\r\n  '.pconnect.yaml',\r\n  '.pconnect.json',\r\n];\r\n\r\nconst DEFAULT_PORTS: Record<ResourceType, number> = {\r\n  postgres: 5432,\r\n  mysql: 3306,\r\n  redis: 6379,\r\n  http: 80,\r\n  'generic-tcp': 0,\r\n};\r\n\r\nconst PROTOCOL_SCHEMES: Record<ResourceType, string> = {\r\n  postgres: 'postgres',\r\n  mysql: 'mysql',\r\n  redis: 'redis',\r\n  http: 'http',\r\n  'generic-tcp': 'tcp',\r\n};\r\n\r\nconst ENV_VAR_MAP: Record<ResourceType, string> = {\r\n  postgres: 'DATABASE_URL',\r\n  mysql: 'DATABASE_URL',\r\n  redis: 'REDIS_URL',\r\n  http: 'API_URL',\r\n  'generic-tcp': 'TCP_URL',\r\n};\r\n\r\nfunction findManifest(startDir?: string): string | null {\r\n  let dir = startDir || process.cwd();\r\n\r\n  for (let depth = 0; depth < 4; depth++) {\r\n    for (const filename of MANIFEST_FILENAMES) {\r\n      const filePath = path.join(dir, filename);\r\n      if (fs.existsSync(filePath)) return filePath;\r\n    }\r\n    const parent = path.dirname(dir);\r\n    if (parent === dir) break;\r\n    dir = parent;\r\n  }\r\n\r\n  return null;\r\n}\r\n\r\nfunction parseManifestFile(filePath: string): Record<string, ManifestResourceConfig> {\r\n  const content = fs.readFileSync(filePath, 'utf-8');\r\n  let raw: Record<string, unknown>;\r\n\r\n  if (filePath.endsWith('.json')) {\r\n    raw = JSON.parse(content);\r\n  } else {\r\n    raw = yaml.load(content) as Record<string, unknown>;\r\n  }\r\n\r\n  if (!raw || typeof raw !== 'object' || !raw.resources || typeof raw.resources !== 'object') {\r\n    return {};\r\n  }\r\n\r\n  const resources: Record<string, ManifestResourceConfig> = {};\r\n  const rawResources = raw.resources as Record<string, unknown>;\r\n\r\n  for (const [name, value] of Object.entries(rawResources)) {\r\n    if (!value || typeof value !== 'object') continue;\r\n    const obj = value as Record<string, unknown>;\r\n\r\n    const type = obj.type as string;\r\n    if (!RESOURCE_TYPES.includes(type as ResourceType)) continue;\r\n\r\n    const accessObj = (obj.access && typeof obj.access === 'object')\r\n      ? obj.access as Record<string, unknown>\r\n      : { mode: 'tcp' };\r\n\r\n    resources[name] = {\r\n      type: type as ResourceType,\r\n      host: typeof obj.host === 'string' ? obj.host : undefined,\r\n      port: typeof obj.port === 'number' ? obj.port : undefined,\r\n      targetHost: typeof obj.targetHost === 'string' ? obj.targetHost : undefined,\r\n      targetPort: typeof obj.targetPort === 'number' ? obj.targetPort : undefined,\r\n      url: typeof obj.url === 'string' ? obj.url : undefined,\r\n      access: {\r\n        mode: (ACCESS_MODES.includes(accessObj.mode as AccessMode)\r\n          ? accessObj.mode\r\n          : 'tcp') as AccessMode,\r\n        via: (TRANSPORT_MODES.includes(accessObj.via as TransportVia)\r\n          ? accessObj.via\r\n          : 'direct') as TransportVia,\r\n      },\r\n    };\r\n  }\r\n\r\n  return resources;\r\n}\r\n\r\nfunction resolveManifestResource(name: string, config: ManifestResourceConfig): ManifestResource {\r\n  let host: string;\r\n  let port: number;\r\n\r\n  if (config.type === 'http' && config.url) {\r\n    try {\r\n      const parsed = new URL(config.url);\r\n      host = parsed.hostname;\r\n      port = parsed.port ? parseInt(parsed.port, 10) : (parsed.protocol === 'https:' ? 443 : 80);\r\n    } catch {\r\n      host = config.host || config.targetHost || 'localhost';\r\n      port = config.port || config.targetPort || DEFAULT_PORTS[config.type];\r\n    }\r\n  } else {\r\n    host = config.targetHost || config.host || 'localhost';\r\n    port = config.targetPort || config.port || DEFAULT_PORTS[config.type];\r\n  }\r\n\r\n  const scheme = PROTOCOL_SCHEMES[config.type];\r\n  const connectionString = `${scheme}://${host}:${port}`;\r\n\r\n  const envVar = ENV_VAR_MAP[config.type]\r\n    || `${name.toUpperCase().replace(/-/g, '_')}_URL`;\r\n\r\n  return {\r\n    name,\r\n    type: config.type,\r\n    host,\r\n    port,\r\n    connectionString,\r\n    envVar,\r\n    accessMode: config.access.mode,\r\n    via: config.access.via || 'direct',\r\n  };\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Tracking\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nfunction trackSdkUsage(hubUrl: string): void {\r\n  const data = JSON.stringify({\r\n    os: typeof process !== 'undefined' ? process.platform : 'browser',\r\n    arch: typeof process !== 'undefined' ? (process.arch === 'arm64' ? 'arm64' : 'x64') : 'unknown',\r\n    version: 'sdk',\r\n    source: 'sdk',\r\n  });\r\n\r\n  fetch(`${hubUrl}/v1/events/install`, {\r\n    method: 'POST',\r\n    headers: { 'Content-Type': 'application/json' },\r\n    body: data,\r\n  }).catch(() => {});\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agent ID Detection\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nconst CONFIG_PATH = path.join(os.homedir(), '.private-connect', 'config.json');\r\n\r\n/**\r\n * Detect the agent ID from local config or environment.\r\n * Returns undefined if no agent is configured — callers that need an agent ID\r\n * should surface a clear error rather than using a fake one.\r\n */\r\nfunction detectAgentId(): string | undefined {\r\n  // 1. Environment variables (highest priority — useful in CI)\r\n  const envId = process.env.PRIVATECONNECT_AGENT_ID || process.env.CONNECT_AGENT_ID;\r\n  if (envId) return envId;\r\n\r\n  // 2. Local config file (~/.private-connect/config.json)\r\n  try {\r\n    if (fs.existsSync(CONFIG_PATH)) {\r\n      const raw = fs.readFileSync(CONFIG_PATH, 'utf-8');\r\n      const config = JSON.parse(raw);\r\n      if (config.agentId && typeof config.agentId === 'string') {\r\n        return config.agentId;\r\n      }\r\n    }\r\n  } catch {\r\n    // Config unreadable or malformed — fall through\r\n  }\r\n\r\n  return undefined;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agents API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class AgentsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(options?: { onlineOnly?: boolean }): Promise<Agent[]> {\r\n    const response = await this.client.fetch('/v1/agents/orchestration');\r\n    const data = await response.json();\r\n    let agents = data.agents || [];\r\n\r\n    if (options?.onlineOnly) {\r\n      agents = agents.filter((a: any) => a.isOnline);\r\n    }\r\n\r\n    return agents.map((a: any) => ({\r\n      id: a.id,\r\n      name: a.name,\r\n      label: a.label,\r\n      isOnline: a.isOnline,\r\n      lastSeenAt: a.lastSeenAt,\r\n      capabilities: a.capabilities?.map((c: any) => c.name) || [],\r\n      services: a.services?.map((s: any) => s.name) || [],\r\n    }));\r\n  }\r\n\r\n  async findByCapability(capability: string): Promise<Agent[]> {\r\n    const response = await this.client.fetch(`/v1/agents/by-capability/${encodeURIComponent(capability)}`);\r\n    const data = await response.json();\r\n    return data.agents || [];\r\n  }\r\n\r\n  async registerCapabilities(capabilities: Array<{ name: string; metadata?: Record<string, unknown> }>): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/capabilities`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ capabilities }),\r\n    });\r\n  }\r\n\r\n  async sendMessage(\r\n    toAgentId: string,\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string; type?: 'request' | 'response' | 'event' }\r\n  ): Promise<{ messageId: string }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/send`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ toAgentId, payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async broadcast(\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string }\r\n  ): Promise<{ sent: number }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/broadcast`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async getMessages(options?: { channel?: string; unreadOnly?: boolean; limit?: number }): Promise<Message[]> {\r\n    const agentId = this.client.requireAgentId();\r\n    const params = new URLSearchParams();\r\n    if (options?.channel) params.set('channel', options.channel);\r\n    if (options?.unreadOnly !== undefined) params.set('unreadOnly', String(options.unreadOnly));\r\n    if (options?.limit) params.set('limit', String(options.limit));\r\n\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages?${params}`);\r\n    const data = await response.json();\r\n    return data.messages || [];\r\n  }\r\n\r\n  async markRead(messageIds: string[]): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/messages/read`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ messageIds }),\r\n    });\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Services API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class ServicesAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(): Promise<Service[]> {\r\n    const response = await this.client.fetch('/v1/services');\r\n    return response.json();\r\n  }\r\n\r\n  async get(name: string): Promise<Service | null> {\r\n    const services = await this.list();\r\n    return services.find(s => s.name.toLowerCase() === name.toLowerCase()) || null;\r\n  }\r\n\r\n  /**\r\n   * Get connection details for a service.\r\n   *\r\n   * If `grantToken` is provided, returns a proxied connection via the hub's\r\n   * grant endpoint instead of assuming a local tunnel.\r\n   */\r\n  async getConnection(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    if (options?.grantToken) {\r\n      return this.getGrantConnection(serviceName, options.grantToken);\r\n    }\r\n\r\n    const service = await this.get(serviceName);\r\n    if (!service) {\r\n      throw new Error(`Service \"${serviceName}\" not found`);\r\n    }\r\n\r\n    const port = service.tunnelPort || service.targetPort;\r\n    const host = 'localhost';\r\n\r\n    let connectionString: string;\r\n    let envVar: string;\r\n\r\n    if (service.targetPort === 5432 || service.protocol === 'postgres') {\r\n      connectionString = `postgres://${host}:${port}/postgres`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 3306 || service.protocol === 'mysql') {\r\n      connectionString = `mysql://${host}:${port}`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 6379 || service.protocol === 'redis') {\r\n      connectionString = `redis://${host}:${port}`;\r\n      envVar = 'REDIS_URL';\r\n    } else if (service.targetPort === 27017 || service.protocol === 'mongodb') {\r\n      connectionString = `mongodb://${host}:${port}`;\r\n      envVar = 'MONGODB_URI';\r\n    } else if (service.protocol === 'http' || service.protocol === 'https') {\r\n      connectionString = `http://${host}:${port}`;\r\n      envVar = 'API_URL';\r\n    } else {\r\n      connectionString = `tcp://${host}:${port}`;\r\n      envVar = `${serviceName.toUpperCase().replace(/-/g, '_')}_URL`;\r\n    }\r\n\r\n    return { service: serviceName, host, port, connectionString, envVar };\r\n  }\r\n\r\n  private getGrantConnection(serviceName: string, grantToken: string): Connection {\r\n    const hubUrl = this.client.hubUrl;\r\n    const grantEndpoint = `${hubUrl}/grant/${encodeURIComponent(serviceName)}`;\r\n\r\n    return {\r\n      service: serviceName,\r\n      host: new URL(hubUrl).hostname,\r\n      port: new URL(hubUrl).port ? parseInt(new URL(hubUrl).port, 10) : 443,\r\n      connectionString: grantEndpoint,\r\n      envVar: 'GRANT_URL',\r\n      grantToken,\r\n      grantEndpoint,\r\n    };\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Grants API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class GrantsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  /**\r\n   * Create a time-limited access grant for an AI agent or external consumer.\r\n   */\r\n  async create(options: GrantCreateOptions): Promise<Grant> {\r\n    const response = await this.client.fetch('/v1/grants', {\r\n      method: 'POST',\r\n      body: JSON.stringify(options),\r\n    });\r\n    const data = await response.json();\r\n    return data.grant;\r\n  }\r\n\r\n  /**\r\n   * List active grants in the workspace.\r\n   */\r\n  async list(options?: { includeExpired?: boolean }): Promise<Grant[]> {\r\n    const params = new URLSearchParams();\r\n    if (options?.includeExpired) params.set('includeExpired', 'true');\r\n    const response = await this.client.fetch(`/v1/grants?${params}`);\r\n    const data = await response.json();\r\n    return data.grants || [];\r\n  }\r\n\r\n  /**\r\n   * Revoke an active grant immediately.\r\n   */\r\n  async revoke(grantId: string): Promise<void> {\r\n    await this.client.fetch(`/v1/grants/${grantId}`, { method: 'DELETE' });\r\n  }\r\n\r\n  /**\r\n   * Validate a grant token (public endpoint — no API key required).\r\n   * Returns the grant if valid, null otherwise.\r\n   */\r\n  async validate(token: string): Promise<Grant | null> {\r\n    try {\r\n      const url = `${this.client.hubUrl}/v1/grants/validate`;\r\n      const response = await fetch(url, {\r\n        method: 'POST',\r\n        headers: { 'Content-Type': 'application/json' },\r\n        body: JSON.stringify({ token }),\r\n      });\r\n      if (!response.ok) return null;\r\n      const data = await response.json();\r\n      return data.valid ? data.grant : null;\r\n    } catch {\r\n      return null;\r\n    }\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Main Client\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class PrivateConnect {\r\n  private config: { apiKey?: string; hubUrl: string; agentId?: string };\r\n  private manifest: Map<string, ManifestResource> = new Map();\r\n  private manifestPath?: string;\r\n\r\n  /** Agents API for discovery and orchestration */\r\n  public agents: AgentsAPI;\r\n\r\n  /** Services API for connecting to services */\r\n  public services: ServicesAPI;\r\n\r\n  /** Grants API for managing scoped access tokens (time-limited or persistent) */\r\n  public grants: GrantsAPI;\r\n\r\n  constructor(config: PrivateConnectConfig = {}) {\r\n    this.config = {\r\n      apiKey: config.apiKey,\r\n      hubUrl: config.hubUrl || 'https://api.privateconnect.co',\r\n      agentId: config.agentId || detectAgentId(),\r\n    };\r\n\r\n    this.agents = new AgentsAPI(this);\r\n    this.services = new ServicesAPI(this);\r\n    this.grants = new GrantsAPI(this);\r\n\r\n    if (!config.disableTracking) {\r\n      trackSdkUsage(this.config.hubUrl);\r\n    }\r\n  }\r\n\r\n  // ─────────────────────────────────────────────────────────────────────────\r\n  // Manifest API — the primary way to use the SDK\r\n  // ─────────────────────────────────────────────────────────────────────────\r\n\r\n  /**\r\n   * Load a pconnect.yml manifest and return a configured client.\r\n   *\r\n   * Auto-discovers pconnect.yml in the current directory (or parents) if no\r\n   * path is given. Pass a config to also enable hub API access.\r\n   *\r\n   * @example\r\n   * ```typescript\r\n   * const pc = PrivateConnect.fromManifest();\r\n   * const db = pc.resource('staging-db');\r\n   * console.log(db.connectionString); // postgres://internal-db:5432\r\n   * ```\r\n   */\r\n  static fromManifest(\r\n    manifestPath?: string,\r\n    config?: PrivateConnectConfig,\r\n  ): PrivateConnect {\r\n    const resolvedPath = manifestPath\r\n      ? path.resolve(manifestPath)\r\n      : findManifest();\r\n\r\n    if (!resolvedPath) {\r\n      throw new Error(\r\n        'No pconnect.yml found. Create one in your project root:\\n\\n' +\r\n        '  resources:\\n' +\r\n        '    staging-db:\\n' +\r\n        '      type: postgres\\n' +\r\n        '      host: internal-db\\n' +\r\n        '      port: 5432\\n' +\r\n        '      access:\\n' +\r\n        '        mode: tcp\\n'\r\n      );\r\n    }\r\n\r\n    const instance = new PrivateConnect(config || { disableTracking: true });\r\n    instance.manifestPath = resolvedPath;\r\n\r\n    const rawResources = parseManifestFile(resolvedPath);\r\n    for (const [name, rawConfig] of Object.entries(rawResources)) {\r\n      instance.manifest.set(name, resolveManifestResource(name, rawConfig));\r\n    }\r\n\r\n    return instance;\r\n  }\r\n\r\n  /**\r\n   * Get a resource declared in pconnect.yml by name.\r\n   *\r\n   * Returns its type, host, port, connection string, and suggested env var.\r\n   * When `connect dev` is running, the connection string points to a live\r\n   * local tunnel.\r\n   */\r\n  resource(name: string): ManifestResource {\r\n    const r = this.manifest.get(name);\r\n    if (!r) {\r\n      const available = Array.from(this.manifest.keys());\r\n      const msg = available.length\r\n        ? `Available: ${available.join(', ')}`\r\n        : 'No resources loaded. Did you call PrivateConnect.fromManifest()?';\r\n      throw new Error(`Resource \"${name}\" not found. ${msg}`);\r\n    }\r\n    return r;\r\n  }\r\n\r\n  /**\r\n   * List all resources declared in the loaded manifest.\r\n   */\r\n  resources(): ManifestResource[] {\r\n    return Array.from(this.manifest.values());\r\n  }\r\n\r\n  /**\r\n   * Generate a `.env`-compatible block for all manifest resources.\r\n   *\r\n   * @example\r\n   * ```typescript\r\n   * const pc = PrivateConnect.fromManifest();\r\n   * console.log(pc.envBlock());\r\n   * // DATABASE_URL=postgres://internal-db:5432\r\n   * // REDIS_URL=redis://redis.internal:6379\r\n   * ```\r\n   */\r\n  envBlock(): string {\r\n    return this.resources()\r\n      .map(r => `${r.envVar}=${r.connectionString}`)\r\n      .join('\\n');\r\n  }\r\n\r\n  // ─────────────────────────────────────────────────────────────────────────\r\n  // Existing APIs\r\n  // ─────────────────────────────────────────────────────────────────────────\r\n\r\n  /** The resolved agent ID, or undefined if not configured. */\r\n  get agentId(): string | undefined {\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** The hub URL this client is connected to. */\r\n  get hubUrl(): string {\r\n    return this.config.hubUrl;\r\n  }\r\n\r\n  /**\r\n   * Shorthand: get connection details for a service.\r\n   * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.\r\n   */\r\n  async connect(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    return this.services.getConnection(serviceName, options);\r\n  }\r\n\r\n  /**\r\n   * Returns the agent ID or throws if not configured.\r\n   * Used by APIs that require an authenticated agent identity.\r\n   */\r\n  requireAgentId(): string {\r\n    if (!this.config.agentId) {\r\n      throw new Error(\r\n        'Agent ID not found. Either:\\n' +\r\n        '  1. Run \"connect up\" to register this machine, or\\n' +\r\n        '  2. Set PRIVATECONNECT_AGENT_ID environment variable, or\\n' +\r\n        '  3. Pass agentId in the PrivateConnect constructor.'\r\n      );\r\n    }\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** Requires an API key or throws. */\r\n  private requireApiKey(): string {\r\n    if (!this.config.apiKey) {\r\n      throw new Error(\r\n        'API key required for hub API calls. Either:\\n' +\r\n        '  1. Set PRIVATECONNECT_API_KEY environment variable, or\\n' +\r\n        '  2. Pass apiKey in the config.'\r\n      );\r\n    }\r\n    return this.config.apiKey;\r\n  }\r\n\r\n  /** Internal fetch with API key auth. */\r\n  async fetch(urlPath: string, options?: RequestInit): Promise<Response> {\r\n    const apiKey = this.requireApiKey();\r\n    const url = `${this.config.hubUrl}${urlPath}`;\r\n    const response = await fetch(url, {\r\n      ...options,\r\n      headers: {\r\n        'x-api-key': apiKey,\r\n        'Content-Type': 'application/json',\r\n        ...options?.headers,\r\n      },\r\n    });\r\n\r\n    if (!response.ok) {\r\n      const error = await response.json().catch(() => ({ message: 'Unknown error' }));\r\n      throw new Error(error.message || `Request failed: ${response.status}`);\r\n    }\r\n\r\n    return response;\r\n  }\r\n}\r\n\r\nexport default PrivateConnect;\r\n\r\n/** Convenience function for quick one-off connections via the hub API. */\r\nexport async function connect(\r\n  serviceName: string,\r\n  config?: PrivateConnectConfig & { grantToken?: string },\r\n): Promise<Connection> {\r\n  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;\r\n  if (!apiKey) {\r\n    throw new Error('API key required. Set PRIVATECONNECT_API_KEY or pass config.apiKey');\r\n  }\r\n\r\n  const client = new PrivateConnect({ ...config, apiKey });\r\n  return client.connect(serviceName, { grantToken: config?.grantToken });\r\n}\r\n\r\n/**\r\n * Load pconnect.yml and get a resource by name.\r\n * Shorthand for `PrivateConnect.fromManifest().resource(name)`.\r\n */\r\nexport function fromManifest(manifestPath?: string): PrivateConnect {\r\n  return PrivateConnect.fromManifest(manifestPath);\r\n}\r\n"]}
package/package.json CHANGED
@@ -1,7 +1,7 @@
1
1
  {
2
2
  "name": "@privateconnect/sdk",
3
- "version": "0.7.4",
4
- "description": "TypeScript SDK for Private Connect - programmatic access to services and agent orchestration",
3
+ "version": "0.7.7",
4
+ "description": "Define your connections in pconnect.yml. Access them from anywhere.",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "files": [
@@ -14,10 +14,11 @@
14
14
  },
15
15
  "keywords": [
16
16
  "private-connect",
17
+ "connectivity",
18
+ "manifest",
19
+ "infrastructure",
17
20
  "tunnel",
18
- "networking",
19
- "orchestration",
20
- "agent"
21
+ "orchestration"
21
22
  ],
22
23
  "license": "FSL-1.1-MIT",
23
24
  "repository": {
@@ -26,8 +27,11 @@
26
27
  "directory": "packages/sdk"
27
28
  },
28
29
  "devDependencies": {
30
+ "@types/js-yaml": "^4.0.9",
29
31
  "@types/node": "^20.0.0",
30
32
  "typescript": "^5.0.0"
33
+ },
34
+ "dependencies": {
35
+ "js-yaml": "^4.1.1"
31
36
  }
32
37
  }
33
-