概述:

  本文主要分享一下博主在学习wxpy 的过程中开发的一个小程序。博主在最近有一个监控报警的需求需要完成,然后刚好在学习wxpy 这个东西,因此很巧妙的将工作和学习联系在一起。

  博文中主要使用到的技术设计到Python,Redis,以及Java。涉及到的技术看似很多,但是主要的语言是基于Python进行开发的。

  架构涉及主要采用了 生产者消费者的涉及模式,使用Redis作为消息队列进行解耦操作。

  主要架构涉及如下:

  

  接下来开始介绍一下程序的实现过程,主要讲解wxpy -> python.redis -> Java.redis

1、Wxpy初体验

  项目使用的python 是3.5版本的,因此语法会和2.x版本有所区别,wxpy 支持python3.4-3.6 以及python2.7版本 ,因此在python版本上不用太过于纠结

  1.1 安装wxpy

    在这里默认大家以及安装好了pip,我们需要安装wxpy 以及wechat_sender 两个包,这里推荐使用国内的豆瓣源,如果大家网速过硬 请忽略。。。。

  1. pip install wxpy -i "https://pypi.doubanio.com/simple/"
  2. pip install wechat_sender -i "https://pypi.doubanio.com/simple/"

  

  1.2 wxpy 登陆

    wxpy 使用起来非常简单,我们只需要创建一个bot 对象,程序运行后,会弹出二维码,扫描二维码后显示登陆成功。

    下述代码在登陆完成后,会向我们的文件传输助手发送一个“hello world!”。(每个程序都需要一个hello world)

  1. from wxpy import *
  2.  
  3. bot = Bot()
  4.  
  5. bot.file_helper.send('hello world!')
  6.  
  7. print("ending")

    关于Bot()对象的相关参数说明,我们可以在源码中的注释中看到:

  1. """
    :param cache_path:
    * 设置当前会话的缓存路径,并开启缓存功能;为 `None` (默认) 则不开启缓存功能。
    * 开启缓存后可在短时间内避免重复扫码,缓存失效时会重新要求登陆。
    * 设为 `True` 时,使用默认的缓存路径 'wxpy.pkl'。
    :param console_qr:
    * 在终端中显示登陆二维码,需要安装 pillow 模块 (`pip3 install pillow`)。
    * 可为整数(int),表示二维码单元格的宽度,通常为 2 (当被设为 `True` 时,也将在内部当作 2)。
    * 也可为负数,表示以反色显示二维码,适用于浅底深字的命令行界面。
    * 例如: 在大部分 Linux 系统中可设为 `True` 或 2,而在 macOS Terminal 的默认白底配色中,应设为 -2。
    :param qr_path: 保存二维码的路径
    :param qr_callback: 获得二维码后的回调,可以用来定义二维码的处理方式,接收参数: uuid, status, qrcode
    :param login_callback: 登陆成功后的回调,若不指定,将进行清屏操作,并删除二维码文件
    :param logout_callback: 登出时的回调
    """

    这里介绍一下两个主要使用到的参数:

  1. cache_path: 在开发过程中可以设置为True 避免每次登陆都需要重新扫描,具有缓存的作用。
  2.  
  3. qr_path:用于保存二维码生成图片,主要解决服务器图片展示不方便的问题

    

  

  1.3 wxpy 好友与聊天群

    如代码所示,我们可以通过Bot.friends 以及Bot.groups 来获取到所有的好友以及聊天群,这里需要注意的是,聊天群需要保存到通讯录中,不然可能会出现找不到聊天群的情况。

    在搜索方法中,可以提供的参数有:姓名,city,province,sex 等相关变量。

    关于好友的详细API文档,可以参考---》 微信好友API

  1. from wxpy import *
  2.  
  3. bot = Bot()
  4.  
  5. # 获取所有好友
  6. friends = bot.friends()
  7.  
  8. # 遍历输出好友名称
  9. for friend in friends:
  10. print(friend)
  11.  
  12. # 找到好友
  13. friend = bot.friends.search('被单')[0]
  14. print(friend)
  15. friend.send("hello world!")
  16.  
  17. # 获取所有聊天群
  18. groups = bot.groups()
  19.  
  20. for group in groups:
  21. print(group)
  22.  
  23. # 找到目标群
  24. group = groups.search("")[0]
  25.  
  26. group.send("hello world!")

    

  1.4 wxpy 消息处理

    接下来主要介绍一下用户发送消息的类型,目前wxpy 支持发送文本,图片,视频以及文件。主要的发送方式如代码所示:

    这里比较重要的就是关于 @bot.register() 的使用,该注释主要用于注册消息接收器,我们可以根据特定的需求,配置不一样的消息接收器。

    Bot.register(chats=Nonemsg_types=Noneexcept_self=Truerun_async=Trueenabled=True) 详情可以查看源码中的介绍

    关于消息处理API,读者可以在该地址下查看详细的配置,这里不做过多的描述。

    代码中有使用到:embed() 这个方法, 主要用于阻塞进程,避免由于程序运行结束导致无法接收消息。

  1. from wxpy import *
  2.  
  3. bot = Bot()
  4. # 获取好友
  5. my_friend = bot.friends().search('被单')[0]
  6.  
  7. # 搜索信息
  8. messages = bot.messages.search(keywords='测试', sender=bot.self)
  9.  
  10. for message in messages:
  11. print(message)
  12.  
  13. # 发送文本
  14. my_friend.send('Hello, WeChat!')
  15. # 发送图片
  16. my_friend.send_image('my_picture.png')
  17. # 发送视频
  18. my_friend.send_video('my_video.mov')
  19. # 发送文件
  20. my_friend.send_file('my_file.zip')
  21. # 以动态的方式发送图片
  22. my_friend.send('@img@my_picture.png')
  23.  
  24. # 发送公众号
  25. my_friend.send_raw_msg(
  26. # 名片的原始消息类型
  27. raw_type=42,
  28. # 注意 `username` 在这里应为微信 ID,且被发送的名片必须为自己的好友
  29. raw_content='<msg username="wxpy_bot" nickname="wxpy 机器人"/>'
  30. )
  31.  
  32. # 消息接收监听器
  33. @bot.register()
  34. def print_others(msg):
  35. # 输出监听到的消息
  36. print(msg)
  37. # 回复消息
  38. msg.reply("hello world")
  39.  
  40. embed()

  1.4 wxpy 图灵机器人

    wxpy 接入图灵机器人相当方便,我们首先需要到图灵近期人官网进行注册,哆啦A梦的任意门

    通过注册Tuling 对象,当我们接收到消息的时候,可以直接使用tuling机器人来帮我们进行答复。其他的业务需求各位可以根据自己的需求来完成相应的逻辑。

  1. from wxpy import *
  2.  
  3. bot = Bot()
  4.  
  5. # 获取好友
  6. dear = bot.friends().search('被单')[0]
  7.  
  8. # 注册获得个人的图灵机器人key 填入
  9. tuling = Tuling(api_key='******')
  10.  
  11. # 使用图灵机器人自动与指定好友聊天
  12. @bot.register(dear)
  13. def reply_my_friend(msg):
  14. print(msg)
  15. tuling.do_reply(msg)
  16.  
  17. embed()

  1.5 wechat_sender

    在熟悉了wxpy 的相关操作后,我们接下来介绍一下一个主要使用到的工具。由于wxpy 的设计,导致了一些业务操作并不好进行实现。因此我们在这里引入一个工具类:wechat_sender 。

    首先我们需要像往常一样进行微信登陆,然后使用 listen() 进行对我们的 bot() 对象进行监听。

    在这里我们可以看到了和上面代码的区别,这里使用的是listen(),上面是使用embed()进行监听。 我们再这里使用listen 进行监听对象后,可以设置相应的配置。监听默认设置的接收对象为self.file_helper,通过设置receivers 可以配置消息的接收者。

  1. # login.py
    from wxpy import *
  2. from wechat_sender import *
  3.  
  4. bot = Bot()
  5.  
  6. friend = bot.friends().search('被单')[0]
  7.  
  8. listen(bot, token='test', receivers=[friend])
  1. # sender.py coding: utf-8
  2. from wechat_sender import Sender
  3.  
  4. sender = Sender(token='test')
  5.  
  6. sender.send('hello world!')

    在别的python 文件中,我们只需要创建一个Sender() 对象,然后调用Sender.send()方法,即可对我们设定好的消息接收者发送消息。

    Sender()在创建的时候可以通过特定的参数设定,比如这里使用了 token 用于避免多个listen 导致sender 混淆。还可以在sender中设置receiver 从listen 中选取需要接收消息的对象。

  1.6 wxpy 在监控模块的代码实现

