docs/decisions/0060-jsos-integration.md
Today, SK relies on JSON serialization and schema generation functionality to generate schemas for function parameters and return types, deserialize them from JSON to the target types as part of the marshaling process, serialize AI models to SK and back, etc.
At the moment, the serialization code either uses no JsonSerializerOptions (JSOs) or uses hardcoded predefined ones for specific purposes without the ability to provide custom ones. This works perfectly fine for non-AOT scenarios where JSON serialization uses reflection by default. However, in Native AOT apps, which do not support all required reflection APIs, reflection-based serialization won't work and will crash.
To enable serialization for Native-AOT scenarios, all serialization code should use source-generated context contracts represented by the JsonSerializerContext base class. See the article How to use source generation in System.Text.Json for more details. Additionally, there should be a way to supply those source-generated classes via the SK public API surface down to the JSON serialization functionality.
This ADR outlines potential options for passing JSOs with configured source-generated contracts down to the JSON serialization code of Native-AOT enabled SK components.
This options presumes adding the new JsonSerializerOptions property of JsonSerializerOptions type to Kernel class. All external source-generated context contracts will be registered there and all SK components requiring JSOs will resolve them from there:
public sealed class MyPlugin { public Order CreateOrder() => new(); }
public sealed class Order { public string? Number { get; set; } }
[JsonSerializable(typeof(Order))]
internal sealed partial class OrderJsonSerializerContext : JsonSerializerContext
{
}
public async Task TestAsync()
{
JsonSerializerOptions options = new JsonSerializerOptions();
options.TypeInfoResolverChain.Add(OrderJsonSerializerContext.Default);
Kernel kernel = new Kernel();
kernel.JsonSerializerOptions = options;
// All the following Kernel extension methods use JSOs configured on the `Kernel.JsonSerializerOptions` property
kernel.CreateFunctionFromMethod(() => new Order());
kernel.CreateFunctionFromPrompt("<prompt>");
kernel.CreatePluginFromFunctions("<plugin>", [kernel.CreateFunctionFromMethod(() => new Order())]);
kernel.CreatePluginFromType<MyPlugin>("<plugin>");
kernel.CreatePluginFromPromptDirectory("<directory>", "<plugin>");
kernel.CreatePluginFromObject(new MyPlugin(), "<plugin>");
// AI connectors can use the `Kernel.JsonSerializerOptions` property as well
var onnxService = new OnnxRuntimeGenAIChatCompletionService("<modelId>", "<modelPath>");
var res = await onnxService.GetChatMessageContentsAsync(new ChatHistory(), new PromptExecutionSettings(), kernel);
// The APIs below can't use the `Kernel.JsonSerializerOptions` property because they don't have access to the `Kernel` instance
KernelFunctionFactory.CreateFromMethod(() => new Order(), options);
KernelFunctionFactory.CreateFromPrompt("<prompt>", options);
KernelPluginFactory.CreateFromObject(new MyPlugin(), options, "<plugin>");
KernelPluginFactory.CreateFromType<MyPlugin>(options, "<plugin>");
KernelPluginFactory.CreateFromFunctions("<plugin>", [kernel.CreateFunctionFromMethod(() => new Order())]);
}
Pros:
Cons:
Via Kernel constructor.
private readonly JsonSerializerOptions? _serializerOptions = null;
// Existing AOT incompatible constructor
[RequiresUnreferencedCode("Uses reflection to handle various aspects of JSON serialization in SK, making it incompatible with AOT scenarios.")]
[RequiresDynamicCode("Uses reflection to handle various aspects of JSON serialization in SK, making it incompatible with AOT scenarios.")]
public Kernel(IServiceProvider? services = null,KernelPluginCollection? plugins = null) {}
// New AOT compatible constructor
public Kernel(JsonSerializerOptions jsonSerializerOptions, IServiceProvider? services = null,KernelPluginCollection? plugins = null)
{
this._serializerOptions = jsonSerializerOptions;
this._serializerOptions.MakeReadOnly(); // Prevent mutations that may not be picked up by SK components created with initial JSOs.
}
public JsonSerializerOptions JsonSerializerOptions => this._serializerOptions ??= JsonSerializerOptions.Default;
Pros:
Via the Kernel.JsonSerializerOptions property setter
private readonly JsonSerializerOptions? _serializerOptions = null;
public JsonSerializerOptions JsonSerializerOptions
{
get
{
return this._serializerOptions ??= ??? // JsonSerializerOptions.Default will work for non-AOT scenarios and will fail in AOT ones.
}
set
{
this._serializerOptions = value;
}
}
Cons:
DI TBD after requirements are fleshed out.
This option presumes supplying JSOs at the component's instantiation site or constructor:
public sealed class Order { public string? Number { get; set; } }
[JsonSerializable(typeof(Order))]
internal sealed partial class OrderJsonSerializerContext : JsonSerializerContext
{
}
JsonSerializerOptions options = new JsonSerializerOptions();
options.TypeInfoResolverChain.Add(OrderJsonSerializerContext.Default);
// All the following kernel extension methods accept JSOs explicitly supplied as an argument for the corresponding parameter:
kernel.CreateFunctionFromMethod(() => new Order(), options);
kernel.CreateFunctionFromPrompt("<prompt>", options);
kernel.CreatePluginFromFunctions("<plugin>", [kernel.CreateFunctionFromMethod(() => new Order(), options)]);
kernel.CreatePluginFromType<MyPlugin>("<plugin>", options);
kernel.CreatePluginFromPromptDirectory("<directory>", "<plugin>", options);
kernel.CreatePluginFromObject(new MyPlugin(), "<plugin>", options);
// The AI connectors accept JSOs at the instantiation site rather than at the invocation site.
var onnxService = new OnnxRuntimeGenAIChatCompletionService("<modelId>", "<modelPath>", options);
var res = await onnxService.GetChatMessageContentsAsync(new ChatHistory(), new PromptExecutionSettings());
// The APIs below already accept JSOs at the instantiation site.
KernelFunctionFactory.CreateFromMethod(() => new Order(), options);
KernelFunctionFactory.CreateFromPrompt("<prompt>", options);
KernelPluginFactory.CreateFromObject(new MyPlugin(), options, "<plugin>");
KernelPluginFactory.CreateFromType<MyPlugin>(options, "<plugin>");
KernelPluginFactory.CreateFromFunctions("<plugin>", [kernel.CreateFunctionFromMethod(() => new Order())]);
Pros:
Cons:
AI connectors may accept JSOs as a parameter in the constructor or as an optional property. The decision will be made when one or a few connectors are refactored to be AOT compatible.
This option presumes supplying JSOs at component operation invocation sites rather than at instantiation sites.
Pros:
Cons:
The "Option #2 JSOs per SK component" was preferred over the other options since it provides an explicit, unified, clear, simple, and effective way of supplying JSOs at the component's instantiation/creation sites.