用 Puppeteer 把繁琐工作给自动化了,太爽啦!_世界动态

最近邀请了一位前端大佬 @神说要有光 入驻我们,给大家输出一些干货内容。


(资料图片仅供参考)

结果他遇到了一个令人头疼的问题:

用知识星球的编辑器写 Markdown 也太难受了!

比如他在掘金编辑器里这样的 markdown 内容:

复制到星球编辑器是这样的:

markdown 语法是识别了,但图片没有自动上传。

如果用富文本格式,格式又不对:

而且 gif 没有识别出来,还是需要手动传一次。

这意味着如果文中有几十张图片,那他需要单独把这几十张图片保存到本地,然后光标定位到对应位置,点击上传图片,把图片插进去。

也就是这样:

把每个图片下载下来,保存为不同的后缀名(png、jpg、gif),然后再定位到对应位置,删除原来的链接,插入图片。

然后这样重复十几次,每篇文章都这样来一遍。

是不是想想都觉得很痛苦。。。

那有什么好的办法解决这个问题呢?

于是他想到了 puppeteer。

它是一个网页自动化的 Node.js 工具,基本所有你手动在浏览器里做的事情,都可以用它来自动化完成。

比如点击、移动光标、输入等等。

那前面那个繁琐的问题自然也可以用 puppeteer 自动化来做,解放我们的生产力。

我们来分析下整个流程:

首先打开星球编辑器页面,如果没登录会跳到登录页:

这一步要扫码,没法自动化。

登录之后进入编辑器页面,输入内容:

这时候我们要把其中的图片链接分析出来,自动下载到本地的目录中。

然后记录每个链接所在的行数,把光标移动到对应的行数,点击上传按钮:

上传这一步也要手动来做,选择之前自动下载的图片就行。

然后光标会自动移动到下一个位置,再点击上传按钮,直到所有图片上传完。

文件浏览器这一步是操作系统的功能,没法自动化。

我们把下载图片、在对应位置插入图片的过程给自动化了。只有登录、选择文件这两步还要还要手动做。

但这样已经方便太多了。

流程理清了,我们就来写下代码吧:

importpuppeteerfrom"puppeteer";constbrowser=awaitpuppeteer.launch({headless:false,defaultViewport:{width:0,height:0}});constpage=awaitbrowser.newPage();awaitpage.goto("http://www.baidu.com");awaitpage.focus("#kw");awaitpage.keyboard.type("hello",{delay:200});awaitpage.click("#su");

引入 puppeteer,跑一个 chrome 浏览器,创建一个页面,导航到 baidu,输入 hello,点击搜索。

puppeteer 的 api 还是很容易懂的。

其中 defaultViewport 设置宽高为 0 是让网页充满整个窗口。

然后我们把它跑起来,因为用到了 es module、顶层 await,需要在 package.json 声明 type 为 module:

声明 type 为 module 就是所有的模块都是 es module 的意思。

然后把它跑起来:

可以看到脚本正确执行了。

然后我们让它打开星球编辑器的网址:

importpuppeteerfrom"puppeteer";constbrowser=awaitpuppeteer.launch({headless:false,defaultViewport:{width:0,height:0}});constpage=awaitbrowser.newPage();awaitpage.goto("https://wx.zsxq.com/dweb2/article?groupId=51122858222824");

确实跳到登录了:

扫码登录之后进入星球页面,就可以写文章了。

但是,下次跑脚本还是要再登录。

我们不是登录过了么?为啥还需要登录?

因为 chrome 默认的数据保存在一个目录中,叫 userDataDir,而这个目录默认是临时生成的,所以每次保存数据的目录都不一样。

这就导致了每次都需要登陆。

所以我们指定一个固定的 userDataDir 就好了。

importpuppeteerfrom"puppeteer";importosfrom"os";importpathfrom"path";constbrowser=awaitpuppeteer.launch({headless:false,defaultViewport:{width:0,height:0},userDataDir:path.join(os.homedir(),".puppeteer-data")});constpage=awaitbrowser.newPage();awaitpage.goto("https://wx.zsxq.com/dweb2/article?groupId=51122858222824");

