@privateconnect/sdk 0.7.1 → 0.7.4

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (3) hide show
  1. package/dist/index.d.ts +75 -77
  2. package/dist/index.js +154 -120
  3. package/package.json +33 -33
package/dist/index.d.ts CHANGED
@@ -1,7 +1,7 @@
1
1
  /**
2
2
  * Private Connect SDK
3
3
  *
4
- * Programmatic access to Private Connect services and agent orchestration.
4
+ * Programmatic access to Private Connect services, grants, and agent orchestration.
5
5
  *
6
6
  * @example
7
7
  * ```typescript
@@ -9,15 +9,21 @@
9
9
  *
10
10
  * const pc = new PrivateConnect({ apiKey: 'your-api-key' });
11
11
  *
12
- * // Connect to a service
12
+ * // Connect to a service (assumes tunnel is already open)
13
13
  * const db = await pc.connect('postgres-prod');
14
14
  * console.log(db.connectionString); // postgres://localhost:5432/...
15
15
  *
16
+ * // Grant an AI agent temporary access
17
+ * const grant = await pc.grants.create({
18
+ * agentLabel: 'claude',
19
+ * resourceType: 'db',
20
+ * resourceName: 'postgres',
21
+ * ttl: '5m',
22
+ * });
23
+ * console.log(grant.token); // gnt_...
24
+ *
16
25
  * // List all agents
17
26
  * const agents = await pc.agents.list();
18
- *
19
- * // Send message to another agent
20
- * await pc.agents.sendMessage(targetAgentId, { action: 'deploy' });
21
27
  * ```
22
28
  */
