pdf-lite 1.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (206) hide show
  1. package/.commitlintrc.cjs +25 -0
  2. package/.github/ISSUE_TEMPLATE/bug_report.md +40 -0
  3. package/.github/ISSUE_TEMPLATE/feature_request.md +19 -0
  4. package/.github/workflows/docs.yaml +93 -0
  5. package/.github/workflows/prepare-release.yaml +79 -0
  6. package/.github/workflows/release.yaml +80 -0
  7. package/.github/workflows/test.yaml +35 -0
  8. package/.husky/commit-msg +1 -0
  9. package/.husky/pre-commit +1 -0
  10. package/.prettierignore +4 -0
  11. package/.prettierrc +4 -0
  12. package/CONTRIBUTING.md +109 -0
  13. package/EXAMPLES.md +1515 -0
  14. package/LICENSE +21 -0
  15. package/README.md +285 -0
  16. package/examples/001-create-pdf.ts +112 -0
  17. package/examples/002-create-encrypted-pdf.ts +121 -0
  18. package/examples/003-sign-pdf.ts +347 -0
  19. package/examples/004-incremental-update.ts +206 -0
  20. package/examples/005-modify-acroform.ts +374 -0
  21. package/examples/006-tokeniser-example.ts +131 -0
  22. package/examples/007-decoder-example.ts +197 -0
  23. package/package.json +72 -0
  24. package/packages/pdf-lite/README.md +3 -0
  25. package/packages/pdf-lite/package.json +68 -0
  26. package/packages/pdf-lite/scripts/create-encryption-tests.sh +41 -0
  27. package/packages/pdf-lite/scripts/gen-signing-keys.sh +290 -0
  28. package/packages/pdf-lite/scripts/generate-all-signing-keys.sh +70 -0
  29. package/packages/pdf-lite/src/core/decoder.ts +454 -0
  30. package/packages/pdf-lite/src/core/generators.ts +128 -0
  31. package/packages/pdf-lite/src/core/incremental-parser.ts +221 -0
  32. package/packages/pdf-lite/src/core/index.ts +2 -0
  33. package/packages/pdf-lite/src/core/objects/pdf-array.ts +54 -0
  34. package/packages/pdf-lite/src/core/objects/pdf-boolean.ts +19 -0
  35. package/packages/pdf-lite/src/core/objects/pdf-comment.ts +50 -0
  36. package/packages/pdf-lite/src/core/objects/pdf-date.ts +74 -0
  37. package/packages/pdf-lite/src/core/objects/pdf-dictionary.ts +171 -0
  38. package/packages/pdf-lite/src/core/objects/pdf-hexadecimal.ts +54 -0
  39. package/packages/pdf-lite/src/core/objects/pdf-indirect-object.ts +137 -0
  40. package/packages/pdf-lite/src/core/objects/pdf-name.ts +19 -0
  41. package/packages/pdf-lite/src/core/objects/pdf-null.ts +15 -0
  42. package/packages/pdf-lite/src/core/objects/pdf-number.ts +98 -0
  43. package/packages/pdf-lite/src/core/objects/pdf-object-reference.ts +30 -0
  44. package/packages/pdf-lite/src/core/objects/pdf-object.ts +107 -0
  45. package/packages/pdf-lite/src/core/objects/pdf-start-xref.ts +39 -0
  46. package/packages/pdf-lite/src/core/objects/pdf-stream.ts +687 -0
  47. package/packages/pdf-lite/src/core/objects/pdf-string.ts +38 -0
  48. package/packages/pdf-lite/src/core/objects/pdf-trailer.ts +57 -0
  49. package/packages/pdf-lite/src/core/objects/pdf-xref-table.ts +264 -0
  50. package/packages/pdf-lite/src/core/parser.ts +22 -0
  51. package/packages/pdf-lite/src/core/ref.ts +102 -0
  52. package/packages/pdf-lite/src/core/serializer.ts +68 -0
  53. package/packages/pdf-lite/src/core/streams/object-stream.ts +20 -0
  54. package/packages/pdf-lite/src/core/tokeniser.ts +687 -0
  55. package/packages/pdf-lite/src/core/tokens/boolean-token.ts +20 -0
  56. package/packages/pdf-lite/src/core/tokens/byte-offset-token.ts +20 -0
  57. package/packages/pdf-lite/src/core/tokens/comment-token.ts +32 -0
  58. package/packages/pdf-lite/src/core/tokens/end-array-token.ts +10 -0
  59. package/packages/pdf-lite/src/core/tokens/end-dictionary-token.ts +10 -0
  60. package/packages/pdf-lite/src/core/tokens/end-object-token.ts +10 -0
  61. package/packages/pdf-lite/src/core/tokens/end-stream-token.ts +11 -0
  62. package/packages/pdf-lite/src/core/tokens/hexadecimal-token.ts +22 -0
  63. package/packages/pdf-lite/src/core/tokens/name-token.ts +19 -0
  64. package/packages/pdf-lite/src/core/tokens/null-token.ts +9 -0
  65. package/packages/pdf-lite/src/core/tokens/number-token.ts +164 -0
  66. package/packages/pdf-lite/src/core/tokens/object-reference-token.ts +24 -0
  67. package/packages/pdf-lite/src/core/tokens/start-array-token.ts +10 -0
  68. package/packages/pdf-lite/src/core/tokens/start-dictionary-token.ts +10 -0
  69. package/packages/pdf-lite/src/core/tokens/start-object-token.ts +28 -0
  70. package/packages/pdf-lite/src/core/tokens/start-stream-token.ts +52 -0
  71. package/packages/pdf-lite/src/core/tokens/start-xref-token.ts +10 -0
  72. package/packages/pdf-lite/src/core/tokens/stream-chunk-token.ts +8 -0
  73. package/packages/pdf-lite/src/core/tokens/string-token.ts +17 -0
  74. package/packages/pdf-lite/src/core/tokens/token.ts +43 -0
  75. package/packages/pdf-lite/src/core/tokens/trailer-token.ts +12 -0
  76. package/packages/pdf-lite/src/core/tokens/whitespace-token.ts +43 -0
  77. package/packages/pdf-lite/src/core/tokens/xref-table-entry-token.ts +65 -0
  78. package/packages/pdf-lite/src/core/tokens/xref-table-section-start-token.ts +31 -0
  79. package/packages/pdf-lite/src/core/tokens/xref-table-start-token.ts +13 -0
  80. package/packages/pdf-lite/src/crypto/ciphers/aes128.ts +63 -0
  81. package/packages/pdf-lite/src/crypto/ciphers/aes256.ts +50 -0
  82. package/packages/pdf-lite/src/crypto/ciphers/rc4.ts +82 -0
  83. package/packages/pdf-lite/src/crypto/constants.ts +10 -0
  84. package/packages/pdf-lite/src/crypto/key-derivation/key-derivation-aes256.ts +213 -0
  85. package/packages/pdf-lite/src/crypto/key-derivation/key-derivation.ts +122 -0
  86. package/packages/pdf-lite/src/crypto/key-gen/key-gen-aes256.ts +79 -0
  87. package/packages/pdf-lite/src/crypto/key-gen/key-gen-rc4-128.ts +190 -0
  88. package/packages/pdf-lite/src/crypto/key-gen/key-gen-rc4-40.ts +129 -0
  89. package/packages/pdf-lite/src/crypto/types.ts +6 -0
  90. package/packages/pdf-lite/src/crypto/utils.ts +81 -0
  91. package/packages/pdf-lite/src/filters/ascii85.ts +128 -0
  92. package/packages/pdf-lite/src/filters/asciihex.ts +55 -0
  93. package/packages/pdf-lite/src/filters/flate.ts +39 -0
  94. package/packages/pdf-lite/src/filters/lzw.ts +144 -0
  95. package/packages/pdf-lite/src/filters/pass-through.ts +37 -0
  96. package/packages/pdf-lite/src/filters/runlength.ts +92 -0
  97. package/packages/pdf-lite/src/filters/types.ts +21 -0
  98. package/packages/pdf-lite/src/index.ts +4 -0
  99. package/packages/pdf-lite/src/pdf/errors.ts +5 -0
  100. package/packages/pdf-lite/src/pdf/index.ts +4 -0
  101. package/packages/pdf-lite/src/pdf/pdf-document.ts +924 -0
  102. package/packages/pdf-lite/src/pdf/pdf-reader.ts +57 -0
  103. package/packages/pdf-lite/src/pdf/pdf-revision.ts +234 -0
  104. package/packages/pdf-lite/src/pdf/pdf-xref-lookup.ts +527 -0
  105. package/packages/pdf-lite/src/security/crypt-filters/aesv2.ts +58 -0
  106. package/packages/pdf-lite/src/security/crypt-filters/aesv3.ts +56 -0
  107. package/packages/pdf-lite/src/security/crypt-filters/base.ts +140 -0
  108. package/packages/pdf-lite/src/security/crypt-filters/identity.ts +40 -0
  109. package/packages/pdf-lite/src/security/crypt-filters/v2.ts +59 -0
  110. package/packages/pdf-lite/src/security/handlers/base.ts +625 -0
  111. package/packages/pdf-lite/src/security/handlers/pubSec.ts +413 -0
  112. package/packages/pdf-lite/src/security/handlers/utils.ts +304 -0
  113. package/packages/pdf-lite/src/security/handlers/v1.ts +225 -0
  114. package/packages/pdf-lite/src/security/handlers/v2.ts +128 -0
  115. package/packages/pdf-lite/src/security/handlers/v4.ts +379 -0
  116. package/packages/pdf-lite/src/security/handlers/v5.ts +298 -0
  117. package/packages/pdf-lite/src/security/types.ts +158 -0
  118. package/packages/pdf-lite/src/signing/document-security-store.ts +224 -0
  119. package/packages/pdf-lite/src/signing/index.ts +3 -0
  120. package/packages/pdf-lite/src/signing/signatures/adbe-pkcs7-detached.ts +154 -0
  121. package/packages/pdf-lite/src/signing/signatures/adbe-pkcs7-sha1.ts +161 -0
  122. package/packages/pdf-lite/src/signing/signatures/adbe-x509-rsa-sha1.ts +106 -0
  123. package/packages/pdf-lite/src/signing/signatures/base.ts +229 -0
  124. package/packages/pdf-lite/src/signing/signatures/etsi-cades-detached.ts +229 -0
  125. package/packages/pdf-lite/src/signing/signatures/etsi-rfc3161.ts +92 -0
  126. package/packages/pdf-lite/src/signing/signatures/index.ts +6 -0
  127. package/packages/pdf-lite/src/signing/signer.ts +120 -0
  128. package/packages/pdf-lite/src/signing/types.ts +86 -0
  129. package/packages/pdf-lite/src/signing/utils.ts +71 -0
  130. package/packages/pdf-lite/src/types.ts +44 -0
  131. package/packages/pdf-lite/src/utils/IterableReadableStream.ts +30 -0
  132. package/packages/pdf-lite/src/utils/algos.ts +446 -0
  133. package/packages/pdf-lite/src/utils/assert.ts +42 -0
  134. package/packages/pdf-lite/src/utils/bytesToHex.ts +18 -0
  135. package/packages/pdf-lite/src/utils/bytesToHexBytes.ts +27 -0
  136. package/packages/pdf-lite/src/utils/bytesToString.ts +17 -0
  137. package/packages/pdf-lite/src/utils/concatUint8Arrays.ts +26 -0
  138. package/packages/pdf-lite/src/utils/escapeString.ts +49 -0
  139. package/packages/pdf-lite/src/utils/hexBytesToBytes.ts +22 -0
  140. package/packages/pdf-lite/src/utils/hexBytesToString.ts +21 -0
  141. package/packages/pdf-lite/src/utils/hexToBytes.ts +18 -0
  142. package/packages/pdf-lite/src/utils/padBytes.ts +25 -0
  143. package/packages/pdf-lite/src/utils/predictors.ts +332 -0
  144. package/packages/pdf-lite/src/utils/replaceInBuffer.ts +56 -0
  145. package/packages/pdf-lite/src/utils/stringToBytes.ts +22 -0
  146. package/packages/pdf-lite/src/utils/stringToHexBytes.ts +23 -0
  147. package/packages/pdf-lite/src/utils/unescapeString.ts +123 -0
  148. package/packages/pdf-lite/test/acceptance/__snapshots__/versions.node.test.ts.snap +60766 -0
  149. package/packages/pdf-lite/test/acceptance/fixtures/1.3/basic.pdf +0 -0
  150. package/packages/pdf-lite/test/acceptance/fixtures/1.4/basic-aes-128.pdf +0 -0
  151. package/packages/pdf-lite/test/acceptance/fixtures/1.4/basic-aes-256.pdf +0 -0
  152. package/packages/pdf-lite/test/acceptance/fixtures/1.4/basic-rc4-128.pdf +0 -0
  153. package/packages/pdf-lite/test/acceptance/fixtures/1.4/basic-rc4-40.pdf +0 -0
  154. package/packages/pdf-lite/test/acceptance/fixtures/1.4/basic.pdf +0 -0
  155. package/packages/pdf-lite/test/acceptance/fixtures/1.5/basic.pdf +0 -0
  156. package/packages/pdf-lite/test/acceptance/fixtures/1.6/basic.pdf +0 -0
  157. package/packages/pdf-lite/test/acceptance/fixtures/1.7/basic.pdf +0 -0
  158. package/packages/pdf-lite/test/acceptance/fixtures/2.0/basic-aes-128.pdf +43 -0
  159. package/packages/pdf-lite/test/acceptance/fixtures/2.0/basic-aes-256.pdf +43 -0
  160. package/packages/pdf-lite/test/acceptance/fixtures/2.0/basic-rc4-128.pdf +43 -0
  161. package/packages/pdf-lite/test/acceptance/fixtures/2.0/basic-rc4-40.pdf +44 -0
  162. package/packages/pdf-lite/test/acceptance/fixtures/2.0/basic.pdf +79 -0
  163. package/packages/pdf-lite/test/acceptance/versions.node.test.ts +41 -0
  164. package/packages/pdf-lite/test/unit/__snapshots__/decoder.node.test.ts.snap +86947 -0
  165. package/packages/pdf-lite/test/unit/__snapshots__/tokeniser.node.test.ts.snap +131829 -0
  166. package/packages/pdf-lite/test/unit/ciphers.test.ts +61 -0
  167. package/packages/pdf-lite/test/unit/decoder.node.test.ts +21 -0
  168. package/packages/pdf-lite/test/unit/decoder.test.ts +567 -0
  169. package/packages/pdf-lite/test/unit/filters.test.ts +67 -0
  170. package/packages/pdf-lite/test/unit/fixtures/basic.pdf +0 -0
  171. package/packages/pdf-lite/test/unit/fixtures/encrypted_v1/basic-aes-128.pdf +0 -0
  172. package/packages/pdf-lite/test/unit/fixtures/encrypted_v1/basic-aes-256.pdf +0 -0
  173. package/packages/pdf-lite/test/unit/fixtures/encrypted_v1/basic-rc4-128.pdf +0 -0
  174. package/packages/pdf-lite/test/unit/fixtures/encrypted_v1/basic-rc4-40.pdf +43 -0
  175. package/packages/pdf-lite/test/unit/fixtures/protectedAdobeLivecycle.pdf +0 -0
  176. package/packages/pdf-lite/test/unit/fixtures/rsa-2048/index.ts +187 -0
  177. package/packages/pdf-lite/test/unit/fixtures/template.pdf +0 -0
  178. package/packages/pdf-lite/test/unit/incremental-update.test.ts +0 -0
  179. package/packages/pdf-lite/test/unit/objects.test.ts +0 -0
  180. package/packages/pdf-lite/test/unit/pdf-document-signing.test.ts +0 -0
  181. package/packages/pdf-lite/test/unit/pdf-revision.test.ts +195 -0
  182. package/packages/pdf-lite/test/unit/pdf.browser.test.ts +0 -0
  183. package/packages/pdf-lite/test/unit/predictors.test.ts +226 -0
  184. package/packages/pdf-lite/test/unit/ref.test.ts +158 -0
  185. package/packages/pdf-lite/test/unit/security-handlers.test.ts +645 -0
  186. package/packages/pdf-lite/test/unit/serializer.test.ts +81 -0
  187. package/packages/pdf-lite/test/unit/signature-objects.test.ts +814 -0
  188. package/packages/pdf-lite/test/unit/string-escaping.test.ts +84 -0
  189. package/packages/pdf-lite/test/unit/tokeniser.node.test.ts +38 -0
  190. package/packages/pdf-lite/test/unit/tokeniser.test.ts +1213 -0
  191. package/packages/pdf-lite/test/unit/utils.test.ts +248 -0
  192. package/packages/pdf-lite/test/unit/xref-lookup.test.ts +72 -0
  193. package/packages/pdf-lite/tsconfig.json +4 -0
  194. package/packages/pdf-lite/tsconfig.prod.json +8 -0
  195. package/packages/pdf-lite/typedoc.json +14 -0
  196. package/packages/pdf-lite/vitest.config.ts +43 -0
  197. package/pnpm-workspace.yaml +2 -0
  198. package/renovate.json +34 -0
  199. package/scripts/build-examples.ts +30 -0
  200. package/scripts/bump-version.sh +56 -0
  201. package/scripts/gen-html-docs.sh +21 -0
  202. package/scripts/gen-md-docs.sh +15 -0
  203. package/scripts/prepare-release.sh +33 -0
  204. package/tsconfig.json +22 -0
  205. package/tsconfig.prod.json +12 -0
  206. package/typedoc.json +34 -0