通过 os.homedir() 拿到 home 目录,再下面新建一个 .puppeteer-data 的目录来保存用户数据。

这样登录一次之后,下次就不再需要登录了:

这时候可以看到 userDataDir 下是保存了用户数据的:

接下来就是编辑部分的自动化了。

我们要做的事情有这么两件:

提取文本中的所有链接,自动下载。

光标定位到每个链接的位置,自动点击上传按钮。

执行这俩自动化脚本的过程最好让用户控制,比如输入 download-img 就自动下载图片,输入 upload-next 光标就自动定位到下个位置,点击上传。

所以我们引入 readline 这个内置模块接收用户输入。

importreadlinefrom"readline";constrl=readline.createInterface({input:process.stdin,output:process.stdout});rl.on("line",async(command)=>{switch(command){case"upload-next":awaituploadNext();break;case"download-img":awaitdownloadImg();break;default:break;}});asyncfunctionuploadNext(){console.log("------");}asyncfunctiondownloadImg(){console.log("+++++++");}

调用 creatInterface api,指定 input、output 为标准输入输出。

然后当收到一行的输入的时候,根据内容决定执行什么方法:

我们先实现 download-img 的部分:

可以看到,编辑器部分的内容就是 .ql-editor 下的一个个 p 标签。

那我们只要取出所有的 p 标签,选出 ![]() 格式的内容就好了。

这需要一个正则,我们先把这个正则写出来:

整体格式是这样的:

![]()

但[] 和 () 需要转义:

!\[\]\(\)

中间部分是除了 [] 和 () 的任意字符出现任意次,也就是这样:

[^\[\]\(\)]*

并且 () 里的内容需要提取,需要用小括号包裹。

完整正则就是这样的:

!\[[^\[\]\(\)]*\]\(([^\[\]\(\)]*)\)

我们测试下:

可以看到 () 中的内容被正确提取出来了。

然后在网页里取出所有的 p 标签,根据内容过滤,把链接和行数记录下来:

constlinks=awaitpage.evaluate(()=>{letlinks=[];constlines=document.querySelectorAll(".ql-editorp");for(leti=0;i<lines.length;i++){constmatchRes=lines[i].textContent.trim().match(/!\[[^\[\]\(\)]*\]\(([^\[\]\(\)]*)\)/)if(matchRes){links.push({index:i,link:matchRes&&matchRes[1],});}}returnlinks;})

用 page.evaluate 方法在网页里远程执行一段 js,拿到它的返回结果。

这里拿到的就是所有的图片链接:

其实严格来说这不叫行数,而是第几个 p 标签,想要定位到对应的 p 标签,只要点击它就好了。

我们记录的下标是从 0 开始,而 nth-child 从 1 开始,所以要加 1。

可以看到,光标定位到了正确的位置:

不过先不着急定位光标,我们先把图片下载给搞定。

下载部分的代码如下:

importhttpsfrom"https";importfsfrom"fs";functiondownloadFile(url,destinationPath,progressCallback){letresolve,reject;constpromise=newPromise((x,y)=>{resolve=x;reject=y;});constrequest=https.get(url,response=>{if(response.statusCode!==200){consterror=newError(`Downloadfailed:serverreturnedcode${response.statusCode}.URL:${url}`);response.resume();reject(error);return;}constfile=fs.createWriteStream(destinationPath);file.on("finish",()=>resolve());file.on("error",error=>reject(error));response.pipe(file);consttotalBytes=parseInt(response.headers["content-length"],10);if(progressCallback)response.on("data",onData.bind(null,totalBytes));});request.on("error",error=>reject(error));returnpromise;functiononData(totalBytes,chunk){progressCallback(totalBytes,chunk.length);}}

用 https 模块的 get 方法请求 url,然后把 response 用流的方式写入文件,并且通过 content-length 的响应头拿到总长度。

