doc/development/duo_agent_platform/mcp/graphql_integration.md
This design introduces a reusable pattern for creating Model Context Protocol
(MCP) tools that leverage the GitLab GraphQL API. The solution provides
a two-layer architecture: a reusable GraphqlTool class that handles GraphQL
execution and error processing, and service wrappers extending GraphqlService
that handle validation and response formatting.
This pattern enables developers to create new GraphQL-powered MCP tools with minimal boilerplate while maintaining consistent security and error handling practices.
The implementation allows AI clients (like Duo agent platform, Claude, Cursor) to perform complex operations on GitLab resources through GraphQL mutations and queries, while reusing existing GraphQL schema definitions and authorization logic.
The GitLab MCP implementation uses a route-based architecture with
route_setting :mcp that exposes REST API endpoints as MCP tools.
While effective for REST operations, this approach has limitations for GraphQL
integration:
Introduce a two-layer architecture for GraphQL-based MCP tools:
Layer 1: GraphQL Tool Classes (Mcp::Tools::GraphqlTool)
GitlabSchemaLayer 2: Service Wrappers (Mcp::Tools::*Service < GraphqlService)
GraphqlService which provides user validation and GraphQL tool executionVersionable concernexecute_graphql_toolgraph TB
A[AI Client
Claude/Cursor] -->|MCP Request| B[GraphqlCreateIssueService
GraphqlService]
B -->|params, version| E[CreateIssueTool
GraphqlTool]
E -->|GraphQL Mutation| F[GitlabSchema.execute]
F -->|Response| E
E -->|Structured Result| B
B -->|MCP Response| A
style B fill:#e1f5fe
style E fill:#fff9c4
style F fill:#f3e5f5
1. MCP Client Request
↓
2. Service.execute(params)
├─ Validate current_user exists
├─ Call super (BaseService.execute)
│ ├─ Validate arguments against input_schema
│ └─ Call perform(arguments)
↓
3. perform method calls execute_graphql_tool(arguments)
├─ Build GraphQL query/mutation with static schema
├─ Transform params → GraphQL input variables
├─ Execute GitlabSchema.execute(graphql_operation, variables)
└─ Process result (success/errors)
↓
4. Format response
├─ Success: Response.success(message, payload)
└─ Error: Response.error(message)
↓
5. Return to MCP Client
File: app/services/mcp/tools/graphql_tool.rb
Purpose: Base class for GraphQL-based MCP tools that handles GraphQL execution, error processing, and versioning.
module Mcp
module Tools
class GraphqlTool
include Mcp::Tools::Concerns::Versionable
attr_reader :current_user, :params
def initialize(current_user:, params:, version: nil)
@current_user = current_user
@params = params
initialize_version(version)
end
# Override in subclasses or use version metadata
def graphql_operation
raise NotImplementedError unless self.class.version_metadata(version)[:graphql_operation]
self.class.version_metadata(version)[:graphql_operation]
end
def operation_name
self.class.version_metadata(version)[:operation_name] ||
raise(NotImplementedError, "operation_name must be defined")
end
# Can be overridden with version-specific methods
def build_variables
raise NotImplementedError, "build_variables must be implemented"
end
def execute
result = GitlabSchema.execute(
graphql_operation_for_version,
variables: build_variables_for_version,
context: execution_context
)
process_result(result)
end
private
def execution_context
{
current_user: current_user,
is_sessionless_user: false
}
end
def process_result(result)
# Handle GraphQL-level errors (syntax, validation, etc.)
if result['errors']
error_messages = extract_error_messages(result['errors'])
return ::Mcp::Tools::Response.error(error_messages.join(', '))
end
operation_data = result.dig('data', operation_name)
return ::Mcp::Tools::Response.error("Operation returned no data") if operation_data.nil?
# Check for operation-specific errors
operation_errors = operation_data['errors']
if operation_errors&.any?
error_messages = extract_error_messages(operation_errors)
return ::Mcp::Tools::Response.error(error_messages.join(', '))
end
formatted_content = [{ type: 'text', text: Gitlab::Json.dump(operation_data) }]
::Mcp::Tools::Response.success(formatted_content, operation_data)
end
def extract_error_messages(errors)
errors.map do |error|
if error.is_a?(String)
error
elsif error.is_a?(Hash)
error['message'] || error.to_s
else
error.to_s
end
end
end
end
end
end
Key Design Decisions:
Mcp::Tools::Response objects (success or error)File: app/services/mcp/tools/graphql_service.rb
Purpose: Provides a specialized base service for GraphQL-based MCP tools with user validation, versioning support, and GraphQL tool execution.
module Mcp
module Tools
class GraphqlService < BaseService
include Mcp::Tools::Concerns::Versionable
extend Gitlab::Utils::Override
def initialize(name:, version: nil)
super(name: name)
initialize_version(version)
end
override :set_cred
def set_cred(current_user: nil, access_token: nil)
@current_user = current_user
_ = access_token # access_token is not used in GraphqlService
end
override :execute
def execute(request: nil, params: nil)
return Response.error("#{self.class.name}: current_user is not set") unless current_user.present?
super
end
protected
# Subclasses should override this to return their GraphQL tool class
def graphql_tool_class
raise NotImplementedError, "#{self.class.name}#graphql_tool_class must be implemented"
end
# Default implementation - can be overridden in subclasses
def perform_default(_arguments = {})
raise NoMethodError, "No implementation found for version #{version}"
end
private
def execute_graphql_tool(arguments)
tool = graphql_tool_class.new(
current_user: current_user,
params: arguments,
version: version
)
tool.execute
end
end
end
end
Key Features:
register_versioncurrent_user is set before executionexecute_graphql_tool helper to instantiate and execute GraphQL toolsgraphql_tool_class and version-specific perform_X_Y_Z methodsFile: app/services/mcp/tools/concerns/versionable.rb
GraphQL-Specific Methods:
# Retrieve GraphQL operation from version metadata
def graphql_operation
version_metadata.fetch(:graphql_operation) do
raise NotImplementedError, "GraphQL operation not defined for version #{version}"
end
end
# Retrieve operation name from version metadata
def operation_name
version_metadata.fetch(:operation_name) do
raise NotImplementedError, "operation_name must be defined"
end
end
protected
# Get operation with fallback to method override
def graphql_operation_for_version
version_metadata[:graphql_operation] || graphql_operation
end
# Call version-specific build_variables method or fallback
def build_variables_for_version
method_name = "build_variables_#{version_method_suffix}"
respond_to?(method_name, true) ? send(method_name) : build_variables
end
Version-Specific Variable Building:
Tools can define version-specific variable building methods:
# Default implementation
def build_variables
{ input: { projectPath: params[:project_path] } }
end
# Version 2.0.0 specific implementation
def build_variables_2_0_0
{
input: {
projectPath: params[:project_path],
includeArchived: params[:include_archived]
}.compact
}
end
File: app/services/mcp/tools/graphql_create_issue_service.rb
Purpose: Provides input validation, MCP protocol compliance, and version management. Authorization is delegated to GraphQL layer.
module Mcp
module Tools
class GraphqlCreateIssueService < GraphqlService
# Register version 0.1.0 with metadata
register_version '0.1.0', {
description: 'Create a new issue in a GitLab project using GraphQL mutation',
input_schema: {
type: 'object',
properties: {
project_path: { type: 'string', description: 'Full project path or ID' },
title: { type: 'string', description: 'Issue title' },
description: { type: 'string', description: 'Issue description' }
},
required: ['project_path', 'title']
}
}
protected
# Specify which GraphQL tool class to use
def graphql_tool_class
Mcp::Tools::CreateIssueTool
end
# Version 0.1.0 implementation
def perform_0_1_0(arguments = {})
execute_graphql_tool(arguments)
end
# Fallback to 0.1.0 behavior for any unimplemented versions
override :perform_default
def perform_default(arguments = {})
perform_0_1_0(arguments)
end
end
end
end
Key Design Decisions:
graphql_tool_class to specify which tool to useexecute_graphql_tool(arguments) which handles tool instantiation and executionFile: app/services/mcp/tools/create_issue_tool.rb
Use Case: Create an issue with basic fields.
module Mcp
module Tools
class CreateIssueTool < GraphqlTool
# Register version with GraphQL operation in metadata
register_version '0.1.0', {
operation_name: 'createIssue',
graphql_operation: <<~GRAPHQL
mutation($input: CreateIssueInput!) {
createIssue(input: $input) {
issue {
id
iid
title
description
webUrl
state
}
errors
}
}
GRAPHQL
}
# Can register additional versions with different operations
register_version '0.2.0', {
operation_name: 'createIssue',
graphql_operation: <<~GRAPHQL
mutation($input: CreateIssueInput!) {
createIssue(input: $input) {
issue {
id
iid
title
description
webUrl
state
createdAt
updatedAt
}
errors
}
}
GRAPHQL
}
# Default variable building (used by v0.1.0)
def build_variables
{
input: {
projectPath: params[:project_path],
title: params[:title],
description: params[:description],
confidential: params[:confidential]
}.compact
}
end
private
# Version-specific variable building for v0.2.0
def build_variables_0_2_0
{
input: {
projectPath: params[:project_path],
title: params[:title],
description: params[:description],
confidential: params[:confidential],
labelIds: params[:label_ids]
}.compact
}
end
end
end
end
Composite tools combine multiple related operations into a single, cohesive MCP tool. Instead of creating separate tools for each different resources, a composite tool provides a unified interface with operation-specific parameters.
Important Limitation: You can only perform one mutation operation per tool invocation.
Benefits of composite tools:
Example use cases:
Implemented issues:
Authorization Flow:
service.execute(request:, params:)GraphqlService.execute validates current_user presenceBaseService.execute calls perform(arguments)perform method calls execute_graphql_tool(arguments)authorize directive in mutations/resolversGraphQL Context:
current_user: Set on every GraphQL executionis_sessionless_user: false: Marks as API/MCP requestThree Error Levels:
current_userResponse.error(message)Response.error(joined_messages)Response.error(joined_messages)Error Propagation:
GitlabSchema.execute → GraphqlTool.process_result →
GraphqlService.execute_graphql_tool → Response.error → MCP Client
Multiple Mutation Error Handling:
createIssue succeeds but updateIssue fails:
Response.error(...)"updateIssue: Title can't be blank"Step 1: Define GraphQL Tool Class
# app/services/mcp/tools/your_graphql_tool.rb
module Mcp
module Tools
class YourGraphqlTool < GraphqlTool
# Register version with operation in metadata
register_version '0.1.0', {
operation_name: 'yourMutation',
graphql_operation: <<~GRAPHQL
mutation($input: YourInput!) {
yourMutation(input: $input) {
result { id title }
errors
}
}
GRAPHQL
}
# Implement variable building
def build_variables
{
input: {
projectPath: params[:project_path],
title: params[:title]
}.compact
}
end
# Optional: Version-specific variable building
private
def build_variables_0_2_0
{
input: {
projectPath: params[:project_path],
title: params[:title],
extraField: params[:extra_field]
}.compact
}
end
end
end
end
Step 2: Create Service Wrapper
# app/services/mcp/tools/your_service.rb
module Mcp
module Tools
class YourService < GraphqlService
# Register version with metadata
register_version '0.1.0', {
description: 'Description of what this tool does',
input_schema: {
type: 'object',
properties: {
project_path: { type: 'string', description: '...' },
title: { type: 'string', description: '...' }
},
required: ['project_path', 'title']
}
}
protected
# Specify the GraphQL tool class to use
def graphql_tool_class
Mcp::Tools::YourGraphqlTool
end
# Version 0.1.0 implementation
def perform_0_1_0(arguments = {})
execute_graphql_tool(arguments)
end
# Fallback to 0.1.0 behavior for any unimplemented versions
override :perform_default
def perform_default(arguments = {})
perform_0_1_0(arguments)
end
end
end
end
Step 3: Register Tool in Manager
GraphQL tools are registered separately from custom tools in Mcp::Tools::Manager:
GRAPHQL_TOOLS = {
'your_tool_name' => ::Mcp::Tools::YourService
}.freeze
Step 3: Add Tests
ee/spec/services/ee/mcp/tools/manager_spec.rbspec/requests/api/mcp/handlers/list_tools_spec.rb and ee/spec/requests/api/mcp/handlers/list_tools_spec.rbApproach: Execute GraphQL directly in service classes without abstraction layer.
class GraphqlCreateIssueService < GraphqlService
def perform_0_1_0(params)
result = GitlabSchema.execute(MUTATION, variables: params, context: {...})
# Handle result inline
end
end
Pros:
Cons:
Decision: Rejected due to maintainability concerns.
Approach: Generic service that accepts arbitrary GraphQL queries from MCP clients.
class GraphqlProxyService < GraphqlService
def perform_0_1_0(params)
query = params[:query]
GitlabSchema.execute(query, variables: params[:variables], context: {...})
end
end
Pros:
Cons:
Decision: Rejected due to security and authorization concerns.
Approach: Automatically generate MCP tools by introspecting GraphQL schema.
Pros:
Cons:
Decision: We will consider this in the next iteration. Manual tool creation provides better control and documentation.
Query Complexity: GraphQL tools inherit GitLab query complexity limits (200 by default).
Caching: GraphQL resolver-level caching applies automatically.
Batch Loading: GraphQL's built-in batch loading prevents N+1 queries for nested fields.
Monitoring: All GraphQL executions logged via existing GraphqlLogger.