前言:
而今我们对“net公众号”大约比较关切,小伙伴们都需要剖析一些“net公众号”的相关文章。那么小编同时在网上网罗了一些有关“net公众号””的相关内容,希望你们能喜欢,我们一起来了解一下吧!最近一个朋友在咨询我微信公众号推送消息的问题。因为我在17年的时候做了一年的微信公众号开发,自然有一丢丢经验,但是那时的服务端是用Spring Boot 开发的,这次是.net开发,而且时间也有点久了。所以在开发过程中遇到了一些问题。接下来,我就来和盆友们唠嗑唠嗑。
用.net开发公众号的各位盆友应该知道Senparc这个神器,其封装了微信公众号/小程序等接口,开箱即用。因为我在15年的时,使用ABP+Senparc开发了一个公众号网页,深感其带来的方便快捷。下面,我们进入正题吧。
因为我不能进入到微信公众号后端,所以朋友就把AppId和AppSecret给我,然后就只有这么搞:
var isGLobalDebug = true;//设置全局 Debug 状态var senparcSetting = SenparcSetting.BuildFromWebConfig(isGLobalDebug);var register = RegisterService.Start(senparcSetting).UseSenparcGlobal();//CO2NET全局注册,必须!var isWeixinDebug = true;//设置微信 Debug 状态var senparcWeixinSetting = SenparcWeixinSetting.BuildFromWebConfig(isWeixinDebug);register.UseSenparcWeixin(senparcWeixinSetting, senparcSetting); AccessTokenContainer.RegisterAsync(appId, appSecret); //如果没有注册则进行注册 string linkUrl = ";; //点击详情后跳转后的链接地址,为空则不跳转 //为模版中的各属性赋值 var templateData = new{ first = new TemplateDataItem("您好,您的订单已支付成功!", "#000000"), product = new TemplateDataItem("旺旺大礼包", "#000000"), price = new TemplateDataItem("99.8元", "#000000"), time = new TemplateDataItem("2016-11-09 16:50:38", "#000000"), remark = new TemplateDataItem("感谢您的光临~", "#000000")};string access_token = AccessTokenContainer.GetAccessToken(appId);SendTemplateMessageResult sendResult = TemplateApi.SendTemplateMessage(access_token, openId, templateId, linkUrl, templateData);//发送成功 if (sendResult.errcode.ToString() == "请求成功"){ Response.Write("请求成功");}else{ Response.Write("出现错误:" + sendResult.errmsg);}Response.Write("ok");
肯定是我太菜,所以我就百度一番,顺手又改了几行代码:
var isGLobalDebug = true;//设置全局 Debug 状态var senparcSetting = SenparcSetting.BuildFromWebConfig(isGLobalDebug);var register = RegisterService.Start(senparcSetting).UseSenparcGlobal();//CO2NET全局注册,必须!var isWeixinDebug = true;//设置微信 Debug 状态var senparcWeixinSetting = SenparcWeixinSetting.BuildFromWebConfig(isWeixinDebug);register.UseSenparcWeixin(senparcWeixinSetting, senparcSetting); AccessTokenContainer.RegisterAsync(appId, appSecret); //如果没有注册则进行注册 string linkUrl = ";; //点击详情后跳转后的链接地址,为空则不跳转 //为模版中的各属性赋值 var templateData = new{ first = new TemplateDataItem("您好,您的订单已支付成功!", "#000000"), product = new TemplateDataItem("旺旺大礼包", "#000000"), price = new TemplateDataItem("99.8元", "#000000"), time = new TemplateDataItem("2016-11-09 16:50:38", "#000000"), remark = new TemplateDataItem("感谢您的光临~", "#000000")};string access_token = AccessTokenContainer.GetAccessToken(appId);SendTemplateMessageResult sendResult = TemplateApi.SendTemplateMessage(access_token, openId, templateId, linkUrl, templateData);//发送成功 if (sendResult.errcode.ToString() == "请求成功"){ Response.Write("请求成功");}else{ Response.Write("出现错误:" + sendResult.errmsg);}Response.Write("ok");
但是还是报同样的错误,顿时绝望至极,我到底是又多菜啊?但是这个人虽然菜,可是也不是刚才说的那样脆弱,所以我就去Senparc的github看看。嗯可以看到Senparc非常友好的抛出这个错误,
那么它是什么时候给我们抛出错误呢?这就要看看代码了:
/// <summary> /// 获取可用Token /// </summary> /// <param name="appId"></param> /// <param name="getNewToken">是否强制重新获取新的Token</param> /// <returns></returns> public static string GetAccessToken(string appId, bool getNewToken = false) { return GetAccessTokenResult(appId, getNewToken).access_token; } /// <summary> /// 获取可用AccessTokenResult对象 /// </summary> /// <param name="appId"></param> /// <param name="getNewToken">是否强制重新获取新的Token</param> /// <returns></returns> public static AccessTokenResult GetAccessTokenResult(string appId, bool getNewToken = false) { if (!BaseContainer<AccessTokenBag>.CheckRegistered(appId)) { throw new UnRegisterAppIdException(appId, $"此appId({appId})尚未注册,请先使用AccessTokenContainer.Register完成注册(全局执行一次即可)!", null); } AccessTokenBag accessTokenBag = BaseContainer<AccessTokenBag>.TryGetItem(appId); using (BaseContainer<AccessTokenBag>.Cache.BeginCacheLock("MP.AccessTokenContainer", appId, 0, default(TimeSpan))) { if (getNewToken || accessTokenBag.AccessTokenExpireTime <= SystemTime.Now) { accessTokenBag.AccessTokenResult = CommonApi.GetToken(accessTokenBag.AppId, accessTokenBag.AppSecret, "client_credential"); accessTokenBag.AccessTokenExpireTime = ApiUtility.GetExpireTime(accessTokenBag.AccessTokenResult.expires_in); BaseContainer<AccessTokenBag>.Update(accessTokenBag, null); } } return accessTokenBag.AccessTokenResult; } public class CommonApi { /// <summary> /// 获取凭证接口 /// </summary> /// <param name="grant_type">获取access_token填写client_credential</param> /// <param name="appid">第三方用户唯一凭证</param> /// <param name="secret">第三方用户唯一凭证密钥,既appsecret</param> /// <returns></returns> [ApiBind(Senparc.NeuChar.PlatformType.WeChat_OfficialAccount, "CommonApi.GetToken", true)] public static AccessTokenResult GetToken(string appid, string secret, string grant_type = "client_credential") { AccessTokenResult json = Get.GetJson<AccessTokenResult>(string.Format(Config.ApiMpHost + "/cgi-bin/token?grant_type={0}&appid={1}&secret={2}", grant_type.AsUrlData(), appid.AsUrlData(), secret.AsUrlData()), null, null); if (Config.ThrownWhenJsonResultFaild && json.errcode != 0) { throw new UnRegisterAppIdException(null, $"尚无已经注册的AppId,请先使用AccessTokenContainer.Register完成注册(全局执行一次即可)!模块:{Senparc.NeuChar.PlatformType.WeChat_OfficialAccount}", null); } return json; } }
就是说,只要不是成功状态,都是提示尚无已经注册的AppId,请先使用AccessTokenContainer.Register完成注册(全局执行一次即可)!模块:WeChat_OfficialAccount(我能说这样有点坑吗?),那么我猜想应该是调用接口出错了,那么就看看是怎么请求的接口呢?继续看代码吧:
/// <summary> /// 获取可用Token /// </summary> /// <param name="appId"></param> /// <param name="getNewToken">是否强制重新获取新的Token</param> /// <returns></returns> public static string GetAccessToken(string appId, bool getNewToken = false) { return GetAccessTokenResult(appId, getNewToken).access_token; } /// <summary> /// 获取可用AccessTokenResult对象 /// </summary> /// <param name="appId"></param> /// <param name="getNewToken">是否强制重新获取新的Token</param> /// <returns></returns> public static AccessTokenResult GetAccessTokenResult(string appId, bool getNewToken = false) { if (!BaseContainer<AccessTokenBag>.CheckRegistered(appId)) { throw new UnRegisterAppIdException(appId, $"此appId({appId})尚未注册,请先使用AccessTokenContainer.Register完成注册(全局执行一次即可)!", null); } AccessTokenBag accessTokenBag = BaseContainer<AccessTokenBag>.TryGetItem(appId); using (BaseContainer<AccessTokenBag>.Cache.BeginCacheLock("MP.AccessTokenContainer", appId, 0, default(TimeSpan))) { if (getNewToken || accessTokenBag.AccessTokenExpireTime <= SystemTime.Now) { accessTokenBag.AccessTokenResult = CommonApi.GetToken(accessTokenBag.AppId, accessTokenBag.AppSecret, "client_credential"); accessTokenBag.AccessTokenExpireTime = ApiUtility.GetExpireTime(accessTokenBag.AccessTokenResult.expires_in); BaseContainer<AccessTokenBag>.Update(accessTokenBag, null); } } return accessTokenBag.AccessTokenResult; } public class CommonApi { /// <summary> /// 获取凭证接口 /// </summary> /// <param name="grant_type">获取access_token填写client_credential</param> /// <param name="appid">第三方用户唯一凭证</param> /// <param name="secret">第三方用户唯一凭证密钥,既appsecret</param> /// <returns></returns> [ApiBind(Senparc.NeuChar.PlatformType.WeChat_OfficialAccount, "CommonApi.GetToken", true)] public static AccessTokenResult GetToken(string appid, string secret, string grant_type = "client_credential") { AccessTokenResult json = Get.GetJson<AccessTokenResult>(string.Format(Config.ApiMpHost + "/cgi-bin/token?grant_type={0}&appid={1}&secret={2}", grant_type.AsUrlData(), appid.AsUrlData(), secret.AsUrlData()), null, null); if (Config.ThrownWhenJsonResultFaild && json.errcode != 0) { throw new UnRegisterAppIdException(null, $"尚无已经注册的AppId,请先使用AccessTokenContainer.Register完成注册(全局执行一次即可)!模块:{Senparc.NeuChar.PlatformType.WeChat_OfficialAccount}", null); } return json; } }
根据代码我我就拼接了一个http的地址,直接在浏览器访问,得到了这样的结果:
{"errcode":40164,"errmsg":"invalid ip 183.221.39.24 ipv6 ::ffff:183.221.39.24, not in whitelist hint: [uBiTIa01561501]"}
一看就知道,请求地址没有加入到白名单里面,所以我让朋友去登录公众平台,
开发->基本配置->IP白名单->查看->修改->将IP地址添加进去,最后,OK了。
所以在开发微信接口的时候,还是得多看看文档啊。
标签: #net公众号