微信登陆模块:

  1. from wechat_sender import *
  2. from wxpy import *
  3.  
  4. bot = Bot(qr_path="qr.png")
  5.  
  6. group = bot.groups().search('监控报警')[0]
  7.  
  8. print("微信登陆成功!进行监控报警功能!")
  9. print(group)
  10.  
  11. #
  12. listen(bot, token='test', receivers=[group])

业务处理模块:

  1. import redis
  2. from wechat_sender import *
  3.  
  4. sender = Sender(token='test', receivers='监控报警')

  5. while true:
  6. # do anything
  7. sender.send(message=data)
  8. # do anything
  9.  
  10. p.unsubscribe('cardniu-monitor')
  11. print('取消订阅')

  

2、Python-Redis

    这一模块我们将简单描述一下python 对于Redis 的支持,首先我们需要安装python-redis相关模块:

  2.1 Python-redis安装

  2.2 Python 简单操作Redis

    由于Python 操作Redis 并不是我们这里的主要内容,所以这里简单的过一下Python 对Redis 的支持。

  1. import redis
  2.  
  3. r = redis.Redis(host='ip', port=6379, db=15, password='****')
  4.  
  5. r.set('name', 'Jaycekon')
  6.  
  7. value = r.get('name')
  8.  
  9. print(value)

  2.3 Redis的发布订阅模式

    在为大家讲解Redis 的发布订阅模式前,先为大家科普一下生产者消费者模式:

    大家来领略一下我的灵魂画图,生产者消费者的核心思想是通过一个冰箱来进行解耦,就是我们的厨师不需要出厨房,顾客也不需要去厨房拿饭吃。通过一个冰箱来进行中间的解耦合。

     下面是我们通过python 实现的一个生产者消费者模式,厨师不停的做饭,顾客不停的吃。。大家相互不影响。

  1. from threading import Thread
  2.  
  3. queues = queue.Queue(10)
  4.  
  5. class Producer(Thread):
  6. def run(self):
  7. while True:
  8. elem = random.randrange(9)
  9. queues.put(elem)
  10. print("厨师 {} 做了 {} 饭 --- 还剩 {} 饭没卖完".format(self.name, elem, queues.qsize()))
  11. time.sleep(random.random())
  12.  
  13. class Consumer(Thread):
  14. def run(self):
  15. while True:
  16. elem = queues.get()
  17. print("吃货{} 吃了 {} 饭 --- 还有 {} 饭可以吃".format(self.name, elem, queues.qsize()))
  18. time.sleep(random.random())
  19.  
  20. def main():
  21. for i in range(3):
  22. p = Producer()
  23. p.start()
  24. for i in range(2):
  25. c = Consumer()
  26. c.start()
  27.  
  28. if __name__ == '__main__':
  29. main()

    

    再来说一下为什么使用到Redis 的发布订阅模式。

    Redis在当前程序中,主要担当了一个消息队列的角色,我们并没有使用目前较为热门的RabbitMq,ActiveMq来消息队列进行解耦。主要原因在于我们的服务不大,消息量也比较小,因此在不影响程序的架构基础上,采用了Redis 作为消息队列。

    消息队列的关键点在于,当生产者发布消息后,要确保消费者能够快速的接收消息。发布订阅模式能够很好的帮我们解决,当有消息到达的时候,程序马上能够做出响应操作。

    Redis消息发布:

  1. import redis
  2.  
  3. pool = redis.ConnectionPool(host='ip', port=6379, db=4, password='****')
  4. r = redis.StrictRedis(connection_pool=pool)
  5. while True:
  6. inputs = input("publish:")
  7. r.publish('spub', inputs)
  8. if inputs == 'over':
  9. print('停止发布')
  10. break

   Redis消息订阅:

  1. import redis
  2. pool = redis.ConnectionPool(host='ip', port=6379, db=4, password='***')
  3. r = redis.StrictRedis(connection_pool=pool)
  4. p = r.pubsub()
  5. p.subscribe('cardniu-monitor')
  6. for item in p.listen():
  7. print(item)
  8. if item['type'] == 'message':
  9. data = item['data']
  10. print("消息队列中接收到信息:", data)if item['data'] == 'over':
  11. break
  12. p.unsubscribe('cardniu-monitor')
  13. print('取消订阅')

  2.4 wxpy+Redis 实现监控系统的消费者

    最终,在python 这边实现的监控系统消费者如下:

    微信登陆模块:

  1. from wechat_sender import *
  2. from wxpy import *
  3.  
  4. bot = Bot(qr_path="qr.png")
  5.  
  6. group = bot.groups().search('监控报警')[0]
  7.  
  8. print("微信登陆成功!进行监控报警功能!")
  9. print(group)
  10.  
  11. #
  12. listen(bot, token='test', receivers=[group])

    Redis消息订阅模块:

  1. import redis
  2. from wechat_sender import *
  3.  
  4. sender = Sender(token='test', receivers='监控报警')
  5.  
  6. pool = redis.ConnectionPool(host='ip', port=6379, db=13, password='***')
  7. r = redis.StrictRedis(connection_pool=pool)
  8. p = r.pubsub()
  9. p.subscribe('cardniu-monitor')
  10. for item in p.listen():
  11. print(item)
  12. if item['type'] == 'message':
  13. data = item['data']
  14. print("消息队列中接收到信息:", data)
        
  15. sender.send(message=data)
  16. if item['data'] == 'over':
  17. break
  18. p.unsubscribe('cardniu-monitor')
  19. print('取消订阅')

