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,349 @@
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/error'
35
+ require 'hexapdf/tokenizer'
36
+ require 'hexapdf/stream'
37
+ require 'hexapdf/xref_section'
38
+
39
+ module HexaPDF
40
+
41
+ # Parses an IO stream according to PDF1.7 to get at the contained objects.
42
+ #
43
+ # This class also contains higher-level methods for getting indirect objects and revisions.
44
+ #
45
+ # See: PDF1.7 s7
46
+ class Parser
47
+
48
+ # Creates a new parser for the given IO object.
49
+ #
50
+ # PDF references are resolved using the associated Document object.
51
+ def initialize(io, document)
52
+ @io = io
53
+ @tokenizer = Tokenizer.new(io)
54
+ @document = document
55
+ @object_stream_data = {}
56
+ retrieve_pdf_header_offset_and_version
57
+ end
58
+
59
+ # Loads the indirect (potentially compressed) object specified by the given cross-reference
60
+ # entry.
61
+ #
62
+ # For information about the +xref_entry+ argument, have a look at HexaPDF::XRefSection and
63
+ # HexaPDF::XRefSection::Entry.
64
+ def load_object(xref_entry)
65
+ obj, oid, gen, stream =
66
+ case xref_entry.type
67
+ when :in_use
68
+ parse_indirect_object(xref_entry.pos)
69
+ when :free
70
+ [nil, xref_entry.oid, xref_entry.gen, nil]
71
+ when :compressed
72
+ load_compressed_object(xref_entry)
73
+ else
74
+ raise_malformed("Invalid cross-reference type '#{xref_entry.type}' encountered")
75
+ end
76
+
77
+ if xref_entry.oid != 0 && (oid != xref_entry.oid || gen != xref_entry.gen)
78
+ raise_malformed("The oid,gen (#{oid},#{gen}) values of the indirect object don't match " \
79
+ "the values (#{xref_entry.oid},#{xref_entry.gen}) from the xref")
80
+ end
81
+
82
+ @document.wrap(obj, oid: oid, gen: gen, stream: stream)
83
+ end
84
+
85
+ # Parses the indirect object at the specified offset.
86
+ #
87
+ # This method is used by a PDF Document to load objects. It should **not** be used by any
88
+ # other object because invalid object positions lead to errors.
89
+ #
90
+ # Returns an array containing [object, oid, gen, stream].
91
+ #
92
+ # See: PDF1.7 s7.3.10, s7.3.8
93
+ def parse_indirect_object(offset = nil)
94
+ @tokenizer.pos = offset + @header_offset if offset
95
+ oid = @tokenizer.next_token
96
+ gen = @tokenizer.next_token
97
+ tok = @tokenizer.next_token
98
+ unless oid.kind_of?(Integer) && gen.kind_of?(Integer) &&
99
+ tok.kind_of?(Tokenizer::Token) && tok == 'obj'.freeze
100
+ raise_malformed("No valid object found", pos: offset)
101
+ end
102
+
103
+ if (tok = @tokenizer.peek_token) && tok.kind_of?(Tokenizer::Token) && tok == 'endobj'.freeze
104
+ maybe_raise("No indirect object value between 'obj' and 'endobj'", pos: @tokenizer.pos)
105
+ object = nil
106
+ else
107
+ object = @tokenizer.next_object
108
+ end
109
+
110
+ tok = @tokenizer.next_token
111
+
112
+ if tok.kind_of?(Tokenizer::Token) && tok == 'stream'.freeze
113
+ unless object.kind_of?(Hash)
114
+ raise_malformed("A stream needs a dictionary, not a(n) #{object.class}", pos: offset)
115
+ end
116
+ tok1 = @tokenizer.next_byte
117
+ tok2 = @tokenizer.next_byte if tok1 == 13 # 13=CR, 10=LF
118
+ if tok1 != 10 && tok1 != 13
119
+ raise_malformed("Keyword stream must be followed by LF or CR/LF", pos: @tokenizer.pos)
120
+ elsif tok1 == 13 && tok2 != 10
121
+ maybe_raise("Keyword stream must be followed by LF or CR/LF, not CR alone",
122
+ pos: @tokenizer.pos)
123
+ @tokenizer.pos -= 1
124
+ end
125
+
126
+ # Note that getting :Length might move the IO pointer (when resolving references)
127
+ pos = @tokenizer.pos
128
+ length = if object[:Length].kind_of?(Integer)
129
+ object[:Length]
130
+ elsif object[:Length].kind_of?(Reference)
131
+ @document.deref(object[:Length]).value
132
+ else
133
+ 0
134
+ end
135
+ @tokenizer.pos = pos + length
136
+
137
+ tok = @tokenizer.next_token
138
+ unless tok.kind_of?(Tokenizer::Token) && tok == 'endstream'.freeze
139
+ maybe_raise("Invalid stream length, keyword endstream not found", pos: @tokenizer.pos)
140
+ @tokenizer.pos = pos
141
+ if @tokenizer.scan_until(/(?=\n?endstream)/)
142
+ length = @tokenizer.pos - pos
143
+ tok = @tokenizer.next_token
144
+ else
145
+ raise_malformed("Stream content must be followed by keyword endstream",
146
+ pos: @tokenizer.pos)
147
+ end
148
+ end
149
+ tok = @tokenizer.next_token
150
+
151
+ object[:Length] = length
152
+ stream = StreamData.new(@tokenizer.io, offset: pos, length: length,
153
+ filter: @document.unwrap(object[:Filter]),
154
+ decode_parms: @document.unwrap(object[:DecodeParms]))
155
+ end
156
+
157
+ unless tok.kind_of?(Tokenizer::Token) && tok == 'endobj'.freeze
158
+ maybe_raise("Indirect object must be followed by keyword endobj", pos: @tokenizer.pos)
159
+ end
160
+
161
+ [object, oid, gen, stream]
162
+ end
163
+
164
+ # Loads the compressed object identified by the cross-reference entry.
165
+ def load_compressed_object(xref_entry)
166
+ unless @object_stream_data.key?(xref_entry.objstm)
167
+ obj = @document.object(xref_entry.objstm)
168
+ unless obj.respond_to?(:parse_stream)
169
+ raise_malformed("Object with oid=#{xref_entry.objstm} is not an object stream")
170
+ end
171
+ @object_stream_data[xref_entry.objstm] = obj.parse_stream
172
+ end
173
+
174
+ [*@object_stream_data[xref_entry.objstm].object_by_index(xref_entry.pos), xref_entry.gen, nil]
175
+ end
176
+
177
+ # Loads a single revision whose cross-reference section/stream is located at the given
178
+ # position.
179
+ #
180
+ # Returns an HexaPDF::XRefSection object and the accompanying trailer dictionary.
181
+ def load_revision(pos)
182
+ if xref_section?(pos)
183
+ xref_section, trailer = parse_xref_section_and_trailer(pos)
184
+ else
185
+ obj = load_object(XRefSection.in_use_entry(0, 0, pos))
186
+ unless obj.respond_to?(:xref_section)
187
+ raise_malformed("Object is not a cross-reference stream", pos: pos)
188
+ end
189
+ xref_section = obj.xref_section
190
+ trailer = obj.trailer
191
+ unless xref_section.entry?(obj.oid, obj.gen)
192
+ maybe_raise("Cross-reference stream doesn't contain entry for itself", pos: pos)
193
+ xref_section.add_in_use_entry(obj.oid, obj.gen, pos)
194
+ end
195
+ end
196
+ xref_section.delete(0)
197
+ [xref_section, trailer]
198
+ end
199
+
200
+ # Looks at the given offset and returns +true+ if there is a cross-reference section at that
201
+ # position.
202
+ def xref_section?(offset)
203
+ @tokenizer.pos = offset + @header_offset
204
+ token = @tokenizer.peek_token
205
+ token.kind_of?(Tokenizer::Token) && token == 'xref'
206
+ end
207
+
208
+ # Parses the cross-reference section at the given position and the following trailer and
209
+ # returns them as an array consisting of an HexaPDF::XRefSection instance and a hash.
210
+ #
211
+ # This method can only parse cross-reference sections, not cross-reference streams!
212
+ #
213
+ # See: PDF1.7 s7.5.4, s7.5.5; ADB1.7 sH.3-3.4.3
214
+ def parse_xref_section_and_trailer(offset)
215
+ @tokenizer.pos = offset + @header_offset
216
+ token = @tokenizer.next_token
217
+ unless token.kind_of?(Tokenizer::Token) && token == 'xref'
218
+ raise_malformed("Xref section doesn't start with keyword xref", pos: @tokenizer.pos)
219
+ end
220
+
221
+ xref = XRefSection.new
222
+ start = @tokenizer.next_token
223
+ while start.kind_of?(Integer)
224
+ number_of_entries = @tokenizer.next_token
225
+ unless number_of_entries.kind_of?(Integer)
226
+ raise_malformed("Invalid cross-reference subsection start", pos: @tokenizer.pos)
227
+ end
228
+
229
+ @tokenizer.skip_whitespace
230
+ start.upto(start + number_of_entries - 1) do |oid|
231
+ pos, gen, type = @tokenizer.next_xref_entry
232
+ if xref.entry?(oid)
233
+ next
234
+ elsif type == 'n'.freeze
235
+ if pos == 0 || gen > 65535
236
+ maybe_raise("Invalid in use cross-reference entry in cross-reference section",
237
+ pos: @tokenizer.pos)
238
+ xref.add_free_entry(oid, gen)
239
+ else
240
+ xref.add_in_use_entry(oid, gen, pos)
241
+ end
242
+ else
243
+ xref.add_free_entry(oid, gen)
244
+ end
245
+ end
246
+ start = @tokenizer.next_token
247
+ end
248
+
249
+ unless start.kind_of?(Tokenizer::Token) && start == 'trailer'
250
+ raise_malformed("Trailer doesn't start with keyword trailer", pos: @tokenizer.pos)
251
+ end
252
+
253
+ trailer = @tokenizer.next_object
254
+ unless trailer.kind_of?(Hash)
255
+ raise_malformed("Trailer is a #{trailer.class} instead of a dictionary ", pos: @tokenizer.pos)
256
+ end
257
+
258
+ [xref, trailer]
259
+ end
260
+
261
+ # Returns the offset of the main cross-reference section/stream.
262
+ #
263
+ # Implementation note: Normally, the %%EOF marker has to be on the last line, however, Adobe
264
+ # viewers relax this restriction and so do we.
265
+ #
266
+ # If strict parsing is disabled, the whole file is searched for the offset.
267
+ #
268
+ # See: PDF1.7 s7.5.5, ADB1.7 sH.3-3.4.4
269
+ def startxref_offset
270
+ @io.seek(0, IO::SEEK_END)
271
+ step_size = 1024
272
+ pos = @io.pos
273
+ eof_not_found = startxref_missing = false
274
+
275
+ while pos != 0
276
+ @io.pos = [pos - step_size, 0].max
277
+ pos = @io.pos
278
+ lines = @io.read(step_size + 40).split(/[\r\n]+/)
279
+
280
+ eof_index = lines.rindex {|l| l.strip == '%%EOF' }
281
+ unless eof_index
282
+ eof_not_found = true
283
+ next
284
+ end
285
+ unless eof_index >= 2 && lines[eof_index - 2].strip == "startxref"
286
+ startxref_missing = true
287
+ next
288
+ end
289
+
290
+ break # we found the startxref offset
291
+ end
292
+
293
+ if eof_not_found
294
+ maybe_raise("PDF file trailer with end-of-file marker not found", pos: pos,
295
+ force: !eof_index)
296
+ elsif startxref_missing
297
+ maybe_raise("PDF file trailer is missing startxref keyword", pos: pos,
298
+ force: eof_index < 2 || lines[eof_index - 2].strip != "startxref")
299
+ end
300
+
301
+ lines[eof_index - 1].to_i
302
+ end
303
+
304
+ # Returns the PDF version number that is stored in the file header.
305
+ #
306
+ # See: PDF1.7 s7.5.2
307
+ def file_header_version
308
+ unless @header_version
309
+ raise_malformed("PDF file header is missing or corrupt", pos: 0)
310
+ end
311
+ @header_version
312
+ end
313
+
314
+ private
315
+
316
+ # Retrieves the offset of the PDF header and the PDF version number in it.
317
+ #
318
+ # The PDF header should normally appear on the first line. However, Adobe relaxes this
319
+ # restriction so that the header may appear in the first 1024 bytes. We follow the Adobe
320
+ # convention.
321
+ #
322
+ # See: PDF1.7 s7.5.2, ADB1.7 sH.3-3.4.1
323
+ def retrieve_pdf_header_offset_and_version
324
+ @io.seek(0)
325
+ @header_offset = @io.read(1024).index(/%PDF-(\d\.\d)/) || 0
326
+ @header_version = $1
327
+ end
328
+
329
+ # Raises a HexaPDF::MalformedPDFError with the given message and source position.
330
+ def raise_malformed(msg, pos: nil)
331
+ raise HexaPDF::MalformedPDFError.new(msg, pos: pos)
332
+ end
333
+
334
+ # Calls the block stored in the config option +parser.on_correctable_error+ with the document,
335
+ # the given message and the position. If the returned value is +true+, raises a
336
+ # HexaPDF::MalformedPDFError. Otherwise the error is corrected and parsing continues.
337
+ #
338
+ # If the option +force+ is used, the block is not called and the error is raised immediately.
339
+ def maybe_raise(msg, pos: nil, force: false)
340
+ if force || @document.config['parser.on_correctable_error'].call(@document, msg, pos)
341
+ error = HexaPDF::MalformedPDFError.new(msg, pos: pos)
342
+ error.set_backtrace(caller(1))
343
+ raise error
344
+ end
345
+ end
346
+
347
+ end
348
+
349
+ end
@@ -0,0 +1,99 @@
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/object'
35
+
36
+ module HexaPDF
37
+
38
+ # Implementation of the PDF rectangle data structure.
39
+ #
40
+ # Rectangles are used for describing page and bounding boxes. They are represented by arrays of
41
+ # four numbers specifying the (x,y) coordinates of *any* diagonally opposite corners.
42
+ #
43
+ # This class simplifies the usage of rectangles by automatically normalizing the coordinates so
44
+ # that they are in the order:
45
+ #
46
+ # [llx, lly, urx, ury]
47
+ #
48
+ # where +llx+ is the lower-left x-coordinate, +lly+ is the lower-left y-coordinate, +urx+ is the
49
+ # upper-right x-coordinate and +ury+ is the upper-right y-coordinate.
50
+ #
51
+ # See: PDF1.7 s7.9.5
52
+ class Rectangle < HexaPDF::Object
53
+
54
+ # Returns the x-coordinate of the lower-left corner.
55
+ def left
56
+ value[0]
57
+ end
58
+
59
+ # Returns the x-coordinate of the upper-right corner.
60
+ def right
61
+ value[2]
62
+ end
63
+
64
+ # Returns the y-coordinate of the lower-left corner.
65
+ def bottom
66
+ value[1]
67
+ end
68
+
69
+ # Returns the y-coordinate of the upper-right corner.
70
+ def top
71
+ value[3]
72
+ end
73
+
74
+ # Returns the width of the rectangle.
75
+ def width
76
+ value[2] - value[0]
77
+ end
78
+
79
+ # Returns the height of the rectangle.
80
+ def height
81
+ value[3] - value[1]
82
+ end
83
+
84
+ private
85
+
86
+ # Ensures that the value is an array containing four numbers that specify the lower-left and
87
+ # upper-right corner.
88
+ def after_data_change
89
+ super
90
+ unless value.kind_of?(Array) && value.size == 4
91
+ raise ArgumentError, "A PDF rectangle structure must contain an array of four numbers"
92
+ end
93
+ value[0], value[2] = value[2], value[0] if value[0] > value[2]
94
+ value[1], value[3] = value[3], value[1] if value[1] > value[3]
95
+ end
96
+
97
+ end
98
+
99
+ end
@@ -0,0 +1,98 @@
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/error'
35
+
36
+ module HexaPDF
37
+
38
+ # A reference to an indirect object.
39
+ #
40
+ # The PDF syntax allows for references to existing and non-existing indirect objects. Such
41
+ # references are represented with objects of this class.
42
+ #
43
+ # Note that after initialization changing the object or generation numbers is not possible
44
+ # anymore!
45
+ #
46
+ # The methods #hash and #eql? are implemented so that objects of this class can be used as hash
47
+ # keys. Furthermore the implementation is compatible to the one of Object, i.e. the hash of a
48
+ # Reference object is the same as the hash of an indirect Object.
49
+ #
50
+ # See: PDF1.7 s7.3.10, Object
51
+ class Reference
52
+
53
+ include Comparable
54
+
55
+ # Returns the object number of the referenced indirect object.
56
+ attr_reader :oid
57
+
58
+ # Returns the generation number of the referenced indirect object.
59
+ attr_reader :gen
60
+
61
+ # Creates a new Reference with the given object number and, optionally, generation number.
62
+ def initialize(oid, gen = 0)
63
+ @oid = Integer(oid)
64
+ @gen = Integer(gen)
65
+ end
66
+
67
+ # Compares this object to another object.
68
+ #
69
+ # If the other object does not respond to +oid+ or +gen+, +nil+ is returned. Otherwise objects
70
+ # are ordered first by object number and then by generation number.
71
+ def <=>(other)
72
+ return nil unless other.respond_to?(:oid) && other.respond_to?(:gen)
73
+ (oid == other.oid ? gen <=> other.gen : oid <=> other.oid)
74
+ end
75
+
76
+ # Returns +true+ if the other object is a Reference and has the same object and generation
77
+ # numbers.
78
+ def ==(other)
79
+ other.kind_of?(Reference) && oid == other.oid && gen == other.gen
80
+ end
81
+
82
+ # Returns +true+ if the other object references the same PDF object as this reference object.
83
+ def eql?(other)
84
+ other.respond_to?(:oid) && oid == other.oid && other.respond_to?(:gen) && gen == other.gen
85
+ end
86
+
87
+ # Computes the hash value based on the object and generation numbers.
88
+ def hash
89
+ oid.hash ^ gen.hash
90
+ end
91
+
92
+ def inspect #:nodoc:
93
+ "#<#{self.class.name} [#{oid}, #{gen}]>"
94
+ end
95
+
96
+ end
97
+
98
+ end