169
云计算
负载均衡
产品简介
产品价格
快速入门
用户指南
实践操作
常见问题
购买指南
云服务器
产品简介
计费规则
购买指导
控制台使用指南
常见问题
云数据库
产品简介
计费规则
购买指导
快速入门
连接实例
控制台使用指南
对象存储
产品简介
控制台使用指南
程序员百科
Python
天翼云服务器最佳实践-搭建微信公众号后台

产品推荐:

1、安全稳定的云服务器租用,2核/2G/5M仅37元,点击抢购>>>

2、高防物理服务器20核/16G/50M/500G防御仅350元,点击抢购>>>

3、百度智能建站(五合一网站)仅880元/年,点击抢购>>> 模板建站(PC+手机站)仅480元/年,点击抢购>>>

4、阿里云服务器2核2G3M仅99元/年、2核4G5M仅199元/年,新老同享,点击抢购>>> 

5、腾讯云服务器2核2G4M仅99元/年、新老同享,点击抢购>>>   



点击这里点击这里注册天翼云VIP帐号,立即体验云服务器>>>

本实践将基于天翼云弹性云主机搭建微信公众号处理后台,使用Python语言编写对应的微信消息处理逻辑代码,接收从微信服务端转发过来的消息,并返回处理结果给最终用户。

准备事项

  • 申请微信公众号。

微信公众号申请链接:https://mp.weixin.qq.com/

  • 购买天翼云弹性云主机服务。

如果没有天翼云账号,需先注册天翼云账号并完成实名认证。

本实践中,使用公共镜像CentOS 7.4。

  • 购买弹性IP。

建议同时购买弹性IP,后面需要在微信公众号上配置公网IP的地址。

操作步骤

基础软件安装

本实践中使用Python+Web.py组合完成微信公众号开发,需要安装或升级Python、pip、Web.py框架、WinSCP软件版本。

  1. 升级默认Python版本;

CentOS 7.4自带Python版本比较老,建议升级到Python3。

1)查看Python版本,使用如下命令:

python –version

2)下载Python安装包,这里以Python 3.6.0版本为例,使用命令:

wget https://www.python.org/ftp/python/3.6.0/Python-3.6.0a1.tar.xz

3)解压安装包,使用如下命令:

tar xvf Python-3.6.0a1.tar.xz

如果出现“configure: error: no acceptable C compiler found in $PATH”异常提示,是因为未安装合适的编译器。

解决方法:

执行以下命令,安装/升级gcc及其他依赖的包。

sudo yum install gcc-c++

并在随后提示安装包是否OK时,输入y并回车。出现如下图提示,说明依赖的包安装成功。

在编译器安装完成后,重新执行 ./configure 命令。

4)执行命令:

make && make install

执行成功。但提示pip错误,原因是我的系统中少了openssl-devel包,可以先忽略。

5)查看Python3版本,使用命令:

python3 –version

6)执行命令:

python3

出现如下提示,则说明Python3安装成功。

  1. 升级默认pip版本;

pip是通用的Python包管理工具。提供了对Python包的查找、下载、安装、卸载功能。Python3安装成功后自带pip3,但版本比较老,建议升级到pip最新版本。同时前面安装python3提示“Ignoring ensurepip failure: pip 8.1.1 requires SSL/TLS”错误,导致pip未成功安装,所以需要重新安装pip。

1)安装openssl-devel包,使用命令:

yum install openssl-devel -y

2)执行命令:

make && make install

出现如下提示说明pip安装成功。

3)升级pip3,使用命令:

pip3 install --upgrade pip

出现如下提示说明升级pip到最新版本了。

  1. 安装Web.py框架;

Web.py官方教程地址:http://webpy.org/,使用如下命令安装web.py:

pip3 install web.py==0.40.dev0

  1. 安装WinSCP;

通常情况下,我们在本地Windows操作系统上编辑代码,完成后再上传至ECS上(CentOS Linux系统)。WinSCP 是一个Windows环境下使用的SSH的开源图形化SFTP客户端, 同时支持 SCP 协议。它的主要功能是在本地与远程计算机间安全地复制文件,并且可以直接编辑文件。

WinSCP安装链接:https://winscp.net/eng/docs/lang:chs

上传代码

  1. 新建main.py文件,复制如下代码:
# -*- coding: utf-8 -*-
# filename: main.py
import web
from handle import Handle
urls = (
    '/wx', 'Handle',
)
if __name__ == '__main__':
    app = web.application(urls, globals())
    app.run()
  1. 新建handle.py文件,复制如下代码:
