Skip to content

Commit 956b80d

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 956b80d

File tree

8 files changed

+95
-22
lines changed

8 files changed

+95
-22
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

+49-6
Original file line numberDiff line numberDiff line change
@@ -5,6 +5,7 @@ import {
55
shouldSkipIntegrationTest,
66
} from '../__test__/test-helpers';
77
import { WorkItemExpand } from 'azure-devops-node-api/interfaces/WorkItemTrackingInterfaces';
8+
import { AzureDevOpsResourceNotFoundError } from '../../../shared/errors';
89

910
describe('getWorkItem integration', () => {
1011
let connection: WebApi | null = null;
@@ -15,7 +16,7 @@ describe('getWorkItem integration', () => {
1516
});
1617

1718
// 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+
test('should retrieve a real work item from Azure DevOps with default expand=all', async () => {
1920
// Skip if no connection is available
2021
if (shouldSkipIntegrationTest()) {
2122
return;
@@ -38,11 +39,16 @@ describe('getWorkItem integration', () => {
3839
expect(result).toBeDefined();
3940
expect(result.id).toBe(workItemId);
4041

41-
// Verify fields exist
42+
// Verify expanded fields and data are present
4243
expect(result.fields).toBeDefined();
44+
expect(result._links).toBeDefined();
45+
expect(result.relations).toBeDefined();
4346
if (result.fields) {
44-
// Don't make assumptions about specific field values, just verify structure
47+
// Verify common fields that should be present with expand=all
4548
expect(result.fields['System.Title']).toBeDefined();
49+
expect(result.fields['System.State']).toBeDefined();
50+
expect(result.fields['System.CreatedDate']).toBeDefined();
51+
expect(result.fields['System.ChangedDate']).toBeDefined();
4652
}
4753
});
4854

@@ -73,14 +79,51 @@ describe('getWorkItem integration', () => {
7379
expect(result).toBeDefined();
7480
expect(result.id).toBe(workItemId);
7581

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();
82+
// When using expand=relations, we should get relations data
83+
expect(result.relations).toBeDefined();
7984

8085
// Verify fields exist
8186
expect(result.fields).toBeDefined();
8287
if (result.fields) {
8388
expect(result.fields['System.Title']).toBeDefined();
8489
}
8590
});
91+
92+
test('should retrieve work item with minimal fields when using expand=none', async () => {
93+
if (shouldSkipIntegrationTest() || !connection) {
94+
return;
95+
}
96+
97+
const workItemId = 1;
98+
99+
// Act - get work item with no expansion
100+
const result = await getWorkItem(
101+
connection,
102+
workItemId,
103+
WorkItemExpand.None,
104+
);
105+
106+
// Assert
107+
expect(result).toBeDefined();
108+
expect(result.id).toBe(workItemId);
109+
expect(result.fields).toBeDefined();
110+
111+
// With expand=none, we should only get basic fields
112+
expect(result._links).toBeUndefined();
113+
expect(result.relations).toBeUndefined();
114+
});
115+
116+
test('should throw AzureDevOpsResourceNotFoundError for non-existent work item', async () => {
117+
if (shouldSkipIntegrationTest() || !connection) {
118+
return;
119+
}
120+
121+
// Use a very large ID that's unlikely to exist
122+
const nonExistentId = 999999999;
123+
124+
// Assert that it throws the correct error
125+
await expect(getWorkItem(connection, nonExistentId)).rejects.toThrow(
126+
AzureDevOpsResourceNotFoundError,
127+
);
128+
});
86129
});

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)