@@ -0,0 +1,347 @@
1
+ // PDF signature example
2
+
3
+ import { PdfArray } from 'pdf-lite/core/objects/pdf-array'
4
+ import { PdfDictionary } from 'pdf-lite/core/objects/pdf-dictionary'
5
+ import { PdfIndirectObject } from 'pdf-lite/core/objects/pdf-indirect-object'
6
+ import { PdfName } from 'pdf-lite/core/objects/pdf-name'
7
+ import { PdfNumber } from 'pdf-lite/core/objects/pdf-number'
8
+ import { PdfObjectReference } from 'pdf-lite/core/objects/pdf-object-reference'
9
+ import { PdfStream } from 'pdf-lite/core/objects/pdf-stream'
10
+ import { PdfDocument } from 'pdf-lite/pdf/pdf-document'
11
+ import { PdfString } from 'pdf-lite/core/objects/pdf-string'
12
+ import {
13
+ PdfAdbePkcsX509RsaSha1SignatureObject,
14
+ PdfAdbePkcs7DetachedSignatureObject,
15
+ PdfAdbePkcs7Sha1SignatureObject,
16
+ PdfEtsiCadesDetachedSignatureObject,
17
+ PdfEtsiRfc3161SignatureObject,
18
+ } from 'pdf-lite'
19
+ import { rsaSigningKeys } from '../packages/pdf-lite/test/unit/fixtures/rsa-2048/index'
20
+ import fs from 'fs/promises'
21
+
22
+ function createPage(
23
+ contentStreamRef: PdfObjectReference,
24
+ ): PdfIndirectObject<PdfDictionary> {
25
+ const pageDict = new PdfDictionary()
26
+ pageDict.set('Type', new PdfName('Page'))
27
+ pageDict.set(
28
+ 'MediaBox',
29
+ new PdfArray([
30
+ new PdfNumber(0),
31
+ new PdfNumber(0),
32
+ new PdfNumber(612),
33
+ new PdfNumber(792),
34
+ ]),
35
+ )
36
+ pageDict.set('Contents', contentStreamRef)
37
+ return new PdfIndirectObject({ content: pageDict })
38
+ }
39
+
40
+ function createPages(
41
+ pages: PdfIndirectObject<PdfDictionary>[],
42
+ ): PdfIndirectObject<PdfDictionary> {
43
+ const pagesDict = new PdfDictionary()
44
+ pagesDict.set('Type', new PdfName('Pages'))
45
+ pagesDict.set('Kids', new PdfArray(pages.map((x) => x.reference)))
46
+ pagesDict.set('Count', new PdfNumber(pages.length))
47
+ return new PdfIndirectObject({ content: pagesDict })
48
+ }
49
+
50
+ function createCatalog(
51
+ pagesRef: PdfObjectReference,
52
+ ): PdfIndirectObject<PdfDictionary> {
53
+ const catalogDict = new PdfDictionary()
54
+ catalogDict.set('Type', new PdfName('Catalog'))
55
+ catalogDict.set('Pages', pagesRef)
56
+ return new PdfIndirectObject({ content: catalogDict })
57
+ }
58
+
59
+ function createFont(): PdfIndirectObject<PdfDictionary> {
60
+ const fontDict = new PdfDictionary()
61
+ fontDict.set('Type', new PdfName('Font'))
62
+ fontDict.set('Subtype', new PdfName('Type1'))
63
+ fontDict.set('BaseFont', new PdfName('Helvetica'))
64
+ return new PdfIndirectObject({ content: fontDict })
65
+ }
66
+
67
+ function createResources(
68
+ fontRef: PdfObjectReference,
69
+ ): PdfIndirectObject<PdfDictionary> {
70
+ const resourcesDict = new PdfDictionary()
71
+ const fontDict = new PdfDictionary()
72
+ fontDict.set('F1', fontRef)
73
+ resourcesDict.set('Font', fontDict)
74
+ return new PdfIndirectObject({ content: resourcesDict })
75
+ }
76
+
77
+ function createPageWithSignatureField(
78
+ contentStreamRef: PdfObjectReference,
79
+ signatureAnnotRef: PdfObjectReference,
80
+ ): PdfIndirectObject<PdfDictionary> {
81
+ const pageDict = new PdfDictionary()
82
+ pageDict.set('Type', new PdfName('Page'))
83
+ pageDict.set(
84
+ 'MediaBox',
85
+ new PdfArray([
86
+ new PdfNumber(0),
87
+ new PdfNumber(0),
88
+ new PdfNumber(612),
89
+ new PdfNumber(792),
90
+ ]),
91
+ )
92
+ pageDict.set('Contents', contentStreamRef)
93
+ pageDict.set('Annots', new PdfArray([signatureAnnotRef]))
94
+
95
+ return new PdfIndirectObject({ content: pageDict })
96
+ }
97
+
98
+ function createSignatureAnnotation(
99
+ signatureRef: PdfObjectReference,
100
+ appearanceStreamRef: PdfObjectReference,
101
+ pageRef: PdfObjectReference,
102
+ signatureName: string,
103
+ ): PdfIndirectObject<PdfDictionary> {
104
+ const signatureAnnotation = new PdfDictionary()
105
+ signatureAnnotation.set('Type', new PdfName('Annot'))
106
+ signatureAnnotation.set('Subtype', new PdfName('Widget'))
107
+ signatureAnnotation.set('FT', new PdfName('Sig'))
108
+ signatureAnnotation.set('T', new PdfString(signatureName))
109
+ signatureAnnotation.set(
110
+ 'Rect',
111
+ new PdfArray([
112
+ new PdfNumber(135), // x1: Start after "Signature: " text (~72 + 63)
113
+ new PdfNumber(640), // y1: Bottom of signature area (652 - 12)
114
+ new PdfNumber(400), // x2: End of signature line
115
+ new PdfNumber(665), // y2: Top of signature area (652 + 13)
116
+ ]),
117
+ )
118
+ signatureAnnotation.set('F', new PdfNumber(4))
119
+ signatureAnnotation.set('P', pageRef) // Reference to parent page
120
+ signatureAnnotation.set('V', signatureRef)
121
+
122
+ // Add appearance dictionary
123
+ const appearanceDict = new PdfDictionary()
124
+ appearanceDict.set('N', appearanceStreamRef)
125
+ signatureAnnotation.set('AP', appearanceDict)
126
+
127
+ return new PdfIndirectObject({ content: signatureAnnotation })
128
+ }
129
+
130
+ function createSignatureAppearance(): PdfIndirectObject<PdfStream> {
131
+ // Create font for appearance
132
+ const appearanceFont = new PdfDictionary()
133
+ appearanceFont.set('Type', new PdfName('Font'))
134
+ appearanceFont.set('Subtype', new PdfName('Type1'))
135
+ appearanceFont.set('BaseFont', new PdfName('Helvetica'))
136
+
137
+ const fontDict = new PdfDictionary()
138
+ fontDict.set('F1', appearanceFont)
139
+
140
+ const resourcesDict = new PdfDictionary()
141
+ resourcesDict.set('Font', fontDict)
142
+
143
+ // Create appearance stream header
144
+ const appearanceHeader = new PdfDictionary()
145
+ appearanceHeader.set('Type', new PdfName('XObject'))
146
+ appearanceHeader.set('Subtype', new PdfName('Form'))
147
+ appearanceHeader.set(
148
+ 'BBox',
149
+ new PdfArray([
150
+ new PdfNumber(0),
151
+ new PdfNumber(0),
152
+ new PdfNumber(265), // Width: 400 - 135
153
+ new PdfNumber(25), // Height: 665 - 640
154
+ ]),
155
+ )
156
+ appearanceHeader.set('Resources', resourcesDict)
157
+
158
+ // Create appearance stream for the signature
159
+ return new PdfIndirectObject({
160
+ content: new PdfStream({
161
+ header: appearanceHeader,
162
+ original:
163
+ 'BT /F1 10 Tf 5 14 Td (Digitally signed by: Jake Shirley) Tj ET',
164
+ }),
165
+ })
166
+ }
167
+
168
+ // Create the document
169
+ const document = new PdfDocument()
170
+
171
+ // Create font
172
+ const font = createFont()
173
+ document.add(font)
174
+
175
+ // Create resources with the font
176
+ const resources = createResources(font.reference)
177
+ document.add(resources)
178
+
179
+ // Create content stream for first page
180
+ const contentStream = new PdfIndirectObject({
181
+ content: new PdfStream({
182
+ header: new PdfDictionary(),
183
+ original: 'BT /F1 24 Tf 100 700 Td (Hello, PDF-Lite!) Tj ET',
184
+ }),
185
+ })
186
+ document.add(contentStream)
187
+
188
+ // Create first page
189
+ const page1 = createPage(contentStream.reference)
190
+ page1.content.set('Resources', resources.reference)
191
+ document.add(page1)
192
+
193
+ // Array to hold all pages and signature objects
194
+ const allPages: PdfIndirectObject<PdfDictionary>[] = [page1]
195
+ const allSignatures: any[] = []
196
+ const signatureFields: PdfObjectReference[] = []
197
+
198
+ // Helper function to create a signature page
199
+ function createSignaturePage(
200
+ signatureType: string,
201
+ signatureObj: any,
202
+ pageNumber: number,
203
+ ) {
204
+ const content = new PdfIndirectObject({
205
+ content: new PdfStream({
206
+ header: new PdfDictionary(),
207
+ original: `BT /F1 12 Tf 72 712 Td (Signature Type: ${signatureType}) Tj 0 -60 Td (Signature: ________________________________) Tj ET`,
208
+ }),
209
+ })
210
+ document.add(content)
211
+
212
+ const appearance = createSignatureAppearance()
213
+ document.add(appearance)
214
+
215
+ // Create page first to get its reference
216
+ const page = createPageWithSignatureField(
217
+ content.reference,
218
+ new PdfObjectReference(0, 0), // Temporary placeholder
219
+ )
220
+ page.content.set('Resources', resources.reference)
221
+ document.add(page)
222
+
223
+ // Now create annotation with page reference
224
+ const annotation = createSignatureAnnotation(
225
+ signatureObj.reference,
226
+ appearance.reference,
227
+ page.reference,
228
+ `Signature${pageNumber}`,
229
+ )
230
+ document.add(annotation)
231
+
232
+ // Update page's Annots array with actual annotation reference
233
+ page.content.set('Annots', new PdfArray([annotation.reference]))
234
+
235
+ signatureFields.push(annotation.reference)
236
+ return page
237
+ }
238
+
239
+ // Page 2: Adobe PKCS7 Detached
240
+ const pkcs7DetachedSig = new PdfAdbePkcs7DetachedSignatureObject({
241
+ privateKey: rsaSigningKeys.privateKey,
242
+ certificate: rsaSigningKeys.cert,
243
+ issuerCertificate: rsaSigningKeys.caCert,
244
+ name: 'Jake Shirley',
245
+ location: 'Earth',
246
+ reason: 'PKCS7 Detached Signature',
247
+ contactInfo: 'test@test.com',
248
+ revocationInfo: {
249
+ crls: [rsaSigningKeys.caCrl],
250
+ ocsps: [rsaSigningKeys.ocspResponse],
251
+ },
252
+ })
253
+ allSignatures.push(pkcs7DetachedSig)
254
+ allPages.push(createSignaturePage('Adobe PKCS7 Detached', pkcs7DetachedSig, 2))
255
+
256
+ // Page 3: Adobe PKCS7 SHA1
257
+ const pkcs7Sha1Sig = new PdfAdbePkcs7Sha1SignatureObject({
258
+ privateKey: rsaSigningKeys.privateKey,
259
+ certificate: rsaSigningKeys.cert,
260
+ issuerCertificate: rsaSigningKeys.caCert,
261
+ name: 'Jake Shirley',
262
+ location: 'Earth',
263
+ reason: 'PKCS7 SHA1 Signature',
264
+ contactInfo: 'test@test.com',
265
+ })
266
+ allSignatures.push(pkcs7Sha1Sig)
267
+ allPages.push(createSignaturePage('Adobe PKCS7 SHA1', pkcs7Sha1Sig, 3))
268
+
269
+ // Page 4: Adobe X509 RSA SHA1
270
+ const x509RsaSha1Sig = new PdfAdbePkcsX509RsaSha1SignatureObject({
271
+ privateKey: rsaSigningKeys.privateKey,
272
+ certificate: rsaSigningKeys.cert,
273
+ additionalCertificates: [rsaSigningKeys.caCert],
274
+ name: 'Jake Shirley',
275
+ location: 'Earth',
276
+ reason: 'X509 RSA SHA1 Signature',
277
+ contactInfo: 'test@test.com',
278
+ revocationInfo: {
279
+ crls: [rsaSigningKeys.caCrl],
280
+ ocsps: [rsaSigningKeys.ocspResponse],
281
+ },
282
+ })
283
+ allSignatures.push(x509RsaSha1Sig)
284
+ allPages.push(createSignaturePage('Adobe X509 RSA SHA1', x509RsaSha1Sig, 4))
285
+
286
+ // Page 5: ETSI CAdES Detached
287
+ const cadesDetachedSig = new PdfEtsiCadesDetachedSignatureObject({
288
+ privateKey: rsaSigningKeys.privateKey,
289
+ certificate: rsaSigningKeys.cert,
290
+ issuerCertificate: rsaSigningKeys.caCert,
291
+ name: 'Jake Shirley',
292
+ location: 'Earth',
293
+ reason: 'CAdES Detached Signature',
294
+ contactInfo: 'test@test.com',
295
+ revocationInfo: {
296
+ crls: [rsaSigningKeys.caCrl],
297
+ ocsps: [rsaSigningKeys.ocspResponse],
298
+ },
299
+ })
300
+ allSignatures.push(cadesDetachedSig)
301
+ allPages.push(createSignaturePage('ETSI CAdES Detached', cadesDetachedSig, 5))
302
+
303
+ // Page 6: ETSI RFC3161 (Timestamp)
304
+ const rfc3161Sig = new PdfEtsiRfc3161SignatureObject({
305
+ timeStampAuthority: {
306
+ url: 'https://freetsa.org/tsr',
307
+ },
308
+ })
309
+ allSignatures.push(rfc3161Sig)
310
+ allPages.push(createSignaturePage('ETSI RFC3161 Timestamp', rfc3161Sig, 6))
311
+
312
+ // Create pages collection with all pages
313
+ const pages = createPages(allPages)
314
+ // Set parent reference for all pages
315
+ allPages.forEach((page) => {
316
+ page.content.set('Parent', pages.reference)
317
+ })
318
+ document.add(pages)
319
+
320
+ // Create catalog with AcroForm
321
+ const catalog = createCatalog(pages.reference)
322
+
323
+ // Add AcroForm to catalog with all signature fields
324
+ const acroForm = new PdfDictionary()
325
+ acroForm.set('Fields', new PdfArray(signatureFields))
326
+ acroForm.set('SigFlags', new PdfNumber(3))
327
+ const acroFormObj = new PdfIndirectObject({ content: acroForm })
328
+ document.add(acroFormObj)
329
+ catalog.content.set('AcroForm', acroFormObj.reference)
330
+
331
+ document.add(catalog)
332
+
333
+ // Set the catalog as the root
334
+ document.trailerDict.set('Root', catalog.reference)
335
+
336
+ // IMPORTANT: Add all signatures LAST - after all other objects
337
+ // This ensures the ByteRange is calculated correctly for each signature
338
+ allSignatures.forEach((sig) => {
339
+ document.startNewRevision()
340
+ document.add(sig)
341
+ })
342
+
343
+ await document.commit()
344
+
345
+ const tmpFolder = `${import.meta.dirname}/tmp`
346
+ await fs.mkdir(tmpFolder, { recursive: true })
347
+ await fs.writeFile(`${tmpFolder}/signed-output.pdf`, document.toBytes())
@@ -0,0 +1,206 @@
1
+ // Incremental PDF update example
2
+
3
+ import { PdfArray } from 'pdf-lite/core/objects/pdf-array'
4
+ import { PdfDictionary } from 'pdf-lite/core/objects/pdf-dictionary'
5
+ import { PdfIndirectObject } from 'pdf-lite/core/objects/pdf-indirect-object'
6
+ import { PdfName } from 'pdf-lite/core/objects/pdf-name'
7
+ import { PdfNumber } from 'pdf-lite/core/objects/pdf-number'
8
+ import { PdfObjectReference } from 'pdf-lite/core/objects/pdf-object-reference'
9
+ import { PdfStream } from 'pdf-lite/core/objects/pdf-stream'
10
+ import { PdfDocument } from 'pdf-lite/pdf/pdf-document'
11
+ import fs from 'fs/promises'
12
+
13
+ const tmpFolder = `${import.meta.dirname}/tmp`
14
+ await fs.mkdir(tmpFolder, { recursive: true })
15
+
16
+ // Helper functions for creating PDF objects
17
+ function createPage(
18
+ contentStreamRef: PdfObjectReference,
19
+ ): PdfIndirectObject<PdfDictionary> {
20
+ const pageDict = new PdfDictionary()
21
+ pageDict.set('Type', new PdfName('Page'))
22
+ pageDict.set(
23
+ 'MediaBox',
24
+ new PdfArray([
25
+ new PdfNumber(0),
26
+ new PdfNumber(0),
27
+ new PdfNumber(612),
28
+ new PdfNumber(792),
29
+ ]),
30
+ )
31
+ pageDict.set('Contents', contentStreamRef)
32
+ return new PdfIndirectObject({ content: pageDict })
33
+ }
34
+
35
+ function createPages(
36
+ pages: PdfIndirectObject<PdfDictionary>[],
37
+ ): PdfIndirectObject<PdfDictionary> {
38
+ const pagesDict = new PdfDictionary()
39
+ pagesDict.set('Type', new PdfName('Pages'))
40
+ pagesDict.set('Kids', new PdfArray(pages.map((x) => x.reference)))
41
+ pagesDict.set('Count', new PdfNumber(pages.length))
42
+ return new PdfIndirectObject({ content: pagesDict })
43
+ }
44
+
45
+ function createCatalog(
46
+ pagesRef: PdfObjectReference,
47
+ ): PdfIndirectObject<PdfDictionary> {
48
+ const catalogDict = new PdfDictionary()
49
+ catalogDict.set('Type', new PdfName('Catalog'))
50
+ catalogDict.set('Pages', pagesRef)
51
+ return new PdfIndirectObject({ content: catalogDict })
52
+ }
53
+
54
+ function createFont(): PdfIndirectObject<PdfDictionary> {
55
+ const fontDict = new PdfDictionary()
56
+ fontDict.set('Type', new PdfName('Font'))
57
+ fontDict.set('Subtype', new PdfName('Type1'))
58
+ fontDict.set('BaseFont', new PdfName('Helvetica'))
59
+ return new PdfIndirectObject({ content: fontDict })
60
+ }
61
+
62
+ function createResources(
63
+ fontRef: PdfObjectReference,
64
+ ): PdfIndirectObject<PdfDictionary> {
65
+ const resourcesDict = new PdfDictionary()
66
+ const fontDict = new PdfDictionary()
67
+ fontDict.set('F1', fontRef)
68
+ resourcesDict.set('Font', fontDict)
69
+ return new PdfIndirectObject({ content: resourcesDict })
70
+ }
71
+
72
+ // Step 1: Create an initial PDF document
73
+ console.log('Step 1: Creating initial PDF document...')
74
+ const document = new PdfDocument()
75
+
76
+ const font = createFont()
77
+ document.add(font)
78
+ const resources = createResources(font.reference)
79
+ document.add(resources)
80
+
81
+ const contentStream = new PdfIndirectObject({
82
+ content: new PdfStream({
83
+ header: new PdfDictionary(),
84
+ original:
85
+ 'BT /F1 24 Tf 100 700 Td (Original Document - Revision 1) Tj ET',
86
+ }),
87
+ })
88
+
89
+ const page = createPage(contentStream.reference)
90
+ page.content.set('Resources', resources.reference)
91
+ document.add(page)
92
+
93
+ const pages = createPages([page])
94
+ page.content.set('Parent', pages.reference)
95
+ document.add(pages)
96
+
97
+ const catalog = createCatalog(pages.reference)
98
+ document.add(catalog)
99
+
100
+ document.trailerDict.set('Root', catalog.reference)
101
+ document.add(contentStream)
102
+
103
+ await document.commit()
104
+ // Save the original PDF
105
+ const originalPdfPath = `${tmpFolder}/original.pdf`
106
+ await fs.writeFile(originalPdfPath, document.toBytes())
107
+ console.log(`Original PDF saved to: ${originalPdfPath}`)
108
+ console.log(`Original PDF has ${document.revisions.length} revision(s)`)
109
+
110
+ // Step 2: Load the PDF and perform an incremental update
111
+ console.log('\nStep 2: Loading PDF and performing incremental update...')
112
+
113
+ // Read the existing PDF
114
+ const existingPdfBytes = await fs.readFile(originalPdfPath)
115
+ const loadedDocument = await PdfDocument.fromBytes([existingPdfBytes])
116
+
117
+ // Lock existing revisions to enable incremental mode
118
+ // This ensures changes are added as new revisions instead of modifying existing ones
119
+ loadedDocument.setIncremental(true)
120
+
121
+ // Create new content for the incremental update
122
+ // In a real scenario, this could be adding annotations, form fields, signatures, etc.
123
+ const newContentStream = new PdfIndirectObject({
124
+ objectNumber: contentStream.objectNumber,
125
+ generationNumber: contentStream.generationNumber,
126
+ content: new PdfStream({
127
+ header: new PdfDictionary(),
128
+ original:
129
+ 'BT /F1 18 Tf 100 650 Td (This content was added in Revision 2) Tj ET',
130
+ }),
131
+ })
132
+
133
+ // Add the new content to the document
134
+ loadedDocument.add(newContentStream)
135
+ await loadedDocument.commit()
136
+
137
+ // Save the incrementally updated PDF
138
+ const updatedPdfPath = `${tmpFolder}/incremental-update.pdf`
139
+ await fs.writeFile(updatedPdfPath, loadedDocument.toBytes())
140
+ console.log(`Incrementally updated PDF saved to: ${updatedPdfPath}`)
141
+ console.log(`Updated PDF has ${loadedDocument.revisions.length} revision(s)`)
142
+
143
+ // Step 3: Verify the incremental update preserved the original content
144
+ console.log('\nStep 3: Verifying incremental update...')
145
+
146
+ // Check file sizes to confirm incremental update (new file should be larger)
147
+ const originalStats = await fs.stat(originalPdfPath)
148
+ const updatedStats = await fs.stat(updatedPdfPath)
149
+
150
+ console.log(`Original PDF size: ${originalStats.size} bytes`)
151
+ console.log(`Updated PDF size: ${updatedStats.size} bytes`)
152
+ console.log(
153
+ `Size difference: ${updatedStats.size - originalStats.size} bytes (new revision data)`,
154
+ )
155
+
156
+ // The updated PDF contains the original bytes plus the new revision
157
+ // This is the key feature of incremental updates - the original PDF is preserved
158
+ const updatedPdfBytes = await fs.readFile(updatedPdfPath)
159
+ const originalPdfBytesForComparison = await fs.readFile(originalPdfPath)
160
+
161
+ // Verify that the beginning of the updated PDF matches the original
162
+ const originalBytesMatch = updatedPdfBytes
163
+ .slice(0, originalPdfBytesForComparison.length - 10) // Exclude the %%EOF marker area
164
+ .toString()
165
+ .includes(
166
+ originalPdfBytesForComparison
167
+ .subarray(0, -10)
168
+ .toString()
169
+ .substring(0, 100),
170
+ )
171
+
172
+ console.log(`Original content preserved: ${originalBytesMatch ? 'Yes' : 'No'}`)
173
+
174
+ // Step 4: Add another incremental revision
175
+ console.log('\nStep 4: Adding another incremental revision...')
176
+
177
+ const secondUpdate = await PdfDocument.fromBytes([updatedPdfBytes])
178
+ secondUpdate.setIncremental(true)
179
+
180
+ const thirdRevisionContent = new PdfIndirectObject({
181
+ objectNumber: contentStream.objectNumber,
182
+ generationNumber: contentStream.generationNumber,
183
+ content: new PdfStream(
184
+ 'BT /F1 14 Tf 100 600 Td (Third revision - demonstrates multiple incremental updates) Tj ET',
185
+ ),
186
+ })
187
+
188
+ secondUpdate.add(thirdRevisionContent)
189
+ await secondUpdate.commit()
190
+
191
+ const multiRevisionPdfPath = `${tmpFolder}/multi-revision.pdf`
192
+ await fs.writeFile(multiRevisionPdfPath, secondUpdate.toBytes())
193
+ console.log(`Multi-revision PDF saved to: ${multiRevisionPdfPath}`)
194
+ console.log(
195
+ `Multi-revision PDF has ${secondUpdate.revisions.length} revision(s)`,
196
+ )
197
+
198
+ const multiRevisionStats = await fs.stat(multiRevisionPdfPath)
199
+ console.log(`Multi-revision PDF size: ${multiRevisionStats.size} bytes`)
200
+
201
+ console.log('\n=== Summary ===')
202
+ console.log('Incremental updates allow you to:')
203
+ console.log('1. Preserve the original PDF content (important for signatures)')
204
+ console.log('2. Add new content without modifying existing revisions')
205
+ console.log('3. Maintain a complete history of document changes')
206
+ console.log('4. Stack multiple revisions on top of each other')