File size: 3,178 Bytes
1b756c8
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
import { ClassicPreset } from 'rete';
import { BaseWorkflowNode } from './base-node';
import { pdfSocket } from '../sockets';
import type { SocketData } from '../types';
import { requirePdfInput, processBatch } from '../types';
import { PDFDocument } from 'pdf-lib';
import { initializeQpdf } from '../../utils/helpers.js';

export class EncryptNode extends BaseWorkflowNode {
  readonly category = 'Secure PDF' as const;
  readonly icon = 'ph-lock';
  readonly description = 'Encrypt PDF with password';

  constructor() {
    super('Encrypt');
    this.addInput('pdf', new ClassicPreset.Input(pdfSocket, 'PDF'));
    this.addOutput('pdf', new ClassicPreset.Output(pdfSocket, 'Encrypted PDF'));
    this.addControl(
      'userPassword',
      new ClassicPreset.InputControl('text', { initial: '' })
    );
    this.addControl(
      'ownerPassword',
      new ClassicPreset.InputControl('text', { initial: '' })
    );
  }

  async data(
    inputs: Record<string, SocketData[]>
  ): Promise<Record<string, SocketData>> {
    const pdfInputs = requirePdfInput(inputs, 'Encrypt');

    const getText = (key: string) => {
      const ctrl = this.controls[key] as
        | ClassicPreset.InputControl<'text'>
        | undefined;
      return ctrl?.value || '';
    };

    const userPassword = getText('userPassword');
    const ownerPassword = getText('ownerPassword') || userPassword;
    if (!userPassword)
      throw new Error('User password is required for encryption');

    return {
      pdf: await processBatch(pdfInputs, async (input) => {
        const qpdf = await initializeQpdf();
        const uid = `${Date.now()}_${Math.random().toString(36).slice(2, 9)}`;
        const inputPath = `/tmp/input_encrypt_${uid}.pdf`;
        const outputPath = `/tmp/output_encrypt_${uid}.pdf`;

        let encryptedData: Uint8Array;
        try {
          qpdf.FS.writeFile(inputPath, input.bytes);

          const args = [
            inputPath,
            '--encrypt',
            userPassword,
            ownerPassword,
            '256',
          ];
          if (ownerPassword !== userPassword) {
            args.push(
              '--modify=none',
              '--extract=n',
              '--print=none',
              '--accessibility=n',
              '--annotate=n',
              '--assemble=n',
              '--form=n',
              '--modify-other=n'
            );
          }
          args.push('--', outputPath);
          qpdf.callMain(args);

          encryptedData = qpdf.FS.readFile(outputPath, { encoding: 'binary' });
        } finally {
          try {
            qpdf.FS.unlink(inputPath);
          } catch {
            /* cleanup */
          }
          try {
            qpdf.FS.unlink(outputPath);
          } catch {
            /* cleanup */
          }
        }

        const resultBytes = new Uint8Array(encryptedData);
        const document = await PDFDocument.load(resultBytes, {
          ignoreEncryption: true,
        });

        return {
          type: 'pdf',
          document,
          bytes: resultBytes,
          filename: input.filename.replace(/\.pdf$/i, '_encrypted.pdf'),
        };
      }),
    };
  }
}