@privateconnect/sdk 0.6.24 → 0.7.3

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/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,123 @@ 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;
84
+ agentLabel: string;
85
+ resourceType: string;
86
+ resourceName: string;
87
+ scope: string;
77
88
  expiresAt: string;
78
- metadata?: Record<string, unknown>;
89
+ expiresInMinutes?: number;
90
+ token?: string;
91
+ endpoint?: string;
92
+ }
93
+ export interface GrantCreateOptions {
94
+ agentLabel: string;
95
+ resourceType: 'db' | 'api' | 'path';
96
+ resourceName: string;
97
+ scope?: 'read-only' | 'full';
98
+ /** Duration string: 60s, 5m, 1h, 1d */
99
+ ttl: string;
79
100
  }
80
- /**
81
- * Agents API for discovery and orchestration
82
- */
83
101
  export declare class AgentsAPI {
84
102
  private client;
85
103
  constructor(client: PrivateConnect);
86
- /**
87
- * List all agents in the workspace
88
- */
89
104
  list(options?: {
90
105
  onlineOnly?: boolean;
91
106
  }): Promise<Agent[]>;
92
- /**
93
- * Find agents by capability
94
- */
95
107
  findByCapability(capability: string): Promise<Agent[]>;
96
- /**
97
- * Register capabilities for this agent
98
- */
99
108
  registerCapabilities(capabilities: Array<{
100
109
  name: string;
101
110
  metadata?: Record<string, unknown>;
102
111
  }>): Promise<void>;
103
- /**
104
- * Send a message to another agent
105
- */
106
112
  sendMessage(toAgentId: string, payload: Record<string, unknown>, options?: {
107
113
  channel?: string;
108
114
  type?: 'request' | 'response' | 'event';
109
115
  }): Promise<{
110
116
  messageId: string;
111
117
  }>;
112
- /**
113
- * Broadcast a message to all online agents
114
- */
115
118
  broadcast(payload: Record<string, unknown>, options?: {
116
119
  channel?: string;
117
120
  }): Promise<{
118
121
  sent: number;
119
122
  }>;
120
- /**
121
- * Get messages for this agent
122
- */
123
123
  getMessages(options?: {
124
124
  channel?: string;
125
125
  unreadOnly?: boolean;
126
126
  limit?: number;
127
127
  }): Promise<Message[]>;
128
- /**
129
- * Mark messages as read
130
- */
131
128
  markRead(messageIds: string[]): Promise<void>;
132
129
  }
133
- /**
134
- * Services API for connecting to and managing services
135
- */
136
130
  export declare class ServicesAPI {
137
131
  private client;
138
132
  constructor(client: PrivateConnect);
139
- /**
140
- * List all services
141
- */
142
133
  list(): Promise<Service[]>;
143
- /**
144
- * Get a specific service by name
145
- */
146
134
  get(name: string): Promise<Service | null>;
147
135
  /**
148
- * Get connection details for a service
136
+ * Get connection details for a service.
137
+ *
138
+ * If `grantToken` is provided, returns a proxied connection via the hub's
139
+ * grant endpoint instead of assuming a local tunnel.
149
140
  */
150
- getConnection(serviceName: string): Promise<Connection>;
141
+ getConnection(serviceName: string, options?: {
142
+ grantToken?: string;
143
+ }): Promise<Connection>;
144
+ private getGrantConnection;
151
145
  }
