首页
学习
活动
专区
圈层
工具
发布
社区首页 >问答首页 >如何在循环中“等待”来自异步函数的承诺/通过Postmark API发送带有附件的电子邮件

如何在循环中“等待”来自异步函数的承诺/通过Postmark API发送带有附件的电子邮件
EN

Stack Overflow用户
提问于 2021-02-17 20:23:07
回答 1查看 125关注 0票数 1

我想通过邮戳API从我的express服务器发送邮件。我在express服务器上收到一个post请求,其中包含以下json有效负载:

代码语言:javascript
复制
{
    "mail": {
        "from": "frommail@address.com",
        "to": "tomail@adrress.com",
        "subject": "Test mit Datei",
        "textBody": "Test mit Datei",
        "replyTo": "replyto@mail.com",
        "files": [
            {
                "name": "filename.pdf",
                "url": "https://sharelink.de"
            }
        ]
    }
}  

我需要迭代文件数组,下载每个文件并将其转换为base64,然后在一个对象中使用名称和内容类型保护结果,我可以通过Postmark API发送该对象。就像在发送邮件但没有文件时一样。我认为这是因为sendEmail函数没有等待encode2base64函数完成。因此,它发送了一个空数组。但是我不知道如何在这个场景中设置await语法。

到目前为止,我的代码如下:

代码语言:javascript
复制
    // Execution
    const from = req.body.mail.from;
    const to = req.body.mail.to;
    const subject = req.body.mail.subject;
    const htmlBody = req.body.mail.htmlBody;
    const textBody = req.body.mail.textBody;
    const replyTo = req.body.mail.replyTo;
    const files = req.body.mail.files;
    sendEmail(from, to, subject, htmlBody, textBody, replyTo, files)
    .catch((err)=>{
                    res.send({"error": err.message});
     })
    .then((response)=>{
                console.log("prom response ",  response.data );
     res.send( response.data );
     })
    //==========================================
    //===== SEND EMAIL WITH POSTMARK API =======
    //==========================================
    /*  FOR THE API THE ATTACHMENTS MUST BE IN THE FOLLOWING FORMAT
    "Attachments": [{
                        Name: text(filename),
                        Content: 'BASE64 ENCODED FILE',
                        ContentType: "application/octet-stream"
                    }]
            */
    async function sendEmail( from, to, subject, htmlBody, textBody, replyTo, files ){
      var attachments = [];
      if ( files ) {
        console.log("============ EMAIL WITH FILES ==============")
        
        for (i in files ){
          var fileName = files[i].name;
          var shareLink = files[i].url;
          console.log("fileName", fileName );
          console.log("url: ", shareLink );
          encodeFile2Base64( shareLink )
          .catch((err)=>{
            console.log(err)
            })
          .then((response)=>{
              const encoded = new Buffer.from(response.data).toString('base64');
              attachments.push({
                "Name": fileName,
                "Content": encoded,
                "ContentType": "application/octet-stream"
              })
              
            })
    
      }}
      var config = {
        method: 'post',
        url: 'https://api.postmarkapp.com/email',
        headers: {'Content-Type': 'application/json','Accept':'application/json','X- Postmark-Server-Token': process.env.POSTMARK_KEY},
        data:{
                "From": from,
                "To": to,
                "Subject": subject,
                "HtmlBody": htmlBody,
                "TextBody": textBody,
                "ReplyTo": replyTo,
          "Attachments": attachments
            }
      };
      const promSendMail = axios(config);
      const resMailSent = await promSendMail;
      return resMailSent
    }
    //==========================================
    
    //==========================================
    //======== ENCODE TO BASE64 ================
    //==========================================
    async function encodeFile2Base64(url){
      var config = {
        method: 'get',
        url : url,
        headers : { responseType: 'arraybuffer' }
      };
      const prom = axios( config );
      const data = await prom;
      return data
    }

我是否需要将整个循环包装到一个异步函数中?或者我应该先创建attachments对象,然后再调用Postmark的API?但不管怎样,我都需要等待。

EN

回答 1

Stack Overflow用户

发布于 2021-02-17 20:56:23

如果您希望使用async / await,则需要将循环包装在async函数中。

sendEmail函数必须返回一个Promise,理想情况下,这应该在发送失败时拒绝。

您可以通过一些模拟发送代码从概念上测试这一点,这样您就可以了解逻辑/原理。

您可以使用for...offor...infor循环。只是不要使用Array.forEach,这不会很好地发挥等待!

例如:

代码语言:javascript
复制
// Send successfully exept for email #2 (simulate an error)
function sendEmail(from, to, subject, body) {
    return new Promise((resolve, reject) => setTimeout(from === 'from2' ? reject: resolve, 1000, from === 'from2' ? "send failure " : "send success "));
}


let emails = Array.from( {length: 5}, (v,k) => ({ from: 'from' + k, to: "to" + k, subject: "subject" + k, body: "body" + k}));

async function sendEmails(emails, stopOnError) {
    for(let i = 0; i < emails.length; i++) {
        try { 
            console.log(`Sending email ${i+1} of ${emails.length}...`);
            let result = await sendEmail(emails[i].from, emails[i].to, emails[i].subject, emails[i].body);
            console.log("result:", result);
        } catch (error) {
            console.error(`sendEmails: error occurred at email #${i}:`, error);
            if (stopOnError) throw error
        }
    }
}

// Don't stop if we get an error, change to true to abandon sending on error
sendEmails(emails, false);

票数 1
EN
页面原文内容由Stack Overflow提供。腾讯云小微IT领域专用引擎提供翻译支持
原文链接:

https://stackoverflow.com/questions/66241899

复制
相关文章

相似问题

领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档