3、Java-Redis

    最后,在生产者这块,即是我们监控系统的核心部分,当我们的Java系统出现异常时,我们即可向Redis发送消息,最后由消费者那一边完成消息的发送。

    在下面会跟大家简单讲解一下生产者这边的代码,但是由于代码设计公司内容,因此不做过多的描述。

    Spring-redis.xml

  1. <?xml version="1.0" encoding="UTF-8"?>
  2. <beans xmlns="http://www.springframework.org/schema/beans"
  3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"
  4. xmlns:util="http://www.springframework.org/schema/util" xmlns:aop="http://www.springframework.org/schema/aop"
  5. xsi:schemaLocation="
  6. http://www.springframework.org/schema/beans
  7. http://www.springframework.org/schema/beans/spring-beans-3.1.xsd
  8. http://www.springframework.org/schema/context
  9. http://www.springframework.org/schema/context/spring-context-3.1.xsd
  10. http://www.springframework.org/schema/util
  11. http://www.springframework.org/schema/util/spring-util-3.1.xsd
  12. http://www.springframework.org/schema/aop
  13. http://www.springframework.org/schema/aop/spring-aop-3.1.xsd">
  14.  
  15. <!-- redis连接池的配置 -->
  16. <bean id="jedisPoolConfig" class="redis.clients.jedis.JedisPoolConfig">
  17. <property name="maxTotal" value="${redis.pool.maxTotal}" />
  18. <property name="maxIdle" value="${redis.pool.maxIdle}" />
  19. <property name="minIdle" value="${redis.pool.minIdle}" />
  20. <property name="maxWaitMillis" value="${redis.pool.maxWaitMillis}" />
  21. <property name="testOnBorrow" value="${redis.pool.testOnBorrow}" />
  22. <property name="testOnReturn" value="${redis.pool.testOnReturn}" />
  23. </bean>
  24.  
  25. <bean id="sentinelJedisPool" class="redis.clients.jedis.JedisSentinelPool">
  26. <constructor-arg index="0" value="${redis.sentinel.masterName}" />
  27. <constructor-arg index="1"
  28. value="#{'${redis.sentinels}'.split(',')}" />
  29. <constructor-arg index="2" ref="jedisPoolConfig" />
  30. <constructor-arg index="3" value="${redis.sentinel.timeout}"
  31. type="int" />
  32. <constructor-arg index="4" value="${redis.sentinel.password}" />
  33. <constructor-arg index="5" value="${redis.sentinel.database}"
  34. type="int" />
  35. </bean>
  36. </beans>

    JedisUtils.java

  1. @Autowired
  2. private JedisSentinelPool jedisPool;
  3.  
  4. @PostConstruct
  5. private void init() throws Exception {
  6. /* 缓存初始化 */
  7. JedisUtils.setJedisPool(jedisPool);
  8. }
  9.  
  10. public static void setJedisPool(Pool<Jedis> jedisPool) throws Exception {
  11. JedisCache.jedisPool = jedisPool;
  12. Jedis jedis = null;
  13. try {
  14. jedis = jedisPool.getResource();
  15. isInitSuc = true;
  16. logger.info("redis start success!");
  17. } catch (Exception e) {
  18. if (null != jedis)
  19. jedisPool.returnBrokenResource(jedis);
  20. logger.error("redis start exception!!!error:{}", e.getMessage(), e);
  21. if (e instanceof redis.clients.jedis.exceptions.JedisConnectionException) {
  22. throw e;
  23. }
  24. } finally {
  25. if (null != jedis)
  26. jedisPool.returnResource(jedis);
  27. }
  28. }
  29.  
  30. public static Long publish(String chanel, String value) {
  31. Jedis jedis = null;
  32. try {
  33. jedis = jedisPool.getResource();
  34. return jedis.publish(chanel,value);
  35. } catch (Exception e) {
  36. if (null != jedis) {
  37. jedisPool.returnBrokenResource(jedis);
  38. jedis = null;
  39. }
  40. logger.error("redis exception:{}", e.getMessage(), e);
  41. return 0L;
  42. } finally {
  43. if (null != jedis)
  44. jedisPool.returnResource(jedis);
  45.  
  46. }
  47. }

    NoticeTask.java

  1. @Scheduled(cron = "*/5 * * * * ? ")
  2. public void runMonitor() {
  3. try {
  4.  
  5. List<T> notices;
  6. List<EbankNotice> result;
  7. while ((notices = QueueHolder.noticeBlockingQueue.take()) != null) { //消费
  8. if (notices.isEmpty()) {
  9. continue;
  10. }
  11. result = service.calculateNotice(notices);
  12. result.forEach(notice -> {
  13. JedisUtils.publish("cardniu-monitor", notice.getTitle() + "," +
  14. DateUtils.format(notice.getPostTime(), "yyyy-MM-dd hh:mm:ss") + "," + notice.getLinkAddress());
  15. });
  16. }
  17. } catch (InterruptedException e) {
  18. logger.error("发送邮件定时任务异常,{}", e.getMessage(), e);
  19. }
  20. }

    

