Skip to content

Commit 5cb27d6

Browse files
committed
feat: add 'expand' option to get_work_item
Enhanced the get_work_item tool to support the WorkItemExpand option. It will default to WorkItemExpand.All when no specific fields are requested, providing more complete information in the response.
1 parent 175495a commit 5cb27d6

File tree

8 files changed

+168
-51
lines changed

8 files changed

+168
-51
lines changed

docs/tools/work-items.md

+8-1
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ Retrieves a work item by its ID.
1717
| Parameter | Type | Required | Description |
1818
| ------------ | ------ | -------- | --------------------------------------------------------------------------------- |
1919
| `workItemId` | number | Yes | The ID of the work item to retrieve |
20-
| `expand` | string | No | Controls the level of detail in the response (e.g., "All", "Relations", "Fields") |
20+
| `expand` | string | No | Controls the level of detail in the response. Defaults to "All" if not specified. Other values: "Relations", "Fields", "None" |
2121

2222
### Response
2323

@@ -45,9 +45,16 @@ Returns a work item object with the following structure:
4545
### Example Usage
4646

4747
```javascript
48+
// Using default expand="All"
4849
const result = await callTool('get_work_item', {
4950
workItemId: 123,
5051
});
52+
53+
// Explicitly specifying expand
54+
const minimalResult = await callTool('get_work_item', {
55+
workItemId: 123,
56+
expand: 'None'
57+
});
5158
```
5259

5360
## create_work_item
Original file line numberDiff line numberDiff line change
@@ -1 +1,3 @@
1+
## Current Task
12

3+
No task is currently in progress. Please take the next task from todo.md.

project-management/task-management/done.md

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

3+
- [x] **Task 2.8**: Allow `get_work_item` to default to 'Expand All' when no specific fields are requested. There isn't usually enough information on the default Get_work_item response now.
4+
- **Role**: Full-Stack Developer
5+
- **Phase**: Completed
6+
- **Notes**:
7+
- Current implementation in `src/features/work-items/get-work-item/feature.ts` only requested minimal fields by default
8+
- Azure DevOps API supports WorkItemExpand enum with options: None, Relations, Fields, Links, All
9+
- When using expand parameter, we should not specify fields array
10+
- Current schema in `src/features/work-items/schemas.ts` didn't expose expand parameter
11+
- **Implementation**:
12+
- Updated `GetWorkItemSchema` in `src/features/work-items/schemas.ts` to include the optional expand parameter
13+
- Modified `getWorkItem` function in `src/features/work-items/get-work-item/feature.ts` to use `WorkItemExpand.All` by default
14+
- Updated documentation in `docs/tools/work-items.md` to reflect the new default behavior
15+
- Added tests in `src/features/work-items/get-work-item/feature.spec.int.ts` to verify the expanded data retrieval
16+
- **Completed**: March 31, 2024
17+
318
- [x] **Task 1.0**: Implement `manage_work_item_link` handler with tests
419
- **Role**: Full-Stack Developer
520
- **Phase**: Completed

project-management/task-management/todo.md

-2
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,5 @@
11
## Azure DevOps MCP Server Project TODO List (Granular Daily Tasks)
22

3-
- [ ] **Task 2.8**: Implement `get_work_item` handler with tests
4-
- **Role**: Full-Stack Developer
53
- [ ] **Task 2.10**: Implement `add_work_item_comment` handler with tests
64
- **Role**: Full-Stack Developer
75
- [ ] **Task 2.13**: Refactor work item tools for consistency (with regression tests)

src/features/work-items/get-work-item/feature.spec.int.ts

+122-35
Original file line numberDiff line numberDiff line change
@@ -5,82 +5,169 @@ import {
55
shouldSkipIntegrationTest,
66
} from '../__test__/test-helpers';
77
import { WorkItemExpand } from 'azure-devops-node-api/interfaces/WorkItemTrackingInterfaces';
8+
import { AzureDevOpsResourceNotFoundError } from '../../../shared/errors';
9+
import { createWorkItem } from '../create-work-item/feature';
10+
import { manageWorkItemLink } from '../manage-work-item-link/feature';
11+
import { CreateWorkItemOptions } from '../types';
812

