Skip to content

Commit c63b261

Browse files
openhands-agentTiberriver256
authored andcommitted
feat: implement get_project_details core functionality
Implements handler for detailed project information including process, work item types, and teams. Closes #101
1 parent 3fad88b commit c63b261

File tree

9 files changed

+1082
-0
lines changed

9 files changed

+1082
-0
lines changed

project-management/task-management/done.md

+22
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,27 @@
11
## Completed Tasks
22

3+
- [x] **Task**: Implement get_project_details core functionality (GitHub Issue #101)
4+
- **Role**: Full-Stack Developer
5+
- **Phase**: Completed
6+
- **Notes**:
7+
- Implemented a handler for the `get_project_details` functionality
8+
- Added support for retrieving comprehensive project information including:
9+
- Basic project metadata
10+
- Process information
11+
- Work item types and their structure
12+
- Fields applicable to each work item type
13+
- Project teams
14+
- Created unit tests and integration tests
15+
- Registered the handler in the server
16+
- **Sub-tasks**:
17+
- [x] Created schema for get_project_details
18+
- [x] Created tests for get_project_details
19+
- [x] Implemented get_project_details feature
20+
- [x] Registered the handler in server.ts
21+
- [x] Updated exports in projects/index.ts
22+
- **Completed**: April 2, 2025
23+
- **Pull Request**: [#111](https://github.com/Tiberriver256/mcp-server-azure-devops/pull/111)
24+
325
- [x] **Task 3.1**: Implement get_repository_details core functionality
426
- **Role**: Full-Stack Developer
527
- **Phase**: Completed
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,212 @@
1+
import { WebApi } from 'azure-devops-node-api';
2+
import { getProjectDetails } from './feature';
3+
import {
4+
getTestConnection,
5+
shouldSkipIntegrationTest,
6+
} from '@/shared/test/test-helpers';
7+
8+
describe('getProjectDetails integration', () => {
9+
let connection: WebApi | null = null;
10+
let projectName: string;
11+
12+
beforeAll(async () => {
13+
// Get a real connection using environment variables
14+
connection = await getTestConnection();
15+
projectName = process.env.AZURE_DEVOPS_DEFAULT_PROJECT || 'DefaultProject';
16+
});
17+
18+
test('should retrieve basic project details from Azure DevOps', async () => {
19+
// Skip if no connection is available
20+
if (shouldSkipIntegrationTest()) {
21+
return;
22+
}
23+
24+
// This connection must be available if we didn't skip
25+
if (!connection) {
26+
throw new Error(
27+
'Connection should be available when test is not skipped',
28+
);
29+
}
30+
31+
// Act - make an actual API call to Azure DevOps
32+
const result = await getProjectDetails(connection, {
33+
projectId: projectName,
34+
});
35+
36+
// Assert on the actual response
37+
expect(result).toBeDefined();
38+
expect(result.name).toBe(projectName);
39+
expect(result.id).toBeDefined();
40+
expect(result.url).toBeDefined();
41+
expect(result.state).toBeDefined();
42+
43+
// Verify basic project structure
44+
expect(result.visibility).toBeDefined();
45+
expect(result.lastUpdateTime).toBeDefined();
46+
expect(result.capabilities).toBeDefined();
47+
});
48+
49+
test('should retrieve project details with teams from Azure DevOps', async () => {
50+
// Skip if no connection is available
51+
if (shouldSkipIntegrationTest()) {
52+
return;
53+
}
54+
55+
// This connection must be available if we didn't skip
56+
if (!connection) {
57+
throw new Error(
58+
'Connection should be available when test is not skipped',
59+
);
60+
}
61+
62+
// Act - make an actual API call to Azure DevOps
63+
const result = await getProjectDetails(connection, {
64+
projectId: projectName,
65+
includeTeams: true,
66+
});
67+
68+
// Assert on the actual response
69+
expect(result).toBeDefined();
70+
expect(result.teams).toBeDefined();
71+
expect(Array.isArray(result.teams)).toBe(true);
72+
73+
// There should be at least one team (the default team)
74+
if (result.teams && result.teams.length > 0) {
75+
const team = result.teams[0];
76+
expect(team.id).toBeDefined();
77+
expect(team.name).toBeDefined();
78+
expect(team.url).toBeDefined();
79+
}
80+
});
81+
82+
test('should retrieve project details with process information from Azure DevOps', async () => {
83+
// Skip if no connection is available
84+
if (shouldSkipIntegrationTest()) {
85+
return;
86+
}
87+
88+
// This connection must be available if we didn't skip
89+
if (!connection) {
90+
throw new Error(
91+
'Connection should be available when test is not skipped',
92+
);
93+
}
94+
95+
// Act - make an actual API call to Azure DevOps
96+
const result = await getProjectDetails(connection, {
97+
projectId: projectName,
98+
includeProcess: true,
99+
});
100+
101+
// Assert on the actual response
102+
expect(result).toBeDefined();
103+
expect(result.process).toBeDefined();
104+
expect(result.process?.name).toBeDefined();
105+
});
106+
107+
test('should retrieve project details with work item types from Azure DevOps', async () => {
108+
// Skip if no connection is available
109+
if (shouldSkipIntegrationTest()) {
110+
return;
111+
}
112+
113+
// This connection must be available if we didn't skip
114+
if (!connection) {
115+
throw new Error(
116+
'Connection should be available when test is not skipped',
117+
);
118+
}
119+
120+
// Act - make an actual API call to Azure DevOps
121+
const result = await getProjectDetails(connection, {
122+
projectId: projectName,
123+
includeProcess: true,
124+
includeWorkItemTypes: true,
125+
});
126+
127+
// Assert on the actual response
128+
expect(result).toBeDefined();
129+
expect(result.process).toBeDefined();
130+
expect(result.process?.workItemTypes).toBeDefined();
131+
expect(Array.isArray(result.process?.workItemTypes)).toBe(true);
132+
133+
// There should be at least one work item type
134+
if (
135+
result.process?.workItemTypes &&
136+
result.process.workItemTypes.length > 0
137+
) {
138+
const workItemType = result.process.workItemTypes[0];
139+
expect(workItemType.name).toBeDefined();
140+
expect(workItemType.description).toBeDefined();
141+
expect(workItemType.states).toBeDefined();
142+
}
143+
});
144+
145+
test('should retrieve project details with fields from Azure DevOps', async () => {
146+
// Skip if no connection is available
147+
if (shouldSkipIntegrationTest()) {
148+
return;
149+
}
150+
151+
// This connection must be available if we didn't skip
152+
if (!connection) {
153+
throw new Error(
154+
'Connection should be available when test is not skipped',
155+
);
156+
}
157+
158+
// Act - make an actual API call to Azure DevOps
159+
const result = await getProjectDetails(connection, {
160+
projectId: projectName,
161+
includeProcess: true,
162+
includeWorkItemTypes: true,
163+
includeFields: true,
164+
});
165+
166+
// Assert on the actual response
167+
expect(result).toBeDefined();
168+
expect(result.process).toBeDefined();
169+
expect(result.process?.workItemTypes).toBeDefined();
170+
171+
// There should be at least one work item type with fields
172+
if (
173+
result.process?.workItemTypes &&
174+
result.process.workItemTypes.length > 0
175+
) {
176+
const workItemType = result.process.workItemTypes[0];
177+
expect(workItemType.fields).toBeDefined();
178+
expect(Array.isArray(workItemType.fields)).toBe(true);
179+
180+
// There should be at least one field (like Title)
181+
if (workItemType.fields && workItemType.fields.length > 0) {
182+
const field = workItemType.fields[0];
183+
expect(field.name).toBeDefined();
184+
expect(field.referenceName).toBeDefined();
185+
}
186+
}
187+
});
188+
189+
test('should throw error when project is not found', async () => {
190+
// Skip if no connection is available
191+
if (shouldSkipIntegrationTest()) {
192+
return;
193+
}
194+
195+
// This connection must be available if we didn't skip
196+
if (!connection) {
197+
throw new Error(
198+
'Connection should be available when test is not skipped',
199+
);
200+
}
201+
202+
// Use a non-existent project name
203+
const nonExistentProjectName = 'non-existent-project-' + Date.now();
204+
205+
// Act & Assert - should throw an error for non-existent project
206+
await expect(
207+
getProjectDetails(connection, {
208+
projectId: nonExistentProjectName,
209+
}),
210+
).rejects.toThrow(/not found|Failed to get project/);
211+
});
212+
});

0 commit comments

Comments
 (0)