不花钱就能够给企业微信作个提醒机器人

李成熙,Shopee Airpay 前端 Leader。2014年毕业加入腾讯AlloyTeam,前后负责过QQ群、花样直播、腾讯文档等项目。后于2018年加入腾讯云云开发团队。专一于性能优化、工程化和小程序服务。微博 | 知乎 | Github前端

到了新公司以后,发现竟然也是用企业微信。但惋惜的是,外部的企业微信竟然没有机器人。这对之前在鹅厂里习惯用企业微信作提醒的我以为很不方便。终于,7月一开始企业微信终于上线机器人功能。node

右击群聊天卡片,能够添加群机器人。python

悬浮在机器人的头像上,会显示出 Webhook 地址。点击这个地址,会跳到机器人的开发文档。ios

提醒机器人的开发其实很简单,其实就是向这个webhook地址,按文档提供的格式发送请求,就能够实现消息推送了。最简单的示例,能够用 Node.jsaxios 类库:git

const axios = require('axios')

async function bookLunch() {
    let result = await axios.post(baseUrl, {
        msgtype: 'text',
        text: {
            content: '大佬,订午饭啦!',
            mentioned_list: ['@all'] // 可使用邮箱或者手机号码
        }
    })

    return result.data
}

bookLunch.then((res) => {
    console.log(res)
})
复制代码

以上是最简单的例子。除了普通文本内容,还能够发送 markdown,图文等内容,你们能够自行去看文档。github

但问题来了:通常来讲提醒,都是须要定时的,好比说每早提醒你们写计划,每周五傍晚提醒你们写周报,怎么可让机器人在这些时间点出现提醒你们呢?你可能会想到买一台服务器,而后在上面部署 cronjob 服务,定时去调度服务。没错,这当然是最通俗的作法。可是买一台服务器要花钱呀,便宜的也得几十块钱一台虚拟机,并且只在上面跑一个这么简单的服务显然是不值的。有没有性价比高的作法呢?有,用云函数!web

我我的的理解,云函数跟传统的服务主要的区别有几点,一个是它是一种事件型的服务,由不一样的事件触发(HTTP、数据更改、对象存储的变动等),第二个它是非长驻的,运行必定时间后会冷却或者销毁,第三个因为以上两种特性,对于一些负载不是很高的服务,用云函数比较省钱。而对于这种提醒机器人,正正是一种负载不是很高的服务,很是合适。对小型团队的这种提醒服务,在最近各大厂商都在推广的时期,真的能够作到不要钱。npm

这里我对腾讯云的云函数最为熟悉,所以就用它来作实践。axios

首先为了方便,咱们能够用腾讯云提供的 SCF CLI 来初始化咱们的云函数和配置文件。我用的电脑是 Macbook,能够直接安装如下的命令进行安装:小程序

pip install scf
复制代码

若是不是Macbook能够先自行安装 pythonpip

而后就是进行配置:

scf configure set --region ap-guangzhou --appid 1253970223 --secret-id AKIxxxxxxxxxx --secret-key uxxlxxxxxxxx
复制代码

appid, secret-idsecret-key 能够在访问密钥页面里拿到。至于 region,则是你想部署云函数的区域,比方说在云函数的控制台首页,就能看到顶部的区域。选广州就是 ap-guangzhou,选香港的就是 ap-hongkong。基本上是 ap- 加上国内市场的拼音或国外城市的英文。

image.png

而后我们初始化好项目(用node.js 8.9版本写云函数):

# 初始化云函数
scf init --runtime nodejs8.9 --name wework-robot

cd wework-robot

# 初始化 node 项目
npm init -y
复制代码

而后就能获得该云函数:

image.png

此次要用到 axios,那咱们就安装这个依赖:

npm i --save axios
复制代码

打开 index.js 是以下一段代码,async 表示该函数能够用 Node.js 的新特性 async/await

'use strict';
exports.main_handler = async (event, context, callback) => {
    console.log("%j", event);
    return "hello shopee!"
};
复制代码

我进行一些删减后,成这样。将函数名字改成 main,并且因为用 async/await 就能够不用 callback 处理异步了。但改了名字也要改 template.yaml,将 main_handler 改成 main

exports.main = async (event, context) => {
    return "hello shopee!"
};
复制代码

image.png

