An invocation scenario is a way of testing SemanticKernel units, such as plugin functions and kernels, in skUnit. A scenario consists of providing some input to a unit, invoking it, and checking the expected output.
For example, suppose you have a GetSentiment
function that takes a text and returns its sentiment, such as "Happy" or "Sad".
You can test this function with different scenarios, such as:
## PARAMETER input
Such a beautiful day it is
## ANSWER Equals
Happy
This scenario verifies that the function returns "Happy" when the input is "Such a beautiful day it is".
If the function only has one parameter, you can omit the ## PARAMETER input
line and write the input directly, like this:
Such a beautiful day it is
## ANSWER Equals
Happy
This scenario is equivalent to the previous one.
You can use various features to define more complex scenarios. In this section, we will explain some of them with an example.
Let's say you have a function called GetSentiment
that takes two parameters and returns a sentence describing the sentiment of the text:
Parameters:
- input: the text to analyze
- options: the possible sentiment values, such as happy, angry, or sad
Returns: a sentence like "The sentiment is happy" or "The sentiment of this text is sad".
Here is a scenario that tests this function:
# SCENARIO GetSentimentHappy
## PARAMETER input
Such a beautiful day it is
## PARAMETER options
happy, angry
## ANSWER SemanticSimilar
The sentiment is happy
Let's break down this scenario line by line.
# SCENARIO GetSentimentHappy
This line gives a name or description to the scenario, which can be used later to generate xUnit theories.
## PARAMETER input
Such a beautiful day it is
## PARAMETER options
happy, angry
These lines define the values of the parameters that will be passed to the function, such as:
var actual = await function.InvokeAsync<string>(kernel,
arguments: new()
{
["input"] = "Such a beautiful day it is",
["options"] = "happy,angry"
});
## ANSWER SemanticSimilar
The sentiment is happy
This line specifies the expected output of the function and how to compare it with the actual output. In this case, the output should be semantically similar to "The sentiment is happy". This means that the output can have different words or syntax, but the meaning should be the same.
This is a powerful feature of skUnit scenarios, as it allows you to use OpenAI itself to perform semantic comparisons.
This assertion can also be written in an alternative style:
## ANSWER
The sentiment of the sentence is happy
## CHECK SemanticSimilar
The sentiment is happy
In this style, an expected answer is just written to serve as a reminder of the anticipated answer (which will not be used during assertions);
and then a ## CHECK SemanticSimilar
is used to explicitly perform the assertion.
However, SemanticSimilar
is not the only assertion method. There are many more assertion checks available (like SemanticCondition, Equals).
You can see the full list of CHECK statements here: CHECK Statement spec.
When defining skUnit statements such as # SCENARIO
, ## PARAMETER
, and so on, you have the freedom to use as many hashtags as you wish. There's no strict rule that mandates a specific count of hashtags for each statement. This flexibility allows you to format your markdown in a way that enhances readability for you. However, as a best practice, we suggest adhering to the recommended usage to maintain a clear and comprehensible hierarchy.
In certain uncommon instances, the data may contain skUnit expressions that could disrupt the parsing of the scenario. For instance, let's consider a scenario with two parameters: input
and options
. If the first parameter contains a markdown value that disrupts parsing, it could pose a problem:
# SCENARIO
## PARAMETER input
This block is a markdown itself that includes this exact section:
## PARAMETER hello
This could lead skUnit to mistakenly identify it as a `parameter`, which it isn't.
## PARAMETER options
happy, angry
To handle these exceptional cases, you can employ an identifier in your statements, like so:
# sk SCENARIO
## sk PARAMETER input
This block is a markdown itself that includes this exact section:
## PARAMETER hello
This could lead skUnit to mistakenly identify it as a `parameter`, which it isn't.
## sk PARAMETER options
happy, angry
In this example, we used sk
as the identifier. However, you can use any identifier of your choice, such as ~
, *
, etc. The parser will recognize whatever you use in the first statement as the unique identifier for the statements.