rootly-mcp-server 2.0.13__py3-none-any.whl → 2.0.14__py3-none-any.whl
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- rootly_mcp_server/server.py +125 -2
- {rootly_mcp_server-2.0.13.dist-info → rootly_mcp_server-2.0.14.dist-info}/METADATA +5 -1
- {rootly_mcp_server-2.0.13.dist-info → rootly_mcp_server-2.0.14.dist-info}/RECORD +6 -6
- {rootly_mcp_server-2.0.13.dist-info → rootly_mcp_server-2.0.14.dist-info}/WHEEL +0 -0
- {rootly_mcp_server-2.0.13.dist-info → rootly_mcp_server-2.0.14.dist-info}/entry_points.txt +0 -0
- {rootly_mcp_server-2.0.13.dist-info → rootly_mcp_server-2.0.14.dist-info}/licenses/LICENSE +0 -0
rootly_mcp_server/server.py
CHANGED
|
@@ -394,6 +394,7 @@ def create_rootly_mcp_server(
|
|
|
394
394
|
"page[size]": page_size, # Use requested page size (already limited to max 20)
|
|
395
395
|
"page[number]": page_number,
|
|
396
396
|
"include": "",
|
|
397
|
+
"fields[incidents]": "id,title,summary,status,severity,created_at,updated_at,url,started_at",
|
|
397
398
|
}
|
|
398
399
|
if query:
|
|
399
400
|
params["filter[search]"] = query
|
|
@@ -418,6 +419,7 @@ def create_rootly_mcp_server(
|
|
|
418
419
|
"page[size]": effective_page_size,
|
|
419
420
|
"page[number]": current_page,
|
|
420
421
|
"include": "",
|
|
422
|
+
"fields[incidents]": "id,title,summary,status,severity,created_at,updated_at,url,started_at",
|
|
421
423
|
}
|
|
422
424
|
if query:
|
|
423
425
|
params["filter[search]"] = query
|
|
@@ -988,6 +990,66 @@ def _filter_openapi_spec(spec: Dict[str, Any], allowed_paths: List[str]) -> Dict
|
|
|
988
990
|
"description": "Page number to retrieve"
|
|
989
991
|
}
|
|
990
992
|
})
|
|
993
|
+
|
|
994
|
+
# Add sparse fieldsets for alerts endpoints to reduce payload size
|
|
995
|
+
if "alert" in path.lower():
|
|
996
|
+
# Add fields[alerts] parameter with essential fields only - make it required with default
|
|
997
|
+
operation["parameters"].append({
|
|
998
|
+
"name": "fields[alerts]",
|
|
999
|
+
"in": "query",
|
|
1000
|
+
"required": True,
|
|
1001
|
+
"schema": {
|
|
1002
|
+
"type": "string",
|
|
1003
|
+
"default": "id,summary,status,started_at,ended_at,short_id,alert_urgency_id,source,noise",
|
|
1004
|
+
"description": "Comma-separated list of alert fields to include (reduces payload size)"
|
|
1005
|
+
}
|
|
1006
|
+
})
|
|
1007
|
+
|
|
1008
|
+
# Add include parameter for alerts endpoints to minimize relationships
|
|
1009
|
+
if "alert" in path.lower():
|
|
1010
|
+
# Check if include parameter already exists
|
|
1011
|
+
include_param_exists = any(param.get("name") == "include" for param in operation["parameters"])
|
|
1012
|
+
if not include_param_exists:
|
|
1013
|
+
operation["parameters"].append({
|
|
1014
|
+
"name": "include",
|
|
1015
|
+
"in": "query",
|
|
1016
|
+
"required": True,
|
|
1017
|
+
"schema": {
|
|
1018
|
+
"type": "string",
|
|
1019
|
+
"default": "",
|
|
1020
|
+
"description": "Related resources to include (empty for minimal payload)"
|
|
1021
|
+
}
|
|
1022
|
+
})
|
|
1023
|
+
|
|
1024
|
+
# Add sparse fieldsets for incidents endpoints to reduce payload size
|
|
1025
|
+
if "incident" in path.lower():
|
|
1026
|
+
# Add fields[incidents] parameter with essential fields only - make it required with default
|
|
1027
|
+
operation["parameters"].append({
|
|
1028
|
+
"name": "fields[incidents]",
|
|
1029
|
+
"in": "query",
|
|
1030
|
+
"required": True,
|
|
1031
|
+
"schema": {
|
|
1032
|
+
"type": "string",
|
|
1033
|
+
"default": "id,title,summary,status,severity,created_at,updated_at,url,started_at",
|
|
1034
|
+
"description": "Comma-separated list of incident fields to include (reduces payload size)"
|
|
1035
|
+
}
|
|
1036
|
+
})
|
|
1037
|
+
|
|
1038
|
+
# Add include parameter for incidents endpoints to minimize relationships
|
|
1039
|
+
if "incident" in path.lower():
|
|
1040
|
+
# Check if include parameter already exists
|
|
1041
|
+
include_param_exists = any(param.get("name") == "include" for param in operation["parameters"])
|
|
1042
|
+
if not include_param_exists:
|
|
1043
|
+
operation["parameters"].append({
|
|
1044
|
+
"name": "include",
|
|
1045
|
+
"in": "query",
|
|
1046
|
+
"required": True,
|
|
1047
|
+
"schema": {
|
|
1048
|
+
"type": "string",
|
|
1049
|
+
"default": "",
|
|
1050
|
+
"description": "Related resources to include (empty for minimal payload)"
|
|
1051
|
+
}
|
|
1052
|
+
})
|
|
991
1053
|
|
|
992
1054
|
# Also clean up any remaining broken references in components
|
|
993
1055
|
if "components" in filtered_spec and "schemas" in filtered_spec["components"]:
|
|
@@ -1002,6 +1064,52 @@ def _filter_openapi_spec(spec: Dict[str, Any], allowed_paths: List[str]) -> Dict
|
|
|
1002
1064
|
logger.warning(f"Removing schema with broken references: {schema_name}")
|
|
1003
1065
|
del schemas[schema_name]
|
|
1004
1066
|
|
|
1067
|
+
# Clean up any operation-level references to removed schemas
|
|
1068
|
+
removed_schemas = set()
|
|
1069
|
+
if "components" in filtered_spec and "schemas" in filtered_spec["components"]:
|
|
1070
|
+
removed_schemas = {"new_workflow", "update_workflow", "workflow", "workflow_task",
|
|
1071
|
+
"workflow_response", "workflow_list", "new_workflow_task",
|
|
1072
|
+
"update_workflow_task", "workflow_task_response", "workflow_task_list"}
|
|
1073
|
+
|
|
1074
|
+
for path, path_item in filtered_spec.get("paths", {}).items():
|
|
1075
|
+
for method, operation in path_item.items():
|
|
1076
|
+
if method.lower() not in ["get", "post", "put", "delete", "patch"]:
|
|
1077
|
+
continue
|
|
1078
|
+
|
|
1079
|
+
# Clean request body references
|
|
1080
|
+
if "requestBody" in operation:
|
|
1081
|
+
request_body = operation["requestBody"]
|
|
1082
|
+
if "content" in request_body:
|
|
1083
|
+
for content_type, content_info in request_body["content"].items():
|
|
1084
|
+
if "schema" in content_info and "$ref" in content_info["schema"]:
|
|
1085
|
+
ref_path = content_info["schema"]["$ref"]
|
|
1086
|
+
schema_name = ref_path.split("/")[-1]
|
|
1087
|
+
if schema_name in removed_schemas:
|
|
1088
|
+
# Replace with generic object schema
|
|
1089
|
+
content_info["schema"] = {
|
|
1090
|
+
"type": "object",
|
|
1091
|
+
"description": "Request data for this endpoint",
|
|
1092
|
+
"additionalProperties": True
|
|
1093
|
+
}
|
|
1094
|
+
logger.debug(f"Cleaned broken reference in {method.upper()} {path} request body: {ref_path}")
|
|
1095
|
+
|
|
1096
|
+
# Clean response references
|
|
1097
|
+
if "responses" in operation:
|
|
1098
|
+
for status_code, response in operation["responses"].items():
|
|
1099
|
+
if "content" in response:
|
|
1100
|
+
for content_type, content_info in response["content"].items():
|
|
1101
|
+
if "schema" in content_info and "$ref" in content_info["schema"]:
|
|
1102
|
+
ref_path = content_info["schema"]["$ref"]
|
|
1103
|
+
schema_name = ref_path.split("/")[-1]
|
|
1104
|
+
if schema_name in removed_schemas:
|
|
1105
|
+
# Replace with generic object schema
|
|
1106
|
+
content_info["schema"] = {
|
|
1107
|
+
"type": "object",
|
|
1108
|
+
"description": "Response data from this endpoint",
|
|
1109
|
+
"additionalProperties": True
|
|
1110
|
+
}
|
|
1111
|
+
logger.debug(f"Cleaned broken reference in {method.upper()} {path} response: {ref_path}")
|
|
1112
|
+
|
|
1005
1113
|
return filtered_spec
|
|
1006
1114
|
|
|
1007
1115
|
|
|
@@ -1013,8 +1121,23 @@ def _has_broken_references(schema_def: Dict[str, Any]) -> bool:
|
|
|
1013
1121
|
broken_refs = [
|
|
1014
1122
|
"incident_trigger_params",
|
|
1015
1123
|
"new_workflow",
|
|
1016
|
-
"update_workflow",
|
|
1017
|
-
"workflow"
|
|
1124
|
+
"update_workflow",
|
|
1125
|
+
"workflow",
|
|
1126
|
+
"new_workflow_task",
|
|
1127
|
+
"update_workflow_task",
|
|
1128
|
+
"workflow_task",
|
|
1129
|
+
"workflow_task_response",
|
|
1130
|
+
"workflow_task_list",
|
|
1131
|
+
"workflow_response",
|
|
1132
|
+
"workflow_list",
|
|
1133
|
+
"workflow_custom_field_selection_response",
|
|
1134
|
+
"workflow_custom_field_selection_list",
|
|
1135
|
+
"workflow_form_field_condition_response",
|
|
1136
|
+
"workflow_form_field_condition_list",
|
|
1137
|
+
"workflow_group_response",
|
|
1138
|
+
"workflow_group_list",
|
|
1139
|
+
"workflow_run_response",
|
|
1140
|
+
"workflow_runs_list"
|
|
1018
1141
|
]
|
|
1019
1142
|
if any(broken_ref in ref_path for broken_ref in broken_refs):
|
|
1020
1143
|
return True
|
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
Metadata-Version: 2.4
|
|
2
2
|
Name: rootly-mcp-server
|
|
3
|
-
Version: 2.0.
|
|
3
|
+
Version: 2.0.14
|
|
4
4
|
Summary: A Model Context Protocol server for Rootly APIs using OpenAPI spec
|
|
5
5
|
Project-URL: Homepage, https://github.com/Rootly-AI-Labs/Rootly-MCP-server
|
|
6
6
|
Project-URL: Issues, https://github.com/Rootly-AI-Labs/Rootly-MCP-server/issues
|
|
@@ -28,6 +28,10 @@ Description-Content-Type: text/markdown
|
|
|
28
28
|
|
|
29
29
|
# Rootly MCP Server
|
|
30
30
|
|
|
31
|
+
[](https://pypi.org/project/rootly-mcp-server/)
|
|
32
|
+
[](https://pypi.org/project/rootly-mcp-server/)
|
|
33
|
+
[](https://pypi.org/project/rootly-mcp-server/)
|
|
34
|
+
|
|
31
35
|
An MCP server for the [Rootly API](https://docs.rootly.com/api-reference/overview) that integrates seamlessly with MCP-compatible editors like Cursor, Windsurf, and Claude. Resolve production incidents in under a minute without leaving your IDE.
|
|
32
36
|
|
|
33
37
|
[](https://cursor.com/install-mcp?name=rootly&config=eyJjb21tYW5kIjoibnB4IC15IG1jcC1yZW1vdGUgaHR0cHM6Ly9tY3Aucm9vdGx5LmNvbS9zc2UgLS1oZWFkZXIgQXV0aG9yaXphdGlvbjoke1JPT1RMWV9BVVRIX0hFQURFUn0iLCJlbnYiOnsiUk9PVExZX0FVVEhfSEVBREVSIjoiQmVhcmVyIDxZT1VSX1JPT1RMWV9BUElfVE9LRU4%2BIn19)
|
|
@@ -1,12 +1,12 @@
|
|
|
1
1
|
rootly_mcp_server/__init__.py,sha256=rvIuqIyuzgC7b9qSnylrdDP2zPO-7Ou9AoblR6re1co,629
|
|
2
2
|
rootly_mcp_server/__main__.py,sha256=_F4p65_VjnN84RtmEdESVLLH0tO5tL9qBfb2Xdvbj2E,6480
|
|
3
3
|
rootly_mcp_server/client.py,sha256=uit-YijR7OAJtysBoclqnublEDVkFfcb29wSzhpBv44,4686
|
|
4
|
-
rootly_mcp_server/server.py,sha256=
|
|
4
|
+
rootly_mcp_server/server.py,sha256=dlqsLnD3QzRtSlUYzK_PUe5zR9rbQRvySRSjmKbCy5w,51476
|
|
5
5
|
rootly_mcp_server/smart_utils.py,sha256=lvGN9ITyJjBkm7ejpYagd8VWodLKnC6FmwECfCOcGwM,22973
|
|
6
6
|
rootly_mcp_server/utils.py,sha256=NyxdcDiFGlV2a8eBO4lKgZg0D7Gxr6xUIB0YyJGgpPA,4165
|
|
7
7
|
rootly_mcp_server/data/__init__.py,sha256=fO8a0bQnRVEoRMHKvhFzj10bhoaw7VsI51czc2MsUm4,143
|
|
8
|
-
rootly_mcp_server-2.0.
|
|
9
|
-
rootly_mcp_server-2.0.
|
|
10
|
-
rootly_mcp_server-2.0.
|
|
11
|
-
rootly_mcp_server-2.0.
|
|
12
|
-
rootly_mcp_server-2.0.
|
|
8
|
+
rootly_mcp_server-2.0.14.dist-info/METADATA,sha256=XvbzBwNb-XkGeedNz2oZDRebwvBA1cUYSYSrPftB9L4,9078
|
|
9
|
+
rootly_mcp_server-2.0.14.dist-info/WHEEL,sha256=qtCwoSJWgHk21S1Kb4ihdzI2rlJ1ZKaIurTj_ngOhyQ,87
|
|
10
|
+
rootly_mcp_server-2.0.14.dist-info/entry_points.txt,sha256=NE33b8VgigVPGBkboyo6pvN1Vz35HZtLybxMO4Q03PI,70
|
|
11
|
+
rootly_mcp_server-2.0.14.dist-info/licenses/LICENSE,sha256=c9w9ZZGl14r54tsP40oaq5adTVX_HMNHozPIH2ymzmw,11341
|
|
12
|
+
rootly_mcp_server-2.0.14.dist-info/RECORD,,
|
|
File without changes
|
|
File without changes
|
|
File without changes
|