Skip to content

Commit d4109ee

Browse files
authored
Oncall Tools (#50)
Add oncall tools --------- Co-authored-by: Ben Sully <[email protected]>
1 parent 9287a51 commit d4109ee

12 files changed

+726
-16
lines changed

.github/workflows/go.yml

+40-4
Original file line numberDiff line numberDiff line change
@@ -24,8 +24,24 @@ jobs:
2424
with:
2525
version: v1.64
2626

27-
test:
28-
name: Test
27+
test-unit:
28+
name: Test Unit
29+
runs-on: ubuntu-latest
30+
steps:
31+
- name: Checkout code
32+
uses: actions/checkout@v4
33+
34+
- name: Set up Go
35+
uses: actions/setup-go@v5
36+
with:
37+
go-version: '1.24'
38+
cache: true
39+
40+
- name: Run unit tests
41+
run: make test-unit
42+
43+
test-integration:
44+
name: Test Integration
2945
runs-on: ubuntu-latest
3046
steps:
3147
- name: Checkout code
@@ -48,5 +64,25 @@ jobs:
4864
- name: Wait for Grafana server and Prometheus server to start and scrape
4965
run: sleep 30
5066

51-
- name: Run tests
52-
run: make test-all
67+
- name: Run integration tests
68+
run: make test-integration
69+
70+
test-cloud:
71+
name: Test Cloud
72+
runs-on: ubuntu-latest
73+
steps:
74+
- name: Checkout code
75+
uses: actions/checkout@v4
76+
77+
- name: Set up Go
78+
uses: actions/setup-go@v5
79+
with:
80+
go-version: '1.24'
81+
cache: true
82+
83+
- name: Run cloud tests
84+
env:
85+
GRAFANA_URL: ${{ vars.CLOUD_GRAFANA_URL }}
86+
GRAFANA_API_KEY: ${{ secrets.CLOUD_GRAFANA_API_KEY }}
87+
run: make test-cloud
88+

Makefile

+10-5
Original file line numberDiff line numberDiff line change
@@ -16,14 +16,19 @@ build-image: ## Build the Docker image.
1616
lint: ## Lint the Go code.
1717
go tool -modfile go.tools.mod golangci-lint run
1818

19-
.PHONY: test
20-
test: ## Run the Go unit tests.
21-
go test ./...
19+
.PHONY: test test-unit
20+
test-unit: ## Run the unit tests (no external dependencies required).
21+
go test -v -tags unit ./...
22+
test: test-unit
2223

23-
.PHONY: test-all
24-
test-all: ## Run the Go unit and integration tests.
24+
.PHONY: test-integration
25+
test-integration: ## Run only the Docker-based integration tests (Requires docker containers to be up and running).
2526
go test -v -tags integration ./...
2627

28+
.PHONY: test-cloud
29+
test-cloud: ## Run only the cloud-based tests (requires cloud Grafana instance and credentials).
30+
go test -v -tags cloud ./tools
31+
2732
.PHONY: run
2833
run: ## Run the MCP server in stdio mode.
2934
go run ./...

README.md

+28-2
Original file line numberDiff line numberDiff line change
@@ -33,6 +33,12 @@ This provides access to your Grafana instance and the surrounding ecosystem.
3333
- [ ] Create and change alert rules
3434
- [ ] List contact points
3535
- [ ] Create and change contact points
36+
- [x] Access Grafana OnCall functionality
37+
- [x] List and manage schedules
38+
- [x] Get shift details
39+
- [x] Get current on-call users
40+
- [x] List teams and users
41+
- [ ] List alert groups
3642

3743
The list of tools is configurable, so you can choose which tools you want to make available to the MCP client.
3844
This is useful if you don't use certain functionality or if you don't want to take up too much of the context window.
@@ -61,6 +67,11 @@ This is useful if you don't use certain functionality or if you don't want to ta
6167
| `query_loki_stats` | Loki | Get statistics about log streams |
6268
| `list_alert_rules` | Alerting | List alert rules |
6369
| `get_alert_rule_by_uid` | Alerting | Get alert rule by UID |
70+
| `list_oncall_schedules` | OnCall | List schedules from Grafana OnCall |
71+
| `get_oncall_shift` | OnCall | Get details for a specific OnCall shift |
72+
| `get_current_oncall_users` | OnCall | Get users currently on-call for a specific schedule |
73+
| `list_oncall_teams` | OnCall | List teams from Grafana OnCall |
74+
| `list_oncall_users` | OnCall | List users from Grafana OnCall |
6475

6576
## Usage
6677

@@ -122,13 +133,28 @@ docker run -it --rm -p 8000:8000 mcp-grafana:latest
122133

123134
### Testing
124135

125-
To run unit tests, run:
136+
There are three types of tests available:
126137

138+
1. Unit Tests (no external dependencies required):
139+
```bash
140+
make test-unit
141+
```
142+
143+
You can also run unit tests with:
127144
```bash
128145
make test
129146
```
130147

131-
**TODO: add integration tests and cloud tests.**
148+
2. Integration Tests (requires docker containers to be up and running):
149+
```bash
150+
make test-integration
151+
```
152+
153+
3. Cloud Tests (requires cloud Grafana instance and credentials):
154+
```bash
155+
make test-cloud
156+
```
157+
> Note: Cloud tests are automatically configured in CI. For local development, you'll need to set up your own Grafana Cloud instance and credentials.
132158
133159
More comprehensive integration tests will require a Grafana instance to be running locally on port 3000; you can start one with Docker Compose:
134160

cmd/mcp-grafana/main.go

+1
Original file line numberDiff line numberDiff line change
@@ -25,6 +25,7 @@ func newServer() *server.MCPServer {
2525
tools.AddLokiTools(s)
2626
tools.AddAlertingTools(s)
2727
tools.AddDashboardTools(s)
28+
tools.AddOnCallTools(s)
2829
return s
2930
}
3031

go.mod

+4
Original file line numberDiff line numberDiff line change
@@ -4,6 +4,7 @@ go 1.24.0
44

55
require (
66
github.com/go-openapi/strfmt v0.23.0
7+
github.com/grafana/amixr-api-go-client v0.0.20
78
github.com/grafana/grafana-openapi-client-go v0.0.0-20250108132429-8d7e1f158f65
89
github.com/grafana/incident-go v0.0.0-20250211094540-dc6a98fdae43
910
github.com/invopop/jsonschema v0.13.0
@@ -32,8 +33,11 @@ require (
3233
github.com/go-openapi/spec v0.21.0 // indirect
3334
github.com/go-openapi/swag v0.23.0 // indirect
3435
github.com/go-openapi/validate v0.24.0 // indirect
36+
github.com/google/go-querystring v1.1.0 // indirect
3537
github.com/google/uuid v1.6.0 // indirect
3638
github.com/grafana/regexp v0.0.0-20240518133315-a468a5bfb3bc // indirect
39+
github.com/hashicorp/go-cleanhttp v0.5.2 // indirect
40+
github.com/hashicorp/go-retryablehttp v0.7.7 // indirect
3741
github.com/josharian/intern v1.0.0 // indirect
3842
github.com/jpillora/backoff v1.0.0 // indirect
3943
github.com/json-iterator/go v1.1.12 // indirect

go.sum

+36
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,9 @@ github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSs
1212
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
1313
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc h1:U9qPSI2PIWSS1VwoXQT9A3Wy9MM3WgvqSxFWenqJduM=
1414
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
15+
github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk=
16+
github.com/fatih/color v1.16.0 h1:zmkK9Ngbjj+K0yRhTVONQh1p/HknKYSlNT+vZCzyokM=
17+
github.com/fatih/color v1.16.0/go.mod h1:fL2Sau1YI5c0pdGEVCbKQbLXB6edEj1ZgiY4NijnWvE=
1518
github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A=
1619
github.com/go-logr/logr v1.4.2 h1:6pFjapn8bFcIbiKo3XT4j/BhANplGihG6tvd+8rYgrY=
1720
github.com/go-logr/logr v1.4.2/go.mod h1:9T104GzyrTigFIr8wt5mBrctHMim0Nb2HLGrmQ40KvY=
@@ -37,17 +40,29 @@ github.com/go-openapi/swag v0.23.0 h1:vsEVJDUo2hPJ2tu0/Xc+4noaxyEffXNIs3cOULZ+Gr
3740
github.com/go-openapi/swag v0.23.0/go.mod h1:esZ8ITTYEsH1V2trKHjAN8Ai7xHb8RV+YSZ577vPjgQ=
3841
github.com/go-openapi/validate v0.24.0 h1:LdfDKwNbpB6Vn40xhTdNZAnfLECL81w+VX3BumrGD58=
3942
github.com/go-openapi/validate v0.24.0/go.mod h1:iyeX1sEufmv3nPbBdX3ieNviWnOZaJ1+zquzJEf2BAQ=
43+
github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE=
4044
github.com/google/go-cmp v0.6.0 h1:ofyhxvXcZhMsU5ulbFiLKl/XBFqE1GSq7atu8tAmTRI=
4145
github.com/google/go-cmp v0.6.0/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY=
46+
github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck=
47+
github.com/google/go-querystring v1.1.0 h1:AnCroh3fv4ZBgVIf1Iwtovgjaw/GiKJo8M8yD/fhyJ8=
48+
github.com/google/go-querystring v1.1.0/go.mod h1:Kcdr2DB4koayq7X8pmAG4sNG59So17icRSOU623lUBU=
4249
github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg=
4350
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
4451
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
52+
github.com/grafana/amixr-api-go-client v0.0.20 h1:/L44PCP0H8y0Z6NSZmJFCUH3Nc5gRPphKrp2Ck7ufz0=
53+
github.com/grafana/amixr-api-go-client v0.0.20/go.mod h1:u53FF0WSBMx6XvZK58fply91KBl6X+OtIu0aJC07amY=
4554
github.com/grafana/grafana-openapi-client-go v0.0.0-20250108132429-8d7e1f158f65 h1:AnfwjPE8TXJO8CX0Q5PvtzGta9Ls3iRASWVV4jHl4KA=
4655
github.com/grafana/grafana-openapi-client-go v0.0.0-20250108132429-8d7e1f158f65/go.mod h1:hiZnMmXc9KXNUlvkV2BKFsiWuIFF/fF4wGgYWEjBitI=
4756
github.com/grafana/incident-go v0.0.0-20250211094540-dc6a98fdae43 h1:+MCsOKi5BJ1wO3PRj3eDNxCScYwE2IcKNW1t8OcTarE=
4857
github.com/grafana/incident-go v0.0.0-20250211094540-dc6a98fdae43/go.mod h1:3QDfdZOWKRxNhMJFL+0C/+12+jLNHDlt0VKNr/i9Daw=
4958
github.com/grafana/regexp v0.0.0-20240518133315-a468a5bfb3bc h1:GN2Lv3MGO7AS6PrRoT6yV5+wkrOpcszoIsO4+4ds248=
5059
github.com/grafana/regexp v0.0.0-20240518133315-a468a5bfb3bc/go.mod h1:+JKpmjMGhpgPL+rXZ5nsZieVzvarn86asRlBg4uNGnk=
60+
github.com/hashicorp/go-cleanhttp v0.5.2 h1:035FKYIWjmULyFRBKPs8TBQoi0x6d9G4xc9neXJWAZQ=
61+
github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48=
62+
github.com/hashicorp/go-hclog v1.6.3 h1:Qr2kF+eVWjTiYmU7Y31tYlP1h0q/X3Nl3tPGdaB11/k=
63+
github.com/hashicorp/go-hclog v1.6.3/go.mod h1:W4Qnvbt70Wk/zYJryRzDRU/4r0kIg0PVHBcfoyhpF5M=
64+
github.com/hashicorp/go-retryablehttp v0.7.7 h1:C8hUCYzor8PIfXHa4UrZkU4VvK8o9ISHxT2Q8+VepXU=
65+
github.com/hashicorp/go-retryablehttp v0.7.7/go.mod h1:pkQpWZeYWskR+D1tR2O5OcBFOxfA7DoAO6xtkuQnHTk=
5166
github.com/invopop/jsonschema v0.13.0 h1:KvpoAJWEjR3uD9Kbm2HWJmqsEaHt8lBUpd0qHcIi21E=
5267
github.com/invopop/jsonschema v0.13.0/go.mod h1:ffZ5Km5SWWRAIN6wbDXItl95euhFz2uON45H2qjYt+0=
5368
github.com/josharian/intern v1.0.0 h1:vlS4z54oSdjm0bgjRigI+G1HpF+tI+9rE5LLzOg8HmY=
@@ -68,6 +83,15 @@ github.com/mark3labs/mcp-go v0.15.0 h1:lViiC4dk6chJHZccezaTzZLMOQVUXJDGNQPtzExr5
6883
github.com/mark3labs/mcp-go v0.15.0/go.mod h1:xBB350hekQsJAK7gJAii8bcEoWemboLm2mRm5/+KBaU=
6984
github.com/matryer/is v1.4.1 h1:55ehd8zaGABKLXQUe2awZ99BD/PTc2ls+KV/dXphgEQ=
7085
github.com/matryer/is v1.4.1/go.mod h1:8I/i5uYgLzgsgEloJE1U6xx5HkBQpAZvepWuujKwMRU=
86+
github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc=
87+
github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4=
88+
github.com/mattn/go-colorable v0.1.13 h1:fFA4WZxdEF4tXPZVKMLwD8oUnCTTo08duU7wxecdEvA=
89+
github.com/mattn/go-colorable v0.1.13/go.mod h1:7S9/ev0klgBDR4GtXTXX8a3vIGJpMovkB8vQcUbaXHg=
90+
github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU=
91+
github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94=
92+
github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM=
93+
github.com/mattn/go-isatty v0.0.20 h1:xfD0iDuEKnDkl03q4limB+vH+GxLEtL/jb4xVJSWWEY=
94+
github.com/mattn/go-isatty v0.0.20/go.mod h1:W+V8PltTTMOvKvAeJH7IuucS94S2C6jfK/D7dTCTo3Y=
7195
github.com/mitchellh/mapstructure v1.5.0 h1:jeMsZIYE/09sWLaz43PL7Gy6RuMjD2eJVyuac5Z2hdY=
7296
github.com/mitchellh/mapstructure v1.5.0/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo=
7397
github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q=
@@ -100,6 +124,7 @@ github.com/rogpeppe/go-internal v1.13.1 h1:KvO1DLK/DRN07sQ1LQKScxyZJuNnedQ5/wKSR
100124
github.com/rogpeppe/go-internal v1.13.1/go.mod h1:uMEvuHeurkdAXX61udpOXGD/AzZDWNMNyH2VO9fmH0o=
101125
github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME=
102126
github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI=
127+
github.com/stretchr/testify v1.7.2/go.mod h1:R6va5+xMeoiuVRoj+gSkQ7d3FALtqAAGI1FQKckRals=
103128
github.com/stretchr/testify v1.10.0 h1:Xv5erBjTwe/5IxqUQTdXv5kgmIvbHo3QQyRwhJsOfJA=
104129
github.com/stretchr/testify v1.10.0/go.mod h1:r2ic/lqez/lEtzL7wO/rwa5dbSLXVDPFyf8C91i36aY=
105130
github.com/wk8/go-ordered-map/v2 v2.1.8 h1:5h/BUHu93oj4gIdvHHHGsScSTMijfx5PeYkE/fJgbpc=
@@ -124,10 +149,21 @@ golang.org/x/oauth2 v0.25.0 h1:CY4y7XT9v0cRI9oupztF8AgiIu99L/ksR/Xp/6jrZ70=
124149
golang.org/x/oauth2 v0.25.0/go.mod h1:XYTD2NtWslqkgxebSiOHnXEap4TF09sJSc7H1sXbhtI=
125150
golang.org/x/sync v0.10.0 h1:3NQrjDixjgGwUOCaF8w2+VYHv0Ve/vGYSbdkTa98gmQ=
126151
golang.org/x/sync v0.10.0/go.mod h1:Czt+wKu1gCyEFDUtn0jG5QVvpJ6rzVqr5aXyt9drQfk=
152+
golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
153+
golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs=
154+
golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
155+
golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
156+
golang.org/x/sys v0.0.0-20220503163025-988cb79eb6c6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
157+
golang.org/x/sys v0.0.0-20220811171246-fbc7d0a398ab/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
158+
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
159+
golang.org/x/sys v0.14.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
160+
golang.org/x/sys v0.20.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
127161
golang.org/x/sys v0.29.0 h1:TPYlXGxvx1MGTn2GiZDhnjPA9wZzZeGKHHmKhHYvgaU=
128162
golang.org/x/sys v0.29.0/go.mod h1:/VUhepiaJMQUp4+oa/7Zr1D23ma6VTLIYjOOTFZPUcA=
129163
golang.org/x/text v0.21.0 h1:zyQAAkrwaneQ066sspRyJaG9VNi/YJ1NfzcGB3hZ/qo=
130164
golang.org/x/text v0.21.0/go.mod h1:4IBbMaMmOPCJ8SecivzSH54+73PCFmPWxNTLm+vZkEQ=
165+
golang.org/x/time v0.0.0-20200630173020-3af7569d3a1e/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ=
166+
golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0=
131167
google.golang.org/protobuf v1.36.4 h1:6A3ZDJHn/eNqc1i+IdefRzy/9PokBTPvcqMySR7NNIM=
132168
google.golang.org/protobuf v1.36.4/go.mod h1:9fA7Ob0pmnwhb644+1+CVWFRbNajQ6iRojtC/QF5bRE=
133169
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=

mcpgrafana_test.go

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
//go:build unit
2+
// +build unit
3+
14
package mcpgrafana
25

36
import (

tools/incident_integration_test.go

+32-5
Original file line numberDiff line numberDiff line change
@@ -1,26 +1,53 @@
11
// Requires a Cloud or other Grafana instance with Grafana Incident available,
22
// with a Prometheus datasource provisioned.
3-
// Run with `go test -tags integration,cloud`.
4-
//go:build integration && cloud
3+
//go:build cloud
4+
// +build cloud
55

66
package tools
77

88
import (
99
"context"
10+
"os"
1011
"testing"
1112

1213
mcpgrafana "github.com/grafana/mcp-grafana"
1314
"github.com/stretchr/testify/assert"
1415
"github.com/stretchr/testify/require"
1516
)
1617

18+
// This file contains cloud integration tests that run against a dedicated test instance
19+
// at mcptests.grafana-dev.net. This instance is configured with a minimal setup on the Incident side
20+
// with two incidents created, one minor and one major, and both of them resolved.
21+
// These tests expect this configuration to exist and will skip if the required
22+
// environment variables (GRAFANA_URL, GRAFANA_API_KEY) are not set.
23+
24+
func createCloudTestContext(t *testing.T) context.Context {
25+
grafanaURL := os.Getenv("GRAFANA_URL")
26+
if grafanaURL == "" {
27+
t.Skip("GRAFANA_URL environment variable not set, skipping cloud Incident integration tests")
28+
}
29+
30+
grafanaApiKey := os.Getenv("GRAFANA_API_KEY")
31+
if grafanaApiKey == "" {
32+
t.Skip("GRAFANA_API_KEY environment variable not set, skipping cloud Incident integration tests")
33+
}
34+
35+
ctx := context.Background()
36+
ctx = mcpgrafana.WithGrafanaURL(ctx, grafanaURL)
37+
ctx = mcpgrafana.WithGrafanaAPIKey(ctx, grafanaApiKey)
38+
ctx = mcpgrafana.ExtractIncidentClientFromEnv(ctx)
39+
return ctx
40+
}
41+
1742
func TestCloudIncidentTools(t *testing.T) {
1843
t.Run("list incidents", func(t *testing.T) {
19-
ctx := mcpgrafana.ExtractIncidentClientFromEnv(context.Background())
44+
ctx := createCloudTestContext(t)
2045
result, err := listIncidents(ctx, ListIncidentsParams{
21-
Limit: 2,
46+
Limit: 1,
2247
})
2348
require.NoError(t, err)
24-
assert.Len(t, result.IncidentPreviews, 2)
49+
assert.NotNil(t, result, "Result should not be nil")
50+
assert.NotNil(t, result.IncidentPreviews, "IncidentPreviews should not be nil")
51+
assert.LessOrEqual(t, len(result.IncidentPreviews), 1, "Should not return more incidents than the limit")
2552
})
2653
}

tools/incident_test.go

+3
Original file line numberDiff line numberDiff line change
@@ -1,3 +1,6 @@
1+
//go:build unit
2+
// +build unit
3+
14
package tools
25

36
import (

0 commit comments

Comments
 (0)