@agentage/cli 0.1.9 → 0.1.11
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/cli.js +98 -6
- package/dist/cli.js.map +1 -1
- package/dist/cli.test.js +3 -3
- package/dist/cli.test.js.map +1 -1
- package/dist/commands/init.d.ts +1 -1
- package/dist/commands/init.d.ts.map +1 -1
- package/dist/commands/init.js +1 -1
- package/dist/commands/init.js.map +1 -1
- package/dist/commands/init.test.js +2 -2
- package/dist/commands/init.test.js.map +1 -1
- package/dist/commands/install.d.ts +8 -0
- package/dist/commands/install.d.ts.map +1 -0
- package/dist/commands/install.js +90 -0
- package/dist/commands/install.js.map +1 -0
- package/dist/commands/install.test.d.ts +2 -0
- package/dist/commands/install.test.d.ts.map +1 -0
- package/dist/commands/install.test.js +172 -0
- package/dist/commands/install.test.js.map +1 -0
- package/dist/commands/list.d.ts.map +1 -1
- package/dist/commands/list.js +124 -21
- package/dist/commands/list.js.map +1 -1
- package/dist/commands/list.test.js +44 -17
- package/dist/commands/list.test.js.map +1 -1
- package/dist/commands/publish.d.ts +10 -0
- package/dist/commands/publish.d.ts.map +1 -0
- package/dist/commands/publish.js +133 -0
- package/dist/commands/publish.js.map +1 -0
- package/dist/commands/publish.test.d.ts +2 -0
- package/dist/commands/publish.test.d.ts.map +1 -0
- package/dist/commands/publish.test.js +260 -0
- package/dist/commands/publish.test.js.map +1 -0
- package/dist/commands/search.d.ts +8 -0
- package/dist/commands/search.d.ts.map +1 -0
- package/dist/commands/search.js +41 -0
- package/dist/commands/search.js.map +1 -0
- package/dist/commands/search.test.d.ts +2 -0
- package/dist/commands/search.test.d.ts.map +1 -0
- package/dist/commands/search.test.js +138 -0
- package/dist/commands/search.test.js.map +1 -0
- package/dist/index.d.ts +1 -1
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +7 -1
- package/dist/index.js.map +1 -1
- package/dist/index.test.js +9 -2
- package/dist/index.test.js.map +1 -1
- package/dist/schemas/agent.schema.d.ts +2 -2
- package/dist/services/registry.service.d.ts +13 -0
- package/dist/services/registry.service.d.ts.map +1 -0
- package/dist/services/registry.service.js +74 -0
- package/dist/services/registry.service.js.map +1 -0
- package/dist/services/registry.service.test.d.ts +2 -0
- package/dist/services/registry.service.test.d.ts.map +1 -0
- package/dist/services/registry.service.test.js +222 -0
- package/dist/services/registry.service.test.js.map +1 -0
- package/dist/types/registry.types.d.ts +65 -0
- package/dist/types/registry.types.d.ts.map +1 -0
- package/dist/types/registry.types.js +2 -0
- package/dist/types/registry.types.js.map +1 -0
- package/dist/utils/agent-parser.d.ts +26 -0
- package/dist/utils/agent-parser.d.ts.map +1 -0
- package/dist/utils/agent-parser.js +45 -0
- package/dist/utils/agent-parser.js.map +1 -0
- package/dist/utils/agent-parser.test.d.ts +2 -0
- package/dist/utils/agent-parser.test.d.ts.map +1 -0
- package/dist/utils/agent-parser.test.js +102 -0
- package/dist/utils/agent-parser.test.js.map +1 -0
- package/package.json +1 -1
|
@@ -0,0 +1,74 @@
|
|
|
1
|
+
import { getAuthToken, getRegistryUrl } from '../utils/config.js';
|
|
2
|
+
export class RegistryApiError extends Error {
|
|
3
|
+
code;
|
|
4
|
+
statusCode;
|
|
5
|
+
details;
|
|
6
|
+
constructor(message, code, statusCode, details) {
|
|
7
|
+
super(message);
|
|
8
|
+
this.code = code;
|
|
9
|
+
this.statusCode = statusCode;
|
|
10
|
+
this.details = details;
|
|
11
|
+
this.name = 'RegistryApiError';
|
|
12
|
+
}
|
|
13
|
+
}
|
|
14
|
+
const registryFetch = async (path, options = {}) => {
|
|
15
|
+
const registryUrl = await getRegistryUrl();
|
|
16
|
+
const token = await getAuthToken();
|
|
17
|
+
const headers = {
|
|
18
|
+
'Content-Type': 'application/json',
|
|
19
|
+
...options.headers,
|
|
20
|
+
};
|
|
21
|
+
if (token) {
|
|
22
|
+
headers['Authorization'] = `Bearer ${token}`;
|
|
23
|
+
}
|
|
24
|
+
const response = await fetch(`${registryUrl}${path}`, {
|
|
25
|
+
...options,
|
|
26
|
+
headers,
|
|
27
|
+
});
|
|
28
|
+
const data = await response.json();
|
|
29
|
+
if (!response.ok) {
|
|
30
|
+
const error = data;
|
|
31
|
+
throw new RegistryApiError(error.message || 'Request failed', error.error || 'request_failed', response.status, error.details);
|
|
32
|
+
}
|
|
33
|
+
return data;
|
|
34
|
+
};
|
|
35
|
+
export const publishAgent = async (data) => {
|
|
36
|
+
const response = await registryFetch('/api/agents', {
|
|
37
|
+
method: 'POST',
|
|
38
|
+
body: JSON.stringify(data),
|
|
39
|
+
});
|
|
40
|
+
return response.agent;
|
|
41
|
+
};
|
|
42
|
+
export const getAgent = async (owner, name) => {
|
|
43
|
+
const response = await registryFetch(`/api/agents/${encodeURIComponent(owner)}/${encodeURIComponent(name)}`);
|
|
44
|
+
return response.data;
|
|
45
|
+
};
|
|
46
|
+
export const getAgentVersion = async (owner, name, version) => {
|
|
47
|
+
const response = await registryFetch(`/api/agents/${encodeURIComponent(owner)}/${encodeURIComponent(name)}/versions/${encodeURIComponent(version)}`);
|
|
48
|
+
return response.data;
|
|
49
|
+
};
|
|
50
|
+
export const searchAgents = async (query, page = 1, limit = 10) => {
|
|
51
|
+
const params = new URLSearchParams({
|
|
52
|
+
q: query,
|
|
53
|
+
page: String(page),
|
|
54
|
+
limit: String(limit),
|
|
55
|
+
});
|
|
56
|
+
const response = await registryFetch(`/api/agents/search?${params.toString()}`);
|
|
57
|
+
return response.data;
|
|
58
|
+
};
|
|
59
|
+
export const listAgents = async (filters = {}) => {
|
|
60
|
+
const params = new URLSearchParams();
|
|
61
|
+
if (filters.page)
|
|
62
|
+
params.set('page', String(filters.page));
|
|
63
|
+
if (filters.limit)
|
|
64
|
+
params.set('limit', String(filters.limit));
|
|
65
|
+
if (filters.sort)
|
|
66
|
+
params.set('sort', filters.sort);
|
|
67
|
+
if (filters.owner)
|
|
68
|
+
params.set('owner', filters.owner);
|
|
69
|
+
if (filters.tag)
|
|
70
|
+
params.set('tag', filters.tag);
|
|
71
|
+
const response = await registryFetch(`/api/agents?${params.toString()}`);
|
|
72
|
+
return response.data;
|
|
73
|
+
};
|
|
74
|
+
//# sourceMappingURL=registry.service.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.service.js","sourceRoot":"","sources":["../../src/services/registry.service.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.service.test.d.ts","sourceRoot":"","sources":["../../src/services/registry.service.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,222 @@
|
|
|
1
|
+
import { RegistryApiError } from './registry.service.js';
|
|
2
|
+
const mockFetch = jest.fn();
|
|
3
|
+
global.fetch = mockFetch;
|
|
4
|
+
jest.mock('../utils/config.js', () => ({
|
|
5
|
+
getAuthToken: jest.fn().mockResolvedValue('test-token'),
|
|
6
|
+
getRegistryUrl: jest.fn().mockResolvedValue('https://dev.agentage.io'),
|
|
7
|
+
}));
|
|
8
|
+
describe('Registry Service', () => {
|
|
9
|
+
beforeEach(() => {
|
|
10
|
+
mockFetch.mockReset();
|
|
11
|
+
});
|
|
12
|
+
describe('RegistryApiError', () => {
|
|
13
|
+
test('creates error with all properties', () => {
|
|
14
|
+
const error = new RegistryApiError('Test error', 'test_error', 400, {
|
|
15
|
+
field: 'value',
|
|
16
|
+
});
|
|
17
|
+
expect(error.message).toBe('Test error');
|
|
18
|
+
expect(error.code).toBe('test_error');
|
|
19
|
+
expect(error.statusCode).toBe(400);
|
|
20
|
+
expect(error.details).toEqual({ field: 'value' });
|
|
21
|
+
expect(error.name).toBe('RegistryApiError');
|
|
22
|
+
});
|
|
23
|
+
});
|
|
24
|
+
describe('publishAgent', () => {
|
|
25
|
+
test('publishes agent successfully', async () => {
|
|
26
|
+
mockFetch.mockResolvedValueOnce({
|
|
27
|
+
ok: true,
|
|
28
|
+
json: () => Promise.resolve({
|
|
29
|
+
success: true,
|
|
30
|
+
agent: {
|
|
31
|
+
name: 'test-agent',
|
|
32
|
+
owner: 'testuser',
|
|
33
|
+
version: '2025-11-30',
|
|
34
|
+
visibility: 'public',
|
|
35
|
+
publishedAt: '2025-11-30T12:00:00Z',
|
|
36
|
+
},
|
|
37
|
+
}),
|
|
38
|
+
});
|
|
39
|
+
const { publishAgent } = await import('./registry.service.js');
|
|
40
|
+
const result = await publishAgent({
|
|
41
|
+
name: 'test-agent',
|
|
42
|
+
visibility: 'public',
|
|
43
|
+
version: '2025-11-30',
|
|
44
|
+
content: '---\nname: test-agent\n---\nContent',
|
|
45
|
+
});
|
|
46
|
+
expect(result.name).toBe('test-agent');
|
|
47
|
+
expect(result.owner).toBe('testuser');
|
|
48
|
+
expect(mockFetch).toHaveBeenCalledWith('https://dev.agentage.io/api/agents', expect.objectContaining({
|
|
49
|
+
method: 'POST',
|
|
50
|
+
headers: expect.objectContaining({
|
|
51
|
+
Authorization: 'Bearer test-token',
|
|
52
|
+
'Content-Type': 'application/json',
|
|
53
|
+
}),
|
|
54
|
+
}));
|
|
55
|
+
});
|
|
56
|
+
test('throws error on failure', async () => {
|
|
57
|
+
mockFetch.mockResolvedValueOnce({
|
|
58
|
+
ok: false,
|
|
59
|
+
status: 400,
|
|
60
|
+
json: () => Promise.resolve({
|
|
61
|
+
error: 'validation_error',
|
|
62
|
+
message: 'Invalid agent name',
|
|
63
|
+
}),
|
|
64
|
+
});
|
|
65
|
+
const { publishAgent } = await import('./registry.service.js');
|
|
66
|
+
await expect(publishAgent({
|
|
67
|
+
name: 'invalid',
|
|
68
|
+
visibility: 'public',
|
|
69
|
+
version: '1.0.0',
|
|
70
|
+
content: 'test',
|
|
71
|
+
})).rejects.toThrow(RegistryApiError);
|
|
72
|
+
});
|
|
73
|
+
});
|
|
74
|
+
describe('getAgent', () => {
|
|
75
|
+
test('fetches agent details', async () => {
|
|
76
|
+
mockFetch.mockResolvedValueOnce({
|
|
77
|
+
ok: true,
|
|
78
|
+
json: () => Promise.resolve({
|
|
79
|
+
success: true,
|
|
80
|
+
data: {
|
|
81
|
+
name: 'test-agent',
|
|
82
|
+
owner: 'testuser',
|
|
83
|
+
latestVersion: '2025-11-30',
|
|
84
|
+
latestContent: 'content',
|
|
85
|
+
},
|
|
86
|
+
}),
|
|
87
|
+
});
|
|
88
|
+
const { getAgent } = await import('./registry.service.js');
|
|
89
|
+
const result = await getAgent('testuser', 'test-agent');
|
|
90
|
+
expect(result.name).toBe('test-agent');
|
|
91
|
+
expect(mockFetch).toHaveBeenCalledWith('https://dev.agentage.io/api/agents/testuser/test-agent', expect.any(Object));
|
|
92
|
+
});
|
|
93
|
+
});
|
|
94
|
+
describe('searchAgents', () => {
|
|
95
|
+
test('searches with query', async () => {
|
|
96
|
+
mockFetch.mockResolvedValueOnce({
|
|
97
|
+
ok: true,
|
|
98
|
+
json: () => Promise.resolve({
|
|
99
|
+
success: true,
|
|
100
|
+
data: {
|
|
101
|
+
agents: [{ name: 'test', owner: 'user' }],
|
|
102
|
+
total: 1,
|
|
103
|
+
page: 1,
|
|
104
|
+
limit: 10,
|
|
105
|
+
hasMore: false,
|
|
106
|
+
},
|
|
107
|
+
}),
|
|
108
|
+
});
|
|
109
|
+
const { searchAgents } = await import('./registry.service.js');
|
|
110
|
+
const result = await searchAgents('test', 1, 10);
|
|
111
|
+
expect(result.agents).toHaveLength(1);
|
|
112
|
+
expect(mockFetch).toHaveBeenCalledWith('https://dev.agentage.io/api/agents/search?q=test&page=1&limit=10', expect.any(Object));
|
|
113
|
+
});
|
|
114
|
+
});
|
|
115
|
+
describe('getAgentVersion', () => {
|
|
116
|
+
test('fetches specific version', async () => {
|
|
117
|
+
mockFetch.mockResolvedValueOnce({
|
|
118
|
+
ok: true,
|
|
119
|
+
json: () => Promise.resolve({
|
|
120
|
+
success: true,
|
|
121
|
+
data: {
|
|
122
|
+
version: '2025-11-01',
|
|
123
|
+
content: 'version content',
|
|
124
|
+
downloads: 10,
|
|
125
|
+
publishedAt: '2025-11-01T00:00:00Z',
|
|
126
|
+
isLatest: false,
|
|
127
|
+
},
|
|
128
|
+
}),
|
|
129
|
+
});
|
|
130
|
+
const { getAgentVersion } = await import('./registry.service.js');
|
|
131
|
+
const result = await getAgentVersion('testuser', 'test-agent', '2025-11-01');
|
|
132
|
+
expect(result.version).toBe('2025-11-01');
|
|
133
|
+
expect(mockFetch).toHaveBeenCalledWith('https://dev.agentage.io/api/agents/testuser/test-agent/versions/2025-11-01', expect.any(Object));
|
|
134
|
+
});
|
|
135
|
+
});
|
|
136
|
+
describe('listAgents', () => {
|
|
137
|
+
test('lists agents with filters', async () => {
|
|
138
|
+
mockFetch.mockResolvedValueOnce({
|
|
139
|
+
ok: true,
|
|
140
|
+
json: () => Promise.resolve({
|
|
141
|
+
success: true,
|
|
142
|
+
data: {
|
|
143
|
+
agents: [{ name: 'test', owner: 'user' }],
|
|
144
|
+
total: 1,
|
|
145
|
+
page: 1,
|
|
146
|
+
limit: 10,
|
|
147
|
+
hasMore: false,
|
|
148
|
+
},
|
|
149
|
+
}),
|
|
150
|
+
});
|
|
151
|
+
const { listAgents } = await import('./registry.service.js');
|
|
152
|
+
const result = await listAgents({
|
|
153
|
+
page: 1,
|
|
154
|
+
limit: 10,
|
|
155
|
+
sort: 'downloads',
|
|
156
|
+
owner: 'testuser',
|
|
157
|
+
tag: 'ai',
|
|
158
|
+
});
|
|
159
|
+
expect(result.agents).toHaveLength(1);
|
|
160
|
+
expect(mockFetch).toHaveBeenCalledWith(expect.stringContaining('page=1'), expect.any(Object));
|
|
161
|
+
});
|
|
162
|
+
test('lists agents without filters', async () => {
|
|
163
|
+
mockFetch.mockResolvedValueOnce({
|
|
164
|
+
ok: true,
|
|
165
|
+
json: () => Promise.resolve({
|
|
166
|
+
success: true,
|
|
167
|
+
data: {
|
|
168
|
+
agents: [],
|
|
169
|
+
total: 0,
|
|
170
|
+
page: 1,
|
|
171
|
+
limit: 10,
|
|
172
|
+
hasMore: false,
|
|
173
|
+
},
|
|
174
|
+
}),
|
|
175
|
+
});
|
|
176
|
+
const { listAgents } = await import('./registry.service.js');
|
|
177
|
+
const result = await listAgents();
|
|
178
|
+
expect(result.agents).toHaveLength(0);
|
|
179
|
+
});
|
|
180
|
+
});
|
|
181
|
+
describe('registryFetch error handling', () => {
|
|
182
|
+
test('handles missing error message', async () => {
|
|
183
|
+
mockFetch.mockResolvedValueOnce({
|
|
184
|
+
ok: false,
|
|
185
|
+
status: 500,
|
|
186
|
+
json: () => Promise.resolve({}),
|
|
187
|
+
});
|
|
188
|
+
const { publishAgent } = await import('./registry.service.js');
|
|
189
|
+
await expect(publishAgent({
|
|
190
|
+
name: 'test',
|
|
191
|
+
visibility: 'public',
|
|
192
|
+
version: '1.0.0',
|
|
193
|
+
content: 'test',
|
|
194
|
+
})).rejects.toThrow('Request failed');
|
|
195
|
+
});
|
|
196
|
+
test('works without auth token', async () => {
|
|
197
|
+
const { getAuthToken } = require('../utils/config.js');
|
|
198
|
+
getAuthToken.mockResolvedValueOnce(null);
|
|
199
|
+
mockFetch.mockResolvedValueOnce({
|
|
200
|
+
ok: true,
|
|
201
|
+
json: () => Promise.resolve({
|
|
202
|
+
success: true,
|
|
203
|
+
data: {
|
|
204
|
+
agents: [],
|
|
205
|
+
total: 0,
|
|
206
|
+
page: 1,
|
|
207
|
+
limit: 10,
|
|
208
|
+
hasMore: false,
|
|
209
|
+
},
|
|
210
|
+
}),
|
|
211
|
+
});
|
|
212
|
+
const { searchAgents } = await import('./registry.service.js');
|
|
213
|
+
await searchAgents('test');
|
|
214
|
+
expect(mockFetch).toHaveBeenCalledWith(expect.any(String), expect.objectContaining({
|
|
215
|
+
headers: expect.not.objectContaining({
|
|
216
|
+
Authorization: expect.any(String),
|
|
217
|
+
}),
|
|
218
|
+
}));
|
|
219
|
+
});
|
|
220
|
+
});
|
|
221
|
+
});
|
|
222
|
+
//# sourceMappingURL=registry.service.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.service.test.js","sourceRoot":"","sources":["../../src/services/registry.service.test.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,65 @@
|
|
|
1
|
+
export interface PublishRequest {
|
|
2
|
+
name: string;
|
|
3
|
+
description?: string;
|
|
4
|
+
visibility: 'public' | 'private';
|
|
5
|
+
version: string;
|
|
6
|
+
content: string;
|
|
7
|
+
contentType?: 'markdown' | 'plain';
|
|
8
|
+
tags?: string[];
|
|
9
|
+
changelog?: string;
|
|
10
|
+
}
|
|
11
|
+
export interface PublishResponse {
|
|
12
|
+
name: string;
|
|
13
|
+
owner: string;
|
|
14
|
+
version: string;
|
|
15
|
+
visibility: 'public' | 'private';
|
|
16
|
+
publishedAt: string;
|
|
17
|
+
}
|
|
18
|
+
export interface AgentSummary {
|
|
19
|
+
name: string;
|
|
20
|
+
owner: string;
|
|
21
|
+
description?: string;
|
|
22
|
+
visibility: 'public' | 'private';
|
|
23
|
+
tags: string[];
|
|
24
|
+
latestVersion: string;
|
|
25
|
+
totalDownloads: number;
|
|
26
|
+
}
|
|
27
|
+
export interface AgentDetail {
|
|
28
|
+
name: string;
|
|
29
|
+
owner: string;
|
|
30
|
+
description?: string;
|
|
31
|
+
visibility: 'public' | 'private';
|
|
32
|
+
tags: string[];
|
|
33
|
+
latestVersion: string;
|
|
34
|
+
latestContent: string;
|
|
35
|
+
contentType: 'markdown' | 'plain';
|
|
36
|
+
totalDownloads: number;
|
|
37
|
+
versions: AgentVersionInfo[];
|
|
38
|
+
}
|
|
39
|
+
export interface AgentVersionInfo {
|
|
40
|
+
version: string;
|
|
41
|
+
content?: string;
|
|
42
|
+
downloads: number;
|
|
43
|
+
publishedAt: string;
|
|
44
|
+
isLatest: boolean;
|
|
45
|
+
}
|
|
46
|
+
export interface SearchResult {
|
|
47
|
+
agents: AgentSummary[];
|
|
48
|
+
total: number;
|
|
49
|
+
page: number;
|
|
50
|
+
limit: number;
|
|
51
|
+
hasMore: boolean;
|
|
52
|
+
}
|
|
53
|
+
export interface ListFilters {
|
|
54
|
+
page?: number;
|
|
55
|
+
limit?: number;
|
|
56
|
+
sort?: 'downloads' | 'newest' | 'name';
|
|
57
|
+
owner?: string;
|
|
58
|
+
tag?: string;
|
|
59
|
+
}
|
|
60
|
+
export interface RegistryError {
|
|
61
|
+
error: string;
|
|
62
|
+
message: string;
|
|
63
|
+
details?: Record<string, string>;
|
|
64
|
+
}
|
|
65
|
+
//# sourceMappingURL=registry.types.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.types.d.ts","sourceRoot":"","sources":["../../src/types/registry.types.ts"],"names":[],"mappings":"AAIA,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,QAAQ,GAAG,SAAS,CAAC;IACjC,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,EAAE,MAAM,CAAC;IAChB,WAAW,CAAC,EAAE,UAAU,GAAG,OAAO,CAAC;IACnC,IAAI,CAAC,EAAE,MAAM,EAAE,CAAC;IAChB,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,eAAe;IAC9B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,UAAU,EAAE,QAAQ,GAAG,SAAS,CAAC;IACjC,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,MAAM,WAAW,YAAY;IAC3B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,QAAQ,GAAG,SAAS,CAAC;IACjC,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,aAAa,EAAE,MAAM,CAAC;IACtB,cAAc,EAAE,MAAM,CAAC;CACxB;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,UAAU,EAAE,QAAQ,GAAG,SAAS,CAAC;IACjC,IAAI,EAAE,MAAM,EAAE,CAAC;IACf,aAAa,EAAE,MAAM,CAAC;IACtB,aAAa,EAAE,MAAM,CAAC;IACtB,WAAW,EAAE,UAAU,GAAG,OAAO,CAAC;IAClC,cAAc,EAAE,MAAM,CAAC;IACvB,QAAQ,EAAE,gBAAgB,EAAE,CAAC;CAC9B;AAED,MAAM,WAAW,gBAAgB;IAC/B,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,QAAQ,EAAE,OAAO,CAAC;CACnB;AAED,MAAM,WAAW,YAAY;IAC3B,MAAM,EAAE,YAAY,EAAE,CAAC;IACvB,KAAK,EAAE,MAAM,CAAC;IACd,IAAI,EAAE,MAAM,CAAC;IACb,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,OAAO,CAAC;CAClB;AAED,MAAM,WAAW,WAAW;IAC1B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,IAAI,CAAC,EAAE,WAAW,GAAG,QAAQ,GAAG,MAAM,CAAC;IACvC,KAAK,CAAC,EAAE,MAAM,CAAC;IACf,GAAG,CAAC,EAAE,MAAM,CAAC;CACd;AAED,MAAM,WAAW,aAAa;IAC5B,KAAK,EAAE,MAAM,CAAC;IACd,OAAO,EAAE,MAAM,CAAC;IAChB,OAAO,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;CAClC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"registry.types.js","sourceRoot":"","sources":["../../src/types/registry.types.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,26 @@
|
|
|
1
|
+
export interface AgentFrontmatter {
|
|
2
|
+
name?: string;
|
|
3
|
+
description?: string;
|
|
4
|
+
version?: string;
|
|
5
|
+
tools?: string[];
|
|
6
|
+
handoffs?: string[];
|
|
7
|
+
'argument-hint'?: string;
|
|
8
|
+
'mcp-servers'?: Record<string, unknown>;
|
|
9
|
+
[key: string]: unknown;
|
|
10
|
+
}
|
|
11
|
+
export interface ParsedAgent {
|
|
12
|
+
frontmatter: AgentFrontmatter;
|
|
13
|
+
content: string;
|
|
14
|
+
body: string;
|
|
15
|
+
}
|
|
16
|
+
export declare const extractFrontmatter: (content: string) => AgentFrontmatter;
|
|
17
|
+
export declare const parseAgentFile: (content: string) => ParsedAgent;
|
|
18
|
+
export declare const readAgentFile: (filePath: string) => Promise<ParsedAgent>;
|
|
19
|
+
export declare const generateDateVersion: () => string;
|
|
20
|
+
export declare const isValidAgentName: (name: string) => boolean;
|
|
21
|
+
export declare const parseAgentIdentifier: (identifier: string) => {
|
|
22
|
+
owner?: string;
|
|
23
|
+
name?: string;
|
|
24
|
+
version?: string;
|
|
25
|
+
};
|
|
26
|
+
//# sourceMappingURL=agent-parser.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"agent-parser.d.ts","sourceRoot":"","sources":["../../src/utils/agent-parser.ts"],"names":[],"mappings":"AAGA,MAAM,WAAW,gBAAgB;IAC/B,IAAI,CAAC,EAAE,MAAM,CAAC;IACd,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,EAAE,CAAC;IACjB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;IACpB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,aAAa,CAAC,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACxC,CAAC,GAAG,EAAE,MAAM,GAAG,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,WAAW;IAC1B,WAAW,EAAE,gBAAgB,CAAC;IAC9B,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,MAAM,CAAC;CACd;AAKD,eAAO,MAAM,kBAAkB,GAAI,SAAS,MAAM,KAAG,gBAWpD,CAAC;AAKF,eAAO,MAAM,cAAc,GAAI,SAAS,MAAM,KAAG,WAUhD,CAAC;AAKF,eAAO,MAAM,aAAa,GAAU,UAAU,MAAM,KAAG,OAAO,CAAC,WAAW,CAGzE,CAAC;AAKF,eAAO,MAAM,mBAAmB,QAAO,MAGtC,CAAC;AAKF,eAAO,MAAM,gBAAgB,GAAI,MAAM,MAAM,KAAG,OACM,CAAC;AAKvD,eAAO,MAAM,oBAAoB,GAC/B,YAAY,MAAM,KACjB;IAAE,KAAK,CAAC,EAAE,MAAM,CAAC;IAAC,IAAI,CAAC,EAAE,MAAM,CAAC;IAAC,OAAO,CAAC,EAAE,MAAM,CAAA;CAcnD,CAAC"}
|
|
@@ -0,0 +1,45 @@
|
|
|
1
|
+
import { readFile } from 'fs/promises';
|
|
2
|
+
import { parse } from 'yaml';
|
|
3
|
+
export const extractFrontmatter = (content) => {
|
|
4
|
+
const match = content.match(/^---\n([\s\S]*?)\n---/);
|
|
5
|
+
if (!match) {
|
|
6
|
+
return {};
|
|
7
|
+
}
|
|
8
|
+
try {
|
|
9
|
+
return parse(match[1]);
|
|
10
|
+
}
|
|
11
|
+
catch {
|
|
12
|
+
return {};
|
|
13
|
+
}
|
|
14
|
+
};
|
|
15
|
+
export const parseAgentFile = (content) => {
|
|
16
|
+
const frontmatter = extractFrontmatter(content);
|
|
17
|
+
const bodyMatch = content.match(/^---\n[\s\S]*?\n---\n?([\s\S]*)$/);
|
|
18
|
+
const body = bodyMatch ? bodyMatch[1].trim() : content;
|
|
19
|
+
return {
|
|
20
|
+
frontmatter,
|
|
21
|
+
content,
|
|
22
|
+
body,
|
|
23
|
+
};
|
|
24
|
+
};
|
|
25
|
+
export const readAgentFile = async (filePath) => {
|
|
26
|
+
const content = await readFile(filePath, 'utf-8');
|
|
27
|
+
return parseAgentFile(content);
|
|
28
|
+
};
|
|
29
|
+
export const generateDateVersion = () => {
|
|
30
|
+
const now = new Date();
|
|
31
|
+
return now.toISOString().split('T')[0];
|
|
32
|
+
};
|
|
33
|
+
export const isValidAgentName = (name) => /^[a-z0-9][a-z0-9-]*[a-z0-9]$|^[a-z0-9]$/.test(name);
|
|
34
|
+
export const parseAgentIdentifier = (identifier) => {
|
|
35
|
+
const [fullName, version] = identifier.split('@');
|
|
36
|
+
const parts = fullName.split('/');
|
|
37
|
+
if (parts.length === 1) {
|
|
38
|
+
return { name: parts[0], version };
|
|
39
|
+
}
|
|
40
|
+
if (parts.length === 2) {
|
|
41
|
+
return { owner: parts[0], name: parts[1], version };
|
|
42
|
+
}
|
|
43
|
+
return {};
|
|
44
|
+
};
|
|
45
|
+
//# sourceMappingURL=agent-parser.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"agent-parser.js","sourceRoot":"","sources":["../../src/utils/agent-parser.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,MAAM,aAAa,CAAC;AACvC,OAAO,EAAE,KAAK,EAAE,MAAM,MAAM,CAAC;AAsB7B,MAAM,CAAC,MAAM,kBAAkB,GAAG,CAAC,OAAe,EAAoB,EAAE;IACtE,MAAM,KAAK,GAAG,OAAO,CAAC,KAAK,CAAC,uBAAuB,CAAC,CAAC;IACrD,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,CAAC;QACH,OAAO,KAAK,CAAC,KAAK,CAAC,CAAC,CAAC,CAAqB,CAAC;IAC7C,CAAC;IAAC,MAAM,CAAC;QACP,OAAO,EAAE,CAAC;IACZ,CAAC;AACH,CAAC,CAAC;AAKF,MAAM,CAAC,MAAM,cAAc,GAAG,CAAC,OAAe,EAAe,EAAE;IAC7D,MAAM,WAAW,GAAG,kBAAkB,CAAC,OAAO,CAAC,CAAC;IAChD,MAAM,SAAS,GAAG,OAAO,CAAC,KAAK,CAAC,kCAAkC,CAAC,CAAC;IACpE,MAAM,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,OAAO,CAAC;IAEvD,OAAO;QACL,WAAW;QACX,OAAO;QACP,IAAI;KACL,CAAC;AACJ,CAAC,CAAC;AAKF,MAAM,CAAC,MAAM,aAAa,GAAG,KAAK,EAAE,QAAgB,EAAwB,EAAE;IAC5E,MAAM,OAAO,GAAG,MAAM,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAClD,OAAO,cAAc,CAAC,OAAO,CAAC,CAAC;AACjC,CAAC,CAAC;AAKF,MAAM,CAAC,MAAM,mBAAmB,GAAG,GAAW,EAAE;IAC9C,MAAM,GAAG,GAAG,IAAI,IAAI,EAAE,CAAC;IACvB,OAAO,GAAG,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;AACzC,CAAC,CAAC;AAKF,MAAM,CAAC,MAAM,gBAAgB,GAAG,CAAC,IAAY,EAAW,EAAE,CACxD,yCAAyC,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;AAKvD,MAAM,CAAC,MAAM,oBAAoB,GAAG,CAClC,UAAkB,EACmC,EAAE;IACvD,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,GAAG,UAAU,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAClD,MAAM,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;IAElC,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QAEvB,OAAO,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,OAAO,EAAE,CAAC;IACrC,CAAC;IAED,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;QACvB,OAAO,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,KAAK,CAAC,CAAC,CAAC,EAAE,OAAO,EAAE,CAAC;IACtD,CAAC;IAED,OAAO,EAAE,CAAC;AACZ,CAAC,CAAC"}
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"agent-parser.test.d.ts","sourceRoot":"","sources":["../../src/utils/agent-parser.test.ts"],"names":[],"mappings":""}
|
|
@@ -0,0 +1,102 @@
|
|
|
1
|
+
import { extractFrontmatter, generateDateVersion, isValidAgentName, parseAgentFile, parseAgentIdentifier, } from './agent-parser.js';
|
|
2
|
+
describe('agent-parser', () => {
|
|
3
|
+
describe('extractFrontmatter', () => {
|
|
4
|
+
test('extracts valid frontmatter', () => {
|
|
5
|
+
const content = `---
|
|
6
|
+
name: test-agent
|
|
7
|
+
description: A test agent
|
|
8
|
+
version: 2025-11-30
|
|
9
|
+
---
|
|
10
|
+
You are a helpful assistant.`;
|
|
11
|
+
const result = extractFrontmatter(content);
|
|
12
|
+
expect(result.name).toBe('test-agent');
|
|
13
|
+
expect(result.description).toBe('A test agent');
|
|
14
|
+
expect(result.version).toBe('2025-11-30');
|
|
15
|
+
});
|
|
16
|
+
test('returns empty object for missing frontmatter', () => {
|
|
17
|
+
const content = 'Just plain text without frontmatter';
|
|
18
|
+
const result = extractFrontmatter(content);
|
|
19
|
+
expect(result).toEqual({});
|
|
20
|
+
});
|
|
21
|
+
test('returns empty object for invalid YAML', () => {
|
|
22
|
+
const content = `---
|
|
23
|
+
invalid: yaml: content: here
|
|
24
|
+
---
|
|
25
|
+
Body`;
|
|
26
|
+
const result = extractFrontmatter(content);
|
|
27
|
+
expect(result).toEqual({});
|
|
28
|
+
});
|
|
29
|
+
});
|
|
30
|
+
describe('parseAgentFile', () => {
|
|
31
|
+
test('parses full agent file', () => {
|
|
32
|
+
const content = `---
|
|
33
|
+
name: my-agent
|
|
34
|
+
model: gpt-4
|
|
35
|
+
---
|
|
36
|
+
You are a helpful assistant.
|
|
37
|
+
|
|
38
|
+
## Instructions
|
|
39
|
+
Be kind and helpful.`;
|
|
40
|
+
const result = parseAgentFile(content);
|
|
41
|
+
expect(result.frontmatter.name).toBe('my-agent');
|
|
42
|
+
expect(result.frontmatter.model).toBe('gpt-4');
|
|
43
|
+
expect(result.content).toBe(content);
|
|
44
|
+
expect(result.body).toContain('You are a helpful assistant.');
|
|
45
|
+
expect(result.body).toContain('## Instructions');
|
|
46
|
+
});
|
|
47
|
+
test('handles content without frontmatter', () => {
|
|
48
|
+
const content = 'Just plain text';
|
|
49
|
+
const result = parseAgentFile(content);
|
|
50
|
+
expect(result.frontmatter).toEqual({});
|
|
51
|
+
expect(result.body).toBe(content);
|
|
52
|
+
});
|
|
53
|
+
});
|
|
54
|
+
describe('generateDateVersion', () => {
|
|
55
|
+
test('generates date in YYYY-MM-DD format', () => {
|
|
56
|
+
const version = generateDateVersion();
|
|
57
|
+
expect(version).toMatch(/^\d{4}-\d{2}-\d{2}$/);
|
|
58
|
+
});
|
|
59
|
+
});
|
|
60
|
+
describe('isValidAgentName', () => {
|
|
61
|
+
test('accepts valid names', () => {
|
|
62
|
+
expect(isValidAgentName('my-agent')).toBe(true);
|
|
63
|
+
expect(isValidAgentName('agent123')).toBe(true);
|
|
64
|
+
expect(isValidAgentName('a')).toBe(true);
|
|
65
|
+
expect(isValidAgentName('test-agent-v2')).toBe(true);
|
|
66
|
+
});
|
|
67
|
+
test('rejects invalid names', () => {
|
|
68
|
+
expect(isValidAgentName('My-Agent')).toBe(false);
|
|
69
|
+
expect(isValidAgentName('-agent')).toBe(false);
|
|
70
|
+
expect(isValidAgentName('agent-')).toBe(false);
|
|
71
|
+
expect(isValidAgentName('agent_name')).toBe(false);
|
|
72
|
+
expect(isValidAgentName('')).toBe(false);
|
|
73
|
+
});
|
|
74
|
+
});
|
|
75
|
+
describe('parseAgentIdentifier', () => {
|
|
76
|
+
test('parses owner/name format', () => {
|
|
77
|
+
const result = parseAgentIdentifier('owner/my-agent');
|
|
78
|
+
expect(result).toEqual({
|
|
79
|
+
owner: 'owner',
|
|
80
|
+
name: 'my-agent',
|
|
81
|
+
version: undefined,
|
|
82
|
+
});
|
|
83
|
+
});
|
|
84
|
+
test('parses owner/name@version format', () => {
|
|
85
|
+
const result = parseAgentIdentifier('owner/my-agent@2025-11-30');
|
|
86
|
+
expect(result).toEqual({
|
|
87
|
+
owner: 'owner',
|
|
88
|
+
name: 'my-agent',
|
|
89
|
+
version: '2025-11-30',
|
|
90
|
+
});
|
|
91
|
+
});
|
|
92
|
+
test('parses name-only format', () => {
|
|
93
|
+
const result = parseAgentIdentifier('my-agent');
|
|
94
|
+
expect(result).toEqual({ name: 'my-agent', version: undefined });
|
|
95
|
+
});
|
|
96
|
+
test('parses name@version format', () => {
|
|
97
|
+
const result = parseAgentIdentifier('my-agent@1.0.0');
|
|
98
|
+
expect(result).toEqual({ name: 'my-agent', version: '1.0.0' });
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
});
|
|
102
|
+
//# sourceMappingURL=agent-parser.test.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"agent-parser.test.js","sourceRoot":"","sources":["../../src/utils/agent-parser.test.ts"],"names":[],"mappings":"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"}
|