此文将分两篇讲解,主要分为以下几步

  1. 签名校验;
  2. 首次提交验证申请;
  3. 接收消息;
  4. 被动响应消息(返回XML);
  5. 映射图灵消息及微信消息;

此篇为第二篇。

被动响应消息(返回XML)

上一篇中,我们已经可以拿到了微信传入的参数,接下来,我们需要响应消息给用户。

微信要求我们返回XML数据,且格式是规定好的,具体请看

微信公众平台开发者文档

响应的实体类,我们之前已经写好了,因为要求是XML格式。

我们在此使用微软提供的System.Xml.Serialization.XmlSerializer来将我们的数据序列化为XML。

所以我们在类上边标记了XmlRoot特性,在枚举的字段上边标记了XmlEnum特性,NewsMsg中在文章列表上标记了XmlArrayXmlArrayItem特性。而后反序列化出来的便是微信要求的格式了。

序列化方法如下:

public string ResponseXML(object value, Type type)
{
StringWriter sw = new StringWriter();
XmlSerializerNamespaces ns = new XmlSerializerNamespaces(); ns.Add("", "");  //去除命名空间
XmlSerializer serializer = new XmlSerializer(type); serializer.Serialize(sw, value, ns); return sw.ToString();
}

注意:此处必须去除XML的命名空间,不然微信不识别

完整方法奉上:

public HttpResponseMessage Post()
{
var requestContent = Request.Content.ReadAsStreamAsync().Result;
//从正文参数中加载微信的请求参数
XmlDocument xmlDoc = new XmlDocument();
xmlDoc.Load(requestContent); logger.DebugFormat("WX请求XML内容:{0}", xmlDoc.InnerText); string msgTypeStr = xmlDoc.SelectSingleNode("xml/MsgType").InnerText;
string userName = xmlDoc.SelectSingleNode("xml/FromUserName").InnerText;
string efhName = xmlDoc.SelectSingleNode("xml/ToUserName").InnerText; string responseContent;
MsgType msgType; //获取消息类型,若未定义,则返回。
if (!Enum.TryParse(msgTypeStr, true, out msgType))
{
responseContent = MsgService.Instance.ResponseXML(new TextMsg
{
FromUserName = efhName,
MsgType = MsgType.Text,
Content = "俺还小,不知道你在说啥子(⊙_⊙)?",
CreateTime = UnixTimestamp.Now.ToNumeric(),
ToUserName = userName
}, typeof(TextMsg)); return new HttpResponseMessage(HttpStatusCode.OK)
{
Content = new StringContent(responseContent, Encoding.UTF8, "application/xml"),
};
} if (msgType == MsgType.Event)
{
return ProcessEvent(xmlDoc, userName, efhName);
} //图灵消息转换为微信响应消息,下一节奉上
string content = xmlDoc.SelectSingleNode("xml/Content").InnerText;
var requestResult = TuLingService.Instance.GetMsgFromResponse(content, userName, efhName); responseContent = MsgService.Instance.ResponseXML(requestResult.Data, requestResult.DataType);
return new HttpResponseMessage(HttpStatusCode.OK)
{
Content = new StringContent(responseContent, Encoding.UTF8, "application/xml"),
};
} private HttpResponseMessage ProcessEvent(XmlDocument xmlDoc, string userName, string efhName)
{
string eventValue = xmlDoc.SelectSingleNode("xml/Event").InnerText; var responseContent = MsgService.Instance.ResponseXML(new TextMsg
{
FromUserName = efhName,
MsgType = MsgType.Text,
Content = eventValue.ToLower().Equals("subscribe") ? "lei好哇~" : "大爷,奴家会想你的",//其实取消订阅是不会发送消息的
CreateTime = UnixTimestamp.Now.ToNumeric(),
ToUserName = userName
}, typeof(TextMsg)); return new HttpResponseMessage(HttpStatusCode.OK)
{
Content = new StringContent(responseContent, Encoding.UTF8, "application/xml"),
};
}

至此,我们已经完成了微信被动回复消息的响应。

映射图灵消息及微信消息

上边我们已经实现了被动回复消息的功能,接下来我们需要将图灵机器人接口与我们的公众平台关联起来。

分析图灵机器人返回的参数,我们发现所有类型的内容都有codetext参数。又因为我们需要将图灵的消息与微信的响应消息直接对应起来,因此我们定义接口,提供转换方法

public class TuLingResult
{
//消息类型(我们在序列化为XML的时候需要提供类型)
public Type DataType { get; set; } public object Data { get; set; }
}
public interface IResponse
{
TuLingResult ToTuLingResult(string fromUserName, string toUserName);
}