好了。是时候来写提醒逻辑了。逻辑并不难,但主要注意的一点是时间。通过试验,云函数这里的时间统一使用了标准的国际时间,也就是北京时间要比它晚8小时,详细逻辑能够看如下代码的注释:

const axios = require('axios')
const baseUrl =
    'https://qyapi.weixin.qq.com/cgi-bin/webhook/send?key=7f399641-40aa-45c8-ad3d-c46e1ee085a7'

async function bookLunch() {
    let result = await axios.post(baseUrl, {
        msgtype: 'text',
        text: {
            content: '大佬,订午饭啦!',
            mentioned_list: ['@all'] // 提醒全部人
        }
    })

    return result.data
}

async function bookTaxi() {
    let result = await axios.post(baseUrl, {
        msgtype: 'text',
        text: {
            content: '辛苦了,早点回家休息吧。9点打车能够报销哦。',
            mentioned_list: ['@all']
        }
    })

    return result.data
}

async function remindWeeklyReport() {
    let result = await axios.post(baseUrl, {
        msgtype: 'text',
        text: {
            content: '周五了,记得写周报看看你这周有没偷懒!',
            mentioned_list: ['@all']
        }
    })

    return result.data
}

async function remindGoHome() {
    let result = await axios.post(baseUrl, {
        msgtype: 'text',
        text: {
            content: '11点半了,早点休息吧!'
        }
    })

    return result.data
}

// 是否周五
function isFriday(day) {
    return day === 5
}

// 是否工做日
function isWeekDay(day) {
    return day > 0 && day < 6
}

// 是否30分,多预留1分钟以防云函数延迟启动或执行
function isHalfHour(min) {
    return min >= 30 && min <= 31
}

// 是否正点,多预留1分钟以防云函数延迟启动或执行
function isSharp(min) {
    return min >= 0 && min <= 1
}

exports.main = async (event, context) => {
    let d = new Date() // js 时间对象
    let day = d.getDay() // 获取今天是星期几,0 表示周日
    let hour = d.getHours() // 获取当前的 时
    let min = d.getMinutes() // 获取当前的 分

    let hourGap = 8 // 我们在东8区
    hour += hourGap // 获取当前准确的时间数

   // 打一下 log 看看具体时间
    console.log(`day: ${day} hour: ${hour} min: ${min} hourGap: ${hourGap}`)

    // 每周五4点到4点半通知写周报
    if (isFriday(day) && hour === 4 && isHalfHour(min)) {
        return await remindWeeklyReport()
    }

    // 工做日天天11点提醒订餐
    if (isWeekDay(day) && hour === 11 && isSharp(min)) {
        return await bookLunch()
    }

    // 工做日天天晚上9点提醒打车能够报销
    if (isWeekDay(day) && hour === 21 && isSharp(min)) {
        return await bookTaxi()
    }

    // 工做日天天晚上11点半提醒休息
    if (isWeekDay(day) && hour === 23 && isHalfHour(min)) {
        return await remindGoHome()
    }

    return 'hi shopee!'
}
复制代码

逻辑都写好了,可是,咱们须要让它定时执行,好比每30分钟执行一次。这个时候,咱们就须要添加“定时触发器” 了。定时触发咱们能够在 template.yaml 里面添加,能够把注释去掉,而后修改获得:

image.png

CronExpression 具体能够参考这个文档:cloud.tencent.com/document/pr…

请使用推荐的写法:

image.png

这里有些参考的示例,直接套用便可:

image.png

我这里写的:0 */30 * * * MON-FRI *,表示每周一到周五,每30分钟会触发一次云函数的调用。

固然,咱们还想开启一下 HTTP 触发器,来用地址直接访问该云函数进行一些逻辑的调试,看看是否真的能成功发消息。

咱们能够再到 template.yaml 里添加这样的 HTTP 触发器:

image.png

好了,万事俱备,咱们只须要再用 SCF CLI 发布便可。

# 打包
scf package -t template.yaml 
Generate deploy file 'deploy.yaml' success

# 发布
scf deploy -t deploy.yaml 
Deploy function 'wework-robot' success
复制代码

发布完成后,咱们能够到腾讯云的控制台看下,已经存在了:

image.png

点进去看看触发方式,发现分别有一个定时触发器,一个API网关触发器(HTTP触发)

如此,便大功告成了!看看效果:

相关文章
相关标签/搜索