所以他们在EchoBot示例中有一个很好的例子来演示链
public static readonly IDialog<string> dialog = Chain.PostToChain()
.Select(msg => msg.Text)
.Switch(
new Case<string, IDialog<string>>(text =>
{
var regex = new Regex("^reset");
return regex.Match(text).Success;
}, (context, txt) =>
{
return Chain.From(() => new PromptDialog.PromptConfirm("Are you sure you want to reset the count?",
"Didn't get that!", 3)).ContinueWith<bool, string>(async (ctx, res) =>
{
string reply;
if (await res)
{
ctx.UserData.SetValue("count", 0);
reply = "Reset count.";
}
else
{
reply = "Did not reset count.";
}
return Chain.Return(reply);
});
}),
new RegexCase<IDialog<string>>(new Regex("^help", RegexOptions.IgnoreCase), (context, txt) =>
{
return Chain.Return("I am a simple echo dialog with a counter! Reset my counter by typing \"reset\"!");
}),
new DefaultCase<string, IDialog<string>>((context, txt) =>
{
int count;
context.UserData.TryGetValue("count", out count);
context.UserData.SetValue("count", ++count);
string reply = string.Format("{0}: You said {1}", count, txt);
return Chain.Return(reply);
}))
.Unwrap()
.PostToUser();
}
然而,我宁愿使用LUIS Intent而不是使用REGEX来确定我的会话路径。我正在使用这段漂亮的代码来提取LUIS Intent。
public static async Task<LUISQuery> ParseUserInput(string strInput)
{
string strRet = string.Empty;
string strEscaped = Uri.EscapeDataString(strInput);
using (var client = new HttpClient())
{
string uri = Constants.Keys.LUISQueryUrl + strEscaped;
HttpResponseMessage msg = await client.GetAsync(uri);
if (msg.IsSuccessStatusCode)
{
var jsonResponse = await msg.Content.ReadAsStringAsync();
var _Data = JsonConvert.DeserializeObject<LUISQuery>(jsonResponse);
return _Data;
}
}
return null;
}
现在不幸的是,因为这是异步的,所以它不能很好地运行LINQ查询来运行case语句。任何人都可以为我提供一些代码,允许我根据LUIS Intents在我的链中包含一个case语句吗?
答案 0 :(得分:0)
Omg在他的评论中是正确的。
请记住,IDialogs有一个TYPE,意思是,IDialog可以返回一个你自己指定类型的对象:
public class TodoItemDialog : IDialog<TodoItem>
{
// Somewhere, you'll call this to end the dialog
public async Task FinishAsync(IDialogContext context, IMessageActivity activity)
{
var todoItem = _itemRepository.GetItemByTitle(activity.Text);
context.Done(todoItem);
}
}
对 context.Done()的调用会返回您的Dialog要返回的对象。当你正在阅读任何类型的IDialog
的类声明时public class TodoItemDialog : LuisDialog<TodoItem>
有助于将其读作:
&#34; TodoItemDialog是一个Dialog类,它在TodoItem完成后返回TodoItem&#34;
您可以使用 context.Forward()而不是链接,它基本上将相同的messageActivity转发到另一个对话框类。
context.Forward()和 context.Call()之间的区别本质上是 context.Forward()允许您转发一个 messageActivity ,由对话框立即处理,而 context.Call()只是启动一个新的对话框,而不会移交任何内容。
来自您的&#34; Root&#34;对话框,如果您需要使用LUIS来确定意图并返回特定对象,您只需使用转发 向其转发 messageActivity,然后处理结果指定的回调:
await context.Forward(new TodoItemDialog(), AfterTodoItemDialogAsync, messageActivity, CancellationToken.None);
private async Task AfterTodoItemDialogAsync(IDialogContext context, IAwaitable<TodoItem> result)
{
var receivedTodoItem = await result;
// Continue conversation
}
最后,您的LuisDialog类看起来像这样:
[Serializable, LuisModel("[ModelID]", "[SubscriptionKey]")]
public class TodoItemDialog : LuisDialog<TodoItem>
{
[LuisIntent("GetTodoItem")]
public async Task GetTodoItem(IDialogContext context, LuisResult result)
{
await context.PostAsync("Working on it, give me a moment...");
result.TryFindEntity("TodoItemText", out EntityRecommendation entity);
if(entity.Score > 0.9)
{
var todoItem = _todoItemRepository.GetByText(entity.Entity);
context.Done(todoItem);
}
}
}
(为简洁起见,我在示例中没有ELSE语句,这当然是您需要添加的内容)