创建文本类数据的实体作为图灵消息的基类(对应微信的文本消息)

public class TextResult : IResponse
{
public int Code { get; set; } public string Text { get; set; } public virtual TuLingResult ToTuLingResult(string fromUserName, string toUserName)
{
return new TuLingResult
{
DataType = typeof(TextMsg),
Data = new TextMsg
{
FromUserName = fromUserName,
ToUserName = toUserName,
Content = Text,
CreateTime = UnixTimestamp.Now.ToNumeric(),
MsgType = MsgType.Text
}
};
}
}

而后依次创建各种数据的实体类。

如:新闻(对应微信的图文消息)

public class NewsResult : TextResult
{
public List<NewsInfo> List { get; set; } public override TuLingResult ToTuLingResult(string fromUserName, string toUserName)
{
if (List.Count > 10)
{
List = List.Take(10).ToList();
} return new TuLingResult
{
DataType = typeof(NewsMsg),
Data = new NewsMsg
{
FromUserName = fromUserName,
ToUserName = toUserName,
ArticleCount = List.Count,
Articles = List.Select(m => new MsgNewsInfo
{
Title = m.Article,
Description = m.Source,
Url = m.DetailUrl,
PicUrl = m.Icon
}).ToList(),
CreateTime = UnixTimestamp.Now.ToNumeric(),
MsgType=MsgType.News
}
};
}
} public class NewsInfo
{
/// <summary>
/// 标题
/// </summary>
public string Article { get; set; } /// <summary>
/// 来源
/// </summary>
public string Source { get; set; } /// <summary>
/// 详情地址
/// </summary>
public string DetailUrl { get; set; } /// <summary>
/// 图标地址
/// </summary>
public string Icon { get; set; }
}

同理创建图灵机器人提供的各类数据实体类

我们想要支持的数据实体都定义完毕后,我们便可以开始请求图灵接口,获取真实的消息了,在此我们使用HttpClient实现。

private const string TULING_API_URL = "http://www.tuling123.com/openapi/api";
private const string TULING_API_KEY = "XXXXX";//图灵的APIKEY
public TuLingResult GetMsgFromResponse(string keyword, string userFlag, string efhName)
{
string linkString = string.Format("{0}?key={1}&info={2}&userid={3}"
, TULING_API_URL, TULING_API_KEY, keyword, userFlag); string content = string.Empty;
using (HttpClient client = new HttpClient())
{
HttpResponseMessage response = client.GetAsync(linkString).Result; content = response.Content.ReadAsStringAsync().Result;
logger.DebugFormat("图灵机器人响应:{0}", content);
}
return ConvertToMsg(content, userFlag, efhName);
}

图灵返回了code标识消息的类型和错误信息,因此我们先将响应消息解析为TextResult,拿到图灵的类型。

先定义图灵类型枚举

public enum ResultType
{
TL_FORMAT_DATA = 50000,
TL_TEXT_DATA = 100000,
TL_LINK_DATA = 200000,
TL_NOVEL_DATA = 301000,
TL_NEWS_DATA = 302000,
TL_APP_DATA = 304000,
TL_TRAIN_DATA = 305000,
TL_AIRPORT_DATA = 306000,
TL_TUAN_DATA = 307000,
TL_TUWEN_DATA = 308000,
TL_HOTEL_DATA = 309000,
TL_LOTTERY_DATA = 310000,
TL_PRICE_DATA = 311000,
TL_RESTAURANT_DATA = 312000, TL_ERROR_LENGTH = 40001,
TL_ERROR_EMPTY = 40002,
TL_ERROR_INVALID = 40003,
TL_ERROR_OUTLIMIT = 40004,
TL_ERROR_NOTSUPPORT = 40005,
TL_ERROR_SERVERUPDATE = 40006,
TL_ERROR_SERVERERROR = 40007
}

对应于图灵的返回码

100000 	文本类数据
200000 网址类数据
301000 小说
302000 新闻
304000 应用、软件、下载
305000 列车
306000 航班
307000 团购
308000 优惠
309000 酒店
310000 彩票
311000 价格
312000 餐厅
40001 key的长度错误(32位)
40002 请求内容为空
40003 key错误或帐号未激活
40004 当天请求次数已用完
40005 暂不支持该功能
40006 服务器升级中
40007 服务器数据格式异常
50000 机器人设定的“学用户说话”或者“默认回答”

而后拿到消息类型