23
29
  export interface PrivateConnectConfig {
@@ -25,7 +31,7 @@ export interface PrivateConnectConfig {
25
31
  apiKey: string;
26
32
  /** Hub URL (default: https://api.privateconnect.co) */
27
33
  hubUrl?: string;
28
- /** Agent ID (auto-detected from config if not provided) */
34
+ /** Agent ID (auto-detected from local config if not provided) */
29
35
  agentId?: string;
30
36
  /** Disable usage tracking (default: false) */
31
37
  disableTracking?: boolean;
@@ -55,6 +61,10 @@ export interface Connection {
55
61
  port: number;
56
62
  connectionString: string;
57
63
  envVar: string;
64
+ /** Present when using a grant-based connection */
65
+ grantToken?: string;
66
+ /** Present when using a grant-based connection */
67
+ grantEndpoint?: string;
58
68
  }
59
69
  export interface Message {
60
70
  id: string;
@@ -69,137 +79,125 @@ export interface Message {
69
79
  createdAt: string;
70
80
  isRead: boolean;
71
81
  }
72
- export interface Session {
82
+ export interface Grant {
73
83
  id: string;
74
- name: string;
75
- createdBy: string;
76
- createdAt: string;
77
- expiresAt: string;
78
- metadata?: Record<string, unknown>;
84
+ agentLabel: string;
85
+ resourceType: string;
86
+ resourceName: string;
87
+ scope: string;
88
+ tokenPrefix?: string;
89
+ persistent: boolean;
90
+ expiresAt: string | null;
91
+ expiresInMinutes?: number | null;
92
+ token?: string;
93
+ endpoint?: string;
94
+ }
95
+ export interface GrantCreateOptions {
96
+ agentLabel: string;
97
+ resourceType: 'db' | 'api' | 'path';
98
+ resourceName: string;
99
+ scope?: 'read-only' | 'full';
100
+ /** Duration string: 60s, 5m, 1h, 1d. Omit for persistent grant. */
101
+ ttl?: string;
79
102
  }
80
- /**
81
- * Agents API for discovery and orchestration
82
- */
83
103
  export declare class AgentsAPI {
84
104
  private client;
85
105
  constructor(client: PrivateConnect);
86
- /**
87
- * List all agents in the workspace
88
- */
89
106
  list(options?: {
90
107
  onlineOnly?: boolean;
91
108
  }): Promise<Agent[]>;
92
- /**
93
- * Find agents by capability
94
- */
95
109
  findByCapability(capability: string): Promise<Agent[]>;
96
- /**
97
- * Register capabilities for this agent
98
- */
99
110
  registerCapabilities(capabilities: Array<{
100
111
  name: string;
101
112
  metadata?: Record<string, unknown>;
102
113
  }>): Promise<void>;
103
- /**
104
- * Send a message to another agent
105
- */
106
114
  sendMessage(toAgentId: string, payload: Record<string, unknown>, options?: {
107
115
  channel?: string;
108
116
  type?: 'request' | 'response' | 'event';
109
117
  }): Promise<{
110
118
  messageId: string;
111
119
  }>;
112
- /**
113
- * Broadcast a message to all online agents
114
- */
115
120
  broadcast(payload: Record<string, unknown>, options?: {
116
121
  channel?: string;
117
122
  }): Promise<{
118
123
  sent: number;
119
124
  }>;
120
- /**
121
- * Get messages for this agent
122
- */
123
125
  getMessages(options?: {
124
126
  channel?: string;
125
127
  unreadOnly?: boolean;
126
128
  limit?: number;
127
129
  }): Promise<Message[]>;
128
- /**
129
- * Mark messages as read
130
- */
131
130
  markRead(messageIds: string[]): Promise<void>;
132
131
  }
133
- /**
134
- * Services API for connecting to and managing services
135
- */
136
132
  export declare class ServicesAPI {
137
133
  private client;
138
134
  constructor(client: PrivateConnect);
139
- /**
140
- * List all services
141
- */
142
135
  list(): Promise<Service[]>;
143
- /**
144
- * Get a specific service by name
145
- */
146
136
  get(name: string): Promise<Service | null>;
147
137
  /**
148
- * Get connection details for a service
138
+ * Get connection details for a service.
139
+ *
140
+ * If `grantToken` is provided, returns a proxied connection via the hub's
141
+ * grant endpoint instead of assuming a local tunnel.
149
142
  */
150
- getConnection(serviceName: string): Promise<Connection>;
143
+ getConnection(serviceName: string, options?: {
144
+ grantToken?: string;
145
+ }): Promise<Connection>;
146
+ private getGrantConnection;
151
147
  }
152
- /**
153
- * Sessions API for ephemeral orchestration sessions
154
- */
155
- export declare class SessionsAPI {
148
+ export declare class GrantsAPI {
156
149
  private client;
157
- private activeSessions;
158
150
  constructor(client: PrivateConnect);
159
151
  /**
160
- * Create an orchestration session
152
+ * Create a time-limited access grant for an AI agent or external consumer.
161
153
  */
162
- create(name: string, options?: {
163
- ttlMinutes?: number;
164
- metadata?: Record<string, unknown>;
165
- }): Promise<Session>;
154
+ create(options: GrantCreateOptions): Promise<Grant>;
166
155
  /**
167
- * End an orchestration session
156
+ * List active grants in the workspace.
168
157
  */
169
- end(sessionId: string): Promise<void>;
158
+ list(options?: {
159
+ includeExpired?: boolean;
160
+ }): Promise<Grant[]>;
161
+ /**
162
+ * Revoke an active grant immediately.
163
+ */
164
+ revoke(grantId: string): Promise<void>;
170
165
  /**
171
- * Get active sessions
166
+ * Validate a grant token (public endpoint — no API key required).
167
+ * Returns the grant if valid, null otherwise.
172
168
  */
173
- getActive(): Session[];
169
+ validate(token: string): Promise<Grant | null>;
174
170
  }
175
- /**
176
- * Main Private Connect SDK client
177
- */
178
171
  export declare class PrivateConnect {
179
172
  private config;
180
173
  /** Agents API for discovery and orchestration */
181
174
  agents: AgentsAPI;
182
175
  /** Services API for connecting to services */
183
176
  services: ServicesAPI;
184
- /** Sessions API for ephemeral orchestration */
185
- sessions: SessionsAPI;
177
+ /** Grants API for managing scoped access tokens (time-limited or persistent) */
178
+ grants: GrantsAPI;
186
179
  constructor(config: PrivateConnectConfig);
187
- /**
188
- * Get the agent ID
189
- */
180
+ /** The resolved agent ID, or undefined if not configured. */
190
181
  get agentId(): string | undefined;
182
+ /** The hub URL this client is connected to. */
183
+ get hubUrl(): string;
191
184
  /**
192
- * Connect to a service and get connection details
185
+ * Shorthand: get connection details for a service.
186
+ * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.
193
187
  */
194
- connect(serviceName: string): Promise<Connection>;
188
+ connect(serviceName: string, options?: {
189
+ grantToken?: string;
190
+ }): Promise<Connection>;
195
191
  /**
196
- * Internal fetch helper
192
+ * Returns the agent ID or throws if not configured.
193
+ * Used by APIs that require an authenticated agent identity.
197
194
  */
195
+ requireAgentId(): string;
196
+ /** Internal fetch with API key auth. */
198
197
  fetch(path: string, options?: RequestInit): Promise<Response>;
199
- /**
200
- * Try to detect agent ID from local config
201
- */
202
- private detectAgentId;
203
198
  }
204
199
  export default PrivateConnect;
205
- export declare function connect(serviceName: string, config?: PrivateConnectConfig): Promise<Connection>;
200
+ /** Convenience function for quick one-off connections. */
201
+ export declare function connect(serviceName: string, config?: PrivateConnectConfig & {
202
+ grantToken?: string;
203
+ }): Promise<Connection>;
package/dist/index.js CHANGED
@@ -2,7 +2,7 @@
2
2
  /**
3
3
  * Private Connect SDK
4
4
  *
5
- * Programmatic access to Private Connect services and agent orchestration.
5
+ * Programmatic access to Private Connect services, grants, and agent orchestration.
6
6
  *
7
7
  * @example
8
8
  * ```typescript
@@ -10,21 +10,32 @@
10
10
  *
11
11
  * const pc = new PrivateConnect({ apiKey: 'your-api-key' });
12
12
  *
13
- * // Connect to a service
13
+ * // Connect to a service (assumes tunnel is already open)
14
14
  * const db = await pc.connect('postgres-prod');
15
15
  * console.log(db.connectionString); // postgres://localhost:5432/...
16
16
  *
17
+ * // Grant an AI agent temporary access
18
+ * const grant = await pc.grants.create({
19
+ * agentLabel: 'claude',
20
+ * resourceType: 'db',
21
+ * resourceName: 'postgres',
22
+ * ttl: '5m',
23
+ * });
24
+ * console.log(grant.token); // gnt_...
25
+ *
17
26
  * // List all agents
18
27
  * const agents = await pc.agents.list();
19
- *
20
- * // Send message to another agent
21
- * await pc.agents.sendMessage(targetAgentId, { action: 'deploy' });
22
28
  * ```
23
29
  */
24
30
  Object.defineProperty(exports, "__esModule", { value: true });
25
- exports.PrivateConnect = exports.SessionsAPI = exports.ServicesAPI = exports.AgentsAPI = void 0;
31
+ exports.PrivateConnect = exports.GrantsAPI = exports.ServicesAPI = exports.AgentsAPI = void 0;
26
32
  exports.connect = connect;
27
- // Track SDK usage (fire and forget)
33
+ const fs = require("fs");
34
+ const path = require("path");
35
+ const os = require("os");
36
+ // ─────────────────────────────────────────────────────────────────────────────
37
+ // Tracking
38
+ // ─────────────────────────────────────────────────────────────────────────────
28
39
  function trackSdkUsage(hubUrl) {
29
40
  const data = JSON.stringify({
30
41
  os: typeof process !== 'undefined' ? process.platform : 'browser',
@@ -36,18 +47,44 @@ function trackSdkUsage(hubUrl) {
36
47
  method: 'POST',
37
48
  headers: { 'Content-Type': 'application/json' },
38
49
  body: data,
39
- }).catch(() => { }); // Silently ignore errors
50
+ }).catch(() => { });
40
51
  }
52
+ // ─────────────────────────────────────────────────────────────────────────────
53
+ // Agent ID Detection
54
+ // ─────────────────────────────────────────────────────────────────────────────
55
+ const CONFIG_PATH = path.join(os.homedir(), '.private-connect', 'config.json');
41
56
  /**
42
- * Agents API for discovery and orchestration
57
+ * Detect the agent ID from local config or environment.
58
+ * Returns undefined if no agent is configured — callers that need an agent ID
59
+ * should surface a clear error rather than using a fake one.
43
60
  */
61
+ function detectAgentId() {
62
+ // 1. Environment variables (highest priority — useful in CI)
63
+ const envId = process.env.PRIVATECONNECT_AGENT_ID || process.env.CONNECT_AGENT_ID;
64
+ if (envId)
65
+ return envId;
66
+ // 2. Local config file (~/.private-connect/config.json)
67
+ try {
68
+ if (fs.existsSync(CONFIG_PATH)) {
69
+ const raw = fs.readFileSync(CONFIG_PATH, 'utf-8');
70
+ const config = JSON.parse(raw);
71
+ if (config.agentId && typeof config.agentId === 'string') {
72
+ return config.agentId;
73
+ }
74
+ }
75
+ }
76
+ catch {
77
+ // Config unreadable or malformed — fall through
78
+ }
79
+ return undefined;
80
+ }
81
+ // ─────────────────────────────────────────────────────────────────────────────
82
+ // Agents API
83
+ // ─────────────────────────────────────────────────────────────────────────────
44
84
  class AgentsAPI {
45
85
  constructor(client) {
46
86
  this.client = client;
47
87
  }
48
- /**
49
- * List all agents in the workspace
50
- */
51
88
  async list(options) {
52
89
  const response = await this.client.fetch('/v1/agents/orchestration');
53
90
  const data = await response.json();
@@ -65,47 +102,36 @@ class AgentsAPI {
65
102
  services: a.services?.map((s) => s.name) || [],
66
103
  }));
67
104
  }
68
- /**
69
- * Find agents by capability
70
- */
71
105
  async findByCapability(capability) {
72
106
  const response = await this.client.fetch(`/v1/agents/by-capability/${encodeURIComponent(capability)}`);
73
107
  const data = await response.json();
74
108
  return data.agents || [];
75
109
  }
76
- /**
77
- * Register capabilities for this agent
78
- */
79
110
  async registerCapabilities(capabilities) {
80
- await this.client.fetch(`/v1/agents/${this.client.agentId}/capabilities`, {
111
+ const agentId = this.client.requireAgentId();
112
+ await this.client.fetch(`/v1/agents/${agentId}/capabilities`, {
81
113
  method: 'POST',
82
114
  body: JSON.stringify({ capabilities }),
83
115
  });
84
116
  }
85
- /**
86
- * Send a message to another agent
87
- */
88
117
  async sendMessage(toAgentId, payload, options) {
89
- const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/send`, {
118
+ const agentId = this.client.requireAgentId();
119
+ const response = await this.client.fetch(`/v1/agents/${agentId}/messages/send`, {
90
120
  method: 'POST',
91
121
  body: JSON.stringify({ toAgentId, payload, ...options }),
92
122
  });
93
123
  return response.json();
94
124
  }
95
- /**
96
- * Broadcast a message to all online agents
97
- */
98
125
  async broadcast(payload, options) {
99
- const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/broadcast`, {
126
+ const agentId = this.client.requireAgentId();
127
+ const response = await this.client.fetch(`/v1/agents/${agentId}/messages/broadcast`, {
100
128
  method: 'POST',
101
129
  body: JSON.stringify({ payload, ...options }),
102
130
  });
103
131
  return response.json();
104
132
  }
105
- /**
106
- * Get messages for this agent
107
- */
108
133
  async getMessages(options) {
134
+ const agentId = this.client.requireAgentId();
109
135
  const params = new URLSearchParams();
110
136
  if (options?.channel)
111
137
  params.set('channel', options.channel);
@@ -113,46 +139,44 @@ class AgentsAPI {
113
139
  params.set('unreadOnly', String(options.unreadOnly));
114
140
  if (options?.limit)
115
141
  params.set('limit', String(options.limit));
116
- const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages?${params}`);
142
+ const response = await this.client.fetch(`/v1/agents/${agentId}/messages?${params}`);
117
143
  const data = await response.json();
118
144
  return data.messages || [];
119
145
  }
120
- /**
121
- * Mark messages as read
122
- */
123
146
  async markRead(messageIds) {
124
- await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/read`, {
147
+ const agentId = this.client.requireAgentId();
148
+ await this.client.fetch(`/v1/agents/${agentId}/messages/read`, {
125
149
  method: 'POST',
126
150
  body: JSON.stringify({ messageIds }),
127
151
  });
128
152
  }
129
153
  }
130
154
  exports.AgentsAPI = AgentsAPI;
131
- /**
132
- * Services API for connecting to and managing services
133
- */
155
+ // ─────────────────────────────────────────────────────────────────────────────
156
+ // Services API
157
+ // ─────────────────────────────────────────────────────────────────────────────
134
158
  class ServicesAPI {
135
159
  constructor(client) {
136
160
  this.client = client;
137
161
  }
138
- /**
139
- * List all services
140
- */
141
162
  async list() {
142
163
  const response = await this.client.fetch('/v1/services');
143
164
  return response.json();
144
165
  }
145
- /**
146
- * Get a specific service by name
147
- */
148
166
  async get(name) {
149
167
  const services = await this.list();
150
168
  return services.find(s => s.name.toLowerCase() === name.toLowerCase()) || null;
151
169
  }
152
170
  /**
153
- * Get connection details for a service
171
+ * Get connection details for a service.
172
+ *
173
+ * If `grantToken` is provided, returns a proxied connection via the hub's
174
+ * grant endpoint instead of assuming a local tunnel.
154
175
  */
155
- async getConnection(serviceName) {
176
+ async getConnection(serviceName, options) {
177
+ if (options?.grantToken) {
178
+ return this.getGrantConnection(serviceName, options.grantToken);
179
+ }
156
180
  const service = await this.get(serviceName);
157
181
  if (!service) {
158
182
  throw new Error(`Service "${serviceName}" not found`);
@@ -185,108 +209,127 @@ class ServicesAPI {
185
209
  connectionString = `tcp://${host}:${port}`;
186
210
  envVar = `${serviceName.toUpperCase().replace(/-/g, '_')}_URL`;
187
211
  }
212
+ return { service: serviceName, host, port, connectionString, envVar };
213
+ }
214
+ getGrantConnection(serviceName, grantToken) {
215
+ const hubUrl = this.client.hubUrl;
216
+ const grantEndpoint = `${hubUrl}/grant/${encodeURIComponent(serviceName)}`;
188
217
  return {
189
218
  service: serviceName,
190
- host,
191
- port,
192
- connectionString,
193
- envVar,
219
+ host: new URL(hubUrl).hostname,
220
+ port: new URL(hubUrl).port ? parseInt(new URL(hubUrl).port, 10) : 443,
221
+ connectionString: grantEndpoint,
222
+ envVar: 'GRANT_URL',
223
+ grantToken,
224
+ grantEndpoint,
194
225
  };
195
226
  }
196
227
  }
197
228
  exports.ServicesAPI = ServicesAPI;
198
- /**
199
- * Sessions API for ephemeral orchestration sessions
200
- */
201
- class SessionsAPI {
229
+ // ─────────────────────────────────────────────────────────────────────────────
230
+ // Grants API
231
+ // ─────────────────────────────────────────────────────────────────────────────
232
+ class GrantsAPI {
202
233
  constructor(client) {
203
234
  this.client = client;
204
- this.activeSessions = new Map();
205
235
  }
206
236
  /**
207
- * Create an orchestration session
237
+ * Create a time-limited access grant for an AI agent or external consumer.
208
238
  */
209
- async create(name, options) {
210
- const ttlMinutes = options?.ttlMinutes || 60;
211
- const sessionId = `${this.client.agentId}-${Date.now()}-${Math.random().toString(36).substring(2, 8)}`;
212
- const expiresAt = new Date(Date.now() + ttlMinutes * 60 * 1000);
213
- const session = {
214
- id: sessionId,
215
- name,
216
- createdBy: this.client.agentId,
217
- createdAt: new Date().toISOString(),
218
- expiresAt: expiresAt.toISOString(),
219
- metadata: options?.metadata,
220
- };
221
- this.activeSessions.set(sessionId, session);
222
- // Broadcast session creation
223
- await this.client.agents.broadcast({ type: 'session:created', session }, { channel: 'orchestration' });
224
- return session;
239
+ async create(options) {
240
+ const response = await this.client.fetch('/v1/grants', {
241
+ method: 'POST',
242
+ body: JSON.stringify(options),
243
+ });
244
+ const data = await response.json();
245
+ return data.grant;
225
246
  }
226
247
  /**
227
- * End an orchestration session
248
+ * List active grants in the workspace.
228
249
  */
229
- async end(sessionId) {
230
- this.activeSessions.delete(sessionId);
231
- // Broadcast session end
232
- await this.client.agents.broadcast({
233
- type: 'session:ended',
234
- sessionId,
235
- endedBy: this.client.agentId,
236
- endedAt: new Date().toISOString(),
237
- }, { channel: 'orchestration' });
250
+ async list(options) {
251
+ const params = new URLSearchParams();
252
+ if (options?.includeExpired)
253
+ params.set('includeExpired', 'true');
254
+ const response = await this.client.fetch(`/v1/grants?${params}`);
255
+ const data = await response.json();
256
+ return data.grants || [];
238
257
  }
239
258
  /**
240
- * Get active sessions
259
+ * Revoke an active grant immediately.
241
260
  */
242
- getActive() {
243
- const now = new Date();
244
- const active = [];
245
- for (const [id, session] of this.activeSessions) {
246
- if (new Date(session.expiresAt) > now) {
247
- active.push(session);
248
- }
249
- else {
250
- this.activeSessions.delete(id);
251
- }
261
+ async revoke(grantId) {
262
+ await this.client.fetch(`/v1/grants/${grantId}`, { method: 'DELETE' });
263
+ }
264
+ /**
265
+ * Validate a grant token (public endpoint — no API key required).
266
+ * Returns the grant if valid, null otherwise.
267
+ */
268
+ async validate(token) {
269
+ try {
270
+ const url = `${this.client.hubUrl}/v1/grants/validate`;
271
+ const response = await fetch(url, {
272
+ method: 'POST',
273
+ headers: { 'Content-Type': 'application/json' },
274
+ body: JSON.stringify({ token }),
275
+ });
276
+ if (!response.ok)
277
+ return null;
278
+ const data = await response.json();
279
+ return data.valid ? data.grant : null;
280
+ }
281
+ catch {
282
+ return null;
252
283
  }
253
- return active;
254
284
  }
255
285
  }
256
- exports.SessionsAPI = SessionsAPI;
257
- /**
258
- * Main Private Connect SDK client
259
- */
286
+ exports.GrantsAPI = GrantsAPI;
287
+ // ─────────────────────────────────────────────────────────────────────────────
288
+ // Main Client
289
+ // ─────────────────────────────────────────────────────────────────────────────
260
290
  class PrivateConnect {
261
291
  constructor(config) {
262
292
  this.config = {
263
293
  apiKey: config.apiKey,
264
294
  hubUrl: config.hubUrl || 'https://api.privateconnect.co',
265
- agentId: config.agentId || this.detectAgentId(),
295
+ agentId: config.agentId || detectAgentId(),
266
296
  };
267
297
  this.agents = new AgentsAPI(this);
268
298
  this.services = new ServicesAPI(this);
269
- this.sessions = new SessionsAPI(this);
270
- // Track SDK usage (non-blocking)
299
+ this.grants = new GrantsAPI(this);
271
300
  if (!config.disableTracking) {
272
301
  trackSdkUsage(this.config.hubUrl);
273
302
  }
274
303
  }
275
- /**
276
- * Get the agent ID
277
- */
304
+ /** The resolved agent ID, or undefined if not configured. */
278
305
  get agentId() {
279
306
  return this.config.agentId;
280
307
  }
308
+ /** The hub URL this client is connected to. */
309
+ get hubUrl() {
310
+ return this.config.hubUrl;
311
+ }
281
312
  /**
282
- * Connect to a service and get connection details
313
+ * Shorthand: get connection details for a service.
314
+ * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.
283
315
  */
284
- async connect(serviceName) {
285
- return this.services.getConnection(serviceName);
316
+ async connect(serviceName, options) {
317
+ return this.services.getConnection(serviceName, options);
286
318
  }
287
319
  /**
288
- * Internal fetch helper
320
+ * Returns the agent ID or throws if not configured.
321
+ * Used by APIs that require an authenticated agent identity.
289
322
  */
323
+ requireAgentId() {
324
+ if (!this.config.agentId) {
325
+ throw new Error('Agent ID not found. Either:\n' +
326
+ ' 1. Run "connect up" to register this machine, or\n' +
327
+ ' 2. Set PRIVATECONNECT_AGENT_ID environment variable, or\n' +
328
+ ' 3. Pass agentId in the PrivateConnect constructor.');
329
+ }
330
+ return this.config.agentId;
331
+ }
332
+ /** Internal fetch with API key auth. */
290
333
  async fetch(path, options) {
291
334
  const url = `${this.config.hubUrl}${path}`;
292
335
  const response = await fetch(url, {
@@ -303,25 +346,16 @@ class PrivateConnect {
303
346
  }
304
347
  return response;
305
348
  }
306
- /**
307
- * Try to detect agent ID from local config
308
- */
309
- detectAgentId() {
310
- // In a real implementation, this would read from ~/.connect/config.json
311
- // For now, generate a default ID
312
- return `sdk-${Date.now()}`;
313
- }
314
349
  }
315
350
  exports.PrivateConnect = PrivateConnect;
316
- // Default export
317
351
  exports.default = PrivateConnect;
318
- // Convenience function
352
+ /** Convenience function for quick one-off connections. */
319
353
  async function connect(serviceName, config) {
320
354
  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;
321
355
  if (!apiKey) {
322
356
  throw new Error('API key required. Set PRIVATECONNECT_API_KEY or pass config.apiKey');
323
357
  }
324
358
  const client = new PrivateConnect({ ...config, apiKey });
325
- return client.connect(serviceName);
359
+ return client.connect(serviceName, { grantToken: config?.grantToken });
326
360
  }
327
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":";AAAA;;;;;;;;;;;;;;;;;;;;;GAqBG;;;AAiZH,0BAQC;AA5YD,oCAAoC;AACpC,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,CAAC,yBAAyB;AAC/C,CAAC;AAkDD;;GAEG;AACH,MAAa,SAAS;IACpB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C;;OAEG;IACH,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;;OAEG;IACH,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;;OAEG;IACH,KAAK,CAAC,oBAAoB,CAAC,YAAyE;QAClG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,MAAM,CAAC,OAAO,eAAe,EAAE;YACxE,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,YAAY,EAAE,CAAC;SACvC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,WAAW,CACf,SAAiB,EACjB,OAAgC,EAChC,OAAuE;QAEvE,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,MAAM,CAAC,OAAO,gBAAgB,EAAE;YAC1F,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;;OAEG;IACH,KAAK,CAAC,SAAS,CACb,OAAgC,EAChC,OAA8B;QAE9B,MAAM,QAAQ,GAAG,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,MAAM,CAAC,OAAO,qBAAqB,EAAE;YAC/F,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;;OAEG;IACH,KAAK,CAAC,WAAW,CAAC,OAAoE;QACpF,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,IAAI,CAAC,MAAM,CAAC,OAAO,aAAa,MAAM,EAAE,CAAC,CAAC;QACjG,MAAM,IAAI,GAAG,MAAM,QAAQ,CAAC,IAAI,EAAE,CAAC;QACnC,OAAO,IAAI,CAAC,QAAQ,IAAI,EAAE,CAAC;IAC7B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,QAAQ,CAAC,UAAoB;QACjC,MAAM,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,cAAc,IAAI,CAAC,MAAM,CAAC,OAAO,gBAAgB,EAAE;YACzE,MAAM,EAAE,MAAM;YACd,IAAI,EAAE,IAAI,CAAC,SAAS,CAAC,EAAE,UAAU,EAAE,CAAC;SACrC,CAAC,CAAC;IACL,CAAC;CACF;AAjGD,8BAiGC;AAED;;GAEG;AACH,MAAa,WAAW;IACtB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;IAAG,CAAC;IAE9C;;OAEG;IACH,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;;OAEG;IACH,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;;OAEG;IACH,KAAK,CAAC,aAAa,CAAC,WAAmB;QACrC,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;YACL,OAAO,EAAE,WAAW;YACpB,IAAI;YACJ,IAAI;YACJ,gBAAgB;YAChB,MAAM;SACP,CAAC;IACJ,CAAC;CACF;AA9DD,kCA8DC;AAED;;GAEG;AACH,MAAa,WAAW;IAGtB,YAAoB,MAAsB;QAAtB,WAAM,GAAN,MAAM,CAAgB;QAFlC,mBAAc,GAAG,IAAI,GAAG,EAAmB,CAAC;IAEP,CAAC;IAE9C;;OAEG;IACH,KAAK,CAAC,MAAM,CAAC,IAAY,EAAE,OAAqE;QAC9F,MAAM,UAAU,GAAG,OAAO,EAAE,UAAU,IAAI,EAAE,CAAC;QAC7C,MAAM,SAAS,GAAG,GAAG,IAAI,CAAC,MAAM,CAAC,OAAO,IAAI,IAAI,CAAC,GAAG,EAAE,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,CAAC;QACvG,MAAM,SAAS,GAAG,IAAI,IAAI,CAAC,IAAI,CAAC,GAAG,EAAE,GAAG,UAAU,GAAG,EAAE,GAAG,IAAI,CAAC,CAAC;QAEhE,MAAM,OAAO,GAAY;YACvB,EAAE,EAAE,SAAS;YACb,IAAI;YACJ,SAAS,EAAE,IAAI,CAAC,MAAM,CAAC,OAAQ;YAC/B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;YACnC,SAAS,EAAE,SAAS,CAAC,WAAW,EAAE;YAClC,QAAQ,EAAE,OAAO,EAAE,QAAQ;SAC5B,CAAC;QAEF,IAAI,CAAC,cAAc,CAAC,GAAG,CAAC,SAAS,EAAE,OAAO,CAAC,CAAC;QAE5C,6BAA6B;QAC7B,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CAChC,EAAE,IAAI,EAAE,iBAAiB,EAAE,OAAO,EAAE,EACpC,EAAE,OAAO,EAAE,eAAe,EAAE,CAC7B,CAAC;QAEF,OAAO,OAAO,CAAC;IACjB,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,GAAG,CAAC,SAAiB;QACzB,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,SAAS,CAAC,CAAC;QAEtC,wBAAwB;QACxB,MAAM,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,SAAS,CAChC;YACE,IAAI,EAAE,eAAe;YACrB,SAAS;YACT,OAAO,EAAE,IAAI,CAAC,MAAM,CAAC,OAAO;YAC5B,OAAO,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;SAClC,EACD,EAAE,OAAO,EAAE,eAAe,EAAE,CAC7B,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,SAAS;QACP,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;QACvB,MAAM,MAAM,GAAc,EAAE,CAAC;QAE7B,KAAK,MAAM,CAAC,EAAE,EAAE,OAAO,CAAC,IAAI,IAAI,CAAC,cAAc,EAAE,CAAC;YAChD,IAAI,IAAI,IAAI,CAAC,OAAO,CAAC,SAAS,CAAC,GAAG,GAAG,EAAE,CAAC;gBACtC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YACvB,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YACjC,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;CACF;AApED,kCAoEC;AAED;;GAEG;AACH,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,IAAI,CAAC,aAAa,EAAE;SAChD,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,QAAQ,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,CAAC;QAEtC,iCAAiC;QACjC,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,CAAC;YAC5B,aAAa,CAAC,IAAI,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;QACpC,CAAC;IACH,CAAC;IAED;;OAEG;IACH,IAAI,OAAO;QACT,OAAO,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC;IAC7B,CAAC;IAED;;OAEG;IACH,KAAK,CAAC,OAAO,CAAC,WAAmB;QAC/B,OAAO,IAAI,CAAC,QAAQ,CAAC,aAAa,CAAC,WAAW,CAAC,CAAC;IAClD,CAAC;IAED;;OAEG;IACH,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;IAED;;OAEG;IACK,aAAa;QACnB,wEAAwE;QACxE,iCAAiC;QACjC,OAAO,OAAO,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC;IAC7B,CAAC;CACF;AAzED,wCAyEC;AAED,iBAAiB;AACjB,kBAAe,cAAc,CAAC;AAE9B,uBAAuB;AAChB,KAAK,UAAU,OAAO,CAAC,WAAmB,EAAE,MAA6B;IAC9E,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,CAAC,CAAC;AACrC,CAAC","sourcesContent":["/**\n * Private Connect SDK\n * \n * Programmatic access to Private Connect services and agent orchestration.\n * \n * @example\n * ```typescript\n * import { PrivateConnect } from '@privateconnect/sdk';\n * \n * const pc = new PrivateConnect({ apiKey: 'your-api-key' });\n * \n * // Connect to a service\n * const db = await pc.connect('postgres-prod');\n * console.log(db.connectionString); // postgres://localhost:5432/...\n * \n * // List all agents\n * const agents = await pc.agents.list();\n * \n * // Send message to another agent\n * await pc.agents.sendMessage(targetAgentId, { action: 'deploy' });\n * ```\n */\n\nexport interface PrivateConnectConfig {\n  /** API key for authentication */\n  apiKey: string;\n  /** Hub URL (default: https://api.privateconnect.co) */\n  hubUrl?: string;\n  /** Agent ID (auto-detected from config if not provided) */\n  agentId?: string;\n  /** Disable usage tracking (default: false) */\n  disableTracking?: boolean;\n}\n\n// Track SDK usage (fire and forget)\nfunction trackSdkUsage(hubUrl: string): void {\n  const data = JSON.stringify({\n    os: typeof process !== 'undefined' ? process.platform : 'browser',\n    arch: typeof process !== 'undefined' ? (process.arch === 'arm64' ? 'arm64' : 'x64') : 'unknown',\n    version: 'sdk',\n    source: 'sdk',\n  });\n  \n  fetch(`${hubUrl}/v1/events/install`, {\n    method: 'POST',\n    headers: { 'Content-Type': 'application/json' },\n    body: data,\n  }).catch(() => {}); // Silently ignore errors\n}\n\nexport interface Service {\n  id: string;\n  name: string;\n  targetHost: string;\n  targetPort: number;\n  tunnelPort?: number;\n  protocol: string;\n  status: string;\n  agentLabel?: string;\n}\n\nexport interface Agent {\n  id: string;\n  name?: string;\n  label: string;\n  isOnline: boolean;\n  lastSeenAt: string;\n  capabilities: string[];\n  services: string[];\n}\n\nexport interface Connection {\n  service: string;\n  host: string;\n  port: number;\n  connectionString: string;\n  envVar: string;\n}\n\nexport interface Message {\n  id: string;\n  from: { id: string; name?: string; label?: string };\n  channel: string;\n  type: string;\n  payload: Record<string, unknown>;\n  createdAt: string;\n  isRead: boolean;\n}\n\nexport interface Session {\n  id: string;\n  name: string;\n  createdBy: string;\n  createdAt: string;\n  expiresAt: string;\n  metadata?: Record<string, unknown>;\n}\n\n/**\n * Agents API for discovery and orchestration\n */\nexport class AgentsAPI {\n  constructor(private client: PrivateConnect) {}\n\n  /**\n   * List all agents in the workspace\n   */\n  async list(options?: { onlineOnly?: boolean }): Promise<Agent[]> {\n    const response = await this.client.fetch('/v1/agents/orchestration');\n    const data = await response.json();\n    let agents = data.agents || [];\n    \n    if (options?.onlineOnly) {\n      agents = agents.filter((a: any) => a.isOnline);\n    }\n    \n    return agents.map((a: any) => ({\n      id: a.id,\n      name: a.name,\n      label: a.label,\n      isOnline: a.isOnline,\n      lastSeenAt: a.lastSeenAt,\n      capabilities: a.capabilities?.map((c: any) => c.name) || [],\n      services: a.services?.map((s: any) => s.name) || [],\n    }));\n  }\n\n  /**\n   * Find agents by capability\n   */\n  async findByCapability(capability: string): Promise<Agent[]> {\n    const response = await this.client.fetch(`/v1/agents/by-capability/${encodeURIComponent(capability)}`);\n    const data = await response.json();\n    return data.agents || [];\n  }\n\n  /**\n   * Register capabilities for this agent\n   */\n  async registerCapabilities(capabilities: Array<{ name: string; metadata?: Record<string, unknown> }>): Promise<void> {\n    await this.client.fetch(`/v1/agents/${this.client.agentId}/capabilities`, {\n      method: 'POST',\n      body: JSON.stringify({ capabilities }),\n    });\n  }\n\n  /**\n   * Send a message to another agent\n   */\n  async sendMessage(\n    toAgentId: string,\n    payload: Record<string, unknown>,\n    options?: { channel?: string; type?: 'request' | 'response' | 'event' }\n  ): Promise<{ messageId: string }> {\n    const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/send`, {\n      method: 'POST',\n      body: JSON.stringify({ toAgentId, payload, ...options }),\n    });\n    return response.json();\n  }\n\n  /**\n   * Broadcast a message to all online agents\n   */\n  async broadcast(\n    payload: Record<string, unknown>,\n    options?: { channel?: string }\n  ): Promise<{ sent: number }> {\n    const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/broadcast`, {\n      method: 'POST',\n      body: JSON.stringify({ payload, ...options }),\n    });\n    return response.json();\n  }\n\n  /**\n   * Get messages for this agent\n   */\n  async getMessages(options?: { channel?: string; unreadOnly?: boolean; limit?: number }): Promise<Message[]> {\n    const params = new URLSearchParams();\n    if (options?.channel) params.set('channel', options.channel);\n    if (options?.unreadOnly !== undefined) params.set('unreadOnly', String(options.unreadOnly));\n    if (options?.limit) params.set('limit', String(options.limit));\n    \n    const response = await this.client.fetch(`/v1/agents/${this.client.agentId}/messages?${params}`);\n    const data = await response.json();\n    return data.messages || [];\n  }\n\n  /**\n   * Mark messages as read\n   */\n  async markRead(messageIds: string[]): Promise<void> {\n    await this.client.fetch(`/v1/agents/${this.client.agentId}/messages/read`, {\n      method: 'POST',\n      body: JSON.stringify({ messageIds }),\n    });\n  }\n}\n\n/**\n * Services API for connecting to and managing services\n */\nexport class ServicesAPI {\n  constructor(private client: PrivateConnect) {}\n\n  /**\n   * List all services\n   */\n  async list(): Promise<Service[]> {\n    const response = await this.client.fetch('/v1/services');\n    return response.json();\n  }\n\n  /**\n   * Get a specific service by name\n   */\n  async get(name: string): Promise<Service | null> {\n    const services = await this.list();\n    return services.find(s => s.name.toLowerCase() === name.toLowerCase()) || null;\n  }\n\n  /**\n   * Get connection details for a service\n   */\n  async getConnection(serviceName: string): Promise<Connection> {\n    const service = await this.get(serviceName);\n    if (!service) {\n      throw new Error(`Service \"${serviceName}\" not found`);\n    }\n\n    const port = service.tunnelPort || service.targetPort;\n    const host = 'localhost';\n    \n    let connectionString = '';\n    let envVar = 'SERVICE_URL';\n    \n    if (service.targetPort === 5432 || service.protocol === 'postgres') {\n      connectionString = `postgres://${host}:${port}/postgres`;\n      envVar = 'DATABASE_URL';\n    } else if (service.targetPort === 3306 || service.protocol === 'mysql') {\n      connectionString = `mysql://${host}:${port}`;\n      envVar = 'DATABASE_URL';\n    } else if (service.targetPort === 6379 || service.protocol === 'redis') {\n      connectionString = `redis://${host}:${port}`;\n      envVar = 'REDIS_URL';\n    } else if (service.targetPort === 27017 || service.protocol === 'mongodb') {\n      connectionString = `mongodb://${host}:${port}`;\n      envVar = 'MONGODB_URI';\n    } else if (service.protocol === 'http' || service.protocol === 'https') {\n      connectionString = `http://${host}:${port}`;\n      envVar = 'API_URL';\n    } else {\n      connectionString = `tcp://${host}:${port}`;\n      envVar = `${serviceName.toUpperCase().replace(/-/g, '_')}_URL`;\n    }\n\n    return {\n      service: serviceName,\n      host,\n      port,\n      connectionString,\n      envVar,\n    };\n  }\n}\n\n/**\n * Sessions API for ephemeral orchestration sessions\n */\nexport class SessionsAPI {\n  private activeSessions = new Map<string, Session>();\n\n  constructor(private client: PrivateConnect) {}\n\n  /**\n   * Create an orchestration session\n   */\n  async create(name: string, options?: { ttlMinutes?: number; metadata?: Record<string, unknown> }): Promise<Session> {\n    const ttlMinutes = options?.ttlMinutes || 60;\n    const sessionId = `${this.client.agentId}-${Date.now()}-${Math.random().toString(36).substring(2, 8)}`;\n    const expiresAt = new Date(Date.now() + ttlMinutes * 60 * 1000);\n\n    const session: Session = {\n      id: sessionId,\n      name,\n      createdBy: this.client.agentId!,\n      createdAt: new Date().toISOString(),\n      expiresAt: expiresAt.toISOString(),\n      metadata: options?.metadata,\n    };\n\n    this.activeSessions.set(sessionId, session);\n\n    // Broadcast session creation\n    await this.client.agents.broadcast(\n      { type: 'session:created', session },\n      { channel: 'orchestration' }\n    );\n\n    return session;\n  }\n\n  /**\n   * End an orchestration session\n   */\n  async end(sessionId: string): Promise<void> {\n    this.activeSessions.delete(sessionId);\n\n    // Broadcast session end\n    await this.client.agents.broadcast(\n      {\n        type: 'session:ended',\n        sessionId,\n        endedBy: this.client.agentId,\n        endedAt: new Date().toISOString(),\n      },\n      { channel: 'orchestration' }\n    );\n  }\n\n  /**\n   * Get active sessions\n   */\n  getActive(): Session[] {\n    const now = new Date();\n    const active: Session[] = [];\n    \n    for (const [id, session] of this.activeSessions) {\n      if (new Date(session.expiresAt) > now) {\n        active.push(session);\n      } else {\n        this.activeSessions.delete(id);\n      }\n    }\n    \n    return active;\n  }\n}\n\n/**\n * Main Private Connect SDK client\n */\nexport class PrivateConnect {\n  private config: { apiKey: string; hubUrl: string; agentId: string };\n  \n  /** Agents API for discovery and orchestration */\n  public agents: AgentsAPI;\n  \n  /** Services API for connecting to services */\n  public services: ServicesAPI;\n  \n  /** Sessions API for ephemeral orchestration */\n  public sessions: SessionsAPI;\n\n  constructor(config: PrivateConnectConfig) {\n    this.config = {\n      apiKey: config.apiKey,\n      hubUrl: config.hubUrl || 'https://api.privateconnect.co',\n      agentId: config.agentId || this.detectAgentId(),\n    };\n\n    this.agents = new AgentsAPI(this);\n    this.services = new ServicesAPI(this);\n    this.sessions = new SessionsAPI(this);\n\n    // Track SDK usage (non-blocking)\n    if (!config.disableTracking) {\n      trackSdkUsage(this.config.hubUrl);\n    }\n  }\n\n  /**\n   * Get the agent ID\n   */\n  get agentId(): string | undefined {\n    return this.config.agentId;\n  }\n\n  /**\n   * Connect to a service and get connection details\n   */\n  async connect(serviceName: string): Promise<Connection> {\n    return this.services.getConnection(serviceName);\n  }\n\n  /**\n   * Internal fetch helper\n   */\n  async fetch(path: string, options?: RequestInit): Promise<Response> {\n    const url = `${this.config.hubUrl}${path}`;\n    const response = await fetch(url, {\n      ...options,\n      headers: {\n        'x-api-key': this.config.apiKey,\n        'Content-Type': 'application/json',\n        ...options?.headers,\n      },\n    });\n\n    if (!response.ok) {\n      const error = await response.json().catch(() => ({ message: 'Unknown error' }));\n      throw new Error(error.message || `Request failed: ${response.status}`);\n    }\n\n    return response;\n  }\n\n  /**\n   * Try to detect agent ID from local config\n   */\n  private detectAgentId(): string {\n    // In a real implementation, this would read from ~/.connect/config.json\n    // For now, generate a default ID\n    return `sdk-${Date.now()}`;\n  }\n}\n\n// Default export\nexport default PrivateConnect;\n\n// Convenience function\nexport async function connect(serviceName: string, config?: PrivateConnectConfig): Promise<Connection> {\n  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;\n  if (!apiKey) {\n    throw new Error('API key required. Set PRIVATECONNECT_API_KEY or pass config.apiKey');\n  }\n  \n  const client = new PrivateConnect({ ...config, apiKey });\n  return client.connect(serviceName);\n}\n\n"]}
361
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\r\n * Private Connect SDK\r\n *\r\n * Programmatic access to Private Connect services, grants, and agent orchestration.\r\n *\r\n * @example\r\n * ```typescript\r\n * import { PrivateConnect } from '@privateconnect/sdk';\r\n *\r\n * const pc = new PrivateConnect({ apiKey: 'your-api-key' });\r\n *\r\n * // Connect to a service (assumes tunnel is already open)\r\n * const db = await pc.connect('postgres-prod');\r\n * console.log(db.connectionString); // postgres://localhost:5432/...\r\n *\r\n * // Grant an AI agent temporary access\r\n * const grant = await pc.grants.create({\r\n *   agentLabel: 'claude',\r\n *   resourceType: 'db',\r\n *   resourceName: 'postgres',\r\n *   ttl: '5m',\r\n * });\r\n * console.log(grant.token); // gnt_...\r\n *\r\n * // List all agents\r\n * const agents = await pc.agents.list();\r\n * ```\r\n */\r\n\r\nimport * as fs from 'fs';\r\nimport * as path from 'path';\r\nimport * as os from 'os';\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Types\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport interface PrivateConnectConfig {\r\n  /** API key for authentication */\r\n  apiKey: string;\r\n  /** Hub URL (default: https://api.privateconnect.co) */\r\n  hubUrl?: string;\r\n  /** Agent ID (auto-detected from local config if not provided) */\r\n  agentId?: string;\r\n  /** Disable usage tracking (default: false) */\r\n  disableTracking?: boolean;\r\n}\r\n\r\nexport interface Service {\r\n  id: string;\r\n  name: string;\r\n  targetHost: string;\r\n  targetPort: number;\r\n  tunnelPort?: number;\r\n  protocol: string;\r\n  status: string;\r\n  agentLabel?: string;\r\n}\r\n\r\nexport interface Agent {\r\n  id: string;\r\n  name?: string;\r\n  label: string;\r\n  isOnline: boolean;\r\n  lastSeenAt: string;\r\n  capabilities: string[];\r\n  services: string[];\r\n}\r\n\r\nexport interface Connection {\r\n  service: string;\r\n  host: string;\r\n  port: number;\r\n  connectionString: string;\r\n  envVar: string;\r\n  /** Present when using a grant-based connection */\r\n  grantToken?: string;\r\n  /** Present when using a grant-based connection */\r\n  grantEndpoint?: string;\r\n}\r\n\r\nexport interface Message {\r\n  id: string;\r\n  from: { id: string; name?: string; label?: string };\r\n  channel: string;\r\n  type: string;\r\n  payload: Record<string, unknown>;\r\n  createdAt: string;\r\n  isRead: boolean;\r\n}\r\n\r\nexport interface Grant {\r\n  id: string;\r\n  agentLabel: string;\r\n  resourceType: string;\r\n  resourceName: string;\r\n  scope: string;\r\n  tokenPrefix?: string;\r\n  persistent: boolean;\r\n  expiresAt: string | null;\r\n  expiresInMinutes?: number | null;\r\n  token?: string;\r\n  endpoint?: string;\r\n}\r\n\r\nexport interface GrantCreateOptions {\r\n  agentLabel: string;\r\n  resourceType: 'db' | 'api' | 'path';\r\n  resourceName: string;\r\n  scope?: 'read-only' | 'full';\r\n  /** Duration string: 60s, 5m, 1h, 1d. Omit for persistent grant. */\r\n  ttl?: string;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Tracking\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nfunction trackSdkUsage(hubUrl: string): void {\r\n  const data = JSON.stringify({\r\n    os: typeof process !== 'undefined' ? process.platform : 'browser',\r\n    arch: typeof process !== 'undefined' ? (process.arch === 'arm64' ? 'arm64' : 'x64') : 'unknown',\r\n    version: 'sdk',\r\n    source: 'sdk',\r\n  });\r\n\r\n  fetch(`${hubUrl}/v1/events/install`, {\r\n    method: 'POST',\r\n    headers: { 'Content-Type': 'application/json' },\r\n    body: data,\r\n  }).catch(() => {});\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agent ID Detection\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nconst CONFIG_PATH = path.join(os.homedir(), '.private-connect', 'config.json');\r\n\r\n/**\r\n * Detect the agent ID from local config or environment.\r\n * Returns undefined if no agent is configured — callers that need an agent ID\r\n * should surface a clear error rather than using a fake one.\r\n */\r\nfunction detectAgentId(): string | undefined {\r\n  // 1. Environment variables (highest priority — useful in CI)\r\n  const envId = process.env.PRIVATECONNECT_AGENT_ID || process.env.CONNECT_AGENT_ID;\r\n  if (envId) return envId;\r\n\r\n  // 2. Local config file (~/.private-connect/config.json)\r\n  try {\r\n    if (fs.existsSync(CONFIG_PATH)) {\r\n      const raw = fs.readFileSync(CONFIG_PATH, 'utf-8');\r\n      const config = JSON.parse(raw);\r\n      if (config.agentId && typeof config.agentId === 'string') {\r\n        return config.agentId;\r\n      }\r\n    }\r\n  } catch {\r\n    // Config unreadable or malformed — fall through\r\n  }\r\n\r\n  return undefined;\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Agents API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class AgentsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(options?: { onlineOnly?: boolean }): Promise<Agent[]> {\r\n    const response = await this.client.fetch('/v1/agents/orchestration');\r\n    const data = await response.json();\r\n    let agents = data.agents || [];\r\n\r\n    if (options?.onlineOnly) {\r\n      agents = agents.filter((a: any) => a.isOnline);\r\n    }\r\n\r\n    return agents.map((a: any) => ({\r\n      id: a.id,\r\n      name: a.name,\r\n      label: a.label,\r\n      isOnline: a.isOnline,\r\n      lastSeenAt: a.lastSeenAt,\r\n      capabilities: a.capabilities?.map((c: any) => c.name) || [],\r\n      services: a.services?.map((s: any) => s.name) || [],\r\n    }));\r\n  }\r\n\r\n  async findByCapability(capability: string): Promise<Agent[]> {\r\n    const response = await this.client.fetch(`/v1/agents/by-capability/${encodeURIComponent(capability)}`);\r\n    const data = await response.json();\r\n    return data.agents || [];\r\n  }\r\n\r\n  async registerCapabilities(capabilities: Array<{ name: string; metadata?: Record<string, unknown> }>): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/capabilities`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ capabilities }),\r\n    });\r\n  }\r\n\r\n  async sendMessage(\r\n    toAgentId: string,\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string; type?: 'request' | 'response' | 'event' }\r\n  ): Promise<{ messageId: string }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/send`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ toAgentId, payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async broadcast(\r\n    payload: Record<string, unknown>,\r\n    options?: { channel?: string }\r\n  ): Promise<{ sent: number }> {\r\n    const agentId = this.client.requireAgentId();\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/broadcast`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ payload, ...options }),\r\n    });\r\n    return response.json();\r\n  }\r\n\r\n  async getMessages(options?: { channel?: string; unreadOnly?: boolean; limit?: number }): Promise<Message[]> {\r\n    const agentId = this.client.requireAgentId();\r\n    const params = new URLSearchParams();\r\n    if (options?.channel) params.set('channel', options.channel);\r\n    if (options?.unreadOnly !== undefined) params.set('unreadOnly', String(options.unreadOnly));\r\n    if (options?.limit) params.set('limit', String(options.limit));\r\n\r\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages?${params}`);\r\n    const data = await response.json();\r\n    return data.messages || [];\r\n  }\r\n\r\n  async markRead(messageIds: string[]): Promise<void> {\r\n    const agentId = this.client.requireAgentId();\r\n    await this.client.fetch(`/v1/agents/${agentId}/messages/read`, {\r\n      method: 'POST',\r\n      body: JSON.stringify({ messageIds }),\r\n    });\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Services API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class ServicesAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  async list(): Promise<Service[]> {\r\n    const response = await this.client.fetch('/v1/services');\r\n    return response.json();\r\n  }\r\n\r\n  async get(name: string): Promise<Service | null> {\r\n    const services = await this.list();\r\n    return services.find(s => s.name.toLowerCase() === name.toLowerCase()) || null;\r\n  }\r\n\r\n  /**\r\n   * Get connection details for a service.\r\n   *\r\n   * If `grantToken` is provided, returns a proxied connection via the hub's\r\n   * grant endpoint instead of assuming a local tunnel.\r\n   */\r\n  async getConnection(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    if (options?.grantToken) {\r\n      return this.getGrantConnection(serviceName, options.grantToken);\r\n    }\r\n\r\n    const service = await this.get(serviceName);\r\n    if (!service) {\r\n      throw new Error(`Service \"${serviceName}\" not found`);\r\n    }\r\n\r\n    const port = service.tunnelPort || service.targetPort;\r\n    const host = 'localhost';\r\n\r\n    let connectionString = '';\r\n    let envVar = 'SERVICE_URL';\r\n\r\n    if (service.targetPort === 5432 || service.protocol === 'postgres') {\r\n      connectionString = `postgres://${host}:${port}/postgres`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 3306 || service.protocol === 'mysql') {\r\n      connectionString = `mysql://${host}:${port}`;\r\n      envVar = 'DATABASE_URL';\r\n    } else if (service.targetPort === 6379 || service.protocol === 'redis') {\r\n      connectionString = `redis://${host}:${port}`;\r\n      envVar = 'REDIS_URL';\r\n    } else if (service.targetPort === 27017 || service.protocol === 'mongodb') {\r\n      connectionString = `mongodb://${host}:${port}`;\r\n      envVar = 'MONGODB_URI';\r\n    } else if (service.protocol === 'http' || service.protocol === 'https') {\r\n      connectionString = `http://${host}:${port}`;\r\n      envVar = 'API_URL';\r\n    } else {\r\n      connectionString = `tcp://${host}:${port}`;\r\n      envVar = `${serviceName.toUpperCase().replace(/-/g, '_')}_URL`;\r\n    }\r\n\r\n    return { service: serviceName, host, port, connectionString, envVar };\r\n  }\r\n\r\n  private getGrantConnection(serviceName: string, grantToken: string): Connection {\r\n    const hubUrl = this.client.hubUrl;\r\n    const grantEndpoint = `${hubUrl}/grant/${encodeURIComponent(serviceName)}`;\r\n\r\n    return {\r\n      service: serviceName,\r\n      host: new URL(hubUrl).hostname,\r\n      port: new URL(hubUrl).port ? parseInt(new URL(hubUrl).port, 10) : 443,\r\n      connectionString: grantEndpoint,\r\n      envVar: 'GRANT_URL',\r\n      grantToken,\r\n      grantEndpoint,\r\n    };\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Grants API\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class GrantsAPI {\r\n  constructor(private client: PrivateConnect) {}\r\n\r\n  /**\r\n   * Create a time-limited access grant for an AI agent or external consumer.\r\n   */\r\n  async create(options: GrantCreateOptions): Promise<Grant> {\r\n    const response = await this.client.fetch('/v1/grants', {\r\n      method: 'POST',\r\n      body: JSON.stringify(options),\r\n    });\r\n    const data = await response.json();\r\n    return data.grant;\r\n  }\r\n\r\n  /**\r\n   * List active grants in the workspace.\r\n   */\r\n  async list(options?: { includeExpired?: boolean }): Promise<Grant[]> {\r\n    const params = new URLSearchParams();\r\n    if (options?.includeExpired) params.set('includeExpired', 'true');\r\n    const response = await this.client.fetch(`/v1/grants?${params}`);\r\n    const data = await response.json();\r\n    return data.grants || [];\r\n  }\r\n\r\n  /**\r\n   * Revoke an active grant immediately.\r\n   */\r\n  async revoke(grantId: string): Promise<void> {\r\n    await this.client.fetch(`/v1/grants/${grantId}`, { method: 'DELETE' });\r\n  }\r\n\r\n  /**\r\n   * Validate a grant token (public endpoint — no API key required).\r\n   * Returns the grant if valid, null otherwise.\r\n   */\r\n  async validate(token: string): Promise<Grant | null> {\r\n    try {\r\n      const url = `${this.client.hubUrl}/v1/grants/validate`;\r\n      const response = await fetch(url, {\r\n        method: 'POST',\r\n        headers: { 'Content-Type': 'application/json' },\r\n        body: JSON.stringify({ token }),\r\n      });\r\n      if (!response.ok) return null;\r\n      const data = await response.json();\r\n      return data.valid ? data.grant : null;\r\n    } catch {\r\n      return null;\r\n    }\r\n  }\r\n}\r\n\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n// Main Client\r\n// ─────────────────────────────────────────────────────────────────────────────\r\n\r\nexport class PrivateConnect {\r\n  private config: { apiKey: string; hubUrl: string; agentId?: string };\r\n\r\n  /** Agents API for discovery and orchestration */\r\n  public agents: AgentsAPI;\r\n\r\n  /** Services API for connecting to services */\r\n  public services: ServicesAPI;\r\n\r\n  /** Grants API for managing scoped access tokens (time-limited or persistent) */\r\n  public grants: GrantsAPI;\r\n\r\n  constructor(config: PrivateConnectConfig) {\r\n    this.config = {\r\n      apiKey: config.apiKey,\r\n      hubUrl: config.hubUrl || 'https://api.privateconnect.co',\r\n      agentId: config.agentId || detectAgentId(),\r\n    };\r\n\r\n    this.agents = new AgentsAPI(this);\r\n    this.services = new ServicesAPI(this);\r\n    this.grants = new GrantsAPI(this);\r\n\r\n    if (!config.disableTracking) {\r\n      trackSdkUsage(this.config.hubUrl);\r\n    }\r\n  }\r\n\r\n  /** The resolved agent ID, or undefined if not configured. */\r\n  get agentId(): string | undefined {\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** The hub URL this client is connected to. */\r\n  get hubUrl(): string {\r\n    return this.config.hubUrl;\r\n  }\r\n\r\n  /**\r\n   * Shorthand: get connection details for a service.\r\n   * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.\r\n   */\r\n  async connect(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\r\n    return this.services.getConnection(serviceName, options);\r\n  }\r\n\r\n  /**\r\n   * Returns the agent ID or throws if not configured.\r\n   * Used by APIs that require an authenticated agent identity.\r\n   */\r\n  requireAgentId(): string {\r\n    if (!this.config.agentId) {\r\n      throw new Error(\r\n        'Agent ID not found. Either:\\n' +\r\n        '  1. Run \"connect up\" to register this machine, or\\n' +\r\n        '  2. Set PRIVATECONNECT_AGENT_ID environment variable, or\\n' +\r\n        '  3. Pass agentId in the PrivateConnect constructor.'\r\n      );\r\n    }\r\n    return this.config.agentId;\r\n  }\r\n\r\n  /** Internal fetch with API key auth. */\r\n  async fetch(path: string, options?: RequestInit): Promise<Response> {\r\n    const url = `${this.config.hubUrl}${path}`;\r\n    const response = await fetch(url, {\r\n      ...options,\r\n      headers: {\r\n        'x-api-key': this.config.apiKey,\r\n        'Content-Type': 'application/json',\r\n        ...options?.headers,\r\n      },\r\n    });\r\n\r\n    if (!response.ok) {\r\n      const error = await response.json().catch(() => ({ message: 'Unknown error' }));\r\n      throw new Error(error.message || `Request failed: ${response.status}`);\r\n    }\r\n\r\n    return response;\r\n  }\r\n}\r\n\r\nexport default PrivateConnect;\r\n\r\n/** Convenience function for quick one-off connections. */\r\nexport async function connect(\r\n  serviceName: string,\r\n  config?: PrivateConnectConfig & { grantToken?: string },\r\n): Promise<Connection> {\r\n  const apiKey = config?.apiKey || process.env.PRIVATECONNECT_API_KEY;\r\n  if (!apiKey) {\r\n    throw new Error('API key required. Set PRIVATECONNECT_API_KEY or pass config.apiKey');\r\n  }\r\n\r\n  const client = new PrivateConnect({ ...config, apiKey });\r\n  return client.connect(serviceName, { grantToken: config?.grantToken });\r\n}\r\n"]}
package/package.json CHANGED
@@ -1,33 +1,33 @@
1
- {
2
- "name": "@privateconnect/sdk",
3
- "version": "0.7.1",
4
- "description": "TypeScript SDK for Private Connect - programmatic access to services and agent orchestration",
5
- "main": "dist/index.js",
6
- "types": "dist/index.d.ts",
7
- "files": [
8
- "dist"
9
- ],
10
- "scripts": {
11
- "build": "tsc",
12
- "dev": "tsc -w",
13
- "clean": "rm -rf dist"
14
- },
15
- "keywords": [
16
- "private-connect",
17
- "tunnel",
18
- "networking",
19
- "orchestration",
20
- "agent"
21
- ],
22
- "license": "FSL-1.1-MIT",
23
- "repository": {
24
- "type": "git",
25
- "url": "git+https://github.com/treadiehq/private-connect.git",
26
- "directory": "packages/sdk"
27
- },
28
- "devDependencies": {
29
- "@types/node": "^20.0.0",
30
- "typescript": "^5.0.0"
31
- }
32
- }
33
-
1
+ {
2
+ "name": "@privateconnect/sdk",
3
+ "version": "0.7.4",
4
+ "description": "TypeScript SDK for Private Connect - programmatic access to services and agent orchestration",
5
+ "main": "dist/index.js",
6
+ "types": "dist/index.d.ts",
7
+ "files": [
8
+ "dist"
9
+ ],
10
+ "scripts": {
11
+ "build": "tsc",
12
+ "dev": "tsc -w",
13
+ "clean": "rm -rf dist"
14
+ },
15
+ "keywords": [
16
+ "private-connect",
17
+ "tunnel",
18
+ "networking",
19
+ "orchestration",
20
+ "agent"
21
+ ],
22
+ "license": "FSL-1.1-MIT",
23
+ "repository": {
24
+ "type": "git",
25
+ "url": "git+https://github.com/treadiehq/private-connect.git",
26
+ "directory": "packages/sdk"
27
+ },
28
+ "devDependencies": {
29
+ "@types/node": "^20.0.0",
30
+ "typescript": "^5.0.0"
31
+ }
32
+ }
33
+