# -*- coding: utf-8 -*-
# filename: handle.py
import hashlib
import web
import receive
import time
import os
class Handle(object):
    def __init__(self):
        self.app_root = os.path.dirname(__file__)
        self.templates_root = os.path.join(self.app_root, 'templates')
        self.render = web.template.render(self.templates_root)
     def GET(self):
        try:
            data = web.input()
            if len(data) == 0:
                return "hello, this is handle view"
            signature = data.signature
            timestamp = data.timestamp
            nonce = data.nonce
            echostr = data.echostr
            token = "此处内容与公众号基本配置里Token字段取值保持一致"
            list = [token, timestamp, nonce]
            list.sort()
            s = list[0] + list[1] + list[2]
            hashcode = hashlib.sha1(s.encode('utf-8')).hexdigest()
            print( "handle/GET func: hashcode, signature: ", hashcode, signature)
            if hashcode == signature:
                return echostr
            else:
                return echostr
        except (Exception) as Argument:
            return Argument
    def POST(self):
        try:
            webData = web.data()
            print("Handle Post webdata is:\n", webData)
            #打印消息体日志
            recMsg = receive.parse_xml(webData)
            if isinstance(recMsg, receive.Msg) and recMsg.MsgType == 'text':
                toUser = recMsg.FromUserName
                fromUser = recMsg.ToUserName
                content = "欢迎" + str(recMsg.Content)
                print('Reply message info:\n')
                print('toUser =', toUser)
                print('fromUser = ', fromUser)
                print('content = ', content)
                return self.render.reply_text(toUser, fromUser, int(time.time()), content)
            else:
                print("不支持的消息类型:",recMsg.MsgType)
                return "success"
        except (Exception) as Argment:
            return Argment
  1. 新建receive.py文件,复制如下代码:
# -*- coding: utf-8 -*-
# filename: receive.py
import xml.etree.ElementTree as ET
def parse_xml(web_data):
    if len(web_data) == 0:
        return None
    xmlData = ET.fromstring(web_data)
    msg_type = xmlData.find('MsgType').text
    if msg_type == 'text':
        return TextMsg(xmlData)
    elif msg_type == 'image':
        return ImageMsg(xmlData)
    elif msg_type == 'location':
        return LocationMsg(xmlData)
    elif msg_type == 'event':
        return EventMsg(xmlData)
class Event(object):
    def __init__(self, xmlData):
        self.ToUserName = xmlData.find('ToUserName').text
        self.FromUserName = xmlData.find('FromUserName').text
        self.CreateTime = xmlData.find('CreateTime').text
        self.MsgType = xmlData.find('MsgType').text
        self.Eventkey = xmlData.find('EventKey').text
class Msg(object):
    def __init__(self, xmlData):
        self.ToUserName = xmlData.find('ToUserName').text
        self.FromUserName = xmlData.find('FromUserName').text
        self.CreateTime = xmlData.find('CreateTime').text
        self.MsgType = xmlData.find('MsgType').text
        self.MsgId = xmlData.find('MsgId').text
class TextMsg(Msg):
    def __init__(self, xmlData):
        Msg.__init__(self, xmlData)
        self.Content = xmlData.find('Content').text
class ImageMsg(Msg):
    def __init__(self, xmlData):
        Msg.__init__(self, xmlData)
        self.PicUrl = xmlData.find('PicUrl').text
        self.MediaId = xmlData.find('MediaId').text
class LocationMsg(Msg):
    def __init__(self, xmlData):
        Msg.__init__(self, xmlData)
        self.Location_X = xmlData.find('Location_X').text
        self.Location_Y = xmlData.find('Location_Y').text
class EventMsg(Msg):
    def __init__(self, xmlData):
        Event.__init__(self, xmlData)
        self.Event = xmlData.find('Event').text
  1. 新建templates文件夹,在文件夹下新建reply_text.xml文件,复制如下代码:
$def with (toUser,fromUser,createTime,content)
<xml>
<ToUserName><![CDATA[$toUser]]></ToUserName>
<FromUserName><![CDATA[$fromUser]]></FromUserName>
<CreateTime>$createTime</CreateTime>
<MsgType><![CDATA[text]]></MsgType>
<Content><![CDATA[$content]]></Content>
</xml>
  1. 最终本地代码文件形成如下:

  1. 通过WinSCP工具将上述文件与目录上传至ECS指定目录下:

启动服务

使用如下命令启动服务:

python3 main.py 80

启用开发者模式

  1. 登录微信公众平台,选择“开发 > 基本配置”,单击“修改配置”。
  2. 填写配置信息,单击“提交”。
  • URL:https://ECS的弹性公网IP/wx,不用添加80端口。
  • Token:需要与handle.py中对应token取值完全一致。
  • EncodingAESKey:随机生成。
  • 消息加解密方式:此为示例,选择简单的“明文模式”。
  1. 验证token成功,单击“启用”。

**说明:**如果token验证失败,请检查Token配置与handle.py中GET消息处理代码是否一致。

验证

使用微信关注公众号,任意发送一条文本消息,看是否能够收到回复。如能收到回复则表明系统处理正常。


这条帮助是否解决了您的问题? 已解决 未解决

提交成功!非常感谢您的反馈,我们会继续努力做到更好! 很抱歉未能解决您的疑问。我们已收到您的反馈意见,同时会及时作出反馈处理!