Skip to content

Commit 0680102

Browse files
committed
feat(search): add code search functionality for Azure DevOps repos
Implement search_code tool for searching across repositories in Azure DevOps. This includes API integration, documentation, schemas, and comprehensive tests. The search tool supports filtering by repository, path, branch, and code element types, as well as pagination and content retrieval options.
1 parent 6a7b4eb commit 0680102

File tree

11 files changed

+1218
-0
lines changed

11 files changed

+1218
-0
lines changed

.husky/pre-commit

100644100755
File mode changed.

README.md

+6
Original file line numberDiff line numberDiff line change
@@ -168,6 +168,12 @@ For repository-specific tool documentation, see the [Repositories Tools Guide](d
168168
- `list_work_items`: List work items in a project
169169
- `manage_work_item_link`: Add, remove, or update links between work items
170170

171+
### Search Tools
172+
173+
- `search_code`: Search for code across repositories in a project
174+
175+
For search-specific tool documentation, see the [Search Tools Guide](docs/tools/search.md).
176+
171177
## Contributing
172178

173179
Contributions are welcome! Please see [CONTRIBUTING.md](CONTRIBUTING.md) for contribution guidelines.

docs/tools/search.md

+94
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
# Search Tools
2+
3+
This document describes the search tools available in the Azure DevOps MCP server.
4+
5+
## search_code
6+
7+
The `search_code` tool allows you to search for code across repositories in an Azure DevOps project. It uses the Azure DevOps Search API to find code matching your search criteria and can optionally include the full content of the files in the results.
8+
9+
### Parameters
10+
11+
| Parameter | Type | Required | Description |
12+
|-----------|------|----------|-------------|
13+
| searchText | string | Yes | The text to search for in the code |
14+
| projectId | string | Yes | The ID or name of the project to search in |
15+
| filters | object | No | Optional filters to narrow search results |
16+
| filters.Repository | string[] | No | Filter by repository names |
17+
| filters.Path | string[] | No | Filter by file paths |
18+
| filters.Branch | string[] | No | Filter by branch names |
19+
| filters.CodeElement | string[] | No | Filter by code element types (function, class, etc.) |
20+
| top | number | No | Number of results to return (default: 100, max: 1000) |
21+
| skip | number | No | Number of results to skip for pagination (default: 0) |
22+
| includeSnippet | boolean | No | Whether to include code snippets in results (default: true) |
23+
| includeContent | boolean | No | Whether to include full file content in results (default: true) |
24+
25+
### Response
26+
27+
The response includes:
28+
29+
- `count`: The total number of matching files
30+
- `results`: An array of search results, each containing:
31+
- `fileName`: The name of the file
32+
- `path`: The path to the file
33+
- `content`: The full content of the file (if `includeContent` is true)
34+
- `matches`: Information about where the search text was found in the file
35+
- `collection`: Information about the collection
36+
- `project`: Information about the project
37+
- `repository`: Information about the repository
38+
- `versions`: Information about the versions of the file
39+
- `facets`: Aggregated information about the search results, such as counts by repository, path, etc.
40+
41+
### Examples
42+
43+
#### Basic Search
44+
45+
```json
46+
{
47+
"searchText": "function searchCode",
48+
"projectId": "MyProject"
49+
}
50+
```
51+
52+
#### Search with Filters
53+
54+
```json
55+
{
56+
"searchText": "function searchCode",
57+
"projectId": "MyProject",
58+
"filters": {
59+
"Repository": ["MyRepo"],
60+
"Path": ["/src"],
61+
"Branch": ["main"],
62+
"CodeElement": ["function", "class"]
63+
}
64+
}
65+
```
66+
67+
#### Search with Pagination
68+
69+
```json
70+
{
71+
"searchText": "function",
72+
"projectId": "MyProject",
73+
"top": 10,
74+
"skip": 20
75+
}
76+
```
77+
78+
#### Search without File Content
79+
80+
```json
81+
{
82+
"searchText": "function",
83+
"projectId": "MyProject",
84+
"includeContent": false
85+
}
86+
```
87+
88+
### Notes
89+
90+
- The search is performed using the Azure DevOps Search API, which is separate from the core Azure DevOps API.
91+
- The search API uses a different base URL (`almsearch.dev.azure.com`) than the regular Azure DevOps API.
92+
- When `includeContent` is true, the tool makes additional API calls to fetch the full content of each file in the search results.
93+
- The search API supports a variety of search syntax, including wildcards, exact phrases, and boolean operators. See the [Azure DevOps Search documentation](https://learn.microsoft.com/en-us/azure/devops/project/search/get-started-search?view=azure-devops) for more information.
94+
- The `CodeElement` filter allows you to filter by code element types such as `function`, `class`, `method`, `property`, `variable`, `comment`, etc.

src/features/search/index.ts

+3
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,3 @@
1+
export * from './schemas';
2+
export * from './types';
3+
export * from './search-code';

src/features/search/schemas.ts

+47
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,47 @@
1+
import { z } from 'zod';
2+
3+
/**
4+
* Schema for searching code in Azure DevOps repositories
5+
*/
6+
export const SearchCodeSchema = z.object({
7+
searchText: z.string().describe('The text to search for'),
8+
projectId: z.string().describe('The ID or name of the project to search in'),
9+
filters: z
10+
.object({
11+
Repository: z
12+
.array(z.string())
13+
.optional()
14+
.describe('Filter by repository names'),
15+
Path: z.array(z.string()).optional().describe('Filter by file paths'),
16+
Branch: z.array(z.string()).optional().describe('Filter by branch names'),
17+
CodeElement: z
18+
.array(z.string())
19+
.optional()
20+
.describe('Filter by code element types (function, class, etc.)'),
21+
})
22+
.optional()
23+
.describe('Optional filters to narrow search results'),
24+
top: z
25+
.number()
26+
.int()
27+
.min(1)
28+
.max(1000)
29+
.default(100)
30+
.describe('Number of results to return (default: 100, max: 1000)'),
31+
skip: z
32+
.number()
33+
.int()
34+
.min(0)
35+
.default(0)
36+
.describe('Number of results to skip for pagination (default: 0)'),
37+
includeSnippet: z
38+
.boolean()
39+
.default(true)
40+
.describe('Whether to include code snippets in results (default: true)'),
41+
includeContent: z
42+
.boolean()
43+
.default(true)
44+
.describe(
45+
'Whether to include full file content in results (default: true)',
46+
),
47+
});

0 commit comments

Comments
 (0)