mirror of
https://github.com/pomerium/pomerium.git
synced 2025-07-17 16:48:13 +02:00
## Summary Adds support for extending authorization log with Model Context Protocol details. i.e. ```json { "level": "info", "server-name": "all", "service": "authorize", "mcp-method": "tools/call", "mcp-tool": "describe_table", "mcp-tool-parameters": { "table_name": "Categories" }, "allow": true, "allow-why-true": ["email-ok", "mcp-tool-ok"], "deny": false, "deny-why-false": [], "time": "2025-06-24T17:40:41-04:00", "message": "authorize check" } ``` ## Related issues Fixes https://linear.app/pomerium/issue/ENG-2393/mcp-authorize-each-incoming-request-to-an-mcp-route ## User Explanation <!-- How would you explain this change to the user? If this change doesn't create any user-facing changes, you can leave this blank. If filled out, add the `docs` label --> ## Checklist - [x] reference any related issues - [x] updated unit tests - [x] add appropriate label (`enhancement`, `bug`, `breaking`, `dependencies`, `ci`) - [x] ready for review
60 lines
1.4 KiB
Go
60 lines
1.4 KiB
Go
package criteria
|
|
|
|
import (
|
|
"github.com/open-policy-agent/opa/ast"
|
|
|
|
"github.com/pomerium/pomerium/pkg/policy/generator"
|
|
"github.com/pomerium/pomerium/pkg/policy/parser"
|
|
)
|
|
|
|
type mcpToolCriterion struct {
|
|
g *Generator
|
|
}
|
|
|
|
func (mcpToolCriterion) DataType() CriterionDataType {
|
|
return CriterionDataTypeStringMatcher
|
|
}
|
|
|
|
func (mcpToolCriterion) Name() string {
|
|
return "mcp_tool"
|
|
}
|
|
|
|
func (c mcpToolCriterion) GenerateRule(_ string, data parser.Value) (*ast.Rule, []*ast.Rule, error) {
|
|
r1 := c.g.NewRule(c.Name())
|
|
r1.Head.Value = NewCriterionTerm(true, ReasonMCPNotAToolCall)
|
|
r1.Body = ast.Body{
|
|
ast.MustParseExpr(`input.mcp.method != "tools/call"`),
|
|
}
|
|
|
|
r2 := &ast.Rule{
|
|
Head: generator.NewHead("", NewCriterionTerm(true, ReasonMCPToolOK)),
|
|
Body: ast.Body{
|
|
ast.MustParseExpr(`input.mcp.method == "tools/call"`),
|
|
},
|
|
}
|
|
toolRef := ast.RefTerm(ast.VarTerm("input"), ast.VarTerm("mcp"), ast.VarTerm("tool_call"), ast.VarTerm("name"))
|
|
err := matchString(&r2.Body, toolRef, data)
|
|
if err != nil {
|
|
return nil, nil, err
|
|
}
|
|
r1.Else = r2
|
|
|
|
r3 := &ast.Rule{
|
|
Head: generator.NewHead("", NewCriterionTerm(false, ReasonMCPToolUnauthorized)),
|
|
Body: ast.Body{
|
|
ast.NewExpr(ast.BooleanTerm(true)),
|
|
},
|
|
}
|
|
r2.Else = r3
|
|
|
|
return r1, nil, nil
|
|
}
|
|
|
|
// MCPTool returns a Criterion which matches an MCP tool name.
|
|
func MCPTool(generator *Generator) Criterion {
|
|
return mcpToolCriterion{g: generator}
|
|
}
|
|
|
|
func init() {
|
|
Register(MCPTool)
|
|
}
|