这样,在每次 data 方法里就能根据总长度,当前 chunk 的长度,算出下载进度。

我们测试下:

consturl="https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/66399947ea6b45289c8d77b6d4568cc5~tplv-k3u1fbpfcp-watermark.image"letcurrentTotal=0;downloadFile(url,"./1.gif",(totalBytes,chunkBytes)=>{constpercent=(currentTotal/totalBytes*100).toFixed(1);console.log("总长度:"+totalBytes+"B","当前已下载:"+currentTotal+"B","进度"+percent+"%");currentTotal+=chunkBytes;})

可以看到,图片下载成功了!

但是,我们现在是知道这是个 gif 才给它加上 .gif 后缀,要是任意一个链接,怎么知道它的格式呢?

这个可以用 image-size 这个包:

importsizeOffrom"image-size";importfsfrom"fs";constbuffer=fs.readFileSync("./1.image");constdimensions=sizeOf(buffer);console.log(dimensions);

它能拿到图片的类型和宽高信息:

这样我们在下载完改下名就可以了。

consturl="https://p6-juejin.byteimg.com/tos-cn-i-k3u1fbpfcp/66399947ea6b45289c8d77b6d4568cc5~tplv-k3u1fbpfcp-watermark.image"letcurrentTotal=0;letfilePath="./1.image";downloadFile(url,filePath,(totalBytes,chunkBytes)=>{constpercent=(currentTotal/totalBytes*100).toFixed(1);console.log("总长度:"+totalBytes+"B","当前已下载:"+currentTotal+"B","进度"+percent+"%");currentTotal+=chunkBytes;if(currentTotal>=totalBytes){const{type}=sizeOf(fs.readFileSync(filePath));fs.renameSync(filePath,filePath+"."+type)}})

当下载完之后,拿到图片信息,重命名一下,把后缀名改成新的。

注意下图中文件名字的变化:

这样,下载图片就搞定了。

我们把它集成到自动化流程中。

先指定下文件保存位置和文件名:

我们在 home 目录下创建一个 .img 目录吧,然后文件名是 1.image、2.image 的形式。

constimgPath=path.join(os.homedir(),".img");fs.rmSync(imgPath,{recursive:true});fs.mkdirSync(imgPath);for(leti=0;i<links.length;i++){constfilePath=path.join(imgPath,(i+1)+".image");fs.writeFileSync(filePath,"aaaa")}

每次先清空 .img 目录,再创建。

执行之后,确实在 .img 目录下创建了对应的图片文件:

然后把下载图片和重命名的逻辑集成进来:

fs.rmSync(imgPath,{recursive:true});fs.mkdirSync(imgPath);for(leti=0;i<links.length;i++){constfilePath=path.join(imgPath,(i+1)+".image");letcurrentTotal=0;downloadFile(links[i].link,filePath,(totalBytes,chunkBytes)=>{currentTotal+=chunkBytes;if(currentTotal>=totalBytes){setTimeout(()=>{const{type}=sizeOf(fs.readFileSync(filePath));fs.renameSync(filePath,filePath+"."+type)console.log(`${filePath}下载完成,重命名为${filePath+"."+type}`);},1000);}})}

这里加了一个 setTimeout,1s 之后执行重命名的逻辑,保证在文件下载完之后再重命名。

效果是这样的:

在 .img 下可以看到所有的图片都下载并重命名成功了:

有 png 也有 gif

下一步只要在不同的位置插入就好了。

我们再来做光标定位的部分。

这部分前面演示过,就是触发对应 p 标签的 click 就好了。

letcursor=0;asyncfunctionuploadNext(){if(cursor>=links.length){return;}awaitpage.click(`.ql-editorp:nth-child(${links[cursor].index+1})`);awaitpage.evaluate((index)=>{constp=document.querySelector(`.ql-editorp:nth-child(${index+1})`);p.textContent="";},links[cursor].index);awaitpage.click(".ql-image");cursor++;}

