feat(openai): 新增 v2 版机器人对话相关接口

This commit is contained in:
Fu Diwei 2024-06-05 19:39:23 +08:00
parent ea25c0bf24
commit 1aed793b08
10 changed files with 289 additions and 134 deletions

View File

@ -73,5 +73,27 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI
return await client.SendFlurlRequestAsJsonAsync<Models.BotEffectiveProgressV2Response>(flurlReq, data: request, cancellationToken: cancellationToken).ConfigureAwait(false);
}
/// <summary>
/// <para>异步调用 [POST] /v2/bot/query 接口。</para>
/// <para>
/// REF: <br/>
/// <![CDATA[ https://developers.weixin.qq.com/doc/aispeech/confapi/dialog/bot/query.html ]]>
/// </para>
/// </summary>
/// <param name="client"></param>
/// <param name="request"></param>
/// <param name="cancellationToken"></param>
/// <returns></returns>
public static async Task<Models.BotQueryV2Response> ExecuteBotQueryV2Async(this WechatOpenAIClient client, Models.BotQueryV2Request request, CancellationToken cancellationToken = default)
{
if (client is null) throw new ArgumentNullException(nameof(client));
if (request is null) throw new ArgumentNullException(nameof(request));
IFlurlRequest flurlReq = client
.CreateFlurlRequest(request, HttpMethod.Post, "v2", "bot", "query");
return await client.SendFlurlRequestAsJsonAsync<Models.BotQueryV2Response>(flurlReq, data: request, cancellationToken: cancellationToken).ConfigureAwait(false);
}
}
}

View File

@ -9,6 +9,7 @@ using Flurl.Http;
namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Interceptors
{
using SKIT.FlurlHttpClient.Internal;
using SKIT.FlurlHttpClient.Primitives;
internal class WechatOpenAIRequestEncryptionInterceptor : HttpInterceptor
{
@ -90,19 +91,19 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Interceptors
if (context.FlurlCall.HttpResponseMessage.StatusCode != HttpStatusCode.OK)
return;
byte[] respBytes = Array.Empty<byte>();
string respBody = string.Empty;
if (context.FlurlCall.HttpResponseMessage.Content is not null)
{
HttpContent httpContent = context.FlurlCall.HttpResponseMessage.Content;
respBytes = await
respBody = await
#if NET5_0_OR_GREATER
httpContent.ReadAsByteArrayAsync(cancellationToken).ConfigureAwait(false);
httpContent.ReadAsStringAsync(cancellationToken).ConfigureAwait(false);
#else
_AsyncEx.RunTaskWithCancellationTokenAsync(httpContent.ReadAsByteArrayAsync(), cancellationToken).ConfigureAwait(false);
_AsyncEx.RunTaskWithCancellationTokenAsync(httpContent.ReadAsStringAsync(), cancellationToken).ConfigureAwait(false);
#endif
}
byte[] respBytesDecrypted;
string respBodyDecrypted;
try
{
const int AES_BLOCK_SIZE = 16;
@ -110,10 +111,10 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Interceptors
byte[] ivBytes = new byte[AES_BLOCK_SIZE]; // iv 是 key 的前 16 个字节
Buffer.BlockCopy(keyBytes, 0, ivBytes, 0, ivBytes.Length);
respBytesDecrypted = Utilities.AESUtility.DecryptWithCBC(
keyBytes: keyBytes,
ivBytes: ivBytes,
cipherBytes: respBytes
respBodyDecrypted = Utilities.AESUtility.DecryptWithCBC(
encodingKey: EncodedString.ToBase64String(keyBytes),
encodingIV: EncodedString.ToBase64String(ivBytes),
encodingCipher: new EncodedString(respBody, EncodingKinds.Base64)
)!;
}
catch (Exception ex)
@ -122,7 +123,7 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Interceptors
}
context.FlurlCall.HttpResponseMessage!.Content?.Dispose();
context.FlurlCall.HttpResponseMessage!.Content = new ByteArrayContent(respBytesDecrypted);
context.FlurlCall.HttpResponseMessage!.Content = new StringContent(respBodyDecrypted);
}
private string GetRequestUrlPath(Uri uri)
@ -144,7 +145,7 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Interceptors
return false;
}
return false;
return true;
}
}
}

View File

