hexapdf 0.1.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 (346) hide show
  1. checksums.yaml +7 -0
  2. data/CONTRIBUTERS +3 -0
  3. data/LICENSE +26 -0
  4. data/README.md +88 -0
  5. data/Rakefile +121 -0
  6. data/VERSION +1 -0
  7. data/agpl-3.0.txt +661 -0
  8. data/bin/hexapdf +6 -0
  9. data/data/hexapdf/afm/Courier-Bold.afm +342 -0
  10. data/data/hexapdf/afm/Courier-BoldOblique.afm +342 -0
  11. data/data/hexapdf/afm/Courier-Oblique.afm +342 -0
  12. data/data/hexapdf/afm/Courier.afm +342 -0
  13. data/data/hexapdf/afm/Helvetica-Bold.afm +2827 -0
  14. data/data/hexapdf/afm/Helvetica-BoldOblique.afm +2827 -0
  15. data/data/hexapdf/afm/Helvetica-Oblique.afm +3051 -0
  16. data/data/hexapdf/afm/Helvetica.afm +3051 -0
  17. data/data/hexapdf/afm/MustRead.html +1 -0
  18. data/data/hexapdf/afm/Symbol.afm +213 -0
  19. data/data/hexapdf/afm/Times-Bold.afm +2588 -0
  20. data/data/hexapdf/afm/Times-BoldItalic.afm +2384 -0
  21. data/data/hexapdf/afm/Times-Italic.afm +2667 -0
  22. data/data/hexapdf/afm/Times-Roman.afm +2419 -0
  23. data/data/hexapdf/afm/ZapfDingbats.afm +225 -0
  24. data/data/hexapdf/encoding/glyphlist.txt +4305 -0
  25. data/data/hexapdf/encoding/zapfdingbats.txt +225 -0
  26. data/examples/arc.rb +50 -0
  27. data/examples/graphics.rb +274 -0
  28. data/examples/hello_world.rb +16 -0
  29. data/examples/machupicchu.jpg +0 -0
  30. data/examples/merging.rb +24 -0
  31. data/examples/optimizing.rb +20 -0
  32. data/examples/show_char_bboxes.rb +55 -0
  33. data/examples/standard_pdf_fonts.rb +72 -0
  34. data/examples/truetype.rb +45 -0
  35. data/lib/hexapdf/cli/extract.rb +128 -0
  36. data/lib/hexapdf/cli/info.rb +121 -0
  37. data/lib/hexapdf/cli/inspect.rb +157 -0
  38. data/lib/hexapdf/cli/modify.rb +218 -0
  39. data/lib/hexapdf/cli.rb +121 -0
  40. data/lib/hexapdf/configuration.rb +392 -0
  41. data/lib/hexapdf/content/canvas.rb +1974 -0
  42. data/lib/hexapdf/content/color_space.rb +364 -0
  43. data/lib/hexapdf/content/graphic_object/arc.rb +267 -0
  44. data/lib/hexapdf/content/graphic_object/endpoint_arc.rb +208 -0
  45. data/lib/hexapdf/content/graphic_object/solid_arc.rb +173 -0
  46. data/lib/hexapdf/content/graphic_object.rb +81 -0
  47. data/lib/hexapdf/content/graphics_state.rb +579 -0
  48. data/lib/hexapdf/content/operator.rb +1072 -0
  49. data/lib/hexapdf/content/parser.rb +204 -0
  50. data/lib/hexapdf/content/processor.rb +451 -0
  51. data/lib/hexapdf/content/transformation_matrix.rb +172 -0
  52. data/lib/hexapdf/content.rb +47 -0
  53. data/lib/hexapdf/data_dir.rb +51 -0
  54. data/lib/hexapdf/dictionary.rb +303 -0
  55. data/lib/hexapdf/dictionary_fields.rb +382 -0
  56. data/lib/hexapdf/document.rb +589 -0
  57. data/lib/hexapdf/document_utils.rb +209 -0
  58. data/lib/hexapdf/encryption/aes.rb +206 -0
  59. data/lib/hexapdf/encryption/arc4.rb +93 -0
  60. data/lib/hexapdf/encryption/fast_aes.rb +79 -0
  61. data/lib/hexapdf/encryption/fast_arc4.rb +67 -0
  62. data/lib/hexapdf/encryption/identity.rb +63 -0
  63. data/lib/hexapdf/encryption/ruby_aes.rb +447 -0
  64. data/lib/hexapdf/encryption/ruby_arc4.rb +96 -0
  65. data/lib/hexapdf/encryption/security_handler.rb +494 -0
  66. data/lib/hexapdf/encryption/standard_security_handler.rb +616 -0
  67. data/lib/hexapdf/encryption.rb +94 -0
  68. data/lib/hexapdf/error.rb +73 -0
  69. data/lib/hexapdf/filter/ascii85_decode.rb +160 -0
  70. data/lib/hexapdf/filter/ascii_hex_decode.rb +87 -0
  71. data/lib/hexapdf/filter/dct_decode.rb +57 -0
  72. data/lib/hexapdf/filter/encryption.rb +59 -0
  73. data/lib/hexapdf/filter/flate_decode.rb +93 -0
  74. data/lib/hexapdf/filter/jpx_decode.rb +56 -0
  75. data/lib/hexapdf/filter/lzw_decode.rb +191 -0
  76. data/lib/hexapdf/filter/predictor.rb +266 -0
  77. data/lib/hexapdf/filter/run_length_decode.rb +108 -0
  78. data/lib/hexapdf/filter.rb +176 -0
  79. data/lib/hexapdf/font/cmap/parser.rb +146 -0
  80. data/lib/hexapdf/font/cmap/writer.rb +176 -0
  81. data/lib/hexapdf/font/cmap.rb +90 -0
  82. data/lib/hexapdf/font/encoding/base.rb +77 -0
  83. data/lib/hexapdf/font/encoding/difference_encoding.rb +64 -0
  84. data/lib/hexapdf/font/encoding/glyph_list.rb +150 -0
  85. data/lib/hexapdf/font/encoding/mac_expert_encoding.rb +221 -0
  86. data/lib/hexapdf/font/encoding/mac_roman_encoding.rb +265 -0
  87. data/lib/hexapdf/font/encoding/standard_encoding.rb +205 -0
  88. data/lib/hexapdf/font/encoding/symbol_encoding.rb +244 -0
  89. data/lib/hexapdf/font/encoding/win_ansi_encoding.rb +280 -0
  90. data/lib/hexapdf/font/encoding/zapf_dingbats_encoding.rb +250 -0
  91. data/lib/hexapdf/font/encoding.rb +68 -0
  92. data/lib/hexapdf/font/true_type/font.rb +179 -0
  93. data/lib/hexapdf/font/true_type/table/cmap.rb +103 -0
  94. data/lib/hexapdf/font/true_type/table/cmap_subtable.rb +384 -0
  95. data/lib/hexapdf/font/true_type/table/directory.rb +92 -0
  96. data/lib/hexapdf/font/true_type/table/glyf.rb +166 -0
  97. data/lib/hexapdf/font/true_type/table/head.rb +143 -0
  98. data/lib/hexapdf/font/true_type/table/hhea.rb +109 -0
  99. data/lib/hexapdf/font/true_type/table/hmtx.rb +79 -0
  100. data/lib/hexapdf/font/true_type/table/loca.rb +79 -0
  101. data/lib/hexapdf/font/true_type/table/maxp.rb +112 -0
  102. data/lib/hexapdf/font/true_type/table/name.rb +218 -0
  103. data/lib/hexapdf/font/true_type/table/os2.rb +200 -0
  104. data/lib/hexapdf/font/true_type/table/post.rb +230 -0
  105. data/lib/hexapdf/font/true_type/table.rb +155 -0
  106. data/lib/hexapdf/font/true_type.rb +48 -0
  107. data/lib/hexapdf/font/true_type_wrapper.rb +240 -0
  108. data/lib/hexapdf/font/type1/afm_parser.rb +230 -0
  109. data/lib/hexapdf/font/type1/character_metrics.rb +67 -0
  110. data/lib/hexapdf/font/type1/font.rb +123 -0
  111. data/lib/hexapdf/font/type1/font_metrics.rb +117 -0
  112. data/lib/hexapdf/font/type1/pfb_parser.rb +71 -0
  113. data/lib/hexapdf/font/type1.rb +52 -0
  114. data/lib/hexapdf/font/type1_wrapper.rb +193 -0
  115. data/lib/hexapdf/font_loader/from_configuration.rb +70 -0
  116. data/lib/hexapdf/font_loader/standard14.rb +98 -0
  117. data/lib/hexapdf/font_loader.rb +85 -0
  118. data/lib/hexapdf/font_utils.rb +89 -0
  119. data/lib/hexapdf/image_loader/jpeg.rb +166 -0
  120. data/lib/hexapdf/image_loader/pdf.rb +89 -0
  121. data/lib/hexapdf/image_loader/png.rb +410 -0
  122. data/lib/hexapdf/image_loader.rb +68 -0
  123. data/lib/hexapdf/importer.rb +139 -0
  124. data/lib/hexapdf/name_tree_node.rb +78 -0
  125. data/lib/hexapdf/number_tree_node.rb +67 -0
  126. data/lib/hexapdf/object.rb +363 -0
  127. data/lib/hexapdf/parser.rb +349 -0
  128. data/lib/hexapdf/rectangle.rb +99 -0
  129. data/lib/hexapdf/reference.rb +98 -0
  130. data/lib/hexapdf/revision.rb +206 -0
  131. data/lib/hexapdf/revisions.rb +194 -0
  132. data/lib/hexapdf/serializer.rb +326 -0
  133. data/lib/hexapdf/stream.rb +279 -0
  134. data/lib/hexapdf/task/dereference.rb +109 -0
  135. data/lib/hexapdf/task/optimize.rb +230 -0
  136. data/lib/hexapdf/task.rb +68 -0
  137. data/lib/hexapdf/tokenizer.rb +406 -0
  138. data/lib/hexapdf/type/catalog.rb +107 -0
  139. data/lib/hexapdf/type/embedded_file.rb +87 -0
  140. data/lib/hexapdf/type/file_specification.rb +232 -0
  141. data/lib/hexapdf/type/font.rb +81 -0
  142. data/lib/hexapdf/type/font_descriptor.rb +109 -0
  143. data/lib/hexapdf/type/font_simple.rb +190 -0
  144. data/lib/hexapdf/type/font_true_type.rb +47 -0
  145. data/lib/hexapdf/type/font_type1.rb +162 -0
  146. data/lib/hexapdf/type/form.rb +103 -0
  147. data/lib/hexapdf/type/graphics_state_parameter.rb +79 -0
  148. data/lib/hexapdf/type/image.rb +73 -0
  149. data/lib/hexapdf/type/info.rb +70 -0
  150. data/lib/hexapdf/type/names.rb +69 -0
  151. data/lib/hexapdf/type/object_stream.rb +224 -0
  152. data/lib/hexapdf/type/page.rb +355 -0
  153. data/lib/hexapdf/type/page_tree_node.rb +269 -0
  154. data/lib/hexapdf/type/resources.rb +212 -0
  155. data/lib/hexapdf/type/trailer.rb +128 -0
  156. data/lib/hexapdf/type/viewer_preferences.rb +73 -0
  157. data/lib/hexapdf/type/xref_stream.rb +204 -0
  158. data/lib/hexapdf/type.rb +67 -0
  159. data/lib/hexapdf/utils/bit_field.rb +87 -0
  160. data/lib/hexapdf/utils/bit_stream.rb +148 -0
  161. data/lib/hexapdf/utils/lru_cache.rb +65 -0
  162. data/lib/hexapdf/utils/math_helpers.rb +55 -0
  163. data/lib/hexapdf/utils/object_hash.rb +130 -0
  164. data/lib/hexapdf/utils/pdf_doc_encoding.rb +93 -0
  165. data/lib/hexapdf/utils/sorted_tree_node.rb +339 -0
  166. data/lib/hexapdf/version.rb +39 -0
  167. data/lib/hexapdf/writer.rb +199 -0
  168. data/lib/hexapdf/xref_section.rb +152 -0
  169. data/lib/hexapdf.rb +34 -0
  170. data/man/man1/hexapdf.1 +249 -0
  171. data/test/data/aes-test-vectors/CBCGFSbox-128-decrypt.data.gz +0 -0
  172. data/test/data/aes-test-vectors/CBCGFSbox-128-encrypt.data.gz +0 -0
  173. data/test/data/aes-test-vectors/CBCGFSbox-192-decrypt.data.gz +0 -0
  174. data/test/data/aes-test-vectors/CBCGFSbox-192-encrypt.data.gz +0 -0
  175. data/test/data/aes-test-vectors/CBCGFSbox-256-decrypt.data.gz +0 -0
  176. data/test/data/aes-test-vectors/CBCGFSbox-256-encrypt.data.gz +0 -0
  177. data/test/data/aes-test-vectors/CBCKeySbox-128-decrypt.data.gz +0 -0
  178. data/test/data/aes-test-vectors/CBCKeySbox-128-encrypt.data.gz +0 -0
  179. data/test/data/aes-test-vectors/CBCKeySbox-192-decrypt.data.gz +0 -0
  180. data/test/data/aes-test-vectors/CBCKeySbox-192-encrypt.data.gz +0 -0
  181. data/test/data/aes-test-vectors/CBCKeySbox-256-decrypt.data.gz +0 -0
  182. data/test/data/aes-test-vectors/CBCKeySbox-256-encrypt.data.gz +0 -0
  183. data/test/data/aes-test-vectors/CBCVarKey-128-decrypt.data.gz +0 -0
  184. data/test/data/aes-test-vectors/CBCVarKey-128-encrypt.data.gz +0 -0
  185. data/test/data/aes-test-vectors/CBCVarKey-192-decrypt.data.gz +0 -0
  186. data/test/data/aes-test-vectors/CBCVarKey-192-encrypt.data.gz +0 -0
  187. data/test/data/aes-test-vectors/CBCVarKey-256-decrypt.data.gz +0 -0
  188. data/test/data/aes-test-vectors/CBCVarKey-256-encrypt.data.gz +0 -0
  189. data/test/data/aes-test-vectors/CBCVarTxt-128-decrypt.data.gz +0 -0
  190. data/test/data/aes-test-vectors/CBCVarTxt-128-encrypt.data.gz +0 -0
  191. data/test/data/aes-test-vectors/CBCVarTxt-192-decrypt.data.gz +0 -0
  192. data/test/data/aes-test-vectors/CBCVarTxt-192-encrypt.data.gz +0 -0
  193. data/test/data/aes-test-vectors/CBCVarTxt-256-decrypt.data.gz +0 -0
  194. data/test/data/aes-test-vectors/CBCVarTxt-256-encrypt.data.gz +0 -0
  195. data/test/data/fonts/Ubuntu-Title.ttf +0 -0
  196. data/test/data/images/cmyk.jpg +0 -0
  197. data/test/data/images/fillbytes.jpg +0 -0
  198. data/test/data/images/gray.jpg +0 -0
  199. data/test/data/images/greyscale-1bit.png +0 -0
  200. data/test/data/images/greyscale-2bit.png +0 -0
  201. data/test/data/images/greyscale-4bit.png +0 -0
  202. data/test/data/images/greyscale-8bit.png +0 -0
  203. data/test/data/images/greyscale-alpha-8bit.png +0 -0
  204. data/test/data/images/greyscale-trns-8bit.png +0 -0
  205. data/test/data/images/greyscale-with-gamma1.0.png +0 -0
  206. data/test/data/images/greyscale-with-gamma1.5.png +0 -0
  207. data/test/data/images/indexed-1bit.png +0 -0
  208. data/test/data/images/indexed-2bit.png +0 -0
  209. data/test/data/images/indexed-4bit.png +0 -0
  210. data/test/data/images/indexed-8bit.png +0 -0
  211. data/test/data/images/indexed-alpha-4bit.png +0 -0
  212. data/test/data/images/indexed-alpha-8bit.png +0 -0
  213. data/test/data/images/rgb.jpg +0 -0
  214. data/test/data/images/truecolour-8bit.png +0 -0
  215. data/test/data/images/truecolour-alpha-8bit.png +0 -0
  216. data/test/data/images/truecolour-gama-chrm-8bit.png +0 -0
  217. data/test/data/images/truecolour-srgb-8bit.png +0 -0
  218. data/test/data/minimal.pdf +44 -0
  219. data/test/data/standard-security-handler/README +9 -0
  220. data/test/data/standard-security-handler/bothpwd-aes-128bit-V4.pdf +44 -0
  221. data/test/data/standard-security-handler/bothpwd-aes-256bit-V5.pdf +0 -0
  222. data/test/data/standard-security-handler/bothpwd-arc4-128bit-V2.pdf +43 -0
  223. data/test/data/standard-security-handler/bothpwd-arc4-128bit-V4.pdf +43 -0
  224. data/test/data/standard-security-handler/bothpwd-arc4-40bit-V1.pdf +0 -0
  225. data/test/data/standard-security-handler/nopwd-aes-128bit-V4.pdf +43 -0
  226. data/test/data/standard-security-handler/nopwd-aes-256bit-V5.pdf +0 -0
  227. data/test/data/standard-security-handler/nopwd-arc4-128bit-V2.pdf +43 -0
  228. data/test/data/standard-security-handler/nopwd-arc4-128bit-V4.pdf +43 -0
  229. data/test/data/standard-security-handler/nopwd-arc4-40bit-V1.pdf +43 -0
  230. data/test/data/standard-security-handler/ownerpwd-aes-128bit-V4.pdf +0 -0
  231. data/test/data/standard-security-handler/ownerpwd-aes-256bit-V5.pdf +43 -0
  232. data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V2.pdf +43 -0
  233. data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V4.pdf +43 -0
  234. data/test/data/standard-security-handler/ownerpwd-arc4-40bit-V1.pdf +43 -0
  235. data/test/data/standard-security-handler/userpwd-aes-128bit-V4.pdf +43 -0
  236. data/test/data/standard-security-handler/userpwd-aes-256bit-V5.pdf +43 -0
  237. data/test/data/standard-security-handler/userpwd-arc4-128bit-V2.pdf +0 -0
  238. data/test/data/standard-security-handler/userpwd-arc4-128bit-V4.pdf +0 -0
  239. data/test/data/standard-security-handler/userpwd-arc4-40bit-V1.pdf +43 -0
  240. data/test/hexapdf/common_tokenizer_tests.rb +204 -0
  241. data/test/hexapdf/content/common.rb +31 -0
  242. data/test/hexapdf/content/graphic_object/test_arc.rb +93 -0
  243. data/test/hexapdf/content/graphic_object/test_endpoint_arc.rb +91 -0
  244. data/test/hexapdf/content/graphic_object/test_solid_arc.rb +86 -0
  245. data/test/hexapdf/content/test_canvas.rb +1113 -0
  246. data/test/hexapdf/content/test_color_space.rb +97 -0
  247. data/test/hexapdf/content/test_graphics_state.rb +138 -0
  248. data/test/hexapdf/content/test_operator.rb +619 -0
  249. data/test/hexapdf/content/test_parser.rb +66 -0
  250. data/test/hexapdf/content/test_processor.rb +156 -0
  251. data/test/hexapdf/content/test_transformation_matrix.rb +64 -0
  252. data/test/hexapdf/encryption/common.rb +87 -0
  253. data/test/hexapdf/encryption/test_aes.rb +121 -0
  254. data/test/hexapdf/encryption/test_arc4.rb +39 -0
  255. data/test/hexapdf/encryption/test_fast_aes.rb +17 -0
  256. data/test/hexapdf/encryption/test_fast_arc4.rb +12 -0
  257. data/test/hexapdf/encryption/test_identity.rb +21 -0
  258. data/test/hexapdf/encryption/test_ruby_aes.rb +23 -0
  259. data/test/hexapdf/encryption/test_ruby_arc4.rb +20 -0
  260. data/test/hexapdf/encryption/test_security_handler.rb +356 -0
  261. data/test/hexapdf/encryption/test_standard_security_handler.rb +274 -0
  262. data/test/hexapdf/filter/common.rb +53 -0
  263. data/test/hexapdf/filter/test_ascii85_decode.rb +60 -0
  264. data/test/hexapdf/filter/test_ascii_hex_decode.rb +33 -0
  265. data/test/hexapdf/filter/test_encryption.rb +24 -0
  266. data/test/hexapdf/filter/test_flate_decode.rb +35 -0
  267. data/test/hexapdf/filter/test_lzw_decode.rb +52 -0
  268. data/test/hexapdf/filter/test_predictor.rb +183 -0
  269. data/test/hexapdf/filter/test_run_length_decode.rb +32 -0
  270. data/test/hexapdf/font/cmap/test_parser.rb +67 -0
  271. data/test/hexapdf/font/cmap/test_writer.rb +58 -0
  272. data/test/hexapdf/font/encoding/test_base.rb +35 -0
  273. data/test/hexapdf/font/encoding/test_difference_encoding.rb +21 -0
  274. data/test/hexapdf/font/encoding/test_glyph_list.rb +59 -0
  275. data/test/hexapdf/font/encoding/test_zapf_dingbats_encoding.rb +16 -0
  276. data/test/hexapdf/font/test_encoding.rb +27 -0
  277. data/test/hexapdf/font/test_true_type_wrapper.rb +110 -0
  278. data/test/hexapdf/font/test_type1_wrapper.rb +66 -0
  279. data/test/hexapdf/font/true_type/common.rb +19 -0
  280. data/test/hexapdf/font/true_type/table/test_cmap.rb +59 -0
  281. data/test/hexapdf/font/true_type/table/test_cmap_subtable.rb +133 -0
  282. data/test/hexapdf/font/true_type/table/test_directory.rb +35 -0
  283. data/test/hexapdf/font/true_type/table/test_glyf.rb +58 -0
  284. data/test/hexapdf/font/true_type/table/test_head.rb +76 -0
  285. data/test/hexapdf/font/true_type/table/test_hhea.rb +40 -0
  286. data/test/hexapdf/font/true_type/table/test_hmtx.rb +38 -0
  287. data/test/hexapdf/font/true_type/table/test_loca.rb +43 -0
  288. data/test/hexapdf/font/true_type/table/test_maxp.rb +62 -0
  289. data/test/hexapdf/font/true_type/table/test_name.rb +95 -0
  290. data/test/hexapdf/font/true_type/table/test_os2.rb +65 -0
  291. data/test/hexapdf/font/true_type/table/test_post.rb +89 -0
  292. data/test/hexapdf/font/true_type/test_font.rb +120 -0
  293. data/test/hexapdf/font/true_type/test_table.rb +41 -0
  294. data/test/hexapdf/font/type1/test_afm_parser.rb +51 -0
  295. data/test/hexapdf/font/type1/test_font.rb +68 -0
  296. data/test/hexapdf/font/type1/test_pfb_parser.rb +37 -0
  297. data/test/hexapdf/font_loader/test_from_configuration.rb +28 -0
  298. data/test/hexapdf/font_loader/test_standard14.rb +22 -0
  299. data/test/hexapdf/image_loader/test_jpeg.rb +83 -0
  300. data/test/hexapdf/image_loader/test_pdf.rb +47 -0
  301. data/test/hexapdf/image_loader/test_png.rb +258 -0
  302. data/test/hexapdf/task/test_dereference.rb +46 -0
  303. data/test/hexapdf/task/test_optimize.rb +137 -0
  304. data/test/hexapdf/test_configuration.rb +82 -0
  305. data/test/hexapdf/test_data_dir.rb +32 -0
  306. data/test/hexapdf/test_dictionary.rb +284 -0
  307. data/test/hexapdf/test_dictionary_fields.rb +185 -0
  308. data/test/hexapdf/test_document.rb +574 -0
  309. data/test/hexapdf/test_document_utils.rb +144 -0
  310. data/test/hexapdf/test_filter.rb +96 -0
  311. data/test/hexapdf/test_font_utils.rb +47 -0
  312. data/test/hexapdf/test_importer.rb +78 -0
  313. data/test/hexapdf/test_object.rb +177 -0
  314. data/test/hexapdf/test_parser.rb +394 -0
  315. data/test/hexapdf/test_rectangle.rb +36 -0
  316. data/test/hexapdf/test_reference.rb +41 -0
  317. data/test/hexapdf/test_revision.rb +139 -0
  318. data/test/hexapdf/test_revisions.rb +93 -0
  319. data/test/hexapdf/test_serializer.rb +169 -0
  320. data/test/hexapdf/test_stream.rb +262 -0
  321. data/test/hexapdf/test_tokenizer.rb +30 -0
  322. data/test/hexapdf/test_writer.rb +120 -0
  323. data/test/hexapdf/test_xref_section.rb +35 -0
  324. data/test/hexapdf/type/test_catalog.rb +30 -0
  325. data/test/hexapdf/type/test_embedded_file.rb +16 -0
  326. data/test/hexapdf/type/test_file_specification.rb +148 -0
  327. data/test/hexapdf/type/test_font.rb +35 -0
  328. data/test/hexapdf/type/test_font_descriptor.rb +51 -0
  329. data/test/hexapdf/type/test_font_simple.rb +190 -0
  330. data/test/hexapdf/type/test_font_type1.rb +128 -0
  331. data/test/hexapdf/type/test_form.rb +60 -0
  332. data/test/hexapdf/type/test_info.rb +14 -0
  333. data/test/hexapdf/type/test_names.rb +9 -0
  334. data/test/hexapdf/type/test_object_stream.rb +84 -0
  335. data/test/hexapdf/type/test_page.rb +260 -0
  336. data/test/hexapdf/type/test_page_tree_node.rb +255 -0
  337. data/test/hexapdf/type/test_resources.rb +167 -0
  338. data/test/hexapdf/type/test_trailer.rb +109 -0
  339. data/test/hexapdf/type/test_xref_stream.rb +131 -0
  340. data/test/hexapdf/utils/test_bit_field.rb +47 -0
  341. data/test/hexapdf/utils/test_lru_cache.rb +22 -0
  342. data/test/hexapdf/utils/test_object_hash.rb +115 -0
  343. data/test/hexapdf/utils/test_pdf_doc_encoding.rb +18 -0
  344. data/test/hexapdf/utils/test_sorted_tree_node.rb +232 -0
  345. data/test/test_helper.rb +56 -0
  346. metadata +427 -0