我们定义一个游标,从 0 开始,先点击第一个 link 的 p 标签,把它的内容清空,插入下载的图片。

然后再次执行就是插入下一个。

这样依次插入。

我们来试试:

首先,打开编辑器页面,自己登录和输入 markdown 内容:

然后输入 download-img 来下载图片:

之后执行 upload-next 插入第一张图片:

再执行 upload-next 插入第二张图片:

插入的位置非常正确!

依次 upload-next 就能把所有图片插入完成。

对比下之前的体验:

一张张下载图片,根据不同的格式来重命名,然后一张张找到对应的位置,删除原来的链接,插入图片。

现在的体验:

输入 download-img 自动下载图片,不断执行 upload-next 选择图片,自动插入到正确的位置。

这体验差距很明显吧!

这就是用 puppeteer 自动化以后的工作流。

演示视频:

全部代码如下:

importpuppeteerfrom"puppeteer";importosfrom"os";importpathfrom"path";importfsfrom"fs";importreadlinefrom"readline";importsizeOffrom"image-size";importdownloadFilefrom"./download.js";constbrowser=awaitpuppeteer.launch({headless:false,defaultViewport:{width:0,height:0},userDataDir:path.join(os.homedir(),".puppeteer-data")});constpage=awaitbrowser.newPage();awaitpage.goto("https://wx.zsxq.com/dweb2/article?groupId=51122858222824");constrl=readline.createInterface({input:process.stdin,output:process.stdout});rl.on("line",async(command)=>{switch(command){case"upload-next":awaituploadNext();break;case"download-img":awaitdownloadImg();break;default:break;}});letlinks=[];asyncfunctiondownloadImg(){links=awaitpage.evaluate(()=>{letlinks=[];constlines=document.querySelectorAll(".ql-editorp");for(leti=0;i<lines.length;i++){constmatchRes=lines[i].textContent.trim().match(/!\[[^\[\]\(\)]*\]\(([^\[\]\(\)]*)\)/)if(matchRes){links.push({index:i,link:matchRes&&matchRes[1],});}}returnlinks;})constimgPath=path.join(os.homedir(),".img");fs.rmSync(imgPath,{recursive:true});fs.mkdirSync(imgPath);for(leti=0;i<links.length;i++){constfilePath=path.join(imgPath,(i+1)+".image");letcurrentTotal=0;downloadFile(links[i].link,filePath,(totalBytes,chunkBytes)=>{currentTotal+=chunkBytes;if(currentTotal>=totalBytes){setTimeout(()=>{const{type}=sizeOf(fs.readFileSync(filePath));fs.renameSync(filePath,filePath+"."+type)console.log(`${filePath}下载完成,重命名为${filePath+"."+type}`);},1000);}})}console.log(links);}letcursor=0;asyncfunctionuploadNext(){if(cursor>=links.length){return;}awaitpage.click(`.ql-editorp:nth-child(${links[cursor].index+1})`);awaitpage.evaluate((index)=>{constp=document.querySelector(`.ql-editorp:nth-child(${index+1})`);p.textContent="";},links[cursor].index);awaitpage.click(".ql-image");cursor++;}
importhttpsfrom"https";importfsfrom"fs";exportdefaultfunctiondownloadFile(url,destinationPath,progressCallback){letresolve,reject;constpromise=newPromise((x,y)=>{resolve=x;reject=y;});constrequest=https.get(url,response=>{if(response.statusCode!==200){consterror=newError(`Downloadfailed:serverreturnedcode${response.statusCode}.URL:${url}`);response.resume();reject(error);return;}constfile=fs.createWriteStream(destinationPath);file.on("finish",()=>resolve());file.on("error",error=>reject(error));response.pipe(file);consttotalBytes=parseInt(response.headers["content-length"],10);if(progressCallback)response.on("data",onData.bind(null,totalBytes));});request.on("error",error=>reject(error));returnpromise;functiononData(totalBytes,chunk){progressCallback(totalBytes,chunk.length);}}

总结