4、总结

     这个项目的核心在于wxpy 的运用,以及生产者消费者的设计思想。语言的话,核心的python这一块的wxpy,在生产者这边,无论是其他的什么语言,都可以作为我们的生产者。

     项目github地址:https://github.com/jaycekon/WxpyDemo

     希望各位喜欢的朋友可以fork 或者start一下~~~~~

参考:

wxpy API:http://wxpy.readthedocs.io/zh/latest/messages.html

wechat_sender Api:https://pypi.python.org/pypi/wechat-sender/0.1.3

python-redis :https://pypi.python.org/pypi/redis

Java-Redis:http://docs.spring.io/spring-data/redis/docs/2.0.0.M4/reference/html/  

Python-WXPY实现微信监控报警的更多相关文章

  1. zabbix实现企业微信监控报警

    一.zabbix基本说明 简介:zabbix基于Web界面的分布式系统监控的企业级开源软件.可以监控各种系统与设备,网络参数,保证服务器设备安全运营:提供灵活的通知机制.如果检测到的指标不达标,就实现 ...

  2. zabbix之微信告警(python版):微信个人报警,微信企业号告警脚本

    微信个人告警脚本 微信个人告警:使用个人微信,发送到微信群组,微信好友 两个脚本执行: 1)能连接网络2)先执行server.py,扫描登录微信,登录之后没有报错,打开新终端查看端口是否起来了3)在z ...

  3. Python+wxpy 实现微信消息轰炸

    需要导入wxpy,在终端中输入以下命令即可 pip install wxpy 如果没有pip先安装pip,安装好了的直接输入命令即可,安装好了但是显示没有安装的可能是没有将pip添加到PATH中,需要 ...

  4. Zabbix 微信监控报警

    Zabbix-Server 设置 # 使脚本目录生效 [root@zabbix ~]# grep 'script' /etc/zabbix/zabbix_server.conf # AlertScri ...

  5. Zabbix 微信报警Python版(带监控项波动图片)

    #!/usr/bin/python # -*- coding: UTF- -*- #Function: 微信报警python版(带波动图) #Environment: python import ur ...

  6. python实现微信发送服务器监控报警消息代码实现

    这篇文章主要介绍了python3.8 微信发送服务器监控报警消息代码实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友可以参考下 ! python版本 > ...

  7. python实现服务器监控报警消息用微信发送(附代码)

    前言 文的文字及图片来源于网络,仅供学习.交流使用,不具有任何商业用途,版权归原作者所有,如有问题请及时联系我们以作处理. 作者:NicePython PS:如有需要Python学习资料的小伙伴可以加 ...

  8. Kubernetes集群的监控报警策略最佳实践

    版权声明:本文为博主原创文章,未经博主同意不得转载. https://blog.csdn.net/M2l0ZgSsVc7r69eFdTj/article/details/79652064 本文为Kub ...

  9. Python玩转微信小程序

    用Python玩转微信   Python玩转微信 大家每天都在用微信,有没有想过用python来控制我们的微信,不多说,直接上干货!  这个是在 itchat上做的封装  http://itchat. ...