@ -33,101 +33,6 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Models
public bool IsLatestValid { get; set; }
}
public class Message
{
/// <summary>
/// 获取或设置技能 ID。
/// </summary>
[Newtonsoft.Json.JsonProperty("ans_node_id")]
[System.Text.Json.Serialization.JsonPropertyName("ans_node_id")]
[System.Text.Json.Serialization.JsonNumberHandling(System.Text.Json.Serialization.JsonNumberHandling.AllowReadingFromString)]
public int AnswerNodeId { get; set; }
/// <summary>
/// 获取或设置技能名称。
/// </summary>
[Newtonsoft.Json.JsonProperty("ans_node_name")]
[System.Text.Json.Serialization.JsonPropertyName("ans_node_name")]
public string AnswerNodeName { get; set; } = default!;
/// <summary>
/// 获取或设置置信度。
/// </summary>
[Newtonsoft.Json.JsonProperty("confidence")]
[System.Text.Json.Serialization.JsonPropertyName("confidence")]
public decimal Confidence { get; set; } = default!;
/// <summary>
/// 获取或设置消息类型。
/// </summary>
[Newtonsoft.Json.JsonProperty("msg_type")]
[System.Text.Json.Serialization.JsonPropertyName("msg_type")]
public string MessageType { get; set; } = default!;
/// <summary>
/// 获取或设置消息内容。
/// </summary>
[Newtonsoft.Json.JsonProperty("content")]
[System.Text.Json.Serialization.JsonPropertyName("content")]
public string Content { get; set; } = default!;
/// <summary>
/// 获取或设置消息状态。
/// </summary>
[Newtonsoft.Json.JsonProperty("status")]
[System.Text.Json.Serialization.JsonPropertyName("status")]
public string Status { get; set; } = default!;
/// <summary>
/// 获取或设置是否是列表选择。
/// </summary>
[Newtonsoft.Json.JsonProperty("list_options")]
[System.Text.Json.Serialization.JsonPropertyName("list_options")]
public bool IsListOptions { get; set; }
/// <summary>
/// 获取或设置是否仅选择。
/// </summary>
[Newtonsoft.Json.JsonProperty("take_options_only")]
[System.Text.Json.Serialization.JsonPropertyName("take_options_only")]
public bool IsTakeOptionsOnly { get; set; }
/// <summary>
/// 获取或设置当前事件。
/// </summary>
[Newtonsoft.Json.JsonProperty("event")]
[System.Text.Json.Serialization.JsonPropertyName("event")]
public string? Event { get; set; }
/// <summary>
/// 获取或设置调试信息。
/// </summary>
[Newtonsoft.Json.JsonProperty("debug_info")]
[System.Text.Json.Serialization.JsonPropertyName("debug_info")]
public string? DebugInfo { get; set; }
/// <summary>
/// 获取或设置窗口标题。
/// </summary>
[Newtonsoft.Json.JsonProperty("resp_title")]
[System.Text.Json.Serialization.JsonPropertyName("resp_title")]
public string? ResponseTitle { get; set; }
/// <summary>
/// 获取或设置场景状态。
/// </summary>
[Newtonsoft.Json.JsonProperty("scene_status")]
[System.Text.Json.Serialization.JsonPropertyName("scene_status")]
public string? SceneStatus { get; set; }
/// <summary>
/// 获取或设置会话 ID。
/// </summary>
[Newtonsoft.Json.JsonProperty("session_id")]
[System.Text.Json.Serialization.JsonPropertyName("session_id")]
public string? SessionId { get; set; }
}
public class Option
{
/// <summary>
@ -208,14 +113,14 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Models
/// </summary>
[Newtonsoft.Json.JsonProperty("norm")]
[System.Text.Json.Serialization.JsonPropertyName("norm")]
public string? Norm { get; set; }
public string? NormalizedValue { get; set; }
/// <summary>
/// 获取或设置归一化的值详细信息。
/// </summary>
[Newtonsoft.Json.JsonProperty("norm_detail")]
[System.Text.Json.Serialization.JsonPropertyName("norm_detail")]
public string? NormDetail { get; set; }
public string? NormalizedValueDetail { get; set; }
/// <summary>
/// 获取或设置回复的意图名称。
@ -311,14 +216,6 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Models
[System.Text.Json.Serialization.JsonPropertyName("msgtype")]
public string MessageType { get; set; } = default!;
/// <summary>
/// 获取或设置回答详细信息。
/// </summary>
[Obsolete("相关接口或字段于 2022-04-15 下线。")]
[Newtonsoft.Json.JsonProperty("msg")]
[System.Text.Json.Serialization.JsonPropertyName("msg")]
public Types.Message[]? MessageList { get; set; }
/// <summary>
/// 获取或设置机器人回复的状态。
/// </summary>

View File

@ -0,0 +1,60 @@
using System.Collections.Generic;
namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Models
{
/// <summary>
/// <para>表示 [POST] /v2/bot/query 接口的请求。</para>
/// </summary>
public class BotQueryV2Request : WechatOpenAIRequest
{
/// <summary>
/// 获取或设置查询语句。
/// </summary>
[Newtonsoft.Json.JsonProperty("query")]
[System.Text.Json.Serialization.JsonPropertyName("query")]
public string QueryString { get; set; } = string.Empty;
/// <summary>
/// 获取或设置查询环境。
/// <para>默认值:"online"</para>
/// </summary>
[Newtonsoft.Json.JsonProperty("env")]
[System.Text.Json.Serialization.JsonPropertyName("env")]
public string Environment { get; set; } = "online";
/// <summary>
/// 获取或设置第一优先级的限定技能命中范围列表。
/// </summary>
[Newtonsoft.Json.JsonProperty("first_priority_skills")]
[System.Text.Json.Serialization.JsonPropertyName("first_priority_skills")]
public IList<string>? FirstPrioritySkillList { get; set; }
/// <summary>
/// 获取或设置第二优先级的限定技能命中范围列表。
/// </summary>
[Newtonsoft.Json.JsonProperty("second_priority_skills")]
[System.Text.Json.Serialization.JsonPropertyName("second_priority_skills")]
public IList<string>? SecondPrioritySkillList { get; set; }
/// <summary>
/// 获取或设置用户标识。
/// </summary>
[Newtonsoft.Json.JsonProperty("userid")]
[System.Text.Json.Serialization.JsonPropertyName("userid")]
public string? UserId { get; set; }
/// <summary>
/// 获取或设置用户昵称。
/// </summary>
[Newtonsoft.Json.JsonProperty("user_name")]
[System.Text.Json.Serialization.JsonPropertyName("user_name")]
public string? UserName { get; set; }
/// <summary>
/// 获取或设置用户头像 URL。
/// </summary>
[Newtonsoft.Json.JsonProperty("avatar")]
[System.Text.Json.Serialization.JsonPropertyName("avatar")]
public string? AvatarUrl { get; set; }
}
}

View File

@ -0,0 +1,128 @@
namespace SKIT.FlurlHttpClient.Wechat.OpenAI.Models
{
/// <summary>
/// <para>表示 [POST] /v2/bot/query 接口的响应。</para>
/// </summary>
public class BotQueryV2Response : WechatOpenAIResponse<BotQueryV2Response.Types.Data>
{
public static class Types
{
public class Data : WechatOpenAIResponseData
{
public static class Types
{
public class Option
{
/// <summary>
/// 获取或设置推荐分类。
/// </summary>
[Newtonsoft.Json.JsonProperty("ans_node_name")]
[System.Text.Json.Serialization.JsonPropertyName("ans_node_name")]
public string AnswerNodeName { get; set; } = default!;
/// <summary>
/// 获取或设置推荐标准问题。
/// </summary>
[Newtonsoft.Json.JsonProperty("title")]
[System.Text.Json.Serialization.JsonPropertyName("title")]
public string Title { get; set; } = default!;
/// <summary>
/// 获取或设置推荐回答。
/// </summary>
[Newtonsoft.Json.JsonProperty("answer")]
[System.Text.Json.Serialization.JsonPropertyName("answer")]
public string Answer { get; set; } = default!;
/// <summary>
/// 获取或设置推荐信息指数。
/// </summary>
[Newtonsoft.Json.JsonProperty("confidence")]
[System.Text.Json.Serialization.JsonPropertyName("confidence")]
public decimal Confidence { get; set; }
}
public class Slot
{
/// <summary>
/// 获取或设置槽位名称。
/// </summary>
[Newtonsoft.Json.JsonProperty("name")]
[System.Text.Json.Serialization.JsonPropertyName("name")]
public string Name { get; set; } = default!;
/// <summary>
/// 获取或设置槽位值。
/// </summary>
[Newtonsoft.Json.JsonProperty("value")]
[System.Text.Json.Serialization.JsonPropertyName("value")]
public string Value { get; set; } = default!;
/// <summary>
/// 获取或设置归一化的值。
/// </summary>
[Newtonsoft.Json.JsonProperty("norm")]
[System.Text.Json.Serialization.JsonPropertyName("norm")]
public string? NormalizedValue { get; set; }
}
}
/// <summary>
/// 获取或设置回答的类型。
/// </summary>
[Newtonsoft.Json.JsonProperty("answer_type")]
[System.Text.Json.Serialization.JsonPropertyName("answer_type")]
public string AnswerType { get; set; } = default!;
/// <summary>
/// 获取或设置命中的回答。
/// </summary>
[Newtonsoft.Json.JsonProperty("answer")]
[System.Text.Json.Serialization.JsonPropertyName("answer")]
public string Answer { get; set; } = default!;
/// <summary>
/// 获取或设置机器人回复状态。
/// </summary>
[Newtonsoft.Json.JsonProperty("status")]
[System.Text.Json.Serialization.JsonPropertyName("status")]
public string Status { get; set; } = default!;
/// <summary>
/// 获取或设置消息 ID。
/// </summary>
[Newtonsoft.Json.JsonProperty("msg_id")]
[System.Text.Json.Serialization.JsonPropertyName("msg_id")]
public string MessageId { get; set; } = default!;
/// <summary>
/// 获取或设置技能名称。
/// </summary>
[Newtonsoft.Json.JsonProperty("skill_name")]
[System.Text.Json.Serialization.JsonPropertyName("skill_name")]
public string? SkillName { get; set; }
/// <summary>
/// 获取或设置意图名称。
/// </summary>
[Newtonsoft.Json.JsonProperty("intent_name")]
[System.Text.Json.Serialization.JsonPropertyName("intent_name")]
public string? IntentName { get; set; }
/// <summary>
/// 获取或设置推荐问题列表。
/// </summary>
[Newtonsoft.Json.JsonProperty("options")]
[System.Text.Json.Serialization.JsonPropertyName("options")]
public Types.Option[]? OptionList { get; set; }
/// <summary>
/// 获取或设置槽位数据列表。
/// </summary>
[Newtonsoft.Json.JsonProperty("slots")]
[System.Text.Json.Serialization.JsonPropertyName("slots")]
public Types.Slot[]? SlotList { get; set; }
}
}
}
}

View File

@ -12,23 +12,6 @@
"intent_confirm_status": "",
"is_default_answer": false,
"list_options": false,
"msg": [
{
"ans_node_id": 17130001,
"ans_node_name": "猜拳",
"confidence": 1,
"content": "2",
"debug_info": "",
"event": "",
"list_options": false,
"msg_type": "text",
"resp_title": "开始猜拳",
"scene_status": "",
"session_id": "",
"status": "FAQ",
"take_options_only": false
}
],
"msg_id": "31904d53",
"ret": 0,
"scene_status": "",

View File

@ -0,0 +1,9 @@
{
"query": "我叫我妈妈的姐姐叫什么?",
"env": "online",
"first_priority_skills": [],
"second_priority_skills": [],
"user_name": "测试用户名",
"avatar": "https://example.com/img/avatar.png",
"userid": "123"
}

View File

@ -0,0 +1,26 @@
{
"code": 0,
"data": {
"answer": "姨妈",
"answer_type": "text",
"intent_name": "亲戚关系问询",
"msg_id": "212140c2-b358-4d72-8049-cd4f9053fcc7",
"options": null,
"skill_name": "亲戚关系",
"slots": [
{
"name": "亲戚称呼1",
"norm": "妈妈",
"value": "妈妈"
},
{
"name": "亲戚称呼2",
"norm": "姐姐",
"value": "姐姐"
}
],
"status": "GENERAL_FAQ"
},
"msg": "success",
"request_id": "212140c2-b358-4d72-8049-cd4f9053fcc7"
}

View File

@ -0,0 +1,28 @@
using System.Threading.Tasks;
using Xunit;
namespace SKIT.FlurlHttpClient.Wechat.OpenAI.UnitTests
{
public class TestCase_ApiExecuteBotTests
{
[Fact(DisplayName = "测试用例:调用 API [POST] /v2/bot/query")]
public async Task TestExecuteTokenV2()
{
var request = new Models.BotQueryV2Request()
{
QueryString = "我叫我妈妈的姐姐叫什么?",
Environment = "debug",
UserId = "TEST_USERID",
UserName = "TEST_USERNAME",
AccessToken = TestConfigs.WechatAccessToken
};
var response = await TestClients.OpenAIInstance.ExecuteBotQueryV2Async(request);
Assert.NotNull(response.Data);
Assert.NotNull(response.Data.MessageId);
Assert.NotNull(response.Data.AnswerType);
Assert.NotNull(response.Data.Answer);
Assert.NotNull(response.Data.Status);
}
}
}

View File

@ -11,7 +11,8 @@ namespace SKIT.FlurlHttpClient.Wechat.OpenAI.UnitTests
var request = new Models.TokenV2Request() { AccountId = TestConfigs.WechatAccountId };
var response = await TestClients.OpenAIInstance.ExecuteTokenV2Async(request);
Assert.NotNull(response.Data?.AccessToken);
Assert.NotNull(response.Data);
Assert.NotNull(response.Data.AccessToken);
}
}
}