private ResultType GetResultType(string response)
{
var result = JsonConvert.DeserializeObject<TextResult>(response); return (ResultType)result.Code;
}

之后,我们便可以按照不同类型返回相对应的TuLingResult。

public TuLingResult ConvertToMsg(string response, string userFlag, string efhName)
{
IResponse result = null; var resultType = GetResultType(response); switch (resultType)
{
case ResultType.TL_TEXT_DATA:
result = JsonConvert.DeserializeObject<TextResult>(response);
break; case ResultType.TL_LINK_DATA:
result = JsonConvert.DeserializeObject<LinkResult>(response);
break; case ResultType.TL_NEWS_DATA:
result = JsonConvert.DeserializeObject<NewsResult>(response);
break; case ResultType.TL_TUWEN_DATA:
result = JsonConvert.DeserializeObject<TuWenResult>(response);
break; case ResultType.TL_TRAIN_DATA:
result = JsonConvert.DeserializeObject<TrainResult>(response);
break; case ResultType.TL_AIRPORT_DATA:
result = JsonConvert.DeserializeObject<AirportResult>(response);
break; case ResultType.TL_APP_DATA:
result = JsonConvert.DeserializeObject<AppResult>(response);
break; case ResultType.TL_HOTEL_DATA:
result = JsonConvert.DeserializeObject<HotelResult>(response);
break; case ResultType.TL_PRICE_DATA:
result = JsonConvert.DeserializeObject<PriceResult>(response);
break; case ResultType.TL_ERROR_LENGTH:
case ResultType.TL_ERROR_INVALID:
case ResultType.TL_ERROR_EMPTY:
case ResultType.TL_ERROR_OUTLIMIT:
result = new TextResult { Text = "您的输入有误" };
break; case ResultType.TL_ERROR_SERVERERROR:
case ResultType.TL_ERROR_SERVERUPDATE:
result = new TextResult { Text = "服务器忙,暂时无法为您提供服务" };
break; case ResultType.TL_ERROR_NOTSUPPORT:
result = new TextResult { Text = "俺还小,您说的这个还得慢慢学习,以后再来试吧" };
break; default:
result = new TextResult { Text = "俺还小,不知道你在说啥子(⊙_⊙)?" };
break;
} return result.ToTuLingResult(efhName, userFlag);
}

而后,我们便可以将我们拿到的TuLingResult中的Data序列化为微信需要的XML

var requestResult = TuLingService.Instance.GetMsgFromResponse(content, userName, efhName);

responseContent = MsgService.Instance.ResponseXML(requestResult.Data, requestResult.DataType);
return new HttpResponseMessage(HttpStatusCode.OK)
{
Content = new StringContent(responseContent, Encoding.UTF8, "application/xml"),
};

至此,我们的微信公众号调用图灵机器人接口就搞定了!


源码托管在csdn上,欢迎批评指正。

https://code.csdn.net/yanhuiqiang/efh-blog

这个微信公众号项目是做在本人的博客项目中的,博客是还在学校的时候就做的,前一阵简单做了个发文章的后台(也只能发文章了),经历了从webform+nhibernate转monorail+mybatis,再转mvc+mybatis的一个过程,代码都是直接Copy简单改完的,懒得优化,博客的部分各位看官将就着看吧。

博客地址:http://efenghuo.com

