forked from microsoft/semantic-kernel
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMethodFunctions_Advanced.cs
148 lines (123 loc) · 5.15 KB
/
MethodFunctions_Advanced.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
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
// Copyright (c) Microsoft. All rights reserved.
using System.ComponentModel;
using System.Globalization;
using System.Reflection;
using System.Text.Json;
using Microsoft.SemanticKernel;
namespace Functions;
/// <summary>
/// These samples show advanced usage of method functions.
/// </summary>
public class MethodFunctions_Advanced(ITestOutputHelper output) : BaseTest(output)
{
/// <summary>
/// This example executes Function1, which in turn executes Function2.
/// </summary>
[Fact]
public async Task MethodFunctionsChaining()
{
Console.WriteLine("Running Method Function Chaining example...");
var kernel = new Kernel();
var functions = kernel.ImportPluginFromType<Plugin>();
var customType = await kernel.InvokeAsync<MyCustomType>(functions["Function1"]);
Console.WriteLine($"CustomType.Number: {customType!.Number}"); // 2
Console.WriteLine($"CustomType.Text: {customType.Text}"); // From Function1 + From Function2
}
/// <summary>
/// This example shows how to access the custom <see cref="InvocationSettingsAttribute"/> attribute the underlying method wrapped by Kernel Function is annotated with.
/// </summary>
[Fact]
public async Task AccessUnderlyingMethodAttributes()
{
// Import the plugin containing the method with the InvocationSettingsAttribute custom attribute
var kernel = new Kernel();
var functions = kernel.ImportPluginFromType<Plugin>();
// Get the kernel function wrapping the method with the InvocationSettingsAttribute
var kernelFunction = functions[nameof(Plugin.FunctionWithInvocationSettingsAttribute)];
// Access the custom attribute the underlying method is annotated with
var invocationSettingsAttribute = kernelFunction.UnderlyingMethod!.GetCustomAttribute<InvocationSettingsAttribute>();
Console.WriteLine($"Priority: {invocationSettingsAttribute?.Priority}");
}
private sealed class Plugin
{
private const string PluginName = nameof(Plugin);
[KernelFunction]
public async Task<MyCustomType> Function1Async(Kernel kernel)
{
// Execute another function
var value = await kernel.InvokeAsync<MyCustomType>(PluginName, "Function2");
return new MyCustomType
{
Number = 2 * value?.Number ?? 0,
Text = "From Function1 + " + value?.Text
};
}
[KernelFunction]
public static MyCustomType Function2()
{
return new MyCustomType
{
Number = 1,
Text = "From Function2"
};
}
[KernelFunction, InvocationSettingsAttribute(priority: Priority.High)]
public static void FunctionWithInvocationSettingsAttribute()
{
}
}
/// <summary>
/// In order to use custom types, <see cref="TypeConverter"/> should be specified,
/// that will convert object instance to string representation.
/// </summary>
/// <remarks>
/// <see cref="TypeConverter"/> is used to represent complex object as meaningful string, so
/// it can be passed to AI for further processing using prompt functions.
/// It's possible to choose any format (e.g. XML, JSON, YAML) to represent your object.
/// </remarks>
[TypeConverter(typeof(MyCustomTypeConverter))]
private sealed class MyCustomType
{
public int Number { get; set; }
public string? Text { get; set; }
}
/// <summary>
/// Implementation of <see cref="TypeConverter"/> for <see cref="MyCustomType"/>.
/// In this example, object instance is serialized with <see cref="JsonSerializer"/> from System.Text.Json,
/// but it's possible to convert object to string using any other serialization logic.
/// </summary>
private sealed class MyCustomTypeConverter : TypeConverter
{
public override bool CanConvertFrom(ITypeDescriptorContext? context, Type sourceType) => true;
/// <summary>
/// This method is used to convert object from string to actual type. This will allow to pass object to
/// method function which requires it.
/// </summary>
public override object? ConvertFrom(ITypeDescriptorContext? context, CultureInfo? culture, object value)
{
return JsonSerializer.Deserialize<MyCustomType>((string)value);
}
/// <summary>
/// This method is used to convert actual type to string representation, so it can be passed to AI
/// for further processing.
/// </summary>
public override object? ConvertTo(ITypeDescriptorContext? context, CultureInfo? culture, object? value, Type destinationType)
{
return JsonSerializer.Serialize(value);
}
}
[AttributeUsage(AttributeTargets.Method)]
private sealed class InvocationSettingsAttribute : Attribute
{
public InvocationSettingsAttribute(Priority priority = Priority.Normal)
{
this.Priority = priority;
}
public Priority Priority { get; }
}
private enum Priority
{
Normal,
High,
}
}