@agent-relay/dashboard-server 0.1.0
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/health-worker-manager.d.ts +62 -0
- package/dist/health-worker-manager.d.ts.map +1 -0
- package/dist/health-worker-manager.js +144 -0
- package/dist/health-worker-manager.js.map +1 -0
- package/dist/health-worker.d.ts +9 -0
- package/dist/health-worker.d.ts.map +1 -0
- package/dist/health-worker.js +79 -0
- package/dist/health-worker.js.map +1 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +17 -0
- package/dist/index.js.map +1 -0
- package/dist/metrics.d.ts +105 -0
- package/dist/metrics.d.ts.map +1 -0
- package/dist/metrics.js +193 -0
- package/dist/metrics.js.map +1 -0
- package/dist/needs-attention.d.ts +24 -0
- package/dist/needs-attention.d.ts.map +1 -0
- package/dist/needs-attention.js +78 -0
- package/dist/needs-attention.js.map +1 -0
- package/dist/server.d.ts +25 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +5099 -0
- package/dist/server.js.map +1 -0
- package/dist/start.d.ts +6 -0
- package/dist/start.d.ts.map +1 -0
- package/dist/start.js +13 -0
- package/dist/start.js.map +1 -0
- package/dist/types/threading.d.ts +8 -0
- package/dist/types/threading.d.ts.map +1 -0
- package/dist/types/threading.js +2 -0
- package/dist/types/threading.js.map +1 -0
- package/dist/user-bridge.d.ts +154 -0
- package/dist/user-bridge.d.ts.map +1 -0
- package/dist/user-bridge.js +372 -0
- package/dist/user-bridge.js.map +1 -0
- package/package.json +49 -0
|
@@ -0,0 +1,372 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* User Bridge - Bridges dashboard WebSocket users to the relay daemon.
|
|
3
|
+
*
|
|
4
|
+
* This module allows human users connected via WebSocket to:
|
|
5
|
+
* - Register as "user" entities in the relay daemon
|
|
6
|
+
* - Join/leave channels
|
|
7
|
+
* - Send/receive messages through the relay daemon
|
|
8
|
+
* - Communicate with agents and other users
|
|
9
|
+
*/
|
|
10
|
+
/**
|
|
11
|
+
* UserBridge manages the connection between dashboard WebSocket users
|
|
12
|
+
* and the relay daemon.
|
|
13
|
+
*/
|
|
14
|
+
export class UserBridge {
|
|
15
|
+
socketPath;
|
|
16
|
+
createRelayClient;
|
|
17
|
+
loadPersistedChannels;
|
|
18
|
+
lookupUserInfo;
|
|
19
|
+
users = new Map();
|
|
20
|
+
constructor(options) {
|
|
21
|
+
this.socketPath = options.socketPath;
|
|
22
|
+
this.createRelayClient = options.createRelayClient;
|
|
23
|
+
this.loadPersistedChannels = options.loadPersistedChannels;
|
|
24
|
+
this.lookupUserInfo = options.lookupUserInfo;
|
|
25
|
+
}
|
|
26
|
+
/**
|
|
27
|
+
* Get the relay client for a user if they are registered.
|
|
28
|
+
* This allows external code to reuse the userBridge's relay client
|
|
29
|
+
* instead of creating a duplicate connection.
|
|
30
|
+
*/
|
|
31
|
+
getRelayClient(username) {
|
|
32
|
+
return this.users.get(username)?.relayClient;
|
|
33
|
+
}
|
|
34
|
+
/**
|
|
35
|
+
* Register a user with the relay daemon.
|
|
36
|
+
* Creates a relay client connection for the user.
|
|
37
|
+
*/
|
|
38
|
+
async registerUser(username, webSocket, options) {
|
|
39
|
+
// If user already registered, unregister first
|
|
40
|
+
if (this.users.has(username)) {
|
|
41
|
+
this.unregisterUser(username);
|
|
42
|
+
}
|
|
43
|
+
// Create relay client for this user
|
|
44
|
+
const relayClient = await this.createRelayClient({
|
|
45
|
+
socketPath: this.socketPath,
|
|
46
|
+
agentName: username,
|
|
47
|
+
entityType: 'user',
|
|
48
|
+
displayName: options?.displayName,
|
|
49
|
+
avatarUrl: options?.avatarUrl,
|
|
50
|
+
});
|
|
51
|
+
// Connect to daemon
|
|
52
|
+
await relayClient.connect();
|
|
53
|
+
// Set up message handler to forward direct messages to WebSocket
|
|
54
|
+
relayClient.onMessage = (from, payload, _messageId, _meta, _originalTo) => {
|
|
55
|
+
const body = typeof payload === 'object' && payload !== null && 'body' in payload
|
|
56
|
+
? payload.body
|
|
57
|
+
: String(payload);
|
|
58
|
+
this.handleIncomingDirectMessage(username, from, body, payload);
|
|
59
|
+
};
|
|
60
|
+
// Set up channel message handler to forward channel messages to WebSocket
|
|
61
|
+
relayClient.onChannelMessage = (from, channel, body, envelope) => {
|
|
62
|
+
this.handleIncomingChannelMessage(username, from, channel, body, envelope);
|
|
63
|
+
};
|
|
64
|
+
// Create session
|
|
65
|
+
const session = {
|
|
66
|
+
username,
|
|
67
|
+
relayClient,
|
|
68
|
+
webSocket,
|
|
69
|
+
channels: new Set(),
|
|
70
|
+
avatarUrl: options?.avatarUrl,
|
|
71
|
+
};
|
|
72
|
+
this.users.set(username, session);
|
|
73
|
+
console.log(`[user-bridge] User registered: ${username} (total: ${this.users.size})`);
|
|
74
|
+
// Auto-join user to #general channel
|
|
75
|
+
// Note: The daemon auto-joins on connect, but we need to track locally too
|
|
76
|
+
session.channels.add('#general');
|
|
77
|
+
if (this.loadPersistedChannels) {
|
|
78
|
+
try {
|
|
79
|
+
const persistedChannels = await this.loadPersistedChannels(username);
|
|
80
|
+
for (const channel of persistedChannels) {
|
|
81
|
+
if (channel === '#general')
|
|
82
|
+
continue;
|
|
83
|
+
if (session.channels.has(channel))
|
|
84
|
+
continue;
|
|
85
|
+
session.relayClient.joinChannel(channel, username);
|
|
86
|
+
session.channels.add(channel);
|
|
87
|
+
}
|
|
88
|
+
}
|
|
89
|
+
catch (err) {
|
|
90
|
+
console.error(`[user-bridge] Failed to restore persisted channels for ${username}:`, err);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
// Set up WebSocket close handler
|
|
94
|
+
webSocket.on('close', () => {
|
|
95
|
+
this.unregisterUser(username);
|
|
96
|
+
});
|
|
97
|
+
console.log(`[user-bridge] User ${username} registered with relay daemon`);
|
|
98
|
+
}
|
|
99
|
+
/**
|
|
100
|
+
* Unregister a user and disconnect their relay client.
|
|
101
|
+
*/
|
|
102
|
+
unregisterUser(username) {
|
|
103
|
+
const session = this.users.get(username);
|
|
104
|
+
if (!session)
|
|
105
|
+
return;
|
|
106
|
+
session.relayClient.disconnect();
|
|
107
|
+
this.users.delete(username);
|
|
108
|
+
console.log(`[user-bridge] User ${username} unregistered from relay daemon`);
|
|
109
|
+
}
|
|
110
|
+
/**
|
|
111
|
+
* Check if a user is registered.
|
|
112
|
+
*/
|
|
113
|
+
isUserRegistered(username) {
|
|
114
|
+
return this.users.has(username);
|
|
115
|
+
}
|
|
116
|
+
/**
|
|
117
|
+
* Update the WebSocket for an existing user session.
|
|
118
|
+
* This is needed when a user reconnects or opens a new tab,
|
|
119
|
+
* so messages are forwarded to the active WebSocket.
|
|
120
|
+
*/
|
|
121
|
+
updateWebSocket(username, newWebSocket) {
|
|
122
|
+
const session = this.users.get(username);
|
|
123
|
+
if (!session) {
|
|
124
|
+
console.log(`[user-bridge] Cannot update WebSocket - user ${username} not registered`);
|
|
125
|
+
return false;
|
|
126
|
+
}
|
|
127
|
+
// Remove the close handler from old WebSocket to prevent auto-unregister
|
|
128
|
+
session.webSocket.removeAllListeners('close');
|
|
129
|
+
// Update to new WebSocket
|
|
130
|
+
session.webSocket = newWebSocket;
|
|
131
|
+
// Set up close handler on new WebSocket
|
|
132
|
+
newWebSocket.on('close', () => {
|
|
133
|
+
// Note: The server manages multi-tab connections and will call
|
|
134
|
+
// unregisterUser when all connections are closed
|
|
135
|
+
});
|
|
136
|
+
console.log(`[user-bridge] Updated WebSocket for user ${username}`);
|
|
137
|
+
return true;
|
|
138
|
+
}
|
|
139
|
+
/**
|
|
140
|
+
* Get list of all registered users.
|
|
141
|
+
*/
|
|
142
|
+
getRegisteredUsers() {
|
|
143
|
+
return Array.from(this.users.keys());
|
|
144
|
+
}
|
|
145
|
+
/**
|
|
146
|
+
* Join a channel.
|
|
147
|
+
*/
|
|
148
|
+
async joinChannel(username, channel) {
|
|
149
|
+
const session = this.users.get(username);
|
|
150
|
+
if (!session) {
|
|
151
|
+
console.warn(`[user-bridge] Cannot join channel - user ${username} not registered`);
|
|
152
|
+
return false;
|
|
153
|
+
}
|
|
154
|
+
// Send CHANNEL_JOIN via relay client
|
|
155
|
+
const success = session.relayClient.joinChannel(channel, username);
|
|
156
|
+
if (success) {
|
|
157
|
+
// Track membership
|
|
158
|
+
session.channels.add(channel);
|
|
159
|
+
}
|
|
160
|
+
return success;
|
|
161
|
+
}
|
|
162
|
+
/**
|
|
163
|
+
* Leave a channel.
|
|
164
|
+
*/
|
|
165
|
+
async leaveChannel(username, channel) {
|
|
166
|
+
const session = this.users.get(username);
|
|
167
|
+
if (!session) {
|
|
168
|
+
console.warn(`[user-bridge] Cannot leave channel - user ${username} not registered`);
|
|
169
|
+
return false;
|
|
170
|
+
}
|
|
171
|
+
// Send CHANNEL_LEAVE via relay client
|
|
172
|
+
const success = session.relayClient.leaveChannel(channel);
|
|
173
|
+
if (success) {
|
|
174
|
+
// Update membership
|
|
175
|
+
session.channels.delete(channel);
|
|
176
|
+
console.log(`[user-bridge] User ${username} left channel ${channel}`);
|
|
177
|
+
}
|
|
178
|
+
return success;
|
|
179
|
+
}
|
|
180
|
+
/**
|
|
181
|
+
* Get channels a user has joined.
|
|
182
|
+
*/
|
|
183
|
+
getUserChannels(username) {
|
|
184
|
+
const session = this.users.get(username);
|
|
185
|
+
return session ? Array.from(session.channels) : [];
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Send a message to a channel.
|
|
189
|
+
*/
|
|
190
|
+
async sendChannelMessage(username, channel, body, options) {
|
|
191
|
+
const session = this.users.get(username);
|
|
192
|
+
if (!session) {
|
|
193
|
+
console.warn(`[user-bridge] Cannot send - user ${username} not registered`);
|
|
194
|
+
return false;
|
|
195
|
+
}
|
|
196
|
+
return session.relayClient.sendChannelMessage(channel, body, {
|
|
197
|
+
thread: options?.thread,
|
|
198
|
+
data: options?.data,
|
|
199
|
+
attachments: options?.attachments,
|
|
200
|
+
});
|
|
201
|
+
}
|
|
202
|
+
/**
|
|
203
|
+
* Send a direct message to another user or agent.
|
|
204
|
+
*/
|
|
205
|
+
async sendDirectMessage(fromUsername, toName, body, options) {
|
|
206
|
+
const session = this.users.get(fromUsername);
|
|
207
|
+
if (!session) {
|
|
208
|
+
console.warn(`[user-bridge] Cannot send DM - user ${fromUsername} not registered`);
|
|
209
|
+
return false;
|
|
210
|
+
}
|
|
211
|
+
return session.relayClient.sendMessage(toName, body, 'message', options?.data, options?.thread);
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Handle incoming direct message from relay daemon.
|
|
215
|
+
*/
|
|
216
|
+
handleIncomingDirectMessage(username, from, body, payload) {
|
|
217
|
+
// Skip channel messages - they are handled by handleIncomingChannelMessage
|
|
218
|
+
// The relay client calls both onMessage and onChannelMessage for channel messages,
|
|
219
|
+
// with _isChannelMessage flag set in the data for onMessage calls
|
|
220
|
+
const payloadObj = payload;
|
|
221
|
+
if (payloadObj?.data?._isChannelMessage) {
|
|
222
|
+
return; // Skip - will be handled by onChannelMessage callback
|
|
223
|
+
}
|
|
224
|
+
const session = this.users.get(username);
|
|
225
|
+
if (!session)
|
|
226
|
+
return;
|
|
227
|
+
const ws = session.webSocket;
|
|
228
|
+
if (ws.readyState !== 1)
|
|
229
|
+
return; // Not OPEN
|
|
230
|
+
// Look up sender's avatar if lookup function is available
|
|
231
|
+
const senderInfo = this.lookupUserInfo?.(from);
|
|
232
|
+
const fromAvatarUrl = senderInfo?.avatarUrl;
|
|
233
|
+
// Determine entity type: user if they have info, agent otherwise
|
|
234
|
+
const fromEntityType = senderInfo ? 'user' : 'agent';
|
|
235
|
+
// Direct message (DELIVER)
|
|
236
|
+
ws.send(JSON.stringify({
|
|
237
|
+
type: 'direct_message',
|
|
238
|
+
from,
|
|
239
|
+
fromAvatarUrl,
|
|
240
|
+
fromEntityType,
|
|
241
|
+
body: payloadObj?.body || body,
|
|
242
|
+
timestamp: new Date().toISOString(),
|
|
243
|
+
}));
|
|
244
|
+
}
|
|
245
|
+
/**
|
|
246
|
+
* Handle incoming channel message from relay daemon.
|
|
247
|
+
*/
|
|
248
|
+
handleIncomingChannelMessage(username, from, channel, body, envelope) {
|
|
249
|
+
const session = this.users.get(username);
|
|
250
|
+
if (!session)
|
|
251
|
+
return;
|
|
252
|
+
const ws = session.webSocket;
|
|
253
|
+
if (ws.readyState !== 1)
|
|
254
|
+
return; // Not OPEN
|
|
255
|
+
// Look up sender's avatar if lookup function is available
|
|
256
|
+
const senderInfo = this.lookupUserInfo?.(from);
|
|
257
|
+
const fromAvatarUrl = senderInfo?.avatarUrl;
|
|
258
|
+
// Determine entity type: user if they have info, agent otherwise
|
|
259
|
+
const fromEntityType = senderInfo ? 'user' : 'agent';
|
|
260
|
+
// Channel message
|
|
261
|
+
const env = envelope;
|
|
262
|
+
ws.send(JSON.stringify({
|
|
263
|
+
type: 'channel_message',
|
|
264
|
+
channel,
|
|
265
|
+
from,
|
|
266
|
+
fromAvatarUrl,
|
|
267
|
+
fromEntityType,
|
|
268
|
+
body,
|
|
269
|
+
thread: env?.payload?.thread,
|
|
270
|
+
mentions: env?.payload?.mentions,
|
|
271
|
+
timestamp: new Date().toISOString(),
|
|
272
|
+
}));
|
|
273
|
+
}
|
|
274
|
+
/**
|
|
275
|
+
* Admin: Add a member to a channel (does not require member to be connected).
|
|
276
|
+
* Used to sync channel memberships from database.
|
|
277
|
+
* Uses the first available user session or creates a temporary one.
|
|
278
|
+
*/
|
|
279
|
+
async adminJoinChannel(channel, member) {
|
|
280
|
+
// Try to use an existing session
|
|
281
|
+
const sessions = Array.from(this.users.values());
|
|
282
|
+
if (sessions.length > 0) {
|
|
283
|
+
const session = sessions[0];
|
|
284
|
+
if (session.relayClient.adminJoinChannel) {
|
|
285
|
+
console.log(`[user-bridge] Admin join: ${member} -> ${channel} (via ${session.username})`);
|
|
286
|
+
return session.relayClient.adminJoinChannel(channel, member);
|
|
287
|
+
}
|
|
288
|
+
}
|
|
289
|
+
// No sessions available - create a temporary system client
|
|
290
|
+
try {
|
|
291
|
+
console.log(`[user-bridge] Admin join: ${member} -> ${channel} (creating temp client)`);
|
|
292
|
+
const tempClient = await this.createRelayClient({
|
|
293
|
+
socketPath: this.socketPath,
|
|
294
|
+
agentName: '__system__',
|
|
295
|
+
entityType: 'user',
|
|
296
|
+
});
|
|
297
|
+
await tempClient.connect();
|
|
298
|
+
// Give daemon time to complete handshake before sending admin commands.
|
|
299
|
+
// 100ms is sufficient for local Unix socket handshake (typically <10ms),
|
|
300
|
+
// but provides margin for the daemon to process the HELLO message and
|
|
301
|
+
// set up internal state. This is a temporary client created just for
|
|
302
|
+
// the admin operation, not a long-lived session.
|
|
303
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
304
|
+
if (tempClient.adminJoinChannel) {
|
|
305
|
+
const result = tempClient.adminJoinChannel(channel, member);
|
|
306
|
+
// Disconnect after a short delay to allow message to be sent
|
|
307
|
+
setTimeout(() => tempClient.disconnect(), 200);
|
|
308
|
+
return result;
|
|
309
|
+
}
|
|
310
|
+
tempClient.disconnect();
|
|
311
|
+
return false;
|
|
312
|
+
}
|
|
313
|
+
catch (err) {
|
|
314
|
+
console.error('[user-bridge] Failed to create temp client for admin join:', err);
|
|
315
|
+
return false;
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
/**
|
|
319
|
+
* Admin: Remove a member from a channel (does not require member to be connected).
|
|
320
|
+
* Used to remove channel members from dashboard.
|
|
321
|
+
* Uses the first available user session or creates a temporary one.
|
|
322
|
+
*/
|
|
323
|
+
async adminRemoveMember(channel, member) {
|
|
324
|
+
// Try to use an existing session
|
|
325
|
+
const sessions = Array.from(this.users.values());
|
|
326
|
+
if (sessions.length > 0) {
|
|
327
|
+
const session = sessions[0];
|
|
328
|
+
if (session.relayClient.adminRemoveMember) {
|
|
329
|
+
console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (via ${session.username})`);
|
|
330
|
+
return session.relayClient.adminRemoveMember(channel, member);
|
|
331
|
+
}
|
|
332
|
+
}
|
|
333
|
+
// No sessions available - create a temporary system client
|
|
334
|
+
try {
|
|
335
|
+
console.log(`[user-bridge] Admin remove: ${member} <- ${channel} (creating temp client)`);
|
|
336
|
+
const tempClient = await this.createRelayClient({
|
|
337
|
+
socketPath: this.socketPath,
|
|
338
|
+
agentName: '__system__',
|
|
339
|
+
entityType: 'user',
|
|
340
|
+
});
|
|
341
|
+
await tempClient.connect();
|
|
342
|
+
// Give daemon time to complete handshake before sending admin commands.
|
|
343
|
+
// 100ms is sufficient for local Unix socket handshake (typically <10ms),
|
|
344
|
+
// but provides margin for the daemon to process the HELLO message and
|
|
345
|
+
// set up internal state. This is a temporary client created just for
|
|
346
|
+
// the admin operation, not a long-lived session.
|
|
347
|
+
await new Promise(resolve => setTimeout(resolve, 100));
|
|
348
|
+
if (tempClient.adminRemoveMember) {
|
|
349
|
+
const result = tempClient.adminRemoveMember(channel, member);
|
|
350
|
+
// Disconnect after a short delay to allow message to be sent
|
|
351
|
+
setTimeout(() => tempClient.disconnect(), 200);
|
|
352
|
+
return result;
|
|
353
|
+
}
|
|
354
|
+
tempClient.disconnect();
|
|
355
|
+
return false;
|
|
356
|
+
}
|
|
357
|
+
catch (err) {
|
|
358
|
+
console.error('[user-bridge] Failed to create temp client for admin remove:', err);
|
|
359
|
+
return false;
|
|
360
|
+
}
|
|
361
|
+
}
|
|
362
|
+
/**
|
|
363
|
+
* Dispose of all user sessions.
|
|
364
|
+
*/
|
|
365
|
+
dispose() {
|
|
366
|
+
for (const [username] of this.users) {
|
|
367
|
+
this.unregisterUser(username);
|
|
368
|
+
}
|
|
369
|
+
console.log('[user-bridge] Disposed all user sessions');
|
|
370
|
+
}
|
|
371
|
+
}
|
|
372
|
+
//# sourceMappingURL=user-bridge.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"user-bridge.js","sourceRoot":"","sources":["../src/user-bridge.ts"],"names":[],"mappings":"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"}
|
package/package.json
ADDED
|
@@ -0,0 +1,49 @@
|
|
|
1
|
+
{
|
|
2
|
+
"name": "@agent-relay/dashboard-server",
|
|
3
|
+
"version": "0.1.0",
|
|
4
|
+
"description": "Relay dashboard server - HTTP/WebSocket server for agent coordination",
|
|
5
|
+
"type": "module",
|
|
6
|
+
"main": "dist/index.js",
|
|
7
|
+
"types": "dist/index.d.ts",
|
|
8
|
+
"exports": {
|
|
9
|
+
".": {
|
|
10
|
+
"types": "./dist/index.d.ts",
|
|
11
|
+
"import": "./dist/index.js"
|
|
12
|
+
}
|
|
13
|
+
},
|
|
14
|
+
"files": [
|
|
15
|
+
"dist",
|
|
16
|
+
"README.md"
|
|
17
|
+
],
|
|
18
|
+
"scripts": {
|
|
19
|
+
"build": "tsc",
|
|
20
|
+
"clean": "rm -rf dist",
|
|
21
|
+
"test": "vitest run",
|
|
22
|
+
"test:watch": "vitest"
|
|
23
|
+
},
|
|
24
|
+
"dependencies": {
|
|
25
|
+
"@agent-relay/protocol": "^0.1.2",
|
|
26
|
+
"@agent-relay/config": "^0.1.0",
|
|
27
|
+
"@agent-relay/storage": "^0.1.0",
|
|
28
|
+
"@agent-relay/bridge": "^0.1.0",
|
|
29
|
+
"@agent-relay/utils": "^0.1.0",
|
|
30
|
+
"@agent-relay/resiliency": "^0.1.0",
|
|
31
|
+
"@agent-relay/trajectory": "^0.1.0",
|
|
32
|
+
"@agent-relay/cloud": "^0.1.0",
|
|
33
|
+
"@agent-relay/daemon": "^0.1.0",
|
|
34
|
+
"@agent-relay/user-directory": "^0.1.0",
|
|
35
|
+
"@agent-relay/sdk": "^0.1.0",
|
|
36
|
+
"express": "^4.21.2",
|
|
37
|
+
"ws": "^8.18.3"
|
|
38
|
+
},
|
|
39
|
+
"devDependencies": {
|
|
40
|
+
"@types/express": "^5.0.1",
|
|
41
|
+
"@types/node": "^22.19.3",
|
|
42
|
+
"@types/ws": "^8.18.1",
|
|
43
|
+
"typescript": "^5.9.3",
|
|
44
|
+
"vitest": "^3.2.4"
|
|
45
|
+
},
|
|
46
|
+
"publishConfig": {
|
|
47
|
+
"access": "public"
|
|
48
|
+
}
|
|
49
|
+
}
|