星球编辑器不好用,每次都要把图片手动下载下来然后插入对应位置,我们通过 puppeteer 把这个流程自动化了。

puppeteer 是一个自动化测试工具,基本所有浏览器手动的操作都能自动化。

我们用 readline 模块读取用户输入,当输入 download-img 的时候,拿到所有的 p 标签,过滤出链接的内容,把信息记录下来。

自动下载图片并用 image-size 读取图片类型来重命名。

然后输入 upload-next,会通过点击对应 p 标签实现光标定位,然后点击上传按钮来选择图片。

自动化以后的工作流程简单太多了,繁琐的工作都给自动化了,体验爽翻了!


欢迎学编程的朋友们加入鱼皮的,和上万名学编程的同学共享知识、交流进步,学习原创项目并享有答疑指导服务。

往期推荐

标签:

最近更新

用 Puppeteer 把繁琐工作给自动化了,太爽啦!_世界动态
2023-05-23 13:57:05
传递绿书签  护苗在行动
2023-05-23 12:54:57
环球今头条!智能无人扫地机“上岗”
2023-05-23 12:31:08
博硕科技:5月22日融资买入205.6万元,融资融券余额6190.59万元|每日热议
2023-05-23 11:30:55
天天微动态丨“飞地养牛”的甜头
2023-05-23 11:03:14
世界热门:骨顶鸡是鸡还是鸭?可以人工饲养吗?
2023-05-23 09:59:30
还记得《一起又看流星雨》里的金娜娜吗?她是由谁扮演的
2023-05-23 09:36:00
北京现代快速跌落售车25万辆仅巅峰期22% 获60亿注资名图纯电动车月销不足20台 全球视点
2023-05-23 09:03:14
“二阳”比“首阳”症状轻吗?权威解答
2023-05-23 08:19:31
华胜天成携手天合光能共探新能源未来-环球信息
2023-05-23 07:50:14
3710的基数一个月五险一金是多少_3710
2023-05-23 07:05:04
板式家具的优缺点及区别_板式家具的优缺点_即时看
2023-05-23 06:05:34
世界资讯:什么是会计凭证?它有何重要作用_什么是会计凭证 它有哪些作用 会计凭证分为有哪些凭证
2023-05-23 03:56:16
【新要闻】忍者殺手人物介紹:海德拉(ハイドラ)
2023-05-23 01:15:43
世界播报:最仁村_对于最仁村简单介绍
2023-05-22 22:54:48
行业周期拐点临近,继续向下空间有限,可以关注起来|当前焦点
2023-05-22 21:57:46
*ST未来:上交所决定公司股票终止上市
2023-05-22 20:52:03
养育儿子如树,5种营养受益终身
2023-05-22 19:58:48
【时快讯】相城区幼儿园学区划分2023
2023-05-22 19:17:04
奢华眼镜品牌OLIVER PEOPLES落户上海静安嘉里中心_天天速看料
2023-05-22 18:49:36
【IPO一线】鑫信滕创业板IPO成功过会:客户入股等三大问题待解-天天微头条
2023-05-22 18:20:29
2023年3月重庆计算机等级考试成绩查询入口 焦点滚动
2023-05-22 17:36:11
每日快看:react-naive工作原理
2023-05-22 17:02:24
万联证券:预计A股后续仍以结构性行情为主
2023-05-22 16:42:51
推拉门贵还是平开门贵 热门
2023-05-22 16:00:17
国网千阳县供电公司:“村网共建”聚合力 服务民生“零距离”
2023-05-22 15:31:28
蒺藜的功效与作用怎么用呢_蒺藜的功效与作用
2023-05-22 14:51:36
全球热门:仲景食品:积极拓展中央厨房调味料业务
2023-05-22 14:09:12
差点撞上!错过出口,重型半挂车隧道出口停车掉头……|世界热闻
2023-05-22 13:18:33
以人民为中心 郑州市城管局深入开展“郑点亮、郑路平、郑好停”调研治理活动 环球热门
2023-05-22 12:44:59