5

我遇到了一个问题,当我必须pipe()将创建的文档发送到多个目标时,在我的情况下是 HTTP 响应和使用node-mailer. 在电子邮件的附件中第一次使用后,没有任何内容可以通过管道传输到响应(从客户端调用它时,PDF 有 0 个字节)。

响应控制器:

const doc = await createPdf(course.name, lastRecordDate, ctx);   

// Send a notificiation email with attachment
if (query.hasOwnProperty('sendEmail') && query.sendEmail === 'true') {
  await sendNotificationEmail(doc, course, ctx);   
}

doc.pipe(res);
res.contentType('application/pdf');

发送电子邮件的功能:

async function sendNotificationEmail(doc: any, course: Course, ctx: Context) {
  const attachment = {
    filename: `${course.name}-certificate.pdf`,
    contentType: 'application/pdf',
    content: doc
  };
  return SMTPSendTemplateWithAttachments(
    ctx,
    ['somememail@test.si'],
    `${course.name}`,
    'en-report-created',
    {
      firstName: ctx.user.firstName,
      courseName: course.name
    },
    [attachment]
  );
}

如果我删除发送电子邮件的功能,PDF 通常会通过管道传输到响应,我可以从客户端下载它。

我试图找到一种克隆流的方法(据我所知,PDFKit 的文档是一个流),但没有成功。

任何解决方案都会非常有帮助。

4

1 回答 1

0

PassThrough我使用两个流解决了这个问题,两个流通过管道传输PDFKit文档Stream,然后在data事件中我将其chunks写入两个单独的缓冲区。在end事件中,我通过电子邮件发送数据并创建了一个新PassThrough流并将其传送到响应中。

这是代码。

// Both PassThrough streams are defined before in order to use them in the createPdf function
streamCopy1 = new PassThrough();
streamCopy2 = new PassThrough();

const buffer1 = [];
const buffer2 = [];

streamCopy1
  .on('data', (chunk) => {
    buffer1.push(chunk);
  })
  .on('end', () => {
    const bufferFinished = Buffer.concat(buffer1);
    if (query.hasOwnProperty('sendEmail') && query.sendEmail === 'true') {
      sendNotificationEmail(bufferFinished, course, ctx);
    }
  });

streamCopy2
  .on('data', (chunk) => {
    buffer2.push(chunk);
  })
  .on('end', () => {
    const bufferFinished = Buffer.concat(buffer2);
    const stream = new PassThrough();
    stream.push(bufferFinished);
    stream.end();
    stream.pipe(res);
    res.contentType('application/pdf');
  });

创建 PDF 的功能。

// function declaration
const doc = new pdfDocument();
doc.pipe(streamCopy1).pipe(streamCopy2);
// rest of the code

该解决方案并不是最好的,非常欢迎任何建议。

于 2019-07-22T08:56:11.450 回答