hexapdf 0.1.0

Sign up to get free protection for your applications and to get access to all the features.
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,130 @@
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 Utils
36
+
37
+ # There are some structures in a PDF file, for example cross reference tables, that index data
38
+ # based on object and generation numbers. However, there is a restriction that in such
39
+ # structures the object numbers must be unique, e.g. there may not be entries for [1, 0] and
40
+ # \[1, 1] at the same time.
41
+ #
42
+ # This class can be used for storing/retrieving data for such structures.
43
+ class ObjectHash
44
+
45
+ include Enumerable
46
+
47
+ # The biggest object number that is stored in the object hash or zero if no objects are
48
+ # stored.
49
+ attr_reader :max_oid
50
+
51
+ # Creates a new object hash.
52
+ def initialize
53
+ @table = {}
54
+ @oids = {}
55
+ @max_oid = 0
56
+ end
57
+
58
+ # :call-seq:
59
+ # objhash[oid, gen] = data
60
+ #
61
+ # Sets the data for the given object and generation numbers.
62
+ #
63
+ # If there is already an entry for the given object number (even if the generation number is
64
+ # different), this entry will be removed.
65
+ def []=(oid, gen, data)
66
+ @table[oid] = data
67
+ @oids[oid] = gen
68
+ @max_oid = oid if oid > @max_oid
69
+ end
70
+
71
+ # :call-seq:
72
+ # objhash[oid] -> data or nil
73
+ # objhash[oid, gen] -> data or nil
74
+ #
75
+ # Returns the data for the given object number, or for the given object and generation
76
+ # numbers.
77
+ #
78
+ # If there is no such data, +nil+ is returned.
79
+ def [](oid, gen = nil)
80
+ (gen.nil? || gen_for_oid(oid) == gen || nil) && @table[oid]
81
+ end
82
+
83
+ # :call-seq:
84
+ # objhash.gen_for_oid(oid) -> Integer or nil
85
+ #
86
+ # Returns the generation number that is stored along the given object number, or +nil+ if
87
+ # the object number is not used.
88
+ def gen_for_oid(oid)
89
+ @oids[oid]
90
+ end
91
+
92
+ # :call-seq:
93
+ # objhash.entry?(oid) -> true or false
94
+ # objhash.entry?(oid, gen) -> true or false
95
+ #
96
+ # Returns +true+ if there is an entry for the given object number, or for the given object
97
+ # and generation numbers.
98
+ def entry?(oid, gen = nil)
99
+ (gen ? gen_for_oid(oid) == gen : @oids.key?(oid))
100
+ end
101
+
102
+ # Deletes the entry for the given object number.
103
+ def delete(oid)
104
+ @table.delete(oid)
105
+ @oids.delete(oid)
106
+ @max_oid = oids.max || 0 if oid == @max_oid
107
+ end
108
+
109
+ # :call-seq:
110
+ # objhash.each {|oid, gen, data| block } -> objhash
111
+ # objhash.each -> Enumerator
112
+ #
113
+ # Calls the given block once for every entry, passing an array consisting of the object and
114
+ # generation number and the associated data as arguments.
115
+ def each
116
+ return to_enum(__method__) unless block_given?
117
+ @oids.keys.each {|oid| yield(oid, @oids[oid], @table[oid]) if @table.key?(oid)}
118
+ self
119
+ end
120
+
121
+ # Returns all used object numbers as an array.
122
+ def oids
123
+ @oids.keys
124
+ end
125
+ private :oids
126
+
127
+ end
128
+
129
+ end
130
+ 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 Utils
36
+
37
+ # Implements encoding conversion functions for the PDFDocEncoding.
38
+ #
39
+ # The PDFDocEncoding is used, together with UTF-16BE, for strings outside content streams.
40
+ # When a PDF file is loaded and a text string in a PDF object does not start with the UTF-16BE
41
+ # BOM U+FEFF, it is automatically converted to UTF-8 on access.
42
+ #
43
+ # The same is done for text strings in UTF-16BE encoding. Therefore all text strings can be
44
+ # assumed to be in UTF-8.
45
+ #
46
+ # When a PDF file is written, text strings are automatically encoded in either PDFDocEncoding
47
+ # or UTF-16BE depending on the characters in the text string.
48
+ #
49
+ # See: PDF1.7 s7.9.2, D.1, D.3
50
+ module PDFDocEncoding
51
+
52
+ # :nodoc:
53
+ CHARACTER_MAP = %W[\uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD
54
+ \uFFFD \u0009 \u000A \uFFFD \uFFFD \u000D \uFFFD \uFFFD
55
+ \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD \uFFFD
56
+ \u02D8 \u02C7 \u02C6 \u02D9 \u02DD \u02DB \u02DA \u02DC
57
+ \u0020 \u0021 \u0022 \u0023 \u0024 \u0025 \u0026 \u0027
58
+ \u0028 \u0029 \u002a \u002b \u002c \u002d \u002e \u002f
59
+ \u0030 \u0031 \u0032 \u0033 \u0034 \u0035 \u0036 \u0037
60
+ \u0038 \u0039 \u003a \u003b \u003c \u003d \u003e \u003f
61
+ \u0040 \u0041 \u0042 \u0043 \u0044 \u0045 \u0046 \u0047
62
+ \u0048 \u0049 \u004a \u004b \u004c \u004d \u004e \u004f
63
+ \u0050 \u0051 \u0052 \u0053 \u0054 \u0055 \u0056 \u0057
64
+ \u0058 \u0059 \u005a \u005b \u005c \u005d \u005e \u005f
65
+ \u0060 \u0061 \u0062 \u0063 \u0064 \u0065 \u0066 \u0067
66
+ \u0068 \u0069 \u006a \u006b \u006c \u006d \u006e \u006f
67
+ \u0070 \u0071 \u0072 \u0073 \u0074 \u0075 \u0076 \u0077
68
+ \u0078 \u0079 \u007a \u007b \u007c \u007d \u007e \uFFFD
69
+ \u2022 \u2020 \u2021 \u2026 \u2014 \u2013 \u0192 \u2044
70
+ \u2039 \u203a \u2212 \u2030 \u201e \u201c \u201d \u2018
71
+ \u2019 \u201a \u2122 \ufb01 \ufb02 \u0141 \u0152 \u0160
72
+ \u0178 \u017d \u0131 \u0142 \u0153 \u0161 \u017e \uFFFD
73
+ \u20ac \u00a1 \u00a2 \u00a3 \u00a4 \u00a5 \u00a6 \u00a7
74
+ \u00a8 \u00a9 \u00aa \u00ab \u00ac \uFFFD \u00ae \u00af
75
+ \u00b0 \u00b1 \u00b2 \u00b3 \u00b4 \u00b5 \u00b6 \u00b7
76
+ \u00b8 \u00b9 \u00ba \u00bb \u00bc \u00bd \u00be \u00bf
77
+ \u00c0 \u00c1 \u00c2 \u00c3 \u00c4 \u00c5 \u00c6 \u00c7
78
+ \u00c8 \u00c9 \u00ca \u00cb \u00cc \u00cd \u00ce \u00cf
79
+ \u00d0 \u00d1 \u00d2 \u00d3 \u00d4 \u00d5 \u00d6 \u00d7
80
+ \u00d8 \u00d9 \u00da \u00db \u00dc \u00dd \u00de \u00df
81
+ \u00e0 \u00e1 \u00e2 \u00e3 \u00e4 \u00e5 \u00e6 \u00e7
82
+ \u00e8 \u00e9 \u00ea \u00eb \u00ec \u00ed \u00ee \u00ef
83
+ \u00f0 \u00f1 \u00f2 \u00f3 \u00f4 \u00f5 \u00f6 \u00f7
84
+ \u00f8 \u00f9 \u00fa \u00fb \u00fc \u00fd \u00fe \u00ff]
85
+
86
+ # Converts the given string to UTF-8, assuming it contains bytes in PDFDocEncoding.
87
+ def self.convert_to_utf8(str)
88
+ str.each_byte.with_object('') {|byte, result| result << CHARACTER_MAP[byte]}
89
+ end
90
+
91
+ end
92
+ end
93
+ end
@@ -0,0 +1,339 @@
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 Utils
36
+
37
+ # Provides the convenience methods that are used for name trees and number trees.
38
+ #
39
+ # The provided methods require two methods defined in the including class so that they work
40
+ # correctly:
41
+ #
42
+ # leaf_node_container_name::
43
+ # Defines the dictionary entry name that contains the leaf node entries.
44
+ #
45
+ # For example, for name trees this would be :Names.
46
+ #
47
+ # key_type::
48
+ # Defines the class that is used for the keys in the tree.
49
+ #
50
+ # The class defined this way is used for making sure that only valid keys are used.
51
+ #
52
+ # For example, for name trees this would be String.
53
+ #
54
+ # See: HexaPDF::NameTreeNode, HexaPDF::NumberTreeNode
55
+ module SortedTreeNode
56
+
57
+ # Tree nodes must always be indirect.
58
+ #
59
+ # Note: There is no requirement that the root node of a tree must be indirect. However, making
60
+ # it indirect simplifies the implementation and is not against the spec.
61
+ def must_be_indirect?
62
+ true
63
+ end
64
+
65
+ # :call-seq:
66
+ # tree.add_entry(key, data, overwrite: true) -> true or false
67
+ #
68
+ # Adds a new tree entry (key-data pair) to the sorted tree and returns +true+ if it was
69
+ # successfully added.
70
+ #
71
+ # If the option +overwrite+ is +true+, an existing entry is overwritten. Otherwise an error is
72
+ # raised.
73
+ #
74
+ # This method has to be invoked on the root node of the tree!
75
+ def add_entry(key, data, overwrite: true)
76
+ if key?(:Limits)
77
+ raise HexaPDF::Error, "Adding a new tree entry is only allowed via the root node"
78
+ elsif !key.kind_of?(key_type)
79
+ raise ArgumentError, "A key must be a #{key_type} object, not a #{key.class}"
80
+ end
81
+
82
+ container_name = leaf_node_container_name
83
+
84
+ if (!key?(:Kids) && !key?(container_name)) ||
85
+ (value[:Kids] && self[:Kids].empty?)
86
+ value.delete(:Kids)
87
+ value[container_name] = []
88
+ end
89
+
90
+ if key?(container_name)
91
+ result = insert_pair(self[container_name], key, data, overwrite: overwrite)
92
+ split_if_needed(self, self)
93
+ else
94
+ stack = []
95
+ path_to_key(self, key, stack)
96
+
97
+ result = insert_pair(stack.last[container_name], key, data, overwrite: overwrite)
98
+ stack.last[:Limits] = stack.last[container_name].values_at(0, -2)
99
+ stack.reverse_each.inject do |nested_node, node|
100
+ nested_lower = nested_node[:Limits][0]
101
+ nested_upper = nested_node[:Limits][1]
102
+ if node[:Limits][0] > nested_lower
103
+ node[:Limits][0] = nested_lower
104
+ elsif node[:Limits][1] < nested_upper
105
+ node[:Limits][1] = nested_upper
106
+ end
107
+ node
108
+ end
109
+
110
+ split_if_needed(stack[-2] || self, stack[-1])
111
+ end
112
+
113
+ result
114
+ end
115
+
116
+ # Deletes the entry specified by the +key+ from the tree and returns the data. If the tree
117
+ # doesn't contain the key, +nil+ is returned.
118
+ #
119
+ # This method has to be invoked on the root node of the tree!
120
+ def delete_entry(key)
121
+ if key?(:Limits)
122
+ raise HexaPDF::Error, "Deleting a tree entry is only allowed via the root node"
123
+ end
124
+
125
+ stack = [self]
126
+ path_to_key(self, key, stack)
127
+ container_name = leaf_node_container_name
128
+
129
+ return unless stack.last[container_name]
130
+ index = find_in_leaf_node(stack.last[container_name], key)
131
+ return unless stack.last[container_name][index] == key
132
+
133
+ value = stack.last[container_name].delete_at(index)
134
+ document.delete(value) if value.kind_of?(HexaPDF::Object)
135
+ value = stack.last[container_name].delete_at(index)
136
+
137
+ stack.last[:Limits] = stack.last[container_name].values_at(0, -2) if stack.last[:Limits]
138
+
139
+ stack.reverse_each.inject do |nested_node, node|
140
+ if (!nested_node[container_name] || nested_node[container_name].empty?) &&
141
+ (!nested_node[:Kids] || nested_node[:Kids].empty?)
142
+ node[:Kids].delete_at(node[:Kids].index {|n| document.deref(n) == nested_node})
143
+ document.delete(nested_node)
144
+ end
145
+ if node[:Kids].size > 0 && node[:Limits]
146
+ node[:Limits][0] = document.deref(node[:Kids][0])[:Limits][0]
147
+ node[:Limits][1] = document.deref(node[:Kids][-1])[:Limits][1]
148
+ end
149
+ node
150
+ end
151
+
152
+ value
153
+ end
154
+
155
+ # Finds and returns the associated entry for the key, or returns +nil+ if no such key is
156
+ # found.
157
+ def find_entry(key)
158
+ container_name = leaf_node_container_name
159
+ node = self
160
+ result = nil
161
+
162
+ while result.nil?
163
+ if node.key?(container_name)
164
+ index = find_in_leaf_node(node[container_name], key)
165
+ if node[container_name][index] == key
166
+ result = document.deref(node[container_name][index + 1])
167
+ end
168
+ elsif node.key?(:Kids)
169
+ index = find_in_intermediate_node(node[:Kids], key)
170
+ node = document.deref(node[:Kids][index])
171
+ break unless key >= node[:Limits][0] && key <= node[:Limits][1]
172
+ else
173
+ break
174
+ end
175
+ end
176
+
177
+ result
178
+ end
179
+
180
+ # :call-seq:
181
+ # node.each_entry {|key, data| block } -> node
182
+ # node.each_entry -> Enumerator
183
+ #
184
+ # Calls the given block once for each entry (key-data pair) of the sorted tree.
185
+ def each_entry
186
+ return to_enum(__method__) unless block_given?
187
+
188
+ container_name = leaf_node_container_name
189
+ stack = [self]
190
+ until stack.empty?
191
+ node = document.deref(stack.pop)
192
+ if node.key?(container_name)
193
+ data = node[container_name]
194
+ index = 0
195
+ while index < data.length
196
+ yield(data[index], document.deref(data[index + 1]))
197
+ index += 2
198
+ end
199
+ elsif node.key?(:Kids)
200
+ stack.concat(node[:Kids].reverse)
201
+ end
202
+ end
203
+
204
+ self
205
+ end
206
+
207
+ private
208
+
209
+ # Starting from node traverses the tree to the node where the key is located or, if not
210
+ # present, where it would be located and adds the nodes to the stack.
211
+ def path_to_key(node, key, stack)
212
+ return unless node.key?(:Kids)
213
+ index = find_in_intermediate_node(node[:Kids], key)
214
+ stack << document.deref(node[:Kids][index])
215
+ path_to_key(stack.last, key, stack)
216
+ end
217
+
218
+ # Returns the index into the /Kids array where the entry for +key+ is located or, if not
219
+ # present, where it would be located.
220
+ def find_in_intermediate_node(array, key)
221
+ left = 0
222
+ right = array.length - 1
223
+ while left < right
224
+ mid = (left + right) / 2
225
+ limits = document.deref(array[mid])[:Limits]
226
+ if limits[1] < key
227
+ left = mid + 1
228
+ elsif limits[0] > key
229
+ right = mid - 1
230
+ else
231
+ left = right = mid
232
+ end
233
+ end
234
+ left
235
+ end
236
+
237
+ # Inserts the key-data pair into array at the correct position and returns +true+ if the
238
+ # key-data pair was successfully inserted.
239
+ #
240
+ # An existing entry for the key is only overwritten if the option +overwrite+ is +true+.
241
+ def insert_pair(array, key, data, overwrite: true)
242
+ index = find_in_leaf_node(array, key)
243
+ return false if array[index] == key && !overwrite
244
+
245
+ if array[index] == key
246
+ array[index + 1] = data
247
+ else
248
+ array.insert(index, key, data)
249
+ end
250
+
251
+ true
252
+ end
253
+
254
+ # Returns the index into the array where the entry for +key+ is located or, if not present,
255
+ # where it would be located.
256
+ def find_in_leaf_node(array, key)
257
+ left = 0
258
+ right = array.length - 1
259
+ while left <= right
260
+ mid = ((left + right) / 2) & ~1 # mid must be even because of [key val key val...]
261
+ if array[mid] < key
262
+ left = mid + 2
263
+ elsif array[mid] > key
264
+ right = mid - 2
265
+ else
266
+ left = mid
267
+ right = left - 1
268
+ end
269
+ end
270
+ left
271
+ end
272
+
273
+ # Splits the leaf node if it contains the maximum number of entries.
274
+ def split_if_needed(parent, leaf_node)
275
+ container_name = leaf_node_container_name
276
+ max_size = config['sorted_tree.max_leaf_node_size'] * 2
277
+ return unless leaf_node[container_name].size >= max_size
278
+
279
+ split_point = (max_size / 2) & ~1
280
+ if parent == leaf_node
281
+ node1 = document.add(document.wrap({}, type: self.class))
282
+ node2 = document.add(document.wrap({}, type: self.class))
283
+ node1[container_name] = leaf_node[container_name][0, split_point]
284
+ node1[:Limits] = node1[container_name].values_at(0, -2)
285
+ node2[container_name] = leaf_node[container_name][split_point..-1]
286
+ node2[:Limits] = node2[container_name].values_at(0, -2)
287
+ parent.delete(container_name)
288
+ parent[:Kids] = [node1, node2]
289
+ else
290
+ node1 = document.add(document.wrap({}, type: self.class))
291
+ node1[container_name] = leaf_node[container_name].slice!(split_point..-1)
292
+ node1[:Limits] = node1[container_name].values_at(0, -2)
293
+ leaf_node[:Limits][1] = leaf_node[container_name][-2]
294
+ index = 1 + parent[:Kids].index {|o| document.deref(o) == leaf_node}
295
+ parent[:Kids].insert(index, node1)
296
+ end
297
+ end
298
+
299
+ # Validates the sorted tree node.
300
+ def perform_validation
301
+ super
302
+ container_name = leaf_node_container_name
303
+
304
+ # All kids entries must be indirect objects
305
+ if key?(:Kids)
306
+ self[:Kids].each do |kid|
307
+ unless (kid.kind_of?(HexaPDF::Object) && kid.indirect?) ||
308
+ kid.kind_of?(HexaPDF::Reference)
309
+ yield("Child entries of sorted tree nodes must be indirect objects", false)
310
+ end
311
+ end
312
+ end
313
+
314
+ # All keys of the container must be lexically ordered strings and the container must be
315
+ # correctly formatted
316
+ if key?(container_name)
317
+ container = self[container_name]
318
+ if container.length.odd?
319
+ yield("Sorted tree leaf node contains odd number of entries", false)
320
+ end
321
+ index = 0
322
+ old = nil
323
+ while index < container.length
324
+ key = document.unwrap(container[index])
325
+ if !key.kind_of?(key_type)
326
+ yield("A key must be a #{key_type} object, not a #{key.class}", false)
327
+ elsif old && old > key
328
+ yield("Sorted tree leaf node entries are not correctly sorted", false)
329
+ end
330
+ old = key
331
+ index += 2
332
+ end
333
+ end
334
+ end
335
+
336
+ end
337
+
338
+ end
339
+ end
@@ -0,0 +1,39 @@
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
+
36
+ # The version of HexaPDF.
37
+ VERSION = '0.1.0'
38
+
39
+ end