@privateconnect/sdk 0.7.1 → 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 +72 -76
- package/dist/index.js +154 -120
- package/package.json +1 -1
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
|
|
82
|
+
export interface Grant {
|
|
73
83
|
id: string;
|
|
74
|
-
|
|
75
|
-
|
|
76
|
-
|
|
84
|
+
agentLabel: string;
|
|
85
|
+
resourceType: string;
|
|
86
|
+
resourceName: string;
|
|
87
|
+
scope: string;
|
|
77
88
|
expiresAt: string;
|
|
78
|
-
|
|
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
|
|
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
|
|
150
|
+
* Create a time-limited access grant for an AI agent or external consumer.
|
|
161
151
|
*/
|
|
162
|
-
create(
|
|
163
|
-
ttlMinutes?: number;
|
|
164
|
-
metadata?: Record<string, unknown>;
|
|
165
|
-
}): Promise<Session>;
|
|
152
|
+
create(options: GrantCreateOptions): Promise<Grant>;
|
|
166
153
|
/**
|
|
167
|
-
*
|
|
154
|
+
* List active grants in the workspace.
|
|
168
155
|
*/
|
|
169
|
-
|
|
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
|
-
*
|
|
164
|
+
* Validate a grant token (public endpoint — no API key required).
|
|
165
|
+
* Returns the grant if valid, null otherwise.
|
|
172
166
|
*/
|
|
173
|
-
|
|
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
|
-
/**
|
|
185
|
-
|
|
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
|
-
*
|
|
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
|
|
186
|
+
connect(serviceName: string, options?: {
|
|
187
|
+
grantToken?: string;
|
|
188
|
+
}): Promise<Connection>;
|
|
195
189
|
/**
|
|
196
|
-
*
|
|
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
|
-
|
|
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.
|
|
31
|
+
exports.PrivateConnect = exports.GrantsAPI = exports.ServicesAPI = exports.AgentsAPI = void 0;
|
|
26
32
|
exports.connect = connect;
|
|
27
|
-
|
|
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(() => { });
|
|
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
|
-
*
|
|
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
|
-
|
|
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
|
|
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
|
|
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/${
|
|
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
|
-
|
|
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
|
-
|
|
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
|
-
|
|
200
|
-
|
|
201
|
-
class
|
|
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
|
|
237
|
+
* Create a time-limited access grant for an AI agent or external consumer.
|
|
208
238
|
*/
|
|
209
|
-
async create(
|
|
210
|
-
const
|
|
211
|
-
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
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
|
-
*
|
|
248
|
+
* List active grants in the workspace.
|
|
228
249
|
*/
|
|
229
|
-
async
|
|
230
|
-
|
|
231
|
-
|
|
232
|
-
|
|
233
|
-
|
|
234
|
-
|
|
235
|
-
|
|
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
|
-
*
|
|
259
|
+
* Revoke an active grant immediately.
|
|
241
260
|
*/
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
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.
|
|
257
|
-
|
|
258
|
-
|
|
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 ||
|
|
295
|
+
agentId: config.agentId || detectAgentId(),
|
|
266
296
|
};
|
|
267
297
|
this.agents = new AgentsAPI(this);
|
|
268
298
|
this.services = new ServicesAPI(this);
|
|
269
|
-
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
|
-
*
|
|
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
|
-
*
|
|
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
|
-
|
|
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