Merge pull request #576 from InstaZDLL/feat/timestamp-pdf

feat: add Timestamp PDF tool with RFC 3161 support
This commit is contained in:
Alam
2026-03-27 11:03:57 +05:30
committed by GitHub
31 changed files with 1644 additions and 11 deletions

View File

@@ -35,6 +35,7 @@ import { SanitizeNode } from './sanitize-node';
import { EncryptNode } from './encrypt-node';
import { DecryptNode } from './decrypt-node';
import { DigitalSignNode } from './digital-sign-node';
import { TimestampNode } from './timestamp-node';
import { RedactNode } from './redact-node';
import { RepairNode } from './repair-node';
import { PdfToTextNode } from './pdf-to-text-node';
@@ -509,6 +510,13 @@ export const nodeRegistry: Record<string, NodeRegistryEntry> = {
description: 'Apply a digital signature to PDF',
factory: () => new DigitalSignNode(),
},
TimestampNode: {
label: 'Timestamp',
category: 'Secure PDF',
icon: 'ph-clock',
description: 'Add an RFC 3161 document timestamp',
factory: () => new TimestampNode(),
},
RedactNode: {
label: 'Redact',
category: 'Secure PDF',

View File

@@ -0,0 +1,66 @@
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 { TIMESTAMP_TSA_PRESETS } from '../../config/timestamp-tsa.js';
import { timestampPdf } from '../../logic/digital-sign-pdf.js';
export class TimestampNode extends BaseWorkflowNode {
readonly category = 'Secure PDF' as const;
readonly icon = 'ph-clock';
readonly description = 'Add an RFC 3161 document timestamp';
constructor() {
super('Timestamp');
this.addInput('pdf', new ClassicPreset.Input(pdfSocket, 'PDF'));
this.addOutput(
'pdf',
new ClassicPreset.Output(pdfSocket, 'Timestamped PDF')
);
this.addControl(
'tsaUrl',
new ClassicPreset.InputControl('text', {
initial: TIMESTAMP_TSA_PRESETS[0].url,
})
);
}
getTsaPresets(): { label: string; url: string }[] {
return TIMESTAMP_TSA_PRESETS;
}
async data(
inputs: Record<string, SocketData[]>
): Promise<Record<string, SocketData>> {
const pdfInputs = requirePdfInput(inputs, 'Timestamp');
const tsaUrlCtrl = this.controls['tsaUrl'] as
| ClassicPreset.InputControl<'text'>
| undefined;
const tsaUrl = tsaUrlCtrl?.value || TIMESTAMP_TSA_PRESETS[0].url;
return {
pdf: await processBatch(pdfInputs, async (input) => {
let bytes: Uint8Array;
try {
bytes = await timestampPdf(input.bytes, tsaUrl);
} catch (err) {
throw new Error(
`Failed to timestamp using TSA ${tsaUrl}: ${err instanceof Error ? err.message : err}`,
{ cause: err }
);
}
const document = await PDFDocument.load(bytes);
return {
type: 'pdf',
document,
bytes,
filename: input.filename.replace(/\.pdf$/i, '_timestamped.pdf'),
};
}),
};
}
}