草庐IT

微信小程序开发实战8_1 小程序客服消息的接收

go lang 2023-04-08 原文

为丰富小程序的服务能力,提高服务质量,微信为小程序提供客服消息的功能。小程序用户可以使用客服消息与小程序服务提供方进行沟通。
客服消息会话入口

  • 小程序内:开发者在小程序内添加客服消息按钮组件,用户可在小程序内唤起客服会话页面,给小程序发消息;
  • 已使用过的小程序客服消息的用户会在微信中看到一个微信会话:“小程序客服消息”,进入“小程序客服消息”会话可查看历史客服消息,以及发送客服消息。

客服消息类型

目前支持文本及图片类型消息

客服消息的接收

  • 若小程序没有启用消息推送,则用户发送的消息将会被转发至网页版和移动端小程序客服工具,客服人员可在网页版与移动端小程序客服工具中接入并回复用户。
  • 如小程序启用了消息推送,则所有的客服消息将推送到开发者填写的URL对应的服务器,不再推送到网页版客服工具和移动端小程序客服工具。

发送客服消息

小程序可通过以下两种方式下发客服消息:

  • 调用发送客服消息接口下发客服消息。
  • 使用公众平台网页版客服工具下发客服消息。

下发条件说明

当用户和小程序客服产生特定动作(如用户向小程序客服发送消息、或者进入会话等情况)的交互时,微信将会把消息数据推送给开发者,开发者可以在一段时间内(目前为 48 小时)调用客服接口,给用户发送客服消息。

8.1消息推送的设置

登录小程序管理后台后,在【开发管理】->【开发设置】->【消息推送】中,管理员扫码启用消息服务,填写服务器地址(URL)、令牌(Token) 和消息加密密钥(EncodingAESKey)等信息。

  • URL: 开发者用来接收微信消息和事件的服务器URL地址。开发者所填写的URL 必须以 http:// 或 https:// 开头,分别支持 80 端口和 443 端口。
  • Token: 可由开发者可以任意填写,用于生成签名(该Token会和接口 URL中包含的Token进行比对,从而验证安全性)。
  • EncodingAESKey: 由开发者手动填写或随机生成,将用作消息体加解密密钥。

    开发者提交信息后,微信服务器将给填写的服务器URL地址发送GET请求,GET请求携带参数如下所示:
  • signature:微信加密签名,signature结合了开发者填写的token参数和请求中的timestamp参数、nonce参数。
  • timestamp:时间戳
  • nonce:随机数
  • echostr:随机字符串
    开发者通过 signature对请求进行校验(下面有校验方式)。若确认此次GET请求来自微信服务器,请原样返回echostr参数内容,则接入生效,成为开发者成功,否则接入失败。加密/校验逻辑如下:
  • 将token、timestamp、nonce三个参数进行字典序排序
  • 将三个参数字符串拼接成一个字符串进行sha1加密
  • 开发者把加密后的字符串与signature对比,验证该请求是否来源于微信
    验证URL有效性成功后即接入生效,成为开发者。消息推送参数验证的相关代码如下所示:
func str2sha1(data string)string{
   t:=sha1.New()
   io.WriteString(t,data)
   return fmt.Sprintf("%x",t.Sum(nil))
}
//验证消息推送参数
func HandlerWxSignCheck(w http.ResponseWriter, r *http.Request) {
   signature := r.FormValue("signature")
   timestamp := r.FormValue("timestamp")
   nonce := r.FormValue("nonce")
   echostr := r.FormValue("echostr")

   var token string="demo_token"
   tmps:=[]string{token,timestamp,nonce}
   sort.Strings(tmps)
   tmpStr:=tmps[0]+tmps[1]+tmps[2]
   tmp:=str2sha1(tmpStr)
   if signature != tmp {
      echostr = ""
   }
   w.Write([]byte(echostr))
}

8.2接收客服消息

当某些特定的用户操作(如用户向小程序客服发送消息、或者进入会话等情况)引发事件推送时,微信服务器会将消息(或事件)的数据包以POST请求发送到开发者配置的URL,开发者可以依据自身业务逻辑进行响应。
微信服务器在将用户的消息发给开发者服务器地址后,微信服务器在五秒内收不到响应会断掉连接,并且重新发起请求,总共重试三次。服务器收到请求后必须做出下述回复,这样微信服务器才不会发起重试,否则,将出现严重的错误提示。详见下面说明:

  • 直接回复success(推荐方式)
  • 直接回复空串(指字节长度为0的空字符串,而不是结构体中content字段的内容为空)
    对于客服消息,一旦遇到以下情况,微信会在小程序会话中向用户下发系统提示“该小程序客服暂时无法提供服务,请稍后再试”:
  • 开发者在5秒内未回复任何内容
  • 开发者回复了异常数据

