@privateconnect/sdk 0.7.5 → 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 +192 -174
- package/dist/index.d.ts +92 -16
- package/dist/index.js +251 -20
- package/package.json +10 -6
package/README.md
CHANGED
|
@@ -1,174 +1,192 @@
|
|
|
1
|
-
# Private Connect SDK
|
|
2
|
-
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
##
|
|
6
|
-
|
|
7
|
-
```bash
|
|
8
|
-
npm install @privateconnect/sdk
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
```
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
```
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
51
|
-
|
|
52
|
-
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
|
|
62
|
-
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
69
|
-
|
|
70
|
-
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
87
|
-
|
|
88
|
-
|
|
89
|
-
|
|
90
|
-
|
|
91
|
-
|
|
92
|
-
//
|
|
93
|
-
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
//
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
|
|
153
|
-
|
|
154
|
-
|
|
155
|
-
|
|
156
|
-
|
|
157
|
-
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
163
|
-
|
|
164
|
-
|
|
165
|
-
|
|
166
|
-
|
|
167
|
-
|
|
168
|
-
|
|
169
|
-
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
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
|
-
*
|
|
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
|
-
*
|
|
10
|
+
* // Load your project's connection manifest
|
|
11
|
+
* const pc = PrivateConnect.fromManifest();
|
|
11
12
|
*
|
|
12
|
-
* //
|
|
13
|
-
* const db =
|
|
14
|
-
* console.log(db.connectionString); // postgres://
|
|
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
|
-
* //
|
|
17
|
-
* const
|
|
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
|
|
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
|
|
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(
|
|
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
|
-
*
|
|
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
|
-
*
|
|
11
|
+
* // Load your project's connection manifest
|
|
12
|
+
* const pc = PrivateConnect.fromManifest();
|
|
12
13
|
*
|
|
13
|
-
* //
|
|
14
|
-
* const db =
|
|
15
|
-
* console.log(db.connectionString); // postgres://
|
|
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
|
-
* //
|
|
18
|
-
* const
|
|
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
|
|
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(
|
|
334
|
-
const
|
|
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':
|
|
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":";AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2BG;;;AAmcH,0BAWC;AA5cD,yBAAyB;AACzB,6BAA6B;AAC7B,yBAAyB;AAmFzB,gFAAgF;AAChF,WAAW;AACX,gFAAgF;AAEhF,SAAS,aAAa,CAAC,MAAc;IACnC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC;QAC1B,EAAE,EAAE,OAAO,OAAO,KAAK,WAAW,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS;QACjE,IAAI,EAAE,OAAO,OAAO,KAAK,WAAW,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS;QAC/F,OAAO,EAAE,KAAK;QACd,MAAM,EAAE,KAAK;KACd,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,MAAM,oBAAoB,EAAE;QACnC,MAAM,EAAE,MAAM;QACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;QAC/C,IAAI,EAAE,IAAI;KACX,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;AACrB,CAAC;AAED,gFAAgF;AAChF,qBAAqB;AACrB,gFAAgF;AAEhF,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,kBAAkB,EAAE,aAAa,CAAC,CAAC;AAE/E;;;;GAIG;AACH,SAAS,aAAa;IACpB,6DAA6D;IAC7D,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC;IAClF,IAAI,KAAK;QAAE,OAAO,KAAK,CAAC;IAExB,wDAAwD;IACxD,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YAClD,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC/B,IAAI,MAAM,CAAC,OAAO,IAAI,OAAO,MAAM,CAAC,OAAO,KAAK,QAAQ,EAAE,CAAC;gBACzD,OAAO,MAAM,CAAC,OAAO,CAAC;YACxB,CAAC;QACH,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,gDAAgD;IAClD,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,gFAAgF;AAChF,aAAa;AACb,gFAAgF;AAEhF,MAAa,SAAS;IACpB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C,KAAK,CAAC,IAAI,CAAC,OAAkC;QAC3C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;QACrE,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,IAAI,MAAM,GAAG,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;QAE/B,IAAI,OAAO,EAAE,UAAU,EAAE,CAAC;YACxB,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;YAC7B,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,IAAI,EAAE,CAAC,CAAC,IAAI;YACZ,KAAK,EAAE,CAAC,CAAC,KAAK;YACd,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,UAAU,EAAE,CAAC,CAAC,UAAU;YACxB,YAAY,EAAE,CAAC,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE;YAC3D,QAAQ,EAAE,CAAC,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE;SACpD,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,UAAkB;QACvC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,4BAA4B,kBAAkB,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QACvG,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;IAC3B,CAAC;IAED,KAAK,CAAC,oBAAoB,CAAC,YAAyE;QAClG,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,eAAe,EAAE;YAC5D,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,YAAY,EAAE,CAAC;SACvC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,WAAW,CACf,SAAiB,EACjB,OAAgC,EAChC,OAAuE;QAEvE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,gBAAgB,EAAE;YAC9E,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC;SACzD,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,SAAS,CACb,OAAgC,EAChC,OAA8B;QAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,qBAAqB,EAAE;YACnF,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC;SAC9C,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAoE;QACpF,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QACrC,IAAI,OAAO,EAAE,OAAO;YAAE,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;QAC7D,IAAI,OAAO,EAAE,UAAU,KAAK,SAAS;YAAE,MAAM,CAAC,GAAG,CAAC,YAAY,EAAE,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;QAC5F,IAAI,OAAO,EAAE,KAAK;YAAE,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QAE/D,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,aAAa,MAAM,EAAE,CAAC,CAAC;QACrF,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;IAC7B,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,UAAoB;QACjC,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,gBAAgB,EAAE;YAC7D,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,CAAC;SACrC,CAAC,CAAC;IACL,CAAC;CACF;AAjFD,8BAiFC;AAED,gFAAgF;AAChF,eAAe;AACf,gFAAgF;AAEhF,MAAa,WAAW;IACtB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C,KAAK,CAAC,IAAI;QACR,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACzD,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,GAAG,CAAC,IAAY;QACpB,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,IAAI,CAAC;IACjF,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,aAAa,CAAC,WAAmB,EAAE,OAAiC;QACxE,IAAI,OAAO,EAAE,UAAU,EAAE,CAAC;YACxB,OAAO,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC;QAClE,CAAC;QAED,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,YAAY,WAAW,aAAa,CAAC,CAAC;QACxD,CAAC;QAED,MAAM,IAAI,GAAG,OAAO,CAAC,UAAU,IAAI,OAAO,CAAC,UAAU,CAAC;QACtD,MAAM,IAAI,GAAG,WAAW,CAAC;QAEzB,IAAI,gBAAgB,GAAG,EAAE,CAAC;QAC1B,IAAI,MAAM,GAAG,aAAa,CAAC;QAE3B,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,UAAU,EAAE,CAAC;YACnE,gBAAgB,GAAG,cAAc,IAAI,IAAI,IAAI,WAAW,CAAC;YACzD,MAAM,GAAG,cAAc,CAAC;QAC1B,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,WAAW,IAAI,IAAI,IAAI,EAAE,CAAC;YAC7C,MAAM,GAAG,cAAc,CAAC;QAC1B,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,WAAW,IAAI,IAAI,IAAI,EAAE,CAAC;YAC7C,MAAM,GAAG,WAAW,CAAC;QACvB,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,KAAK,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC1E,gBAAgB,GAAG,aAAa,IAAI,IAAI,IAAI,EAAE,CAAC;YAC/C,MAAM,GAAG,aAAa,CAAC;QACzB,CAAC;aAAM,IAAI,OAAO,CAAC,QAAQ,KAAK,MAAM,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,UAAU,IAAI,IAAI,IAAI,EAAE,CAAC;YAC5C,MAAM,GAAG,SAAS,CAAC;QACrB,CAAC;aAAM,CAAC;YACN,gBAAgB,GAAG,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC;YAC3C,MAAM,GAAG,GAAG,WAAW,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC;QACjE,CAAC;QAED,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE,gBAAgB,EAAE,MAAM,EAAE,CAAC;IACxE,CAAC;IAEO,kBAAkB,CAAC,WAAmB,EAAE,UAAkB;QAChE,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;QAClC,MAAM,aAAa,GAAG,GAAG,MAAM,UAAU,kBAAkB,CAAC,WAAW,CAAC,EAAE,CAAC;QAE3E,OAAO;YACL,OAAO,EAAE,WAAW;YACpB,IAAI,EAAE,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,QAAQ;YAC9B,IAAI,EAAE,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG;YACrE,gBAAgB,EAAE,aAAa;YAC/B,MAAM,EAAE,WAAW;YACnB,UAAU;YACV,aAAa;SACd,CAAC;IACJ,CAAC;CACF;AAxED,kCAwEC;AAED,gFAAgF;AAChF,aAAa;AACb,gFAAgF;AAEhF,MAAa,SAAS;IACpB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAA2B;QACtC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE;YACrD,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;SAC9B,CAAC,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAI,CAAC,OAAsC;QAC/C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QACrC,IAAI,OAAO,EAAE,cAAc;YAAE,MAAM,CAAC,GAAG,CAAC,gBAAgB,EAAE,MAAM,CAAC,CAAC;QAClE,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,MAAM,EAAE,CAAC,CAAC;QACjE,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;IAC3B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAAe;QAC1B,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,EAAE,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC;IACzE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,QAAQ,CAAC,KAAa;QAC1B,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,qBAAqB,CAAC;YACvD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;gBAChC,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;gBAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,CAAC;aAChC,CAAC,CAAC;YACH,IAAI,CAAC,QAAQ,CAAC,EAAE;gBAAE,OAAO,IAAI,CAAC;YAC9B,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;QACxC,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF;AApDD,8BAoDC;AAED,gFAAgF;AAChF,cAAc;AACd,gFAAgF;AAEhF,MAAa,cAAc;IAYzB,YAAY,MAA4B;QACtC,IAAI,CAAC,MAAM,GAAG;YACZ,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,+BAA+B;YACxD,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,aAAa,EAAE;SAC3C,CAAC;QAEF,IAAI,CAAC,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,QAAQ,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,CAAC;QAElC,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;YAC5B,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED,6DAA6D;IAC7D,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;IAC7B,CAAC;IAED,+CAA+C;IAC/C,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,OAAO,CAAC,WAAmB,EAAE,OAAiC;QAClE,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IAC3D,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CACb,+BAA+B;gBAC/B,sDAAsD;gBACtD,6DAA6D;gBAC7D,sDAAsD,CACvD,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;IAC7B,CAAC;IAED,wCAAwC;IACxC,KAAK,CAAC,KAAK,CAAC,IAAY,EAAE,OAAqB;QAC7C,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC;QAC3C,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;YAChC,GAAG,OAAO;YACV,OAAO,EAAE;gBACP,WAAW,EAAE,IAAI,CAAC,MAAM,CAAC,MAAM;gBAC/B,cAAc,EAAE,kBAAkB;gBAClC,GAAG,OAAO,EAAE,OAAO;aACpB;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,KAAK,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,eAAe,EAAE,CAAC,CAAC,CAAC;YAChF,MAAM,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,IAAI,mBAAmB,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC;QACzE,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF;AAjFD,wCAiFC;AAED,kBAAe,cAAc,CAAC;AAE9B,0DAA0D;AACnD,KAAK,UAAU,OAAO,CAC3B,WAAmB,EACnB,MAAuD;IAEvD,MAAM,MAAM,GAAG,MAAM,EAAE,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC;IACpE,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CAAC,oEAAoE,CAAC,CAAC;IACxF,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,cAAc,CAAC,EAAE,GAAG,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACzD,OAAO,MAAM,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,CAAC,CAAC;AACzE,CAAC","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":";AAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;;;AA+tBH,0BAWC;AAMD,oCAEC;AAhvBD,yBAAyB;AACzB,6BAA6B;AAC7B,yBAAyB;AACzB,gCAAgC;AAmFhC,gFAAgF;AAChF,iBAAiB;AACjB,gFAAgF;AAEnE,QAAA,cAAc,GAAG,CAAC,UAAU,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,EAAE,aAAa,CAAU,CAAC;AAGhF,QAAA,YAAY,GAAG,CAAC,KAAK,EAAE,MAAM,CAAU,CAAC;AAGxC,QAAA,eAAe,GAAG,CAAC,QAAQ,EAAE,KAAK,CAAU,CAAC;AAwB1D,gFAAgF;AAChF,mBAAmB;AACnB,gFAAgF;AAEhF,MAAM,kBAAkB,GAAG;IACzB,cAAc;IACd,eAAe;IACf,eAAe;IACf,eAAe;IACf,gBAAgB;IAChB,gBAAgB;CACjB,CAAC;AAEF,MAAM,aAAa,GAAiC;IAClD,QAAQ,EAAE,IAAI;IACd,KAAK,EAAE,IAAI;IACX,KAAK,EAAE,IAAI;IACX,IAAI,EAAE,EAAE;IACR,aAAa,EAAE,CAAC;CACjB,CAAC;AAEF,MAAM,gBAAgB,GAAiC;IACrD,QAAQ,EAAE,UAAU;IACpB,KAAK,EAAE,OAAO;IACd,KAAK,EAAE,OAAO;IACd,IAAI,EAAE,MAAM;IACZ,aAAa,EAAE,KAAK;CACrB,CAAC;AAEF,MAAM,WAAW,GAAiC;IAChD,QAAQ,EAAE,cAAc;IACxB,KAAK,EAAE,cAAc;IACrB,KAAK,EAAE,WAAW;IAClB,IAAI,EAAE,SAAS;IACf,aAAa,EAAE,SAAS;CACzB,CAAC;AAEF,SAAS,YAAY,CAAC,QAAiB;IACrC,IAAI,GAAG,GAAG,QAAQ,IAAI,OAAO,CAAC,GAAG,EAAE,CAAC;IAEpC,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,CAAC,EAAE,KAAK,EAAE,EAAE,CAAC;QACvC,KAAK,MAAM,QAAQ,IAAI,kBAAkB,EAAE,CAAC;YAC1C,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,QAAQ,CAAC,CAAC;YAC1C,IAAI,EAAE,CAAC,UAAU,CAAC,QAAQ,CAAC;gBAAE,OAAO,QAAQ,CAAC;QAC/C,CAAC;QACD,MAAM,MAAM,GAAG,IAAI,CAAC,OAAO,CAAC,GAAG,CAAC,CAAC;QACjC,IAAI,MAAM,KAAK,GAAG;YAAE,MAAM;QAC1B,GAAG,GAAG,MAAM,CAAC;IACf,CAAC;IAED,OAAO,IAAI,CAAC;AACd,CAAC;AAED,SAAS,iBAAiB,CAAC,QAAgB;IACzC,MAAM,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IACnD,IAAI,GAA4B,CAAC;IAEjC,IAAI,QAAQ,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;QAC/B,GAAG,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;IAC5B,CAAC;SAAM,CAAC;QACN,GAAG,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAA4B,CAAC;IACtD,CAAC;IAED,IAAI,CAAC,GAAG,IAAI,OAAO,GAAG,KAAK,QAAQ,IAAI,CAAC,GAAG,CAAC,SAAS,IAAI,OAAO,GAAG,CAAC,SAAS,KAAK,QAAQ,EAAE,CAAC;QAC3F,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,MAAM,SAAS,GAA2C,EAAE,CAAC;IAC7D,MAAM,YAAY,GAAG,GAAG,CAAC,SAAoC,CAAC;IAE9D,KAAK,MAAM,CAAC,IAAI,EAAE,KAAK,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;QACzD,IAAI,CAAC,KAAK,IAAI,OAAO,KAAK,KAAK,QAAQ;YAAE,SAAS;QAClD,MAAM,GAAG,GAAG,KAAgC,CAAC;QAE7C,MAAM,IAAI,GAAG,GAAG,CAAC,IAAc,CAAC;QAChC,IAAI,CAAC,sBAAc,CAAC,QAAQ,CAAC,IAAoB,CAAC;YAAE,SAAS;QAE7D,MAAM,SAAS,GAAG,CAAC,GAAG,CAAC,MAAM,IAAI,OAAO,GAAG,CAAC,MAAM,KAAK,QAAQ,CAAC;YAC9D,CAAC,CAAC,GAAG,CAAC,MAAiC;YACvC,CAAC,CAAC,EAAE,IAAI,EAAE,KAAK,EAAE,CAAC;QAEpB,SAAS,CAAC,IAAI,CAAC,GAAG;YAChB,IAAI,EAAE,IAAoB;YAC1B,IAAI,EAAE,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;YACzD,IAAI,EAAE,OAAO,GAAG,CAAC,IAAI,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS;YACzD,UAAU,EAAE,OAAO,GAAG,CAAC,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,SAAS;YAC3E,UAAU,EAAE,OAAO,GAAG,CAAC,UAAU,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CAAC,SAAS;YAC3E,GAAG,EAAE,OAAO,GAAG,CAAC,GAAG,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,SAAS;YACtD,MAAM,EAAE;gBACN,IAAI,EAAE,CAAC,oBAAY,CAAC,QAAQ,CAAC,SAAS,CAAC,IAAkB,CAAC;oBACxD,CAAC,CAAC,SAAS,CAAC,IAAI;oBAChB,CAAC,CAAC,KAAK,CAAe;gBACxB,GAAG,EAAE,CAAC,uBAAe,CAAC,QAAQ,CAAC,SAAS,CAAC,GAAmB,CAAC;oBAC3D,CAAC,CAAC,SAAS,CAAC,GAAG;oBACf,CAAC,CAAC,QAAQ,CAAiB;aAC9B;SACF,CAAC;IACJ,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,SAAS,uBAAuB,CAAC,IAAY,EAAE,MAA8B;IAC3E,IAAI,IAAY,CAAC;IACjB,IAAI,IAAY,CAAC;IAEjB,IAAI,MAAM,CAAC,IAAI,KAAK,MAAM,IAAI,MAAM,CAAC,GAAG,EAAE,CAAC;QACzC,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,IAAI,GAAG,CAAC,MAAM,CAAC,GAAG,CAAC,CAAC;YACnC,IAAI,GAAG,MAAM,CAAC,QAAQ,CAAC;YACvB,IAAI,GAAG,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,MAAM,CAAC,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC;QAC7F,CAAC;QAAC,MAAM,CAAC;YACP,IAAI,GAAG,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,UAAU,IAAI,WAAW,CAAC;YACvD,IAAI,GAAG,MAAM,CAAC,IAAI,IAAI,MAAM,CAAC,UAAU,IAAI,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;QACxE,CAAC;IACH,CAAC;SAAM,CAAC;QACN,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,IAAI,IAAI,WAAW,CAAC;QACvD,IAAI,GAAG,MAAM,CAAC,UAAU,IAAI,MAAM,CAAC,IAAI,IAAI,aAAa,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACxE,CAAC;IAED,MAAM,MAAM,GAAG,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IAC7C,MAAM,gBAAgB,GAAG,GAAG,MAAM,MAAM,IAAI,IAAI,IAAI,EAAE,CAAC;IAEvD,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC,IAAI,CAAC;WAClC,GAAG,IAAI,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC;IAEpD,OAAO;QACL,IAAI;QACJ,IAAI,EAAE,MAAM,CAAC,IAAI;QACjB,IAAI;QACJ,IAAI;QACJ,gBAAgB;QAChB,MAAM;QACN,UAAU,EAAE,MAAM,CAAC,MAAM,CAAC,IAAI;QAC9B,GAAG,EAAE,MAAM,CAAC,MAAM,CAAC,GAAG,IAAI,QAAQ;KACnC,CAAC;AACJ,CAAC;AAED,gFAAgF;AAChF,WAAW;AACX,gFAAgF;AAEhF,SAAS,aAAa,CAAC,MAAc;IACnC,MAAM,IAAI,GAAG,IAAI,CAAC,SAAS,CAAC;QAC1B,EAAE,EAAE,OAAO,OAAO,KAAK,WAAW,CAAC,CAAC,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC,CAAC,SAAS;QACjE,IAAI,EAAE,OAAO,OAAO,KAAK,WAAW,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,SAAS;QAC/F,OAAO,EAAE,KAAK;QACd,MAAM,EAAE,KAAK;KACd,CAAC,CAAC;IAEH,KAAK,CAAC,GAAG,MAAM,oBAAoB,EAAE;QACnC,MAAM,EAAE,MAAM;QACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;QAC/C,IAAI,EAAE,IAAI;KACX,CAAC,CAAC,KAAK,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;AACrB,CAAC;AAED,gFAAgF;AAChF,qBAAqB;AACrB,gFAAgF;AAEhF,MAAM,WAAW,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,kBAAkB,EAAE,aAAa,CAAC,CAAC;AAE/E;;;;GAIG;AACH,SAAS,aAAa;IACpB,6DAA6D;IAC7D,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,uBAAuB,IAAI,OAAO,CAAC,GAAG,CAAC,gBAAgB,CAAC;IAClF,IAAI,KAAK;QAAE,OAAO,KAAK,CAAC;IAExB,wDAAwD;IACxD,IAAI,CAAC;QACH,IAAI,EAAE,CAAC,UAAU,CAAC,WAAW,CAAC,EAAE,CAAC;YAC/B,MAAM,GAAG,GAAG,EAAE,CAAC,YAAY,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YAClD,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAC/B,IAAI,MAAM,CAAC,OAAO,IAAI,OAAO,MAAM,CAAC,OAAO,KAAK,QAAQ,EAAE,CAAC;gBACzD,OAAO,MAAM,CAAC,OAAO,CAAC;YACxB,CAAC;QACH,CAAC;IACH,CAAC;IAAC,MAAM,CAAC;QACP,gDAAgD;IAClD,CAAC;IAED,OAAO,SAAS,CAAC;AACnB,CAAC;AAED,gFAAgF;AAChF,aAAa;AACb,gFAAgF;AAEhF,MAAa,SAAS;IACpB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C,KAAK,CAAC,IAAI,CAAC,OAAkC;QAC3C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,0BAA0B,CAAC,CAAC;QACrE,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,IAAI,MAAM,GAAG,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;QAE/B,IAAI,OAAO,EAAE,UAAU,EAAE,CAAC;YACxB,MAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC;QACjD,CAAC;QAED,OAAO,MAAM,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;YAC7B,EAAE,EAAE,CAAC,CAAC,EAAE;YACR,IAAI,EAAE,CAAC,CAAC,IAAI;YACZ,KAAK,EAAE,CAAC,CAAC,KAAK;YACd,QAAQ,EAAE,CAAC,CAAC,QAAQ;YACpB,UAAU,EAAE,CAAC,CAAC,UAAU;YACxB,YAAY,EAAE,CAAC,CAAC,YAAY,EAAE,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE;YAC3D,QAAQ,EAAE,CAAC,CAAC,QAAQ,EAAE,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,IAAI,EAAE;SACpD,CAAC,CAAC,CAAC;IACN,CAAC;IAED,KAAK,CAAC,gBAAgB,CAAC,UAAkB;QACvC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,4BAA4B,kBAAkB,CAAC,UAAU,CAAC,EAAE,CAAC,CAAC;QACvG,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;IAC3B,CAAC;IAED,KAAK,CAAC,oBAAoB,CAAC,YAAyE;QAClG,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,eAAe,EAAE;YAC5D,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,YAAY,EAAE,CAAC;SACvC,CAAC,CAAC;IACL,CAAC;IAED,KAAK,CAAC,WAAW,CACf,SAAiB,EACjB,OAAgC,EAChC,OAAuE;QAEvE,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,gBAAgB,EAAE;YAC9E,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,SAAS,EAAE,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC;SACzD,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,SAAS,CACb,OAAgC,EAChC,OAA8B;QAE9B,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,qBAAqB,EAAE;YACnF,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,GAAG,OAAO,EAAE,CAAC;SAC9C,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,WAAW,CAAC,OAAoE;QACpF,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QACrC,IAAI,OAAO,EAAE,OAAO;YAAE,MAAM,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,OAAO,CAAC,CAAC;QAC7D,IAAI,OAAO,EAAE,UAAU,KAAK,SAAS;YAAE,MAAM,CAAC,GAAG,CAAC,YAAY,EAAE,MAAM,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC,CAAC;QAC5F,IAAI,OAAO,EAAE,KAAK;YAAE,MAAM,CAAC,GAAG,CAAC,OAAO,EAAE,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC;QAE/D,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,aAAa,MAAM,EAAE,CAAC,CAAC;QACrF,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;IAC7B,CAAC;IAED,KAAK,CAAC,QAAQ,CAAC,UAAoB;QACjC,MAAM,OAAO,GAAG,IAAI,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC;QAC7C,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,gBAAgB,EAAE;YAC7D,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,CAAC;SACrC,CAAC,CAAC;IACL,CAAC;CACF;AAjFD,8BAiFC;AAED,gFAAgF;AAChF,eAAe;AACf,gFAAgF;AAEhF,MAAa,WAAW;IACtB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C,KAAK,CAAC,IAAI;QACR,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,CAAC,CAAC;QACzD,OAAO,QAAQ,CAAC,IAAI,EAAE,CAAC;IACzB,CAAC;IAED,KAAK,CAAC,GAAG,CAAC,IAAY;QACpB,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,IAAI,CAAC,WAAW,EAAE,CAAC,IAAI,IAAI,CAAC;IACjF,CAAC;IAED;;;;;OAKG;IACH,KAAK,CAAC,aAAa,CAAC,WAAmB,EAAE,OAAiC;QACxE,IAAI,OAAO,EAAE,UAAU,EAAE,CAAC;YACxB,OAAO,IAAI,CAAC,kBAAkB,CAAC,WAAW,EAAE,OAAO,CAAC,UAAU,CAAC,CAAC;QAClE,CAAC;QAED,MAAM,OAAO,GAAG,MAAM,IAAI,CAAC,GAAG,CAAC,WAAW,CAAC,CAAC;QAC5C,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,MAAM,IAAI,KAAK,CAAC,YAAY,WAAW,aAAa,CAAC,CAAC;QACxD,CAAC;QAED,MAAM,IAAI,GAAG,OAAO,CAAC,UAAU,IAAI,OAAO,CAAC,UAAU,CAAC;QACtD,MAAM,IAAI,GAAG,WAAW,CAAC;QAEzB,IAAI,gBAAwB,CAAC;QAC7B,IAAI,MAAc,CAAC;QAEnB,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,UAAU,EAAE,CAAC;YACnE,gBAAgB,GAAG,cAAc,IAAI,IAAI,IAAI,WAAW,CAAC;YACzD,MAAM,GAAG,cAAc,CAAC;QAC1B,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,WAAW,IAAI,IAAI,IAAI,EAAE,CAAC;YAC7C,MAAM,GAAG,cAAc,CAAC;QAC1B,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,IAAI,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,WAAW,IAAI,IAAI,IAAI,EAAE,CAAC;YAC7C,MAAM,GAAG,WAAW,CAAC;QACvB,CAAC;aAAM,IAAI,OAAO,CAAC,UAAU,KAAK,KAAK,IAAI,OAAO,CAAC,QAAQ,KAAK,SAAS,EAAE,CAAC;YAC1E,gBAAgB,GAAG,aAAa,IAAI,IAAI,IAAI,EAAE,CAAC;YAC/C,MAAM,GAAG,aAAa,CAAC;QACzB,CAAC;aAAM,IAAI,OAAO,CAAC,QAAQ,KAAK,MAAM,IAAI,OAAO,CAAC,QAAQ,KAAK,OAAO,EAAE,CAAC;YACvE,gBAAgB,GAAG,UAAU,IAAI,IAAI,IAAI,EAAE,CAAC;YAC5C,MAAM,GAAG,SAAS,CAAC;QACrB,CAAC;aAAM,CAAC;YACN,gBAAgB,GAAG,SAAS,IAAI,IAAI,IAAI,EAAE,CAAC;YAC3C,MAAM,GAAG,GAAG,WAAW,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,IAAI,EAAE,GAAG,CAAC,MAAM,CAAC;QACjE,CAAC;QAED,OAAO,EAAE,OAAO,EAAE,WAAW,EAAE,IAAI,EAAE,IAAI,EAAE,gBAAgB,EAAE,MAAM,EAAE,CAAC;IACxE,CAAC;IAEO,kBAAkB,CAAC,WAAmB,EAAE,UAAkB;QAChE,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;QAClC,MAAM,aAAa,GAAG,GAAG,MAAM,UAAU,kBAAkB,CAAC,WAAW,CAAC,EAAE,CAAC;QAE3E,OAAO;YACL,OAAO,EAAE,WAAW;YACpB,IAAI,EAAE,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,QAAQ;YAC9B,IAAI,EAAE,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,QAAQ,CAAC,IAAI,GAAG,CAAC,MAAM,CAAC,CAAC,IAAI,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,GAAG;YACrE,gBAAgB,EAAE,aAAa;YAC/B,MAAM,EAAE,WAAW;YACnB,UAAU;YACV,aAAa;SACd,CAAC;IACJ,CAAC;CACF;AAxED,kCAwEC;AAED,gFAAgF;AAChF,aAAa;AACb,gFAAgF;AAEhF,MAAa,SAAS;IACpB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAA2B;QACtC,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,YAAY,EAAE;YACrD,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,OAAO,CAAC;SAC9B,CAAC,CAAC;QACH,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,KAAK,CAAC;IACpB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,IAAI,CAAC,OAAsC;QAC/C,MAAM,MAAM,GAAG,IAAI,eAAe,EAAE,CAAC;QACrC,IAAI,OAAO,EAAE,cAAc;YAAE,MAAM,CAAC,GAAG,CAAC,gBAAgB,EAAE,MAAM,CAAC,CAAC;QAClE,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,MAAM,EAAE,CAAC,CAAC;QACjE,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,MAAM,IAAI,EAAE,CAAC;IAC3B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,OAAe;QAC1B,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,OAAO,EAAE,EAAE,EAAE,MAAM,EAAE,QAAQ,EAAE,CAAC,CAAC;IACzE,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,QAAQ,CAAC,KAAa;QAC1B,IAAI,CAAC;YACH,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,qBAAqB,CAAC;YACvD,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;gBAChC,MAAM,EAAE,MAAM;gBACd,OAAO,EAAE,EAAE,cAAc,EAAE,kBAAkB,EAAE;gBAC/C,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,KAAK,EAAE,CAAC;aAChC,CAAC,CAAC;YACH,IAAI,CAAC,QAAQ,CAAC,EAAE;gBAAE,OAAO,IAAI,CAAC;YAC9B,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;YACnC,OAAO,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,IAAI,CAAC;QACxC,CAAC;QAAC,MAAM,CAAC;YACP,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;CACF;AApDD,8BAoDC;AAED,gFAAgF;AAChF,cAAc;AACd,gFAAgF;AAEhF,MAAa,cAAc;IAczB,YAAY,SAA+B,EAAE;QAZrC,aAAQ,GAAkC,IAAI,GAAG,EAAE,CAAC;QAa1D,IAAI,CAAC,MAAM,GAAG;YACZ,MAAM,EAAE,MAAM,CAAC,MAAM;YACrB,MAAM,EAAE,MAAM,CAAC,MAAM,IAAI,+BAA+B;YACxD,OAAO,EAAE,MAAM,CAAC,OAAO,IAAI,aAAa,EAAE;SAC3C,CAAC;QAEF,IAAI,CAAC,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,QAAQ,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,CAAC;QACtC,IAAI,CAAC,MAAM,GAAG,IAAI,SAAS,CAAC,IAAI,CAAC,CAAC;QAElC,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;YAC5B,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED,4EAA4E;IAC5E,gDAAgD;IAChD,4EAA4E;IAE5E;;;;;;;;;;;;OAYG;IACH,MAAM,CAAC,YAAY,CACjB,YAAqB,EACrB,MAA6B;QAE7B,MAAM,YAAY,GAAG,YAAY;YAC/B,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,CAAC;YAC5B,CAAC,CAAC,YAAY,EAAE,CAAC;QAEnB,IAAI,CAAC,YAAY,EAAE,CAAC;YAClB,MAAM,IAAI,KAAK,CACb,6DAA6D;gBAC7D,gBAAgB;gBAChB,mBAAmB;gBACnB,wBAAwB;gBACxB,2BAA2B;gBAC3B,oBAAoB;gBACpB,iBAAiB;gBACjB,qBAAqB,CACtB,CAAC;QACJ,CAAC;QAED,MAAM,QAAQ,GAAG,IAAI,cAAc,CAAC,MAAM,IAAI,EAAE,eAAe,EAAE,IAAI,EAAE,CAAC,CAAC;QACzE,QAAQ,CAAC,YAAY,GAAG,YAAY,CAAC;QAErC,MAAM,YAAY,GAAG,iBAAiB,CAAC,YAAY,CAAC,CAAC;QACrD,KAAK,MAAM,CAAC,IAAI,EAAE,SAAS,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;YAC7D,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,EAAE,uBAAuB,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC,CAAC;QACxE,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;IAED;;;;;;OAMG;IACH,QAAQ,CAAC,IAAY;QACnB,MAAM,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;QAClC,IAAI,CAAC,CAAC,EAAE,CAAC;YACP,MAAM,SAAS,GAAG,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC,CAAC;YACnD,MAAM,GAAG,GAAG,SAAS,CAAC,MAAM;gBAC1B,CAAC,CAAC,cAAc,SAAS,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;gBACtC,CAAC,CAAC,kEAAkE,CAAC;YACvE,MAAM,IAAI,KAAK,CAAC,aAAa,IAAI,gBAAgB,GAAG,EAAE,CAAC,CAAC;QAC1D,CAAC;QACD,OAAO,CAAC,CAAC;IACX,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC;IAC5C,CAAC;IAED;;;;;;;;;;OAUG;IACH,QAAQ;QACN,OAAO,IAAI,CAAC,SAAS,EAAE;aACpB,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,GAAG,CAAC,CAAC,MAAM,IAAI,CAAC,CAAC,gBAAgB,EAAE,CAAC;aAC7C,IAAI,CAAC,IAAI,CAAC,CAAC;IAChB,CAAC;IAED,4EAA4E;IAC5E,gBAAgB;IAChB,4EAA4E;IAE5E,6DAA6D;IAC7D,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;IAC7B,CAAC;IAED,+CAA+C;IAC/C,IAAI,MAAM;QACR,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;IAC5B,CAAC;IAED;;;OAGG;IACH,KAAK,CAAC,OAAO,CAAC,WAAmB,EAAE,OAAiC;QAClE,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;IAC3D,CAAC;IAED;;;OAGG;IACH,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,EAAE,CAAC;YACzB,MAAM,IAAI,KAAK,CACb,+BAA+B;gBAC/B,sDAAsD;gBACtD,6DAA6D;gBAC7D,sDAAsD,CACvD,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;IAC7B,CAAC;IAED,qCAAqC;IAC7B,aAAa;QACnB,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,EAAE,CAAC;YACxB,MAAM,IAAI,KAAK,CACb,+CAA+C;gBAC/C,4DAA4D;gBAC5D,iCAAiC,CAClC,CAAC;QACJ,CAAC;QACD,OAAO,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC;IAC5B,CAAC;IAED,wCAAwC;IACxC,KAAK,CAAC,KAAK,CAAC,OAAe,EAAE,OAAqB;QAChD,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,EAAE,CAAC;QACpC,MAAM,GAAG,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,MAAM,GAAG,OAAO,EAAE,CAAC;QAC9C,MAAM,QAAQ,GAAG,MAAM,KAAK,CAAC,GAAG,EAAE;YAChC,GAAG,OAAO;YACV,OAAO,EAAE;gBACP,WAAW,EAAE,MAAM;gBACnB,cAAc,EAAE,kBAAkB;gBAClC,GAAG,OAAO,EAAE,OAAO;aACpB;SACF,CAAC,CAAC;QAEH,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAE,CAAC;YACjB,MAAM,KAAK,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,CAAC,EAAE,OAAO,EAAE,eAAe,EAAE,CAAC,CAAC,CAAC;YAChF,MAAM,IAAI,KAAK,CAAC,KAAK,CAAC,OAAO,IAAI,mBAAmB,QAAQ,CAAC,MAAM,EAAE,CAAC,CAAC;QACzE,CAAC;QAED,OAAO,QAAQ,CAAC;IAClB,CAAC;CACF;AAhMD,wCAgMC;AAED,kBAAe,cAAc,CAAC;AAE9B,0EAA0E;AACnE,KAAK,UAAU,OAAO,CAC3B,WAAmB,EACnB,MAAuD;IAEvD,MAAM,MAAM,GAAG,MAAM,EAAE,MAAM,IAAI,OAAO,CAAC,GAAG,CAAC,sBAAsB,CAAC;IACpE,IAAI,CAAC,MAAM,EAAE,CAAC;QACZ,MAAM,IAAI,KAAK,CAAC,oEAAoE,CAAC,CAAC;IACxF,CAAC;IAED,MAAM,MAAM,GAAG,IAAI,cAAc,CAAC,EAAE,GAAG,MAAM,EAAE,MAAM,EAAE,CAAC,CAAC;IACzD,OAAO,MAAM,CAAC,OAAO,CAAC,WAAW,EAAE,EAAE,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,CAAC,CAAC;AACzE,CAAC;AAED;;;GAGG;AACH,SAAgB,YAAY,CAAC,YAAqB;IAChD,OAAO,cAAc,CAAC,YAAY,CAAC,YAAY,CAAC,CAAC;AACnD,CAAC","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
|
-
"description": "
|
|
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
|
-
"
|
|
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
|
-
|