forked from microsoft/semantic-kernel
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathResourceTemplateDefinition.cs
100 lines (82 loc) · 3.35 KB
/
ResourceTemplateDefinition.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
// Copyright (c) Microsoft. All rights reserved.
using System.Text.RegularExpressions;
using Microsoft.SemanticKernel;
using ModelContextProtocol.Protocol.Types;
using ModelContextProtocol.Server;
namespace MCPServer.Resources;
/// <summary>
/// Represents a resource template definition.
/// </summary>
public sealed class ResourceTemplateDefinition
{
/// <summary>
/// The regular expression to match the resource template.
/// </summary>
private Regex? _regex = null;
/// <summary>
/// The kernel function to invoke the resource template handler.
/// </summary>
private KernelFunction? _kernelFunction = null;
/// <summary>
/// Gets or sets the MCP resource template.
/// </summary>
public required ResourceTemplate ResourceTemplate { get; init; }
/// <summary>
/// Gets or sets the handler for the MCP resource template.
/// </summary>
public required Delegate Handler { get; init; }
/// <summary>
/// Gets or sets the kernel instance to invoke the resource template handler.
/// If not provided, an instance registered in DI container will be used.
/// </summary>
public Kernel? Kernel { get; set; }
/// <summary>
/// Checks if the given Uri matches the resource template.
/// </summary>
/// <param name="uri">The Uri to check for match.</param>
public bool IsMatch(string uri)
{
return this.GetRegex().IsMatch(uri);
}
/// <summary>
/// Invokes the resource template handler.
/// </summary>
/// <param name="context">The MCP server context.</param>
/// <param name="cancellationToken">The cancellation token.</param>
/// <returns>The result of the invocation.</returns>
public async ValueTask<ReadResourceResult> InvokeHandlerAsync(RequestContext<ReadResourceRequestParams> context, CancellationToken cancellationToken)
{
this._kernelFunction ??= KernelFunctionFactory.CreateFromMethod(this.Handler);
this.Kernel
??= context.Server.Services?.GetRequiredService<Kernel>()
?? throw new InvalidOperationException("Kernel is not available.");
KernelArguments args = new(source: this.GetArguments(context.Params!.Uri!))
{
{ "context", context },
};
FunctionResult result = await this._kernelFunction.InvokeAsync(kernel: this.Kernel, arguments: args, cancellationToken: cancellationToken);
return result.GetValue<ReadResourceResult>() ?? throw new InvalidOperationException("The handler did not return a valid result.");
}
private Regex GetRegex()
{
if (this._regex != null)
{
return this._regex;
}
var pattern = "^" +
Regex.Escape(this.ResourceTemplate.UriTemplate)
.Replace("\\{", "(?<")
.Replace("}", ">[^/]+)") +
"$";
return this._regex = new(pattern, RegexOptions.Compiled);
}
private Dictionary<string, object?> GetArguments(string uri)
{
var match = this.GetRegex().Match(uri);
if (!match.Success)
{
throw new ArgumentException($"The uri '{uri}' does not match the template '{this.ResourceTemplate.UriTemplate}'.");
}
return match.Groups.Cast<Group>().Where(g => g.Name != "0").ToDictionary(g => g.Name, g => (object?)g.Value);
}
}