随机推荐

  1. [C++][OpenGL]自己写GUI(0)——介绍

    文章可转载,转载请注明出处:http://www.cnblogs.com/collectionne/p/6928612.html.文章未完,如果不在博客园(cnblogs)发现本文,请访问前面的链接查 ...

  2. Python教程(1.1)——配置Python环境

    在正式开始学习Python之前我们需要先配置好Python环境. Python Python可以从Python官方网站上,选择适合你的操作系统的版本下载.下载完之后,运行下载的可执行文件进行安装. 这 ...

  3. sql 经典面试题

     如果数据库里两个日期型字段d1,d2,怎样用sql语句列出按月的所有区间,比如表结构如下localid    d1          d21         2014-1-15    2014-3- ...

  4. APUE-文件和目录(七)符号链接

    符号链接 符号链接的用途 符号链接是对一个文件的间接指针,它与前面介绍的硬连接不同,硬连接指向文件的i节点.引入符号链接是为了避开硬连接的一些限制: 硬链接通常要求链接和文件位于同一文件系统中. 只有 ...

  5. 解决laydate时间日期插件定位溢出

    laydate是一款比较好用的网页时间日期插件,不过用起来有一些细节问题需要我们手动去解决!例如:laydate兼容bootstrap 1. 默认情况 laydate弹出层默认对齐input左边框 2 ...

  6. 可选参数、命名参数、.NET的特殊类型、特性

    1.可选参数和命名参数    1.1可选参数        语法:            [修饰符] 返回类型 方法名(必选参数n,可选参数n)        注意:            1.必选参 ...

  7. 【译】Reflection.Emit vs. CodeDOM

    原文:http://ayende.com/blog/1606/reflection-emit-vs-codedom Both technologies allow you to generate ex ...

  8. java 父类构造器

    当创建任何java对象时,程序总会首先调用系统的父类非静态初始化块(隐式执行)和父类构造器(从object开始(java程序中所有类的最终父类都是java.lang.Object类,使用语句super ...

  9. 查找oracle自己用户的表

    查找oracle自己用户的表 select table_name from user_tables;

  10. Android 串口设置校验位、速率、停止位等参数

    Android 串口通讯设置校验位.速率.停止位等参数 最近业余时间有个android项目需要和硬件进行通信,通讯方式都是通过串口,android串口通讯google有个开源的demo 和很多人一样我 ...