@@ -0,0 +1,209 @@
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ #--
4
+ # This file is part of HexaPDF.
5
+ #
6
+ # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
7
+ # Copyright (C) 2016 Thomas Leitner
8
+ #
9
+ # HexaPDF is free software: you can redistribute it and/or modify it
10
+ # under the terms of the GNU Affero General Public License version 3 as
11
+ # published by the Free Software Foundation with the addition of the
12
+ # following permission added to Section 15 as permitted in Section 7(a):
13
+ # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
14
+ # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
15
+ # INFRINGEMENT OF THIRD PARTY RIGHTS.
16
+ #
17
+ # HexaPDF is distributed in the hope that it will be useful, but WITHOUT
18
+ # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19
+ # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
20
+ # License for more details.
21
+ #
22
+ # You should have received a copy of the GNU Affero General Public License
23
+ # along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
24
+ #
25
+ # The interactive user interfaces in modified source and object code
26
+ # versions of HexaPDF must display Appropriate Legal Notices, as required
27
+ # under Section 5 of the GNU Affero General Public License version 3.
28
+ #
29
+ # In accordance with Section 7(b) of the GNU Affero General Public
30
+ # License, a covered work must retain the producer line in every PDF that
31
+ # is created or manipulated using HexaPDF.
32
+ #++
33
+
34
+ require 'hexapdf/configuration'
35
+
36
+ module HexaPDF
37
+
38
+ # This class provides utility functions for PDF documents. It is available through the
39
+ # HexaPDF::Document#utils method.
40
+ #
41
+ # Some functions can't be attributed to a single "manager" object. For example, while embedding a
42
+ # file can be done within a HexaPDF::Type::Filespecification object, loading an image from a file
43
+ # as a PDF object doesn't have such a place. Such functions are available via this class.
44
+ class DocumentUtils
45
+
46
+ # This module provides methods for managing the images embedded in a PDF file; images
47
+ # themselves are represented by the HexaPDF::Type::Image class.
48
+ #
49
+ # Since an image can be used as a mask for another image, not all image objects found in a PDF
50
+ # are really used as images. Such cases are all handled by this class automatically.
51
+ module Images
52
+
53
+ # :call-seq:
54
+ # images.add_image(file) -> image
55
+ # images.add_image(io) -> image
56
+ #
57
+ # Adds the image from the given file or IO to the PDF and returns the image object.
58
+ #
59
+ # If the image has been added to the PDF before (i.e. if there is an image object with the
60
+ # same path name), the already existing image object is returned.
61
+ def add_image(file_or_io)
62
+ name = if file_or_io.kind_of?(String)
63
+ file_or_io
64
+ elsif file_or_io.respond_to?(:to_path)
65
+ file_or_io.to_path
66
+ end
67
+ if name
68
+ name = File.absolute_path(name)
69
+ image = each_image.find {|im| im.source_path == name}
70
+ end
71
+ unless image
72
+ image = image_loader_for(file_or_io).load(@document, file_or_io)
73
+ image.source_path = name
74
+ end
75
+ image
76
+ end
77
+
78
+ # :call-seq:
79
+ # images.each_image {|image| block } -> images
80
+ # images.each_image -> Enumerator
81
+ #
82
+ # Iterates over all images in the PDF.
83
+ #
84
+ # Note that only real images are yielded which means, for example, that images used as soft
85
+ # mask are not.
86
+ def each_image(&block)
87
+ images = @document.each(current: false).select do |obj|
88
+ obj[:Subtype] == :Image && !obj[:ImageMask]
89
+ end
90
+ masks = images.each_with_object([]) do |image, temp|
91
+ temp << image[:Mask] if image[:Mask].kind_of?(Stream)
92
+ temp << image[:SMask] if image[:SMask].kind_of?(Stream)
93
+ end
94
+ (images - masks).each(&block)
95
+ end
96
+
97
+ private
98
+
99
+ # Returns the image loader (see ImageLoader) for the given file or IO stream or raises an
100
+ # error if no suitable image loader is found.
101
+ def image_loader_for(file_or_io)
102
+ GlobalConfiguration['image_loader'].each_index do |index|
103
+ loader = GlobalConfiguration.constantize('image_loader', index) do
104
+ raise HexaPDF::Error, "Couldn't retrieve image loader from configuration"
105
+ end
106
+ return loader if loader.handles?(file_or_io)
107
+ end
108
+
109
+ raise HexaPDF::Error, "Couldn't find suitable image loader"
110
+ end
111
+
112
+ end
113
+
114
+
115
+ # This module provides methods for managing file specification of a PDF file.
116
+ #
117
+ # Note that for a given PDF file not all file specifications may be found, e.g. when a file
118
+ # specification is only a string. Therefore this module can only handle those file
119
+ # specifications that are indirect file specification dictionaries with the /Type key set.
120
+ module Files
121
+
122
+ # :call-seq:
123
+ # files.add_file(filename, name: File.basename(filename), description: nil, embed: true) -> file_spec
124
+ # files.add_file(io, name:, description: nil) -> file_spec
125
+ #
126
+ # Adds the file or IO to the PDF and returns the corresponding file specification object.
127
+ #
128
+ # Options:
129
+ #
130
+ # name::
131
+ # The name that should be used for the file path. This name is also for registering the
132
+ # file in the EmbeddedFiles name tree.
133
+ #
134
+ # description::
135
+ # A description of the file.
136
+ #
137
+ # embed::
138
+ # When an IO object is given, it is always embedded and this option is ignored.
139
+ #
140
+ # When a filename is given and this option is +true+, then the file is embedded. Otherwise
141
+ # only a reference to it is stored.
142
+ #
143
+ # See: HexaPDF::Type::FileSpecification
144
+ def add_file(file_or_io, name: nil, description: nil, embed: true)
145
+ name ||= File.basename(file_or_io) if file_or_io.kind_of?(String)
146
+ if name.nil?
147
+ raise ArgumentError, "The name argument is mandatory when given an IO object"
148
+ end
149
+
150
+ spec = @document.add(Type: :Filespec)
151
+ spec.path = name
152
+ spec[:Desc] = description if description
153
+ spec.embed(file_or_io, name: name, register: true) if embed || !file_or_io.kind_of?(String)
154
+ spec
155
+ end
156
+
157
+ # :call-seq:
158
+ # files.each_file(search: false) {|file_spec| block } -> files
159
+ # files.each_file(search: false) -> Enumerator
160
+ #
161
+ # Iterates over indirect file specification dictionaries of the PDF.
162
+ #
163
+ # By default, only the file specifications in their standard locations, namely in the
164
+ # EmbeddedFiles name tree and in the page annotations, are returned. If the +search+ option is
165
+ # +true+, then all indirect objects are searched for file specification dictionaries which is
166
+ # much slower.
167
+ def each_file(search: false)
168
+ return to_enum(__method__, search: search) unless block_given?
169
+
170
+ if search
171
+ @document.each(current: false) do |obj|
172
+ yield(obj) if obj.type == :Filespec
173
+ end
174
+ else
175
+ seen = {}
176
+ tree = @document.catalog[:Names] && @document.catalog[:Names][:EmbeddedFiles]
177
+ tree.each_entry do |_, spec|
178
+ seen[spec] = true
179
+ yield(spec)
180
+ end if tree
181
+
182
+ @document.pages.each_page do |page|
183
+ next unless page[:Annots]
184
+ page[:Annots].each do |annot|
185
+ annot = @document.deref(annot)
186
+ next unless annot[:Subtype] == :FileAttachment
187
+ spec = @document.deref(annot[:FS])
188
+ yield(spec) unless seen.key?(spec)
189
+ seen[spec] = true
190
+ end
191
+ end
192
+ end
193
+
194
+ self
195
+ end
196
+
197
+ end
198
+
199
+ include Images
200
+ include Files
201
+
202
+ # Creates a new DocumentUtils object for the given PDF document.
203
+ def initialize(document)
204
+ @document = document
205
+ end
206
+
207
+ end
208
+
209
+ end
@@ -0,0 +1,206 @@
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ #--
4
+ # This file is part of HexaPDF.
5
+ #
6
+ # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
7
+ # Copyright (C) 2016 Thomas Leitner
8
+ #
9
+ # HexaPDF is free software: you can redistribute it and/or modify it
10
+ # under the terms of the GNU Affero General Public License version 3 as
11
+ # published by the Free Software Foundation with the addition of the
12
+ # following permission added to Section 15 as permitted in Section 7(a):
13
+ # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
14
+ # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
15
+ # INFRINGEMENT OF THIRD PARTY RIGHTS.
16
+ #
17
+ # HexaPDF is distributed in the hope that it will be useful, but WITHOUT
18
+ # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19
+ # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
20
+ # License for more details.
21
+ #
22
+ # You should have received a copy of the GNU Affero General Public License
23
+ # along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
24
+ #
25
+ # The interactive user interfaces in modified source and object code
26
+ # versions of HexaPDF must display Appropriate Legal Notices, as required
27
+ # under Section 5 of the GNU Affero General Public License version 3.
28
+ #
29
+ # In accordance with Section 7(b) of the GNU Affero General Public
30
+ # License, a covered work must retain the producer line in every PDF that
31
+ # is created or manipulated using HexaPDF.
32
+ #++
33
+
34
+ require 'securerandom'
35
+ require 'hexapdf/error'
36
+
37
+ module HexaPDF
38
+ module Encryption
39
+
40
+ # Common interface for AES algorithms
41
+ #
42
+ # This module defines the common interface that is used by the security handlers to encrypt or
43
+ # decrypt data with AES. It has to be *prepended* by any AES algorithm class.
44
+ #
45
+ # See the ClassMethods module for available class level methods of AES algorithms.
46
+ #
47
+ # == Implementing an AES Class
48
+ #
49
+ # An AES class needs to define at least the following methods:
50
+ #
51
+ # initialize(key, iv, mode)::
52
+ # Initializes the AES algorithm with the given key and initialization vector. The mode
53
+ # determines how the AES algorithm object works: If the mode is :encrypt, the object
54
+ # encrypts the data, if the mode is :decrypt, the object decrypts the data.
55
+ #
56
+ # process(data)::
57
+ # Processes the data and returns the encrypted/decrypted data. The method can assume that
58
+ # the passed in data always has a length that is a multiple of BLOCK_SIZE.
59
+ module AES
60
+
61
+ # Valid AES key lengths
62
+ VALID_KEY_LENGTH = [16, 24, 32]
63
+
64
+ # The AES block size
65
+ BLOCK_SIZE = 16
66
+
67
+ # Convenience methods for decryption and encryption that operate according to the PDF
68
+ # specification.
69
+ #
70
+ # These methods will be available on the class object that prepends the AES module.
71
+ module ClassMethods
72
+
73
+ # Encrypts the given +data+ using the +key+ and a randomly generated initialization
74
+ # vector.
75
+ #
76
+ # The data is padded using the PKCS#5 padding scheme and the initialization vector is
77
+ # prepended to the encrypted data,
78
+ #
79
+ # See: PDF1.7 s7.6.2.
80
+ def encrypt(key, data)
81
+ iv = random_bytes(BLOCK_SIZE)
82
+ iv << new(key, iv, :encrypt).process(pad(data))
83
+ end
84
+
85
+ # Returns a Fiber object that encrypts the data from the given source fiber with the
86
+ # +key+.
87
+ #
88
+ # Padding and the initialization vector are handled like in #encrypt.
89
+ def encryption_fiber(key, source)
90
+ Fiber.new do
91
+ data = random_bytes(BLOCK_SIZE)
92
+ algorithm = new(key, data, :encrypt)
93
+ Fiber.yield(data)
94
+
95
+ data = ''.b
96
+ while source.alive? && (new_data = source.resume)
97
+ data << new_data
98
+ next if data.length < BLOCK_SIZE
99
+ new_data = data.slice!(0, data.length - data.length % BLOCK_SIZE)
100
+ Fiber.yield(algorithm.process(new_data))
101
+ end
102
+
103
+ algorithm.process(pad(data))
104
+ end
105
+ end
106
+
107
+ # Decrypts the given +data+ using the +key+.
108
+ #
109
+ # It is assumed that the initialization vector is included in the first BLOCK_SIZE bytes
110
+ # of the data. After the decryption the PKCS#5 padding is removed.
111
+ #
112
+ # See: PDF1.7 s7.6.2.
113
+ def decrypt(key, data)
114
+ if data.length % BLOCK_SIZE != 0 || data.length < 2 * BLOCK_SIZE
115
+ raise HexaPDF::EncryptionError, "Invalid data for decryption, need 32 + 16*n bytes"
116
+ end
117
+ unpad(new(key, data.slice!(0, BLOCK_SIZE), :decrypt).process(data))
118
+ end
119
+
120
+ # Returns a Fiber object that decrypts the data from the given source fiber with the
121
+ # +key+.
122
+ #
123
+ # Padding and the initialization vector are handled like in #decrypt.
124
+ def decryption_fiber(key, source)
125
+ Fiber.new do
126
+ data = ''.b
127
+ while data.length < BLOCK_SIZE && source.alive? && (new_data = source.resume)
128
+ data << new_data
129
+ end
130
+
131
+ algorithm = new(key, data.slice!(0, BLOCK_SIZE), :decrypt)
132
+
133
+ while source.alive? && (new_data = source.resume)
134
+ data << new_data
135
+ next if data.length < 2 * BLOCK_SIZE
136
+ new_data = data.slice!(0, data.length - BLOCK_SIZE - data.length % BLOCK_SIZE)
137
+ Fiber.yield(algorithm.process(new_data))
138
+ end
139
+
140
+ if data.length < BLOCK_SIZE || data.length % BLOCK_SIZE != 0
141
+ raise HexaPDF::EncryptionError, "Invalid data for decryption, need 32 + 16*n bytes"
142
+ end
143
+
144
+ unpad(algorithm.process(data))
145
+ end
146
+ end
147
+
148
+ # Returns a string of n random bytes.
149
+ #
150
+ # The specific AES algorithm class can override this class method to provide another
151
+ # method for generating random bytes.
152
+ def random_bytes(n)
153
+ SecureRandom.random_bytes(n)
154
+ end
155
+
156
+ private
157
+
158
+ # Pads the data to a muliple of BLOCK_SIZE using the PKCS#5 padding scheme and returns the
159
+ # result.
160
+ #
161
+ # See: PDF1.7 s7.6.2
162
+ def pad(data)
163
+ padding_length = BLOCK_SIZE - data.size % BLOCK_SIZE
164
+ data + padding_length.chr * padding_length
165
+ end
166
+
167
+ # Removes the padding from the data according to the PKCS#5 padding scheme and returns the
168
+ # result.
169
+ #
170
+ # See: PDF1.7 s7.6.2
171
+ def unpad(data)
172
+ padding_length = data.getbyte(-1)
173
+ if padding_length > BLOCK_SIZE || padding_length == 0
174
+ raise HexaPDF::EncryptionError, "Invalid AES padding length #{padding_length}"
175
+ end
176
+ data[0...-padding_length]
177
+ end
178
+
179
+ end
180
+
181
+ # Automatically extends the klass with the necessary class level methods.
182
+ def self.prepended(klass) # :nodoc:
183
+ klass.extend(ClassMethods)
184
+ end
185
+
186
+ # Creates a new AES object using the given encryption key and initialization vector.
187
+ #
188
+ # The mode must either be :encrypt or :decrypt.
189
+ #
190
+ # Classes prepending this module have to have their own initialization method as this method
191
+ # just performs basic checks.
192
+ def initialize(key, iv, mode)
193
+ unless VALID_KEY_LENGTH.include?(key.length)
194
+ raise HexaPDF::EncryptionError, "AES key length must be 128, 192 or 256 bit"
195
+ end
196
+ unless iv.length == BLOCK_SIZE
197
+ raise HexaPDF::EncryptionError, "AES initialization vector length must be 128 bit"
198
+ end
199
+ mode = mode.intern
200
+ super
201
+ end
202
+
203
+ end
204
+
205
+ end
206
+ end
@@ -0,0 +1,93 @@
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ #--
4
+ # This file is part of HexaPDF.
5
+ #
6
+ # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
7
+ # Copyright (C) 2016 Thomas Leitner
8
+ #
9
+ # HexaPDF is free software: you can redistribute it and/or modify it
10
+ # under the terms of the GNU Affero General Public License version 3 as
11
+ # published by the Free Software Foundation with the addition of the
12
+ # following permission added to Section 15 as permitted in Section 7(a):
13
+ # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
14
+ # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
15
+ # INFRINGEMENT OF THIRD PARTY RIGHTS.
16
+ #
17
+ # HexaPDF is distributed in the hope that it will be useful, but WITHOUT
18
+ # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19
+ # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
20
+ # License for more details.
21
+ #
22
+ # You should have received a copy of the GNU Affero General Public License
23
+ # along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
24
+ #
25
+ # The interactive user interfaces in modified source and object code
26
+ # versions of HexaPDF must display Appropriate Legal Notices, as required
27
+ # under Section 5 of the GNU Affero General Public License version 3.
28
+ #
29
+ # In accordance with Section 7(b) of the GNU Affero General Public
30
+ # License, a covered work must retain the producer line in every PDF that
31
+ # is created or manipulated using HexaPDF.
32
+ #++
33
+
34
+ module HexaPDF
35
+ module Encryption
36
+
37
+ # Common interface for ARC4 algorithms
38
+ #
39
+ # This module defines the common interface that is used by the security handlers to encrypt or
40
+ # decrypt data with ARC4. It has to be *prepended* by any ARC4 algorithm class.
41
+ #
42
+ # See the ClassMethods module for available class level methods of ARC4 algorithms.
43
+ #
44
+ # == Implementing an ARC4 Class
45
+ #
46
+ # An ARC4 class needs to define at least the following methods:
47
+ #
48
+ # initialize(key)::
49
+ # Initializes the ARC4 algorithm with the given key.
50
+ #
51
+ # process(data)::
52
+ # Processes the data and returns the encrypted/decrypted data. Since the ARC4 algorithm is
53
+ # symmetric in regards to its inner workings, the same method can be used for encryption and
54
+ # decryption.
55
+ module ARC4
56
+
57
+ # Convenience methods for decryption and encryption that operate according to the PDF
58
+ # specification.
59
+ #
60
+ # These methods will be available on the class object that prepends the ARC4 module.
61
+ module ClassMethods
62
+
63
+ # Encrypts the given +data+ with the +key+.
64
+ #
65
+ # See: PDF1.7 s7.6.2.
66
+ def encrypt(key, data)
67
+ new(key).process(data)
68
+ end
69
+ alias :decrypt :encrypt
70
+
71
+ # Returns a Fiber object that encrypts the data from the given source fiber with the
72
+ # +key+.
73
+ def encryption_fiber(key, source)
74
+ Fiber.new do
75
+ algorithm = new(key)
76
+ while source.alive? && (data = source.resume)
77
+ Fiber.yield(algorithm.process(data)) unless data.empty?
78
+ end
79
+ end
80
+ end
81
+ alias :decryption_fiber :encryption_fiber
82
+
83
+ end
84
+
85
+ # Automatically extends the klass with the necessary class level methods.
86
+ def self.prepended(klass) # :nodoc:
87
+ klass.extend(ClassMethods)
88
+ end
89
+
90
+ end
91
+
92
+ end
93
+ end
@@ -0,0 +1,79 @@
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ #--
4
+ # This file is part of HexaPDF.
5
+ #
6
+ # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
7
+ # Copyright (C) 2016 Thomas Leitner
8
+ #
9
+ # HexaPDF is free software: you can redistribute it and/or modify it
10
+ # under the terms of the GNU Affero General Public License version 3 as
11
+ # published by the Free Software Foundation with the addition of the
12
+ # following permission added to Section 15 as permitted in Section 7(a):
13
+ # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
14
+ # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
15
+ # INFRINGEMENT OF THIRD PARTY RIGHTS.
16
+ #
17
+ # HexaPDF is distributed in the hope that it will be useful, but WITHOUT
18
+ # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19
+ # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
20
+ # License for more details.
21
+ #
22
+ # You should have received a copy of the GNU Affero General Public License
23
+ # along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
24
+ #
25
+ # The interactive user interfaces in modified source and object code
26
+ # versions of HexaPDF must display Appropriate Legal Notices, as required
27
+ # under Section 5 of the GNU Affero General Public License version 3.
28
+ #
29
+ # In accordance with Section 7(b) of the GNU Affero General Public
30
+ # License, a covered work must retain the producer line in every PDF that
31
+ # is created or manipulated using HexaPDF.
32
+ #++
33
+
34
+ require 'openssl'
35
+ require 'hexapdf/error'
36
+ require 'hexapdf/encryption/aes'
37
+
38
+ module HexaPDF
39
+ module Encryption
40
+
41
+ # Implementation of the general encryption algorithm AES using OpenSSL as backend.
42
+ #
43
+ # Since OpenSSL is a native Ruby extension (that comes bundled with Ruby) it is much faster
44
+ # than the pure Ruby version and it can use the AES-NI instruction set on CPUs when available.
45
+ #
46
+ # This implementation is using AES in Cipher Block Chaining (CBC) mode.
47
+ #
48
+ # See: PDF1.7 s7.6.2
49
+ class FastAES
50
+
51
+ prepend AES
52
+
53
+ # Uses OpenSSL to generate the requested random bytes.
54
+ #
55
+ # See AES::ClassMethods#random_bytes for more information.
56
+ def self.random_bytes(n)
57
+ OpenSSL::Random.random_bytes(n)
58
+ end
59
+
60
+ # Creates a new FastAES object using the given encryption key and initialization vector.
61
+ #
62
+ # The mode must either be :encrypt or :decrypt.
63
+ def initialize(key, iv, mode)
64
+ @cipher = OpenSSL::Cipher.new("AES-#{key.length << 3}-CBC")
65
+ @cipher.send(mode)
66
+ @cipher.key = key
67
+ @cipher.iv = iv
68
+ @cipher.padding = 0
69
+ end
70
+
71
+ # Encrypts or decrypts the given data whose length must be a multiple of 16.
72
+ def process(data)
73
+ @cipher.update(data)
74
+ end
75
+
76
+ end
77
+
78
+ end
79
+ end
@@ -0,0 +1,67 @@
1
+ # -*- encoding: utf-8 -*-
2
+ #
3
+ #--
4
+ # This file is part of HexaPDF.
5
+ #
6
+ # HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
7
+ # Copyright (C) 2016 Thomas Leitner
8
+ #
9
+ # HexaPDF is free software: you can redistribute it and/or modify it
10
+ # under the terms of the GNU Affero General Public License version 3 as
11
+ # published by the Free Software Foundation with the addition of the
12
+ # following permission added to Section 15 as permitted in Section 7(a):
13
+ # FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
14
+ # THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
15
+ # INFRINGEMENT OF THIRD PARTY RIGHTS.
16
+ #
17
+ # HexaPDF is distributed in the hope that it will be useful, but WITHOUT
18
+ # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
19
+ # FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
20
+ # License for more details.
21
+ #
22
+ # You should have received a copy of the GNU Affero General Public License
23
+ # along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
24
+ #
25
+ # The interactive user interfaces in modified source and object code
26
+ # versions of HexaPDF must display Appropriate Legal Notices, as required
27
+ # under Section 5 of the GNU Affero General Public License version 3.
28
+ #
29
+ # In accordance with Section 7(b) of the GNU Affero General Public
30
+ # License, a covered work must retain the producer line in every PDF that
31
+ # is created or manipulated using HexaPDF.
32
+ #++
33
+
34
+ require 'openssl'
35
+ require 'hexapdf/encryption/arc4'
36
+
37
+ module HexaPDF
38
+ module Encryption
39
+
40
+ # Implementation of the general encryption algorithm ARC4 using OpenSSL as backend.
41
+ #
42
+ # See: PDF1.7 s7.6.2
43
+ class FastARC4
44
+
45
+ prepend ARC4
46
+
47
+ # Creates a new FastARC4 object using the given encryption key.
48
+ def initialize(key)
49
+ @cipher = OpenSSL::Cipher::RC4.new
50
+ @cipher.key_len = key.length
51
+ @cipher.key = key
52
+ end
53
+
54
+ # Processes the given data.
55
+ #
56
+ # Since this is a symmetric algorithm, the same method can be used for encryption and
57
+ # decryption.
58
+ def process(data)
59
+ @cipher.update(data)
60
+ end
61
+ alias :decrypt :process
62
+ alias :encrypt :process
63
+
64
+ end
65
+
66
+ end
67
+ end