文本消息的结构

用户在客服会话中发送文本消息时将产生如下数据包:

{
  "ToUserName": "toUser",
  "FromUserName": "fromUser",
  "CreateTime": 1482048670,
  "MsgType": "text",
  "Content": "this is a test",
  "MsgId": 1234567890123456
}

参数说明

  • ToUserName:小程序的原始ID
  • FromUserName:发送者的openid
  • Content:文本消息内容

图片消息的结构

用户在客服会话中发送图片消息时将产生如下数据包:

{
  "ToUserName": "toUser",
  "FromUserName": "fromUser",
  "CreateTime": 1482048670,
  "MsgType": "image",
  "PicUrl": "this is a url",
  "MediaId": "media_id",
  "MsgId": 1234567890123456
}

参数说明

  • ToUserName:小程序的原始ID
  • FromUserName:发送者的openid
  • MsgType:image
  • PicUrl:图片链接(由系统生成)
  • MediaId:图片消息媒体id,可以调用【获取临时素材】((getTempMedia)接口拉取数据。
    接收客服消息的示例代码:
//微信用户消息
type WxMsgRecive struct {
   ToUserName        string    `json:"ToUserName"`
   FromUserName   string    `json:"FromUserName"`
   CreateTime        int64     `json:"CreateTime"`
   MsgType       string    `json:"MsgType"`
   MsgId        int64     `json:"MsgId"`
   Content       string    `json:"Content"`
   PicUrl           string    `json:"PicUrl"`
   MediaId       string    `json:"MediaId"`
}
//接收微信发送的消息
func HandlerReciveWxMsg(w http.ResponseWriter, r *http.Request) {
   bdata, _ := ioutil.ReadAll(r.Body)
   var ent wxpay.WxMsgRecive
   err := json.Unmarshal(bdata, &ent)
   if err != nil {
      fmt.Println(err)
   }
   //接受到消息后的业务流程
   //.......
   w.Write([]byte("success"))
}

说明:

  • 上述代码只能用于明文消息的接收,若消息推送培配置中将消息设置为安全模式,则需要对发送的数据进行解密。
  • 由于接收前不能获取消息的类型,因此需要定义一个包含所有字段的结构来解析消息数据:WxMsgRecive。

有关微信小程序开发实战8_1 小程序客服消息的接收的更多相关文章

  1. ruby - 在 Ruby 程序执行时阻止 Windows 7 PC 进入休眠状态 - 2

    我需要在客户计算机上运行Ruby应用程序。通常需要几天才能完成(复制大备份文件)。问题是如果启用sleep,它会中断应用程序。否则,计算机将持续运行数周,直到我下次访问为止。有什么方法可以防止执行期间休眠并让Windows在执行后休眠吗?欢迎任何疯狂的想法;-) 最佳答案 Here建议使用SetThreadExecutionStateWinAPI函数,使应用程序能够通知系统它正在使用中,从而防止系统在应用程序运行时进入休眠状态或关闭显示。像这样的东西:require'Win32API'ES_AWAYMODE_REQUIRED=0x0

  2. ruby - 如何指定 Rack 处理程序 - 2

    Rackup通过Rack的默认处理程序成功运行任何Rack应用程序。例如:classRackAppdefcall(environment)['200',{'Content-Type'=>'text/html'},["Helloworld"]]endendrunRackApp.new但是当最后一行更改为使用Rack的内置CGI处理程序时,rackup给出“NoMethodErrorat/undefinedmethod`call'fornil:NilClass”:Rack::Handler::CGI.runRackApp.newRack的其他内置处理程序也提出了同样的反对意见。例如Rack

  3. ruby - 在 Ruby 中编写命令行实用程序 - 2

    我想用ruby​​编写一个小的命令行实用程序并将其作为gem分发。我知道安装后,Guard、Sass和Thor等某些gem可以从命令行自行运行。为了让gem像二进制文件一样可用,我需要在我的gemspec中指定什么。 最佳答案 Gem::Specification.newdo|s|...s.executable='name_of_executable'...endhttp://docs.rubygems.org/read/chapter/20 关于ruby-在Ruby中编写命令行实用程序

  4. ruby - 使用 C 扩展开发 ruby​​gem 时,如何使用 Rspec 在本地进行测试? - 2

    我正在编写一个包含C扩展的gem。通常当我写一个gem时,我会遵循TDD的过程,我会写一个失败的规范,然后处理代码直到它通过,等等......在“ext/mygem/mygem.c”中我的C扩展和在gemspec的“扩展”中配置的有效extconf.rb,如何运行我的规范并仍然加载我的C扩展?当我更改C代码时,我需要采取哪些步骤来重新编译代码?这可能是个愚蠢的问题,但是从我的gem的开发源代码树中输入“bundleinstall”不会构建任何native扩展。当我手动运行rubyext/mygem/extconf.rb时,我确实得到了一个Makefile(在整个项目的根目录中),然后当

  5. ruby-on-rails - Rails 应用程序之间的通信 - 2

    我构建了两个需要相互通信和发送文件的Rails应用程序。例如,一个Rails应用程序会发送请求以查看其他应用程序数据库中的表。然后另一个应用程序将呈现该表的json并将其发回。我还希望一个应用程序将存储在其公共(public)目录中的文本文件发送到另一个应用程序的公共(public)目录。我从来没有做过这样的事情,所以我什至不知道从哪里开始。任何帮助,将不胜感激。谢谢! 最佳答案 无论Rails是什么,几乎所有Web应用程序都有您的要求,大多数现代Web应用程序都需要相互通信。但是有一个小小的理解需要你坚持下去,网站不应直接访问彼此

  6. ruby - 无法运行 Rails 2.x 应用程序 - 2

    我尝试运行2.x应用程序。我使用rvm并为此应用程序设置其他版本的ruby​​:$rvmuseree-1.8.7-head我尝试运行服务器,然后出现很多错误:$script/serverNOTE:Gem.source_indexisdeprecated,useSpecification.Itwillberemovedonorafter2011-11-01.Gem.source_indexcalledfrom/Users/serg/rails_projects_terminal/work_proj/spohelp/config/../vendor/rails/railties/lib/r

  7. ruby-on-rails - Rails 应用程序中的 Rails : How are you using application_controller. rb 是新手吗? - 2

    刚入门rails,开始慢慢理解。有人可以解释或给我一些关于在application_controller中编码的好处或时间和原因的想法吗?有哪些用例。您如何为Rails应用程序使用应用程序Controller?我不想在那里放太多代码,因为据我了解,每个请求都会调用此Controller。这是真的? 最佳答案 ApplicationController实际上是您应用程序中的每个其他Controller都将从中继承的类(尽管这不是强制性的)。我同意不要用太多代码弄乱它并保持干净整洁的态度,尽管在某些情况下ApplicationContr

  8. ruby-on-rails - 如何在我的 Rails 应用程序 View 中打印 ruby​​ 变量的内容? - 2

    我是一个Rails初学者,但我想从我的RailsView(html.haml文件)中查看Ruby变量的内容。我试图在ruby​​中打印出变量(认为它会在终端中出现),但没有得到任何结果。有什么建议吗?我知道Rails调试器,但更喜欢使用inspect来打印我的变量。 最佳答案 您可以在View中使用puts方法将信息输出到服务器控制台。您应该能够在View中的任何位置使用Haml执行以下操作:-puts@my_variable.inspect 关于ruby-on-rails-如何在我的R

  9. Ruby Sinatra 配置用于生产和开发 - 2

    我已经在Sinatra上创建了应用程序,它代表了一个简单的API。我想在生产和开发上进行部署。我想在部署时选择,是开发还是生产,一些方法的逻辑应该改变,这取决于部署类型。是否有任何想法,如何完成以及解决此问题的一些示例。例子:我有代码get'/api/test'doreturn"Itisdev"end但是在部署到生产环境之后我想在运行/api/test之后看到ItisPROD如何实现? 最佳答案 根据SinatraDocumentation:EnvironmentscanbesetthroughtheRACK_ENVenvironm

  10. ruby-on-rails - 如何在 Rails View 上显示错误消息? - 2

    我是rails的新手,想在form字段上应用验证。myviewsnew.html.erb.....模拟.rbclassSimulation{:in=>1..25,:message=>'Therowmustbebetween1and25'}end模拟Controller.rbclassSimulationsController我想检查模型类中row字段的整数范围,如果不在范围内则返回错误信息。我可以检查上面代码的范围,但无法返回错误消息提前致谢 最佳答案 关键是您使用的是模型表单,一种显示ActiveRecord模型实例属性的表单。c

随机推荐