自从用邮箱注册了很多账号后,便会收到诸如以下类似的邮件,刚开始还以为是一张图片,后来仔细一看不是图片呀,好像还是 HTML 呀,于是好奇宝宝我 Google 一下,查阅多篇资料后总结出怎么用前端知识和 Node 做一个这样的“邮件网页”。
知道怎么实现功能后,思考着我该写什么主题呢,用一个 HTML 模板随便给小伙伴们发个邮件炫个技?不行,作为一个很 cool 的程序员怎么能这么 low 呢,最近天气变化幅度大,温度捉摸不定,女朋友总是抱怨穿少了又冷穿多了又热,嗨呀,要不我就写个每天定时给宝宝发送天气预报的邮件,另外想起宝宝喜欢看 ONE·一个这个 APP 上的每日更新,要不发天气预报的同时,再附赠一个“ONE 的每日订阅”?机智又浪漫,开始搬砖~
本来是想最后放效果图的,怕你们看到一半就没兴趣了,就在前面剧透一下我最后做出来的效果图吧~
1. 如何获取天气预报和 ONE 上的 data?
答:获取 data 有两种方法,第一种方法是获取天气预报和 ONE 的 API,第二种是用 node 爬虫获取天气预报和 ONE 网页的信息。后来找了下,发现 ONE 并没有 API 接口,为了让两者统一,于是决定使用 node 上的一个插件叫cheerio,配合superagent能够很方便地爬取网页上的信息。
2. 如何做出 HTML 的这种邮件?
答:之前学过一段时间的 express 这个框架,接触到模版引擎这个概念,传入 data 便可获得 html 文件,再结合 node 的 fs 模块,获取到这个 html 文件,便可以结合 node 的邮件插件发送 HTML 邮件啦!
3. 如何用 node 发送邮件?
感谢无私的开源开发者,开发了一款发送邮件的 Node 插件nodemailer,兼容主流的 Email 厂商,只需要配置好邮箱账号和 smtp 授权码,便可以用你的邮箱账号在 node 脚本上发文件,很 cool 有没有~
4. 如何做到每日定时发送?
其实可以通过各种 hack 的方式写这么一个定时任务,但是既然 node 社区有这个定时的轮子,那我们直接用就好了,node-schedule是一个有着各种配置的定时任务发生器,可以定时每个月、每个礼拜、每天具体什么时候执行什么任务,这正符合每天早晨定时给宝宝发送邮件的需求。
一切准备就绪,开始做一次浪漫的程序员
这里我们使用到superagent和cheerio组合来实现爬虫:
- 分析网页 DOM 结构,如下图所示:
- 用 superagent 来获取指定网页的所有 DOM:
superagent.get(URL).end(function(err,res){
//
}- 用 cheerio 来筛选 superagent 获取到的 DOM,取出需要的 DOM
imgUrl:$(todayOne).find('.fp-one-imagen').attr('src'),
type:$(todayOne).find('.fp-one-imagen-footer').text().replace(/(^\s*)|(\s*$)/g, ""),
text:$(todayOne).find('.fp-one-cita').text().replace(/(^\s*)|(\s*$)/g, "")以下就是爬取 ONE 的代码,天气预报网页也是一个道理:
const superagent = require('superagent') //发送网络请求获取DOM
const cheerio = require('cheerio') //能够像Jquery一样方便获取DOM节点
const OneUrl = 'http://wufazhuce.com/' //ONE的web版网站
superagent.get(OneUrl).end(function(err, res) {
if (err) {
console.log(err)
}
let $ = cheerio.load(res.text)
let selectItem = $('#carousel-one .carousel-inner .item')
let todayOne = selectItem[0] //获取轮播图第一个页面,也就是当天更新的内容
let todayOneData = {
//保存到一个json中
imgUrl: $(todayOne)
.find('.fp-one-imagen')
.attr('src'),
type: $(todayOne)
.find('.fp-one-imagen-footer')
.text()
.replace(/(^\s*)|(\s*$)/g, ''),
text: $(todayOne)
.find('.fp-one-cita')
.text()
.replace(/(^\s*)|(\s*$)/g, '')
}
console.log(todayOneData)
})通过爬虫获取到了数据,那么我们就能够通过将 date 输入到 EJS 渲染出 HTML,我们在目录下创建 js 脚本和 ejs 模版文件:
- app.js
const ejs = require('ejs') //ejs模版引擎
const fs = require('fs') //文件读写
const path = require('path') //路径配置
//传给EJS的数据
let data = {
title: 'nice to meet you~'
}
//将目录下的mail.ejs获取到,得到一个模版
const template = ejs.compile(
fs.readFileSync(path.resolve(__dirname, 'mail.ejs'), 'utf8')
)
//将数据传入模版中,生成HTML
const html = template(data)
console.log(html)- mail.ejs
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<meta http-equiv="X-UA-Compatible" content="ie=edge">
<title>Document</title>
</head>
<body>
<h1>
<%= title %>
</h1>
</body>
</html>这里我们可以发送纯 text 也可以发送 html,注意的是邮箱密码不是你登录邮箱的密码,而是 smtp 授权码,什么是 smtp 授权码呢?就是你的邮箱账号可以使用这个 smtp 授权码在别的地方发邮件,一般 smtp 授权码在邮箱官网的设置中可以看的到,设置如下注释。
const nodemailer = require('nodemailer') //发送邮件的node插件
let transporter = nodemailer.createTransport({
service: '126', // 发送者的邮箱厂商,支持列表:https://nodemailer.com/smtp/well-known/
port: 465, // SMTP 端口
secureConnection: true, // SSL安全链接
auth: {
//发送者的账户密码
user: '账户@126.com', //账户
pass: 'smtp授权码' //smtp授权码,到邮箱设置下获取
}
})
let mailOptions = {
from: '"发送者昵称" <地址@126.com>', // 发送者昵称和地址
to: 'like@vince.studio', // 接收者的邮箱地址
subject: '一封暖暖的小邮件', // 邮件主题
text: 'test mail' //邮件的text
// html: html //也可以用html发送
}
//发送邮件
transporter.sendMail(mailOptions, (error, info) => {
if (error) {
return console.log(error)
}
console.log('邮件发送成功 ID:', info.messageId)
})这里我们用到了node-schedule来定时执行任务,示例如下:
var schedule = require('node-schedule')
//1. 确定的时间执行
var date = new Date(2017, 12, 10, 15, 50, 0)
schedule.scheduleJob(date, function() {
console.log('执行任务')
})
//2. 秒为单位执行
//比如:每5秒执行一次
var rule1 = new schedule.RecurrenceRule()
var times1 = [1, 6, 11, 16, 21, 26, 31, 36, 41, 46, 51, 56]
rule1.second = times1
schedule.scheduleJob(rule1, function() {
console.log('执行任务')
})
//3.以分为单位执行
//比如:每5分种执行一次
var rule2 = new schedule.RecurrenceRule()
var times2 = [1, 6, 11, 16, 21, 26, 31, 36, 41, 46, 51, 56]
rule2.minute = times2
schedule.scheduleJob(rule2, function() {
console.log('执行任务')
})
//4.以天单位执行
//比如:每天6点30分执行
var rule = new schedule.RecurrenceRule()
rule.dayOfWeek = [0, new schedule.Range(1, 6)]
rule.hour = 6
rule.minute = 30
var j = schedule.scheduleJob(rule, function() {
console.log('执行任务')
getData()
})当所有的问题都解决后,便是开始结合代码成一段完整的程序,思路很简单,我们来逐步分析:
- 由于获取数据是异步的,并且不能判断出哪个先获取到数据,这个是可以将获取数据的函数封装成一个 Promise 对象,最后在一起用 Promise.all 来判断所有数据获取完毕,再发送邮件
// 其中一个数据获取函数,其他的也是类似
function getOneData() {
let p = new Promise(function(resolve, reject) {
superagent.get(OneUrl).end(function(err, res) {
if (err) {
reject(err)
}
let $ = cheerio.load(res.text)
let selectItem = $('#carousel-one .carousel-inner .item')
let todayOne = selectItem[0]
let todayOneData = {
imgUrl: $(todayOne)
.find('.fp-one-imagen')
.attr('src'),
type: $(todayOne)
.find('.fp-one-imagen-footer')
.text()
.replace(/(^\s*)|(\s*$)/g, ''),
text: $(todayOne)
.find('.fp-one-cita')
.text()
.replace(/(^\s*)|(\s*$)/g, '')
}
resolve(todayOneData)
})
})
return p
}- 将爬取数据统一处理,作为 EJS 的参数,发送邮件模板。
function getAllDataAndSendMail() {
let HtmlData = {}
// how long with
let today = new Date()
let initDay = new Date(startDay)
let lastDay = Math.floor((today - initDay) / 1000 / 60 / 60 / 24)
let todaystr =
today.getFullYear() +
' / ' +
(today.getMonth() + 1) +
' / ' +
today.getDate()
HtmlData['lastDay'] = lastDay
HtmlData['todaystr'] = todaystr
Promise.all([getOneData(), getWeatherTips(), getWeatherData()])
.then(function(data) {
HtmlData['todayOneData'] = data[0]
HtmlData['weatherTip'] = data[1]
HtmlData['threeDaysData'] = data[2]
sendMail(HtmlData)
})
.catch(function(err) {
getAllDataAndSendMail() //再次获取
console.log('获取数据失败: ', err)
})
}- 发送邮件具体代码
function sendMail(HtmlData) {
const template = ejs.compile(
fs.readFileSync(path.resolve(__dirname, 'email.ejs'), 'utf8')
)
const html = template(HtmlData)
let transporter = nodemailer.createTransport({
service: EmianService,
port: 465,
secureConnection: true,
auth: EamilAuth
})
let mailOptions = {
from: EmailFrom,
to: EmailTo,
subject: EmailSubject,
html: html
}
transporter.sendMail(mailOptions, (error, info = {}) => {
if (error) {
console.log(error)
sendMail(HtmlData) //再次发送
}
console.log('Message sent: %s', info.messageId)
})
}如果你觉得这封邮件的内容适合你发送的对象,可以按照以下步骤,改少量参数即可运行程序;
- git clone https://github.com/Vincedream/NodeMail
- 打开 main.js,修改配置项
//纪念日
let startDay = '2016/6/24'
//当地拼音,需要在下面的墨迹天气url确认
const local = 'zhejiang/hangzhou'
//发送者邮箱厂家
let EmianService = '163'
//发送者邮箱账户SMTP授权码
let EamilAuth = {
user: 'xxxxxx@163.com',
pass: 'xxxxxx'
}
//发送者昵称与邮箱地址
let EmailFrom = '"name" <xxxxxx@163.com>'
//接收者邮箱地
let EmailTo = 'like@vince.studio'
//邮件主题
let EmailSubject = '一封暖暖的小邮件'
//每日发送时间
let EmailHour = 6
let EmialMinminute = 30- 终端输入
npm install安装依赖,再输入node main.js,运行脚本,当然你的电脑不可能不休眠,建议你部署到你的云服务器上运行。
冬天到了,是不是也该用程序员的专业知识给身边的人带来一些温暖呢,源代码与 demo 已经放到 github 上,要不试一试?
GitHub:https://github.com/Vincedream/NodeMail
使用 mipang 天气来进行更新,注意只支持海外天气站点,因为国内和国外天气页面不一样
所以新增了一个foreign.ejs的模板,来发送海外天气邮件
在main.js中部分代码修改如下
// 爬取数据的url
const OneUrl = 'http://wufazhuce.com/'
//const WeatherUrl = "https://tianqi.moji.com/weather/china/" + local;
const WeatherUrl = 'https://weather.mipang.com/tianqi-315811' //奥斯纳布吕克,请手动打开找自己需要的海外城市天气地址
// 获取国外天气预报(米胖api)
function getWeatherData() {
let p = new Promise(function(resolve, reject) {
superagent.get(WeatherUrl).end(function(err, res) {
if (err) {
reject(err)
}
let threeDaysData = []
let weatherTip = ''
let $ = cheerio.load(res.text)
$('.item').each(function(i, elem) {
const SingleDay = $(this).find('.tt')
threeDaysData.push({
Day:
$(SingleDay[0])
.find('.week')
.text()
.replace(/(^\s*)|(\s*$)/g, '') +
$(SingleDay[0])
.find('.day')
.text(),
WeatherImgUrl: $(SingleDay[2])
.find('img')
.attr('src'),
WeatherText: $(SingleDay[3])
.text()
.replace(/(^\s*)|(\s*$)/g, ''),
Temperature:
$(SingleDay[1])
.find('.temp1')
.text()
.replace(/(^\s*)|(\s*$)/g, '') +
'-' +
$(SingleDay[1])
.find('.temp2')
.text()
.replace(/(^\s*)|(\s*$)/g, ''),
WindDirection: $(SingleDay[4])
.find('img')
.attr('title'),
WindLevel: $(SingleDay[5])
.text()
.replace(/(^\s*)|(\s*$)/g, '')
})
})
resolve(threeDaysData)
})
})
return p
}
// 发动邮件
function sendMail(HtmlData) {
const template = ejs.compile(
//fs.readFileSync(path.resolve(__dirname, "email.ejs"), "utf8")
fs.readFileSync(path.resolve(__dirname, 'foreign.ejs'), 'utf8') //注释掉上面一行,改为这一行利用foreign模板
)
const html = template(HtmlData)
let transporter = nodemailer.createTransport({
service: EmianService,
port: 465,
secureConnection: true,
auth: EamilAuth
})
let mailOptions = {
from: EmailFrom,
to: EmailTo,
subject: EmailSubject,
html: html
}
transporter.sendMail(mailOptions, (error, info = {}) => {
if (error) {
console.log(error)
sendMail(HtmlData) //再次发送
}
console.log('邮件发送成功', info.messageId)
console.log('静等下一次发送')
})
}然后按照上面的正常运行即可
最近发生了一点小事,然后我把收件人改成自己了,每天早晨 6 点都能收到自己的一份“暖暖的邮件”,非常感动,想把这份温暖传递给更多人,于是想到了加一个群发模块
实际上 nodemailer 是有群发功能的,只要在收件人处写一个 array 即可,但我偏偏不这么简单的搞,一定要用上数据库!要学习一下!
技术栈:SQL,SQLite,Node.js
为什么选择SQLite呢,首先因为东西就像Excel一样是一个单独的数据库文件,非常便于携带,同时查询又比较方便,相比MySQL、MongoDB更加的轻量化,此外因为是Pm2挂Node,所以我不想单独安装一个MongoDB,搞MySQL的话权限什么的又特别复杂我又怕和Apache冲突出事,所以就选择了SQLite
什么进入退出程序还是要知道的
由于 macOS 系统自带 sqlite,直接在 terminal 中输入 sqlite3 即可进入
$ sqlite3
SQLite version 3.28.0 2019-04-15 14:49:49
Enter ".help" for usage hints.
Connected to a transient in-memory database.
Use ".open FILENAME" to reopen on a persistent database.
sqlite>
在 sqlite 命令行内输入.quit或者.exit
SQLite version 3.28.0 2019-04-15 14:49:49
Enter ".help" for usage hints.
Connected to a transient in-memory database.
Use ".open FILENAME" to reopen on a persistent database.
sqlite> .exit
SQLite version 3.28.0 2019-04-15 14:49:49
Enter ".help" for usage hints.
Connected to a transient in-memory database.
Use ".open FILENAME" to reopen on a persistent database.
sqlite> .quit
.schema email
CREATE TABLE email(
ID INTEGER PRIMARY KEY AUTOINCREMENT,
name CHAR(20) NOT NULL,
dept CHAR(20) NOT NULL,
email CHAR(50)
);
其他操作和 SQL 操作别无二致,不讲了,感觉基本通用的
sqlite>.header on
sqlite>.mode column
sqlite>.timer on
sqlite>
创建表
sqlite> CREATE TABLE email(
ID INTEGER PRIMARY KEY AUTOINCREMENT,
name CHAR(20) NOT NULL,
dept CHAR(20) NOT NULL,
email CHAR(50)
);
加入基础数据
INSERT INTO email (name,dept,email)
VALUES ( 'A', 'fuwubu', 'a@gmail.com');
INSERT INTO email (name,dept,email)
VALUES ( 'B', 'fuwubu', 'b@gmail.com');
INSERT INTO email (name,dept,email)
VALUES ( 'C', 'fuwubu', 'c@gmail.com');
然后输入select * from email;就可以输出啦
sqlite> select * from email;
select * from email;
ID name dept email
---------- ---------- ---------- -----------
1 A fuwubu a@gmail.com
2 B fuwubu b@gmail.com
3 C fuwubu c@gmail.com
Run Time: real 0.000 user 0.000139 sys 0.000103
var sqlite3 = require('sqlite3').verbose()
var file = './subscribeList.db' //这里写的就是数据库文件的路径
var db = new sqlite3.Database(file)
function getEmail() {
let p = new Promise(function(resolve, reject) {
db.all('select email from email', function(err, row) {
for (let key in row) {
emailList.push(row[key].email)
}
})
setTimeout(() => {
resolve(emailList)
}, 100)
})
return p
}
Promise.all([getEmail()])
.then(function(data) {
console.log(JSON.stringify(data[0]))
})
.catch(function(err) {
console.log('获取数据失败: ', err)
})在最后一个 Promise 这里就需要添加其他函数了,然后就可以获得对应的数据了
输出结果:
[ 'a@gmail.com', 'b@gmail.com', 'c@gmail.com' ]Promise.all([getOneData(), getWeatherTips(), getWeatherData(), getEmail()])
.then(function(data) {
HtmlData['todayOneData'] = data[0]
HtmlData['weatherTip'] = data[1]
HtmlData['threeDaysData'] = data[2]
HtmlData['email'] = data[3]
sendMail(HtmlData)
})
.catch(function(err) {
getAllDataAndSendMail() //再次获取
console.log('获取数据失败: ', err)
})然后修改该修改的邮件以及需要发送的邮件地址即可,用 pm2 执行就行啦~
本 branch 地址:https://github.com/Andrew8102/NodeMail
累死我了,但总之感恩节快乐!!!