使用web api开发微信公众号,调用图灵机器人接口(二)的更多相关文章

  1. 使用web api开发微信公众号,调用图灵机器人接口(一)

    此文将分两篇讲解,主要分为以下几步 签名校验; 首次提交验证申请; 接收消息; 被动响应消息(返回XML); 映射图灵消息及微信消息; 其实图灵机器人搭载微信公众号很简单,只需要把图灵的地址配到公众后 ...

  2. 小机器人自动回复(python,可扩展开发微信公众号的小机器人)

    api来之图灵机器人.我们都知道微信公众号可以有自动回复,我们先用python脚本编写一个简单的自动回复的脚本,利用图灵机器人的api. http://www.tuling123.com/help/h ...

  3. Asp.Net Web API开发微信后台

    如果说用Asp.Net开发微信后台是非主流,那么Asp.Net Web API的微信后台绝对是不走寻常路. 需要说明的是,本人认为Asp.Net Web API在开发很多不同的请求方法的Restful ...

  4. 使用vue开发微信公众号下SPA站点的填坑之旅

    原文发表于本人博客,点击进入使用vue开发微信公众号下SPA站点的填坑之旅 本文为我创业过程中,开发项目的填坑之旅.作为一个技术宅男,我的项目是做一个微信公众号,前后端全部自己搞定,不浪费国家一分钱^ ...

  5. PHP开发微信公众号(一)二维码的获取

    要开发微信公众号,首先进行需要注册一个,然后认证.这就不用多说了. 当然如果没有,也可以去申请一个测试号来使用,地址:https://mp.weixin.qq.com/debug/cgi-bin/sa ...

  6. Java开发微信公众号(五)---微信开发中如何获取access_token以及缓存access_token

    获取access_token是微信api最重要的一个部分,因为调用其他api很多都需要用到access_token.比如自定义菜单接口.客服接口.获取用户信息接口.用户分组接口.群发接口等在请求的时候 ...

  7. vue开发微信公众号--开发准备

    由于工作项目的原因,需要使用vue开发微信公众号,但是这种微信公众号更多是将APP套了一个微信的壳子,除了前面的授权和微信有关系以外,其他的都和微信没多大的关系了,特此记录 开发流程 首先需要在电脑上 ...

  8. spring-boot-route(二十三)开发微信公众号

    在讲微信公众号开发之前,先来大概了解一下微信公众号.微信公众号大体上可以分为服务号和订阅号,订阅号和服务号的区别如下: 服务号可以申请微信支付功能. 服务号只能由企业申请,订阅号可以有企业或个人申请. ...

  9. Java开发微信公众号(四)---微信服务器post消息体的接收及消息的处理

    在前几节文章中我们讲述了微信公众号环境的搭建.如何接入微信公众平台.以及微信服务器请求消息,响应消息,事件消息以及工具处理类的封装:接下来我们重点说一下-微信服务器post消息体的接收及消息的处理,这 ...

随机推荐

  1. 文件夹名为单字符时右击弹出C++错误

    原因:操作系统安装有虚拟光驱软件——WinMount,此错误为WinMount的一个Bug 解决方法: 打开Windows注册表,定位此注册表并删除即可 [HKEY_CLASSES_ROOT\Dire ...

  2. 【转载】 C语言命令行小猪佩奇

    // ASCII Peppa Pig by Milo Yip #include <math.h> #include <stdio.h> #include <stdlib. ...

  3. 解决Oracle11g密码180天过期,账号锁住的问题

    1.查看用户的proifle是哪个,一般是default: sql>SELECT username,PROFILE FROM dba_users; 2.查看指定概要文件(如default)的密码 ...

  4. jQuery插件实例二:年华时代插件ReturnTop回到首页

    这个插件功能在于当网页内容高度很高时,方便用户快速回到顶部.核心在于对屏幕高度的获取,定时器的使用,在引用代码后,只使用$.nhsd.returnTop();即可实现效果 效果图: 代码: ; fun ...

  5. 破解myeclipse10失败的一个奇葩原因

    昨天开发用的myeclipse10突然弹窗提示我要激活,我清楚的记得安装时候已经破解并且看到激活信息了. 翻遍搜索出来的文章,改systemid之类的也试过了,问题依旧存在,很是绝望. 今早过来机灵了 ...

  6. configuration on ubuntu server

    1.network configuration 1.1 static ip sudo vi /etc/network/interfaces auto eth0 iface eth0 inet stat ...

  7. 禁用wps的云文档,恢复到清爽的状态

    wps安装完成后,默认会开启云文档功能,每次打开表格.文档都会显示乱七八糟的一些东西,很麻烦 不得已只好手动为wps“瘦身”: 在wps表格或者文档的快捷方式上右键,选择打开文件所在位置 找到一个最新 ...

  8. 1217. [HNOI2003]消防局的设立【贪心】

    Description 2020年,人类在火星上建立了一个庞大的基地群,总共有n个基地.起初为了节约材料,人类只修建了n-1条道路来 连接这些基地,并且每两个基地都能够通过道路到达,所以所有的基地形成 ...

  9. 1103. [POI2007]MEG-Megalopolis【树链剖分】

    Description 在经济全球化浪潮的影响下,习惯于漫步在清晨的乡间小路的邮递员Blue Mary也开始骑着摩托车传递邮件了. 不过,她经常回忆起以前在乡间漫步的情景.昔日,乡下有依次编号为1.. ...

  10. 【洛谷】【动态规划/二维背包】P1855 榨取kkksc03

    [题目描述:] ... (宣传luogu2的内容被自动省略) 洛谷的运营组决定,如果...,那么他可以浪费掉kkksc03的一些时间的同时消耗掉kkksc03的一些金钱以满足自己的一个愿望. Kkks ...