首页 > 解决方案 > 等待动作完成,然后在 puppeteer 中再次执行

问题描述

我有一个 puppeteer 脚本,可以将一些文本输入到字段中,提交查询并处理结果。

目前,该脚本一次只处理 1 个搜索词,但我需要它能够连续处理一组项目。

我想我只是将代码放在一个循环中(参见下面的代码),但是,它只是一次将数组中的所有项目输入到字段中,并且不为每个搜索词执行代码块:

  for (const search of searchTerms) {
    await Promise.all([
      page.type('input[name="q"]', 'in:spam ' + search + String.fromCharCode(13)),
      page.waitForNavigation({
          waitUntil: 'networkidle2'
        })
    ]);

    const count = await page.evaluate((sel) => {
      return document.querySelectorAll(sel)[1].querySelectorAll('tr').length;
    }, 'table[id^=":"]');

    if (count > 0) {
      const more = await page.$x('//span[contains(@class, "asa") and contains(@class, "bjy")]');
      await more[1].click();

      await page.waitFor(1250);
      const markRead = await page.$x('//div[text()="Mark all as read"]');
      await markRead[0].click();

      const selectAll = await page.$x('//span[@role="checkbox"]');
      await selectAll[1].click();

      const move = await page.$x('//div[@act="8"]');
      await move[0].click();

      await page.waitFor(5000);
    }
  }

我尝试使用Nodejs Synchronous For each loop的递归函数

我还尝试使用带有产量和承诺的函数生成器,甚至尝试了来自这篇文章Nodejs Puppeteer Wait to finish all code from loopeachSeries的包中的函数async

我尝试过的任何事情都没有成功。任何帮助将不胜感激,谢谢!

标签: javascriptnode.jsautomationpuppeteer

解决方案


无法使用相同的选项卡同时访问两个网站。您可以在浏览器上尝试以确保。

撇开玩笑不谈,如果你想搜索多个项目,你必须为此创建一个pagetab

for (const search of searchTerms) {
  const newTab = await browser.newPage()
  // other modified code here
}

...等等,仍然会一一搜索。但是如果你使用具有并发限制的地图,它会很好地工作。

我们可以p-all为此使用。

const pAll = require('p-all');
const actions = []
for (const search of searchTerms) {
  actions.push(async()=>{
    const newTab = await browser.newPage()
    // other modified code here
  })
}
pAll(actions, {concurrency: 2}) // <-- set how many to search at once

所以我们循环遍历每个术语,并在操作列表中添加一个新的承诺。添加功能不会花费太多时间。然后我们可以运行承诺链。

您仍然需要修改上面的代码以获得您想要的。和平!


推荐阅读