152
- /**
153
- * Sessions API for ephemeral orchestration sessions
154
- */
155
- export declare class SessionsAPI {
146
+ export declare class GrantsAPI {
156
147
  private client;
157
- private activeSessions;
158
148
  constructor(client: PrivateConnect);
159
149
  /**
160
- * Create an orchestration session
150
+ * Create a time-limited access grant for an AI agent or external consumer.
161
151
  */
162
- create(name: string, options?: {
163
- ttlMinutes?: number;
164
- metadata?: Record<string, unknown>;
165
- }): Promise<Session>;
152
+ create(options: GrantCreateOptions): Promise<Grant>;
166
153
  /**
167
- * End an orchestration session
154
+ * List active grants in the workspace.
168
155
  */
169
- end(sessionId: string): Promise<void>;
156
+ list(options?: {
157
+ includeExpired?: boolean;
158
+ }): Promise<Grant[]>;
159
+ /**
160
+ * Revoke an active grant immediately.
161
+ */
162
+ revoke(grantId: string): Promise<void>;
170
163
  /**
171
- * Get active sessions
164
+ * Validate a grant token (public endpoint — no API key required).
165
+ * Returns the grant if valid, null otherwise.
172
166
  */
173
- getActive(): Session[];
167
+ validate(token: string): Promise<Grant | null>;
174
168
  }
175
- /**
176
- * Main Private Connect SDK client
177
- */
178
169
  export declare class PrivateConnect {
179
170
  private config;
180
171
  /** Agents API for discovery and orchestration */
181
172
  agents: AgentsAPI;
182
173
  /** Services API for connecting to services */
183
174
  services: ServicesAPI;
184
- /** Sessions API for ephemeral orchestration */
185
- sessions: SessionsAPI;
175
+ /** Grants API for managing time-limited access tokens */
176
+ grants: GrantsAPI;
186
177
  constructor(config: PrivateConnectConfig);
187
- /**
188
- * Get the agent ID
189
- */
178
+ /** The resolved agent ID, or undefined if not configured. */
190
179
  get agentId(): string | undefined;
180
+ /** The hub URL this client is connected to. */
181
+ get hubUrl(): string;
191
182
  /**
192
- * Connect to a service and get connection details
183
+ * Shorthand: get connection details for a service.
184
+ * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.
193
185
  */
194
- connect(serviceName: string): Promise<Connection>;
186
+ connect(serviceName: string, options?: {
187
+ grantToken?: string;
188
+ }): Promise<Connection>;
195
189
  /**
196
- * Internal fetch helper
190
+ * Returns the agent ID or throws if not configured.
191
+ * Used by APIs that require an authenticated agent identity.
197
192
  */
193
+ requireAgentId(): string;
194
+ /** Internal fetch with API key auth. */
198
195
  fetch(path: string, options?: RequestInit): Promise<Response>;
199
- /**
200
- * Try to detect agent ID from local config
201
- */
202
- private detectAgentId;
203
196
  }
204
197
  export default PrivateConnect;
205
- export declare function connect(serviceName: string, config?: PrivateConnectConfig): Promise<Connection>;
198
+ /** Convenience function for quick one-off connections. */
199
+ export declare function connect(serviceName: string, config?: PrivateConnectConfig & {
200
+ grantToken?: string;
201
+ }): 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":"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","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":["/**\n * Private Connect SDK\n *\n * Programmatic access to Private Connect services, grants, 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 (assumes tunnel is already open)\n * const db = await pc.connect('postgres-prod');\n * console.log(db.connectionString); // postgres://localhost:5432/...\n *\n * // Grant an AI agent temporary access\n * const grant = await pc.grants.create({\n *   agentLabel: 'claude',\n *   resourceType: 'db',\n *   resourceName: 'postgres',\n *   ttl: '5m',\n * });\n * console.log(grant.token); // gnt_...\n *\n * // List all agents\n * const agents = await pc.agents.list();\n * ```\n */\n\nimport * as fs from 'fs';\nimport * as path from 'path';\nimport * as os from 'os';\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Types\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 local config if not provided) */\n  agentId?: string;\n  /** Disable usage tracking (default: false) */\n  disableTracking?: boolean;\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  /** Present when using a grant-based connection */\n  grantToken?: string;\n  /** Present when using a grant-based connection */\n  grantEndpoint?: 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 Grant {\n  id: string;\n  agentLabel: string;\n  resourceType: string;\n  resourceName: string;\n  scope: string;\n  expiresAt: string;\n  expiresInMinutes?: number;\n  token?: string;\n  endpoint?: string;\n}\n\nexport interface GrantCreateOptions {\n  agentLabel: string;\n  resourceType: 'db' | 'api' | 'path';\n  resourceName: string;\n  scope?: 'read-only' | 'full';\n  /** Duration string: 60s, 5m, 1h, 1d */\n  ttl: string;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Tracking\n// ─────────────────────────────────────────────────────────────────────────────\n\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(() => {});\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Agent ID Detection\n// ─────────────────────────────────────────────────────────────────────────────\n\nconst CONFIG_PATH = path.join(os.homedir(), '.private-connect', 'config.json');\n\n/**\n * Detect the agent ID from local config or environment.\n * Returns undefined if no agent is configured — callers that need an agent ID\n * should surface a clear error rather than using a fake one.\n */\nfunction detectAgentId(): string | undefined {\n  // 1. Environment variables (highest priority — useful in CI)\n  const envId = process.env.PRIVATECONNECT_AGENT_ID || process.env.CONNECT_AGENT_ID;\n  if (envId) return envId;\n\n  // 2. Local config file (~/.private-connect/config.json)\n  try {\n    if (fs.existsSync(CONFIG_PATH)) {\n      const raw = fs.readFileSync(CONFIG_PATH, 'utf-8');\n      const config = JSON.parse(raw);\n      if (config.agentId && typeof config.agentId === 'string') {\n        return config.agentId;\n      }\n    }\n  } catch {\n    // Config unreadable or malformed — fall through\n  }\n\n  return undefined;\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Agents API\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport class AgentsAPI {\n  constructor(private client: PrivateConnect) {}\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  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  async registerCapabilities(capabilities: Array<{ name: string; metadata?: Record<string, unknown> }>): Promise<void> {\n    const agentId = this.client.requireAgentId();\n    await this.client.fetch(`/v1/agents/${agentId}/capabilities`, {\n      method: 'POST',\n      body: JSON.stringify({ capabilities }),\n    });\n  }\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 agentId = this.client.requireAgentId();\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/send`, {\n      method: 'POST',\n      body: JSON.stringify({ toAgentId, payload, ...options }),\n    });\n    return response.json();\n  }\n\n  async broadcast(\n    payload: Record<string, unknown>,\n    options?: { channel?: string }\n  ): Promise<{ sent: number }> {\n    const agentId = this.client.requireAgentId();\n    const response = await this.client.fetch(`/v1/agents/${agentId}/messages/broadcast`, {\n      method: 'POST',\n      body: JSON.stringify({ payload, ...options }),\n    });\n    return response.json();\n  }\n\n  async getMessages(options?: { channel?: string; unreadOnly?: boolean; limit?: number }): Promise<Message[]> {\n    const agentId = this.client.requireAgentId();\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/${agentId}/messages?${params}`);\n    const data = await response.json();\n    return data.messages || [];\n  }\n\n  async markRead(messageIds: string[]): Promise<void> {\n    const agentId = this.client.requireAgentId();\n    await this.client.fetch(`/v1/agents/${agentId}/messages/read`, {\n      method: 'POST',\n      body: JSON.stringify({ messageIds }),\n    });\n  }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Services API\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport class ServicesAPI {\n  constructor(private client: PrivateConnect) {}\n\n  async list(): Promise<Service[]> {\n    const response = await this.client.fetch('/v1/services');\n    return response.json();\n  }\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   * If `grantToken` is provided, returns a proxied connection via the hub's\n   * grant endpoint instead of assuming a local tunnel.\n   */\n  async getConnection(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\n    if (options?.grantToken) {\n      return this.getGrantConnection(serviceName, options.grantToken);\n    }\n\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 { service: serviceName, host, port, connectionString, envVar };\n  }\n\n  private getGrantConnection(serviceName: string, grantToken: string): Connection {\n    const hubUrl = this.client.hubUrl;\n    const grantEndpoint = `${hubUrl}/grant/${encodeURIComponent(serviceName)}`;\n\n    return {\n      service: serviceName,\n      host: new URL(hubUrl).hostname,\n      port: new URL(hubUrl).port ? parseInt(new URL(hubUrl).port, 10) : 443,\n      connectionString: grantEndpoint,\n      envVar: 'GRANT_URL',\n      grantToken,\n      grantEndpoint,\n    };\n  }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Grants API\n// ─────────────────────────────────────────────────────────────────────────────\n\nexport class GrantsAPI {\n  constructor(private client: PrivateConnect) {}\n\n  /**\n   * Create a time-limited access grant for an AI agent or external consumer.\n   */\n  async create(options: GrantCreateOptions): Promise<Grant> {\n    const response = await this.client.fetch('/v1/grants', {\n      method: 'POST',\n      body: JSON.stringify(options),\n    });\n    const data = await response.json();\n    return data.grant;\n  }\n\n  /**\n   * List active grants in the workspace.\n   */\n  async list(options?: { includeExpired?: boolean }): Promise<Grant[]> {\n    const params = new URLSearchParams();\n    if (options?.includeExpired) params.set('includeExpired', 'true');\n    const response = await this.client.fetch(`/v1/grants?${params}`);\n    const data = await response.json();\n    return data.grants || [];\n  }\n\n  /**\n   * Revoke an active grant immediately.\n   */\n  async revoke(grantId: string): Promise<void> {\n    await this.client.fetch(`/v1/grants/${grantId}`, { method: 'DELETE' });\n  }\n\n  /**\n   * Validate a grant token (public endpoint — no API key required).\n   * Returns the grant if valid, null otherwise.\n   */\n  async validate(token: string): Promise<Grant | null> {\n    try {\n      const url = `${this.client.hubUrl}/v1/grants/validate`;\n      const response = await fetch(url, {\n        method: 'POST',\n        headers: { 'Content-Type': 'application/json' },\n        body: JSON.stringify({ token }),\n      });\n      if (!response.ok) return null;\n      const data = await response.json();\n      return data.valid ? data.grant : null;\n    } catch {\n      return null;\n    }\n  }\n}\n\n// ─────────────────────────────────────────────────────────────────────────────\n// Main Client\n// ─────────────────────────────────────────────────────────────────────────────\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  /** Grants API for managing time-limited access tokens */\n  public grants: GrantsAPI;\n\n  constructor(config: PrivateConnectConfig) {\n    this.config = {\n      apiKey: config.apiKey,\n      hubUrl: config.hubUrl || 'https://api.privateconnect.co',\n      agentId: config.agentId || detectAgentId(),\n    };\n\n    this.agents = new AgentsAPI(this);\n    this.services = new ServicesAPI(this);\n    this.grants = new GrantsAPI(this);\n\n    if (!config.disableTracking) {\n      trackSdkUsage(this.config.hubUrl);\n    }\n  }\n\n  /** The resolved agent ID, or undefined if not configured. */\n  get agentId(): string | undefined {\n    return this.config.agentId;\n  }\n\n  /** The hub URL this client is connected to. */\n  get hubUrl(): string {\n    return this.config.hubUrl;\n  }\n\n  /**\n   * Shorthand: get connection details for a service.\n   * Pass `grantToken` to connect via the grant proxy instead of a local tunnel.\n   */\n  async connect(serviceName: string, options?: { grantToken?: string }): Promise<Connection> {\n    return this.services.getConnection(serviceName, options);\n  }\n\n  /**\n   * Returns the agent ID or throws if not configured.\n   * Used by APIs that require an authenticated agent identity.\n   */\n  requireAgentId(): string {\n    if (!this.config.agentId) {\n      throw new Error(\n        'Agent ID not found. Either:\\n' +\n        '  1. Run \"connect up\" to register this machine, or\\n' +\n        '  2. Set PRIVATECONNECT_AGENT_ID environment variable, or\\n' +\n        '  3. Pass agentId in the PrivateConnect constructor.'\n      );\n    }\n    return this.config.agentId;\n  }\n\n  /** Internal fetch with API key auth. */\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\nexport default PrivateConnect;\n\n/** Convenience function for quick one-off connections. */\nexport async function connect(\n  serviceName: string,\n  config?: PrivateConnectConfig & { grantToken?: string },\n): 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, { grantToken: config?.grantToken });\n}\n"]}
package/package.json CHANGED
@@ -1,12 +1,17 @@
1
1
  {
2
2
  "name": "@privateconnect/sdk",
3
- "version": "0.6.24",
3
+ "version": "0.7.3",
4
4
  "description": "TypeScript SDK for Private Connect - programmatic access to services and agent orchestration",
5
5
  "main": "dist/index.js",
6
6
  "types": "dist/index.d.ts",
7
7
  "files": [
8
8
  "dist"
9
9
  ],
10
+ "scripts": {
11
+ "build": "tsc",
12
+ "dev": "tsc -w",
13
+ "clean": "rm -rf dist"
14
+ },
10
15
  "keywords": [
11
16
  "private-connect",
12
17
  "tunnel",
@@ -23,10 +28,6 @@
23
28
  "devDependencies": {
24
29
  "@types/node": "^20.0.0",
25
30
  "typescript": "^5.0.0"
26
- },
27
- "scripts": {
28
- "build": "tsc",
29
- "dev": "tsc -w",
30
- "clean": "rm -rf dist"
31
31
  }
32
- }
32
+ }
33
+
package/LICENSE DELETED
@@ -1,110 +0,0 @@
1
- # Functional Source License, Version 1.1
2
-
3
- ## Abbreviation
4
-
5
- FSL-1.1-MIT
6
-
7
- ## Notice
8
-
9
- Copyright (c) 2026 Treadie, Inc
10
-
11
- ## Terms and Conditions
12
-
13
- ### Licensor ("We")
14
-
15
- The party offering the Software under these Terms and Conditions.
16
-
17
- ### The Software
18
-
19
- The "Software" is each version of the software that we make available under
20
- these Terms and Conditions, as indicated by our inclusion of these Terms and
21
- Conditions with the Software.
22
-
23
- ### License Grant
24
-
25
- Subject to your compliance with this License Grant and the Patents,
26
- Redistribution and Trademark clauses below, we hereby grant you the right to
27
- use, copy, modify, create derivative works, publicly perform, publicly display
28
- and redistribute the Software for any Permitted Purpose identified below.
29
-
30
- ### Permitted Purpose
31
-
32
- A Permitted Purpose is any purpose other than a Competing Use. A Competing Use
33
- means making the Software available to others in a commercial product or
34
- service that:
35
-
36
- 1. substitutes for the Software;
37
-
38
- 2. substitutes for any other product or service we offer using the Software
39
- that exists as of the date we make the Software available; or
40
-
41
- 3. offers the same or substantially similar functionality as the Software.
42
-
43
- Permitted Purposes specifically include using the Software:
44
-
45
- 1. for your internal use and access;
46
-
47
- 2. for non-commercial education;
48
-
49
- 3. for non-commercial research; and
50
-
51
- 4. in connection with professional services that you provide to a licensee
52
- using the Software in accordance with these Terms and Conditions.
53
-
54
- ### Patents
55
-
56
- To the extent your use for a Permitted Purpose would necessarily infringe our
57
- patents, the license grant above includes a license under our patents. If you
58
- make a claim against any party that the Software infringes or contributes to
59
- the infringement of any patent, then your patent license to the Software ends
60
- immediately.
61
-
62
- ### Redistribution
63
-
64
- The Terms and Conditions apply to all copies, modifications and derivatives of
65
- the Software.
66
-
67
- If you redistribute any copies, modifications or derivatives of the Software,
68
- you must include a copy of or a link to these Terms and Conditions and not
69
- remove any copyright notices provided in or with the Software.
70
-
71
- ### Disclaimer
72
-
73
- THE SOFTWARE IS PROVIDED "AS IS" AND WITHOUT WARRANTIES OF ANY KIND, EXPRESS OR
74
- IMPLIED, INCLUDING WITHOUT LIMITATION WARRANTIES OF FITNESS FOR A PARTICULAR
75
- PURPOSE, MERCHANTABILITY, TITLE OR NON-INFRINGEMENT.
76
-
77
- IN NO EVENT WILL WE HAVE ANY LIABILITY TO YOU ARISING OUT OF OR RELATED TO THE
78
- SOFTWARE, INCLUDING INDIRECT, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES,
79
- EVEN IF WE HAVE BEEN INFORMED OF THEIR POSSIBILITY IN ADVANCE.
80
-
81
- ### Trademarks
82
-
83
- Except for displaying the License Details and identifying us as the origin of
84
- the Software, you have no right under these Terms and Conditions to use our
85
- trademarks, trade names, service marks or product names.
86
-
87
- ## Grant of Future License
88
-
89
- We hereby irrevocably grant you an additional license to use the Software under
90
- the MIT license that is effective on the second anniversary of the date we make
91
- the Software available. On or after that date, you may use the Software under
92
- the MIT license, in which case the following will apply:
93
-
94
- Permission is hereby granted, free of charge, to any person obtaining a copy of
95
- this software and associated documentation files (the "Software"), to deal in
96
- the Software without restriction, including without limitation the rights to
97
- use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies
98
- of the Software, and to permit persons to whom the Software is furnished to do
99
- so, subject to the following conditions:
100
-
101
- The above copyright notice and this permission notice shall be included in all
102
- copies or substantial portions of the Software.
103
-
104
- THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
105
- IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
106
- FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
107
- AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
108
- LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
109
- OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
110
- SOFTWARE.