913
describe('getWorkItem integration', () => {
1014
let connection: WebApi | null = null;
15+
let testWorkItemId: number | null = null;
16+
let linkedWorkItemId: number | null = null;
17+
let projectName: string;
1118

1219
beforeAll(async () => {
1320
// Get a real connection using environment variables
1421
connection = await getTestConnection();
15-
});
22+
projectName = process.env.AZURE_DEVOPS_DEFAULT_PROJECT || 'DefaultProject';
1623

17-
// This test requires that work item #1 exists in the default project
18-
test('should retrieve a real work item from Azure DevOps', async () => {
19-
// Skip if no connection is available
20-
if (shouldSkipIntegrationTest()) {
24+
// Skip setup if integration tests should be skipped
25+
if (shouldSkipIntegrationTest() || !connection) {
2126
return;
2227
}
2328

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',
29+
try {
30+
// Create a test work item
31+
const uniqueTitle = `Test Work Item ${new Date().toISOString()}`;
32+
const options: CreateWorkItemOptions = {
33+
title: uniqueTitle,
34+
description: 'Test work item for get-work-item integration tests',
35+
};
36+
37+
const testWorkItem = await createWorkItem(
38+
connection,
39+
projectName,
40+
'Task',
41+
options,
42+
);
43+
44+
// Create another work item to link to the first one
45+
const linkedItemOptions: CreateWorkItemOptions = {
46+
title: `Linked Work Item ${new Date().toISOString()}`,
47+
description: 'Linked work item for get-work-item integration tests',
48+
};
49+
50+
const linkedWorkItem = await createWorkItem(
51+
connection,
52+
projectName,
53+
'Task',
54+
linkedItemOptions,
2855
);
56+
57+
if (testWorkItem?.id && linkedWorkItem?.id) {
58+
testWorkItemId = testWorkItem.id;
59+
linkedWorkItemId = linkedWorkItem.id;
60+
61+
// Create a link between the two work items
62+
await manageWorkItemLink(connection, projectName, {
63+
sourceWorkItemId: testWorkItemId,
64+
targetWorkItemId: linkedWorkItemId,
65+
operation: 'add',
66+
relationType: 'System.LinkTypes.Related',
67+
comment: 'Link created for get-work-item integration tests',
68+
});
69+
}
70+
} catch (error) {
71+
console.error('Failed to create test work items:', error);
2972
}
73+
});
3074

31-
// For a true integration test, use a known work item ID that exists
32-
const workItemId = 1; // This assumes work item #1 exists in your project
75+
test('should retrieve a real work item from Azure DevOps with default expand=all', async () => {
76+
// Skip if no connection is available
77+
if (shouldSkipIntegrationTest() || !connection || !testWorkItemId) {
78+
return;
79+
}
3380

34-
// Act - make an actual API call to Azure DevOps
35-
const result = await getWorkItem(connection, workItemId);
81+
// Act - get work item by ID
82+
const result = await getWorkItem(connection, testWorkItemId);
3683

37-
// Assert on the actual response
84+
// Assert
3885
expect(result).toBeDefined();
39-
expect(result.id).toBe(workItemId);
86+
expect(result.id).toBe(testWorkItemId);
4087

41-
// Verify fields exist
88+
// Verify expanded fields and data are present
4289
expect(result.fields).toBeDefined();
90+
expect(result._links).toBeDefined();
91+
92+
// With expand=all and a linked item, relations should be defined
93+
expect(result.relations).toBeDefined();
94+
4395
if (result.fields) {
44-
// Don't make assumptions about specific field values, just verify structure
96+
// Verify common fields that should be present with expand=all
4597
expect(result.fields['System.Title']).toBeDefined();
98+
expect(result.fields['System.State']).toBeDefined();
99+
expect(result.fields['System.CreatedDate']).toBeDefined();
100+
expect(result.fields['System.ChangedDate']).toBeDefined();
46101
}
47102
});
48103

49104
test('should retrieve work item with expanded relations', async () => {
50105
// Skip if no connection is available
51-
if (shouldSkipIntegrationTest()) {
106+
if (shouldSkipIntegrationTest() || !connection || !testWorkItemId) {
52107
return;
53108
}
54109

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-
// For a true integration test, use a known work item ID that exists
63-
const workItemId = 1; // This assumes work item #1 exists in your project
64-
65-
// Act - make an actual API call to Azure DevOps with expanded relations
110+
// Act - get work item with relations expansion
66111
const result = await getWorkItem(
67112
connection,
68-
workItemId,
113+
testWorkItemId,
69114
WorkItemExpand.Relations,
70115
);
71116

72-
// Assert on the actual response
117+
// Assert
73118
expect(result).toBeDefined();
74-
expect(result.id).toBe(workItemId);
119+
expect(result.id).toBe(testWorkItemId);
120+
121+
// When using expand=relations on a work item with links, relations should be defined
122+
expect(result.relations).toBeDefined();
75123

76-
// When using expand, we may get additional information beyond just fields
77-
// For example, revision, url, _links, etc.
78-
expect(result._links || result.url || result.rev).toBeTruthy();
124+
// Verify we can access the related work item
125+
if (result.relations && result.relations.length > 0) {
126+
const relation = result.relations[0];
127+
expect(relation.rel).toBe('System.LinkTypes.Related');
128+
expect(relation.url).toContain(linkedWorkItemId?.toString());
129+
}
79130

80131
// Verify fields exist
81132
expect(result.fields).toBeDefined();
82133
if (result.fields) {
83134
expect(result.fields['System.Title']).toBeDefined();
84135
}
85136
});
137+
138+
test('should retrieve work item with minimal fields when using expand=none', async () => {
139+
if (shouldSkipIntegrationTest() || !connection || !testWorkItemId) {
140+
return;
141+
}
142+
143+
// Act - get work item with no expansion
144+
const result = await getWorkItem(
145+
connection,
146+
testWorkItemId,
147+
WorkItemExpand.None,
148+
);
149+
150+
// Assert
151+
expect(result).toBeDefined();
152+
expect(result.id).toBe(testWorkItemId);
153+
expect(result.fields).toBeDefined();
154+
155+
// With expand=none, we should still get _links but no relations
156+
// The Azure DevOps API still returns _links even with expand=none
157+
expect(result.relations).toBeUndefined();
158+
});
159+
160+
test('should throw AzureDevOpsResourceNotFoundError for non-existent work item', async () => {
161+
if (shouldSkipIntegrationTest() || !connection) {
162+
return;
163+
}
164+
165+
// Use a very large ID that's unlikely to exist
166+
const nonExistentId = 999999999;
167+
168+
// Assert that it throws the correct error
169+
await expect(getWorkItem(connection, nonExistentId)).rejects.toThrow(
170+
AzureDevOpsResourceNotFoundError,
171+
);
172+
});
86173
});

src/features/work-items/get-work-item/feature.ts

+9-12
Original file line numberDiff line numberDiff line change
@@ -11,28 +11,25 @@ import { WorkItem } from '../types';
1111
*
1212
* @param connection The Azure DevOps WebApi connection
1313
* @param workItemId The ID of the work item
14-
* @param expand Optional expansion options
14+
* @param expand Optional expansion options (defaults to WorkItemExpand.All)
1515
* @returns The work item details
1616
* @throws {AzureDevOpsResourceNotFoundError} If the work item is not found
1717
*/
1818
export async function getWorkItem(
1919
connection: WebApi,
2020
workItemId: number,
21-
expand?: WorkItemExpand,
21+
expand: WorkItemExpand = WorkItemExpand.All,
2222
): Promise<WorkItem> {
2323
try {
2424
const witApi = await connection.getWorkItemTrackingApi();
25-
const fields = [
26-
'System.Id',
27-
'System.Title',
28-
'System.State',
29-
'System.AssignedTo',
30-
];
3125

32-
// Don't pass fields when using expand parameter
33-
const workItem = expand
34-
? await witApi.getWorkItem(workItemId, undefined, undefined, expand)
35-
: await witApi.getWorkItem(workItemId, fields);
26+
// Always use expand parameter for consistent behavior
27+
const workItem = await witApi.getWorkItem(
28+
workItemId,
29+
undefined,
30+
undefined,
31+
expand,
32+
);
3633

3734
if (!workItem) {
3835
throw new AzureDevOpsResourceNotFoundError(

src/features/work-items/schemas.ts

+7
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,17 @@
11
import { z } from 'zod';
2+
import { WorkItemExpand } from 'azure-devops-node-api/interfaces/WorkItemTrackingInterfaces';
23

34
/**
45
* Schema for getting a work item
56
*/
67
export const GetWorkItemSchema = z.object({
78
workItemId: z.number().describe('The ID of the work item'),
9+
expand: z
10+
.nativeEnum(WorkItemExpand)
11+
.optional()
12+
.describe(
13+
'The level of detail to include in the response. Defaults to "all" if not specified.',
14+
),
815
});
916

1017
/**

src/server.ts

+5-1
Original file line numberDiff line numberDiff line change
@@ -180,7 +180,11 @@ export function createAzureDevOpsServer(config: AzureDevOpsConfig): Server {
180180
// Work item tools
181181
case 'get_work_item': {
182182
const args = GetWorkItemSchema.parse(request.params.arguments);
183-
const result = await getWorkItem(connection, args.workItemId);
183+
const result = await getWorkItem(
184+
connection,
185+
args.workItemId,
186+
args.expand,
187+
);
184188
return {
185189
content: [{ type: 'text', text: JSON.stringify(result, null, 2) }],
186190
};

0 commit comments

Comments
 (0)