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,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