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.
- checksums.yaml +7 -0
- data/CONTRIBUTERS +3 -0
- data/LICENSE +26 -0
- data/README.md +88 -0
- data/Rakefile +121 -0
- data/VERSION +1 -0
- data/agpl-3.0.txt +661 -0
- data/bin/hexapdf +6 -0
- data/data/hexapdf/afm/Courier-Bold.afm +342 -0
- data/data/hexapdf/afm/Courier-BoldOblique.afm +342 -0
- data/data/hexapdf/afm/Courier-Oblique.afm +342 -0
- data/data/hexapdf/afm/Courier.afm +342 -0
- data/data/hexapdf/afm/Helvetica-Bold.afm +2827 -0
- data/data/hexapdf/afm/Helvetica-BoldOblique.afm +2827 -0
- data/data/hexapdf/afm/Helvetica-Oblique.afm +3051 -0
- data/data/hexapdf/afm/Helvetica.afm +3051 -0
- data/data/hexapdf/afm/MustRead.html +1 -0
- data/data/hexapdf/afm/Symbol.afm +213 -0
- data/data/hexapdf/afm/Times-Bold.afm +2588 -0
- data/data/hexapdf/afm/Times-BoldItalic.afm +2384 -0
- data/data/hexapdf/afm/Times-Italic.afm +2667 -0
- data/data/hexapdf/afm/Times-Roman.afm +2419 -0
- data/data/hexapdf/afm/ZapfDingbats.afm +225 -0
- data/data/hexapdf/encoding/glyphlist.txt +4305 -0
- data/data/hexapdf/encoding/zapfdingbats.txt +225 -0
- data/examples/arc.rb +50 -0
- data/examples/graphics.rb +274 -0
- data/examples/hello_world.rb +16 -0
- data/examples/machupicchu.jpg +0 -0
- data/examples/merging.rb +24 -0
- data/examples/optimizing.rb +20 -0
- data/examples/show_char_bboxes.rb +55 -0
- data/examples/standard_pdf_fonts.rb +72 -0
- data/examples/truetype.rb +45 -0
- data/lib/hexapdf/cli/extract.rb +128 -0
- data/lib/hexapdf/cli/info.rb +121 -0
- data/lib/hexapdf/cli/inspect.rb +157 -0
- data/lib/hexapdf/cli/modify.rb +218 -0
- data/lib/hexapdf/cli.rb +121 -0
- data/lib/hexapdf/configuration.rb +392 -0
- data/lib/hexapdf/content/canvas.rb +1974 -0
- data/lib/hexapdf/content/color_space.rb +364 -0
- data/lib/hexapdf/content/graphic_object/arc.rb +267 -0
- data/lib/hexapdf/content/graphic_object/endpoint_arc.rb +208 -0
- data/lib/hexapdf/content/graphic_object/solid_arc.rb +173 -0
- data/lib/hexapdf/content/graphic_object.rb +81 -0
- data/lib/hexapdf/content/graphics_state.rb +579 -0
- data/lib/hexapdf/content/operator.rb +1072 -0
- data/lib/hexapdf/content/parser.rb +204 -0
- data/lib/hexapdf/content/processor.rb +451 -0
- data/lib/hexapdf/content/transformation_matrix.rb +172 -0
- data/lib/hexapdf/content.rb +47 -0
- data/lib/hexapdf/data_dir.rb +51 -0
- data/lib/hexapdf/dictionary.rb +303 -0
- data/lib/hexapdf/dictionary_fields.rb +382 -0
- data/lib/hexapdf/document.rb +589 -0
- data/lib/hexapdf/document_utils.rb +209 -0
- data/lib/hexapdf/encryption/aes.rb +206 -0
- data/lib/hexapdf/encryption/arc4.rb +93 -0
- data/lib/hexapdf/encryption/fast_aes.rb +79 -0
- data/lib/hexapdf/encryption/fast_arc4.rb +67 -0
- data/lib/hexapdf/encryption/identity.rb +63 -0
- data/lib/hexapdf/encryption/ruby_aes.rb +447 -0
- data/lib/hexapdf/encryption/ruby_arc4.rb +96 -0
- data/lib/hexapdf/encryption/security_handler.rb +494 -0
- data/lib/hexapdf/encryption/standard_security_handler.rb +616 -0
- data/lib/hexapdf/encryption.rb +94 -0
- data/lib/hexapdf/error.rb +73 -0
- data/lib/hexapdf/filter/ascii85_decode.rb +160 -0
- data/lib/hexapdf/filter/ascii_hex_decode.rb +87 -0
- data/lib/hexapdf/filter/dct_decode.rb +57 -0
- data/lib/hexapdf/filter/encryption.rb +59 -0
- data/lib/hexapdf/filter/flate_decode.rb +93 -0
- data/lib/hexapdf/filter/jpx_decode.rb +56 -0
- data/lib/hexapdf/filter/lzw_decode.rb +191 -0
- data/lib/hexapdf/filter/predictor.rb +266 -0
- data/lib/hexapdf/filter/run_length_decode.rb +108 -0
- data/lib/hexapdf/filter.rb +176 -0
- data/lib/hexapdf/font/cmap/parser.rb +146 -0
- data/lib/hexapdf/font/cmap/writer.rb +176 -0
- data/lib/hexapdf/font/cmap.rb +90 -0
- data/lib/hexapdf/font/encoding/base.rb +77 -0
- data/lib/hexapdf/font/encoding/difference_encoding.rb +64 -0
- data/lib/hexapdf/font/encoding/glyph_list.rb +150 -0
- data/lib/hexapdf/font/encoding/mac_expert_encoding.rb +221 -0
- data/lib/hexapdf/font/encoding/mac_roman_encoding.rb +265 -0
- data/lib/hexapdf/font/encoding/standard_encoding.rb +205 -0
- data/lib/hexapdf/font/encoding/symbol_encoding.rb +244 -0
- data/lib/hexapdf/font/encoding/win_ansi_encoding.rb +280 -0
- data/lib/hexapdf/font/encoding/zapf_dingbats_encoding.rb +250 -0
- data/lib/hexapdf/font/encoding.rb +68 -0
- data/lib/hexapdf/font/true_type/font.rb +179 -0
- data/lib/hexapdf/font/true_type/table/cmap.rb +103 -0
- data/lib/hexapdf/font/true_type/table/cmap_subtable.rb +384 -0
- data/lib/hexapdf/font/true_type/table/directory.rb +92 -0
- data/lib/hexapdf/font/true_type/table/glyf.rb +166 -0
- data/lib/hexapdf/font/true_type/table/head.rb +143 -0
- data/lib/hexapdf/font/true_type/table/hhea.rb +109 -0
- data/lib/hexapdf/font/true_type/table/hmtx.rb +79 -0
- data/lib/hexapdf/font/true_type/table/loca.rb +79 -0
- data/lib/hexapdf/font/true_type/table/maxp.rb +112 -0
- data/lib/hexapdf/font/true_type/table/name.rb +218 -0
- data/lib/hexapdf/font/true_type/table/os2.rb +200 -0
- data/lib/hexapdf/font/true_type/table/post.rb +230 -0
- data/lib/hexapdf/font/true_type/table.rb +155 -0
- data/lib/hexapdf/font/true_type.rb +48 -0
- data/lib/hexapdf/font/true_type_wrapper.rb +240 -0
- data/lib/hexapdf/font/type1/afm_parser.rb +230 -0
- data/lib/hexapdf/font/type1/character_metrics.rb +67 -0
- data/lib/hexapdf/font/type1/font.rb +123 -0
- data/lib/hexapdf/font/type1/font_metrics.rb +117 -0
- data/lib/hexapdf/font/type1/pfb_parser.rb +71 -0
- data/lib/hexapdf/font/type1.rb +52 -0
- data/lib/hexapdf/font/type1_wrapper.rb +193 -0
- data/lib/hexapdf/font_loader/from_configuration.rb +70 -0
- data/lib/hexapdf/font_loader/standard14.rb +98 -0
- data/lib/hexapdf/font_loader.rb +85 -0
- data/lib/hexapdf/font_utils.rb +89 -0
- data/lib/hexapdf/image_loader/jpeg.rb +166 -0
- data/lib/hexapdf/image_loader/pdf.rb +89 -0
- data/lib/hexapdf/image_loader/png.rb +410 -0
- data/lib/hexapdf/image_loader.rb +68 -0
- data/lib/hexapdf/importer.rb +139 -0
- data/lib/hexapdf/name_tree_node.rb +78 -0
- data/lib/hexapdf/number_tree_node.rb +67 -0
- data/lib/hexapdf/object.rb +363 -0
- data/lib/hexapdf/parser.rb +349 -0
- data/lib/hexapdf/rectangle.rb +99 -0
- data/lib/hexapdf/reference.rb +98 -0
- data/lib/hexapdf/revision.rb +206 -0
- data/lib/hexapdf/revisions.rb +194 -0
- data/lib/hexapdf/serializer.rb +326 -0
- data/lib/hexapdf/stream.rb +279 -0
- data/lib/hexapdf/task/dereference.rb +109 -0
- data/lib/hexapdf/task/optimize.rb +230 -0
- data/lib/hexapdf/task.rb +68 -0
- data/lib/hexapdf/tokenizer.rb +406 -0
- data/lib/hexapdf/type/catalog.rb +107 -0
- data/lib/hexapdf/type/embedded_file.rb +87 -0
- data/lib/hexapdf/type/file_specification.rb +232 -0
- data/lib/hexapdf/type/font.rb +81 -0
- data/lib/hexapdf/type/font_descriptor.rb +109 -0
- data/lib/hexapdf/type/font_simple.rb +190 -0
- data/lib/hexapdf/type/font_true_type.rb +47 -0
- data/lib/hexapdf/type/font_type1.rb +162 -0
- data/lib/hexapdf/type/form.rb +103 -0
- data/lib/hexapdf/type/graphics_state_parameter.rb +79 -0
- data/lib/hexapdf/type/image.rb +73 -0
- data/lib/hexapdf/type/info.rb +70 -0
- data/lib/hexapdf/type/names.rb +69 -0
- data/lib/hexapdf/type/object_stream.rb +224 -0
- data/lib/hexapdf/type/page.rb +355 -0
- data/lib/hexapdf/type/page_tree_node.rb +269 -0
- data/lib/hexapdf/type/resources.rb +212 -0
- data/lib/hexapdf/type/trailer.rb +128 -0
- data/lib/hexapdf/type/viewer_preferences.rb +73 -0
- data/lib/hexapdf/type/xref_stream.rb +204 -0
- data/lib/hexapdf/type.rb +67 -0
- data/lib/hexapdf/utils/bit_field.rb +87 -0
- data/lib/hexapdf/utils/bit_stream.rb +148 -0
- data/lib/hexapdf/utils/lru_cache.rb +65 -0
- data/lib/hexapdf/utils/math_helpers.rb +55 -0
- data/lib/hexapdf/utils/object_hash.rb +130 -0
- data/lib/hexapdf/utils/pdf_doc_encoding.rb +93 -0
- data/lib/hexapdf/utils/sorted_tree_node.rb +339 -0
- data/lib/hexapdf/version.rb +39 -0
- data/lib/hexapdf/writer.rb +199 -0
- data/lib/hexapdf/xref_section.rb +152 -0
- data/lib/hexapdf.rb +34 -0
- data/man/man1/hexapdf.1 +249 -0
- data/test/data/aes-test-vectors/CBCGFSbox-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCGFSbox-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCKeySbox-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarKey-256-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-128-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-128-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-192-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-192-encrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-256-decrypt.data.gz +0 -0
- data/test/data/aes-test-vectors/CBCVarTxt-256-encrypt.data.gz +0 -0
- data/test/data/fonts/Ubuntu-Title.ttf +0 -0
- data/test/data/images/cmyk.jpg +0 -0
- data/test/data/images/fillbytes.jpg +0 -0
- data/test/data/images/gray.jpg +0 -0
- data/test/data/images/greyscale-1bit.png +0 -0
- data/test/data/images/greyscale-2bit.png +0 -0
- data/test/data/images/greyscale-4bit.png +0 -0
- data/test/data/images/greyscale-8bit.png +0 -0
- data/test/data/images/greyscale-alpha-8bit.png +0 -0
- data/test/data/images/greyscale-trns-8bit.png +0 -0
- data/test/data/images/greyscale-with-gamma1.0.png +0 -0
- data/test/data/images/greyscale-with-gamma1.5.png +0 -0
- data/test/data/images/indexed-1bit.png +0 -0
- data/test/data/images/indexed-2bit.png +0 -0
- data/test/data/images/indexed-4bit.png +0 -0
- data/test/data/images/indexed-8bit.png +0 -0
- data/test/data/images/indexed-alpha-4bit.png +0 -0
- data/test/data/images/indexed-alpha-8bit.png +0 -0
- data/test/data/images/rgb.jpg +0 -0
- data/test/data/images/truecolour-8bit.png +0 -0
- data/test/data/images/truecolour-alpha-8bit.png +0 -0
- data/test/data/images/truecolour-gama-chrm-8bit.png +0 -0
- data/test/data/images/truecolour-srgb-8bit.png +0 -0
- data/test/data/minimal.pdf +44 -0
- data/test/data/standard-security-handler/README +9 -0
- data/test/data/standard-security-handler/bothpwd-aes-128bit-V4.pdf +44 -0
- data/test/data/standard-security-handler/bothpwd-aes-256bit-V5.pdf +0 -0
- data/test/data/standard-security-handler/bothpwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/bothpwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/bothpwd-arc4-40bit-V1.pdf +0 -0
- data/test/data/standard-security-handler/nopwd-aes-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-aes-256bit-V5.pdf +0 -0
- data/test/data/standard-security-handler/nopwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/nopwd-arc4-40bit-V1.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-aes-128bit-V4.pdf +0 -0
- data/test/data/standard-security-handler/ownerpwd-aes-256bit-V5.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V2.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/ownerpwd-arc4-40bit-V1.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-aes-128bit-V4.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-aes-256bit-V5.pdf +43 -0
- data/test/data/standard-security-handler/userpwd-arc4-128bit-V2.pdf +0 -0
- data/test/data/standard-security-handler/userpwd-arc4-128bit-V4.pdf +0 -0
- data/test/data/standard-security-handler/userpwd-arc4-40bit-V1.pdf +43 -0
- data/test/hexapdf/common_tokenizer_tests.rb +204 -0
- data/test/hexapdf/content/common.rb +31 -0
- data/test/hexapdf/content/graphic_object/test_arc.rb +93 -0
- data/test/hexapdf/content/graphic_object/test_endpoint_arc.rb +91 -0
- data/test/hexapdf/content/graphic_object/test_solid_arc.rb +86 -0
- data/test/hexapdf/content/test_canvas.rb +1113 -0
- data/test/hexapdf/content/test_color_space.rb +97 -0
- data/test/hexapdf/content/test_graphics_state.rb +138 -0
- data/test/hexapdf/content/test_operator.rb +619 -0
- data/test/hexapdf/content/test_parser.rb +66 -0
- data/test/hexapdf/content/test_processor.rb +156 -0
- data/test/hexapdf/content/test_transformation_matrix.rb +64 -0
- data/test/hexapdf/encryption/common.rb +87 -0
- data/test/hexapdf/encryption/test_aes.rb +121 -0
- data/test/hexapdf/encryption/test_arc4.rb +39 -0
- data/test/hexapdf/encryption/test_fast_aes.rb +17 -0
- data/test/hexapdf/encryption/test_fast_arc4.rb +12 -0
- data/test/hexapdf/encryption/test_identity.rb +21 -0
- data/test/hexapdf/encryption/test_ruby_aes.rb +23 -0
- data/test/hexapdf/encryption/test_ruby_arc4.rb +20 -0
- data/test/hexapdf/encryption/test_security_handler.rb +356 -0
- data/test/hexapdf/encryption/test_standard_security_handler.rb +274 -0
- data/test/hexapdf/filter/common.rb +53 -0
- data/test/hexapdf/filter/test_ascii85_decode.rb +60 -0
- data/test/hexapdf/filter/test_ascii_hex_decode.rb +33 -0
- data/test/hexapdf/filter/test_encryption.rb +24 -0
- data/test/hexapdf/filter/test_flate_decode.rb +35 -0
- data/test/hexapdf/filter/test_lzw_decode.rb +52 -0
- data/test/hexapdf/filter/test_predictor.rb +183 -0
- data/test/hexapdf/filter/test_run_length_decode.rb +32 -0
- data/test/hexapdf/font/cmap/test_parser.rb +67 -0
- data/test/hexapdf/font/cmap/test_writer.rb +58 -0
- data/test/hexapdf/font/encoding/test_base.rb +35 -0
- data/test/hexapdf/font/encoding/test_difference_encoding.rb +21 -0
- data/test/hexapdf/font/encoding/test_glyph_list.rb +59 -0
- data/test/hexapdf/font/encoding/test_zapf_dingbats_encoding.rb +16 -0
- data/test/hexapdf/font/test_encoding.rb +27 -0
- data/test/hexapdf/font/test_true_type_wrapper.rb +110 -0
- data/test/hexapdf/font/test_type1_wrapper.rb +66 -0
- data/test/hexapdf/font/true_type/common.rb +19 -0
- data/test/hexapdf/font/true_type/table/test_cmap.rb +59 -0
- data/test/hexapdf/font/true_type/table/test_cmap_subtable.rb +133 -0
- data/test/hexapdf/font/true_type/table/test_directory.rb +35 -0
- data/test/hexapdf/font/true_type/table/test_glyf.rb +58 -0
- data/test/hexapdf/font/true_type/table/test_head.rb +76 -0
- data/test/hexapdf/font/true_type/table/test_hhea.rb +40 -0
- data/test/hexapdf/font/true_type/table/test_hmtx.rb +38 -0
- data/test/hexapdf/font/true_type/table/test_loca.rb +43 -0
- data/test/hexapdf/font/true_type/table/test_maxp.rb +62 -0
- data/test/hexapdf/font/true_type/table/test_name.rb +95 -0
- data/test/hexapdf/font/true_type/table/test_os2.rb +65 -0
- data/test/hexapdf/font/true_type/table/test_post.rb +89 -0
- data/test/hexapdf/font/true_type/test_font.rb +120 -0
- data/test/hexapdf/font/true_type/test_table.rb +41 -0
- data/test/hexapdf/font/type1/test_afm_parser.rb +51 -0
- data/test/hexapdf/font/type1/test_font.rb +68 -0
- data/test/hexapdf/font/type1/test_pfb_parser.rb +37 -0
- data/test/hexapdf/font_loader/test_from_configuration.rb +28 -0
- data/test/hexapdf/font_loader/test_standard14.rb +22 -0
- data/test/hexapdf/image_loader/test_jpeg.rb +83 -0
- data/test/hexapdf/image_loader/test_pdf.rb +47 -0
- data/test/hexapdf/image_loader/test_png.rb +258 -0
- data/test/hexapdf/task/test_dereference.rb +46 -0
- data/test/hexapdf/task/test_optimize.rb +137 -0
- data/test/hexapdf/test_configuration.rb +82 -0
- data/test/hexapdf/test_data_dir.rb +32 -0
- data/test/hexapdf/test_dictionary.rb +284 -0
- data/test/hexapdf/test_dictionary_fields.rb +185 -0
- data/test/hexapdf/test_document.rb +574 -0
- data/test/hexapdf/test_document_utils.rb +144 -0
- data/test/hexapdf/test_filter.rb +96 -0
- data/test/hexapdf/test_font_utils.rb +47 -0
- data/test/hexapdf/test_importer.rb +78 -0
- data/test/hexapdf/test_object.rb +177 -0
- data/test/hexapdf/test_parser.rb +394 -0
- data/test/hexapdf/test_rectangle.rb +36 -0
- data/test/hexapdf/test_reference.rb +41 -0
- data/test/hexapdf/test_revision.rb +139 -0
- data/test/hexapdf/test_revisions.rb +93 -0
- data/test/hexapdf/test_serializer.rb +169 -0
- data/test/hexapdf/test_stream.rb +262 -0
- data/test/hexapdf/test_tokenizer.rb +30 -0
- data/test/hexapdf/test_writer.rb +120 -0
- data/test/hexapdf/test_xref_section.rb +35 -0
- data/test/hexapdf/type/test_catalog.rb +30 -0
- data/test/hexapdf/type/test_embedded_file.rb +16 -0
- data/test/hexapdf/type/test_file_specification.rb +148 -0
- data/test/hexapdf/type/test_font.rb +35 -0
- data/test/hexapdf/type/test_font_descriptor.rb +51 -0
- data/test/hexapdf/type/test_font_simple.rb +190 -0
- data/test/hexapdf/type/test_font_type1.rb +128 -0
- data/test/hexapdf/type/test_form.rb +60 -0
- data/test/hexapdf/type/test_info.rb +14 -0
- data/test/hexapdf/type/test_names.rb +9 -0
- data/test/hexapdf/type/test_object_stream.rb +84 -0
- data/test/hexapdf/type/test_page.rb +260 -0
- data/test/hexapdf/type/test_page_tree_node.rb +255 -0
- data/test/hexapdf/type/test_resources.rb +167 -0
- data/test/hexapdf/type/test_trailer.rb +109 -0
- data/test/hexapdf/type/test_xref_stream.rb +131 -0
- data/test/hexapdf/utils/test_bit_field.rb +47 -0
- data/test/hexapdf/utils/test_lru_cache.rb +22 -0
- data/test/hexapdf/utils/test_object_hash.rb +115 -0
- data/test/hexapdf/utils/test_pdf_doc_encoding.rb +18 -0
- data/test/hexapdf/utils/test_sorted_tree_node.rb +232 -0
- data/test/test_helper.rb +56 -0
- metadata +427 -0
@@ -0,0 +1,206 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2016 Thomas Leitner
|
8
|
+
#
|
9
|
+
# HexaPDF is free software: you can redistribute it and/or modify it
|
10
|
+
# under the terms of the GNU Affero General Public License version 3 as
|
11
|
+
# published by the Free Software Foundation with the addition of the
|
12
|
+
# following permission added to Section 15 as permitted in Section 7(a):
|
13
|
+
# FOR ANY PART OF THE COVERED WORK IN WHICH THE COPYRIGHT IS OWNED BY
|
14
|
+
# THOMAS LEITNER, THOMAS LEITNER DISCLAIMS THE WARRANTY OF NON
|
15
|
+
# INFRINGEMENT OF THIRD PARTY RIGHTS.
|
16
|
+
#
|
17
|
+
# HexaPDF is distributed in the hope that it will be useful, but WITHOUT
|
18
|
+
# ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
|
19
|
+
# FITNESS FOR A PARTICULAR PURPOSE. See the GNU Affero General Public
|
20
|
+
# License for more details.
|
21
|
+
#
|
22
|
+
# You should have received a copy of the GNU Affero General Public License
|
23
|
+
# along with HexaPDF. If not, see <http://www.gnu.org/licenses/>.
|
24
|
+
#
|
25
|
+
# The interactive user interfaces in modified source and object code
|
26
|
+
# versions of HexaPDF must display Appropriate Legal Notices, as required
|
27
|
+
# under Section 5 of the GNU Affero General Public License version 3.
|
28
|
+
#
|
29
|
+
# In accordance with Section 7(b) of the GNU Affero General Public
|
30
|
+
# License, a covered work must retain the producer line in every PDF that
|
31
|
+
# is created or manipulated using HexaPDF.
|
32
|
+
#++
|
33
|
+
|
34
|
+
require 'hexapdf/error'
|
35
|
+
require 'hexapdf/utils/object_hash'
|
36
|
+
|
37
|
+
module HexaPDF
|
38
|
+
|
39
|
+
# Embodies one revision of a PDF file, either the initial version or an incremental update.
|
40
|
+
#
|
41
|
+
# The purpose of a Revision object is to manage the objects and the trailer of one revision.
|
42
|
+
# These objects can either be added manually or loaded from a cross-reference section or stream.
|
43
|
+
# Since a PDF file can be incrementally updated, it can have multiple revisions.
|
44
|
+
#
|
45
|
+
# If a revision doesn't have an associated cross-reference section, it wasn't created from a PDF
|
46
|
+
# file.
|
47
|
+
#
|
48
|
+
# See: PDF1.7 s7.5.6, Revisions
|
49
|
+
class Revision
|
50
|
+
|
51
|
+
include Enumerable
|
52
|
+
|
53
|
+
# The trailer dictionary
|
54
|
+
attr_reader :trailer
|
55
|
+
|
56
|
+
# The callable object responsible for loading objects.
|
57
|
+
attr_accessor :loader
|
58
|
+
|
59
|
+
# :call-seq:
|
60
|
+
# Revision.new(trailer) -> revision
|
61
|
+
# Revision.new(trailer, xref_section: section, loader: loader) -> revision
|
62
|
+
# Revision.new(trailer, xref_section: section) {|entry| block } -> revision
|
63
|
+
#
|
64
|
+
# Creates a new Revision object.
|
65
|
+
#
|
66
|
+
# Options:
|
67
|
+
#
|
68
|
+
# xref_section::
|
69
|
+
# An XRefSection object that contains information on how to load objects. If this option is
|
70
|
+
# specified, then a +loader+ or a block also needs to be specified!
|
71
|
+
#
|
72
|
+
# loader::
|
73
|
+
# The loader object needs to respond to +call+ taking a cross-reference entry and returning
|
74
|
+
# the loaded object. If no +xref_section+ is supplied, this value is not used.
|
75
|
+
#
|
76
|
+
# If a block is given, it is used instead of the loader object.
|
77
|
+
def initialize(trailer, xref_section: nil, loader: nil, &block)
|
78
|
+
@trailer = trailer
|
79
|
+
@loader = xref_section && (block || loader)
|
80
|
+
@xref_section = xref_section || XRefSection.new
|
81
|
+
@objects = HexaPDF::Utils::ObjectHash.new
|
82
|
+
end
|
83
|
+
|
84
|
+
# Returns the next free object number for adding an object to this revision.
|
85
|
+
def next_free_oid
|
86
|
+
((a = @xref_section.max_oid) < (b = @objects.max_oid) ? b : a) + 1
|
87
|
+
end
|
88
|
+
|
89
|
+
# :call-seq:
|
90
|
+
# revision.object(ref) -> obj or nil
|
91
|
+
# revision.object(oid) -> obj or nil
|
92
|
+
#
|
93
|
+
# Returns the object for the given reference or object number if such an object is available
|
94
|
+
# in this revision, or +nil+ otherwise.
|
95
|
+
#
|
96
|
+
# If the revision has an entry but one that is pointing to a free entry in the cross-reference
|
97
|
+
# section, an object representing PDF null is returned.
|
98
|
+
def object(ref)
|
99
|
+
if ref.respond_to?(:oid)
|
100
|
+
oid = ref.oid
|
101
|
+
gen = ref.gen
|
102
|
+
else
|
103
|
+
oid = ref
|
104
|
+
end
|
105
|
+
|
106
|
+
if @objects.entry?(oid, gen)
|
107
|
+
@objects[oid, gen]
|
108
|
+
elsif (xref_entry = @xref_section[oid, gen])
|
109
|
+
load_object(xref_entry)
|
110
|
+
else
|
111
|
+
nil
|
112
|
+
end
|
113
|
+
end
|
114
|
+
|
115
|
+
# :call-seq:
|
116
|
+
# revision.object?(ref) -> true or false
|
117
|
+
# revision.object?(oid) -> true or false
|
118
|
+
#
|
119
|
+
# Returns +true+ if the revision contains an object
|
120
|
+
#
|
121
|
+
# * for the exact reference if the argument responds to :oid, or else
|
122
|
+
# * for the given object number.
|
123
|
+
def object?(ref)
|
124
|
+
if ref.respond_to?(:oid)
|
125
|
+
@objects.entry?(ref.oid, ref.gen) || @xref_section.entry?(ref.oid, ref.gen)
|
126
|
+
else
|
127
|
+
@objects.entry?(ref) || @xref_section.entry?(ref)
|
128
|
+
end
|
129
|
+
end
|
130
|
+
|
131
|
+
# :call-seq:
|
132
|
+
# revision.add(obj) -> obj
|
133
|
+
#
|
134
|
+
# Adds the given object (needs to be a HexaPDF::Object) to this revision and returns it.
|
135
|
+
def add(obj)
|
136
|
+
if object?(obj.oid)
|
137
|
+
raise HexaPDF::Error, "A revision can only contain one object with a given object number"
|
138
|
+
elsif !obj.indirect?
|
139
|
+
raise HexaPDF::Error, "A revision can only contain indirect objects"
|
140
|
+
end
|
141
|
+
add_without_check(obj)
|
142
|
+
end
|
143
|
+
|
144
|
+
# :call-seq:
|
145
|
+
# revision.delete(ref, mark_as_free: true)
|
146
|
+
# revision.delete(oid, mark_as_free: true)
|
147
|
+
#
|
148
|
+
# Deletes the object specified either by reference or by object number from this revision by
|
149
|
+
# marking it as free.
|
150
|
+
#
|
151
|
+
# If the +mark_as_free+ option is set to +false+, the object is really deleted.
|
152
|
+
def delete(ref_or_oid, mark_as_free: true)
|
153
|
+
return unless object?(ref_or_oid)
|
154
|
+
ref_or_oid = ref_or_oid.oid if ref_or_oid.respond_to?(:oid)
|
155
|
+
|
156
|
+
obj = object(ref_or_oid)
|
157
|
+
obj.data.value = nil
|
158
|
+
if mark_as_free
|
159
|
+
add_without_check(HexaPDF::Object.new(nil, oid: obj.oid, gen: obj.gen))
|
160
|
+
else
|
161
|
+
@xref_section.delete(ref_or_oid)
|
162
|
+
@objects.delete(ref_or_oid)
|
163
|
+
end
|
164
|
+
end
|
165
|
+
|
166
|
+
# :call-seq:
|
167
|
+
# revision.each {|obj| block } -> revision
|
168
|
+
# revision.each -> Enumerator
|
169
|
+
#
|
170
|
+
# Calls the given block once for every object of the revision.
|
171
|
+
#
|
172
|
+
# Objects that are loadable via an associated cross-reference section but are currently not,
|
173
|
+
# are loaded automatically.
|
174
|
+
def each
|
175
|
+
return to_enum(__method__) unless block_given?
|
176
|
+
|
177
|
+
if defined?(@all_objects_loaded)
|
178
|
+
@objects.each {|_oid, _gen, data| yield(data)}
|
179
|
+
else
|
180
|
+
seen = {}
|
181
|
+
@objects.each {|oid, _gen, data| seen[oid] = true; yield(data)}
|
182
|
+
@xref_section.each do |oid, _gen, data|
|
183
|
+
next if seen.key?(oid)
|
184
|
+
yield(@objects[oid] || load_object(data))
|
185
|
+
end
|
186
|
+
@all_objects_loaded = true
|
187
|
+
end
|
188
|
+
|
189
|
+
self
|
190
|
+
end
|
191
|
+
|
192
|
+
private
|
193
|
+
|
194
|
+
# Loads a single object from the associated cross-reference section.
|
195
|
+
def load_object(xref_entry)
|
196
|
+
add_without_check(@loader.call(xref_entry))
|
197
|
+
end
|
198
|
+
|
199
|
+
# Adds the object to the available objects of this revision and returns it.
|
200
|
+
def add_without_check(obj)
|
201
|
+
@objects[obj.oid, obj.gen] = obj
|
202
|
+
end
|
203
|
+
|
204
|
+
end
|
205
|
+
|
206
|
+
end
|
@@ -0,0 +1,194 @@
|
|
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/parser'
|
36
|
+
require 'hexapdf/revision'
|
37
|
+
require 'hexapdf/type/trailer'
|
38
|
+
|
39
|
+
module HexaPDF
|
40
|
+
|
41
|
+
# Manages the revisions of a PDF document.
|
42
|
+
#
|
43
|
+
# A PDF document has one revision when it is created. Later, new revisions are added when changes
|
44
|
+
# are made. This allows for adding information/content to a PDF file without changing the original
|
45
|
+
# content.
|
46
|
+
#
|
47
|
+
# The order of the revisions is important. In HexaPDF the oldest revision always has index 0 and
|
48
|
+
# the newest revision the highest index. This is also the order in which the revisions get
|
49
|
+
# written.
|
50
|
+
#
|
51
|
+
# See: PDF1.7 s7.5.6, HexaPDF::Revision
|
52
|
+
class Revisions
|
53
|
+
|
54
|
+
class << self
|
55
|
+
|
56
|
+
# Loads all revisions for the document from the given IO and returns the created Revisions
|
57
|
+
# object.
|
58
|
+
#
|
59
|
+
# If the +io+ object is +nil+, an empty Revisions object is returned.
|
60
|
+
def from_io(document, io)
|
61
|
+
return new(document) if io.nil?
|
62
|
+
|
63
|
+
parser = Parser.new(io, document)
|
64
|
+
object_loader = lambda {|xref_entry| parser.load_object(xref_entry)}
|
65
|
+
revision_loader = lambda do |offset|
|
66
|
+
xref_section, trailer = parser.load_revision(offset)
|
67
|
+
Revision.new(document.wrap(trailer, type: :XXTrailer), xref_section: xref_section,
|
68
|
+
loader: object_loader)
|
69
|
+
end
|
70
|
+
|
71
|
+
revisions = [revision_loader.call(parser.startxref_offset)]
|
72
|
+
|
73
|
+
i = revisions.length - 1
|
74
|
+
while i >= 0
|
75
|
+
# PDF1.7 s7.5.5 states that :Prev needs to be indirect, Adobe's reference 3.4.4 says it
|
76
|
+
# should be direct. Adobe's POV is followed here. Same with :XRefStm.
|
77
|
+
xrefstm = revisions[i].trailer.value[:XRefStm]
|
78
|
+
prev = revisions[i].trailer.value[:Prev]
|
79
|
+
new_revisions = [(revision_loader.call(prev) if prev),
|
80
|
+
(revision_loader.call(xrefstm) if xrefstm)].compact
|
81
|
+
revisions.insert(i, *new_revisions)
|
82
|
+
i += new_revisions.length - 1
|
83
|
+
end
|
84
|
+
|
85
|
+
document.version = parser.file_header_version
|
86
|
+
new(document, initial_revisions: revisions)
|
87
|
+
end
|
88
|
+
|
89
|
+
end
|
90
|
+
|
91
|
+
include Enumerable
|
92
|
+
|
93
|
+
# Creates a new revisions object for the given PDF document.
|
94
|
+
#
|
95
|
+
# Options:
|
96
|
+
#
|
97
|
+
# initial_revisions::
|
98
|
+
# An array of revisions that should initially be used. If this option is not specified, a
|
99
|
+
# single empty revision is added.
|
100
|
+
def initialize(document, initial_revisions: nil)
|
101
|
+
@document = document
|
102
|
+
@revisions = []
|
103
|
+
if initial_revisions
|
104
|
+
@revisions += initial_revisions
|
105
|
+
else
|
106
|
+
add
|
107
|
+
end
|
108
|
+
end
|
109
|
+
|
110
|
+
# Returns the revision at the specified index.
|
111
|
+
def revision(index)
|
112
|
+
@revisions[index]
|
113
|
+
end
|
114
|
+
alias :[] :revision
|
115
|
+
|
116
|
+
# Returns the current revision.
|
117
|
+
def current
|
118
|
+
@revisions.last
|
119
|
+
end
|
120
|
+
|
121
|
+
# Returns the number of HexaPDF::Revision objects managed by this object.
|
122
|
+
def size
|
123
|
+
@revisions.size
|
124
|
+
end
|
125
|
+
|
126
|
+
# Adds a new empty revision to the document and returns it.
|
127
|
+
def add
|
128
|
+
if @revisions.empty?
|
129
|
+
trailer = {}
|
130
|
+
else
|
131
|
+
trailer = current.trailer.value.dup
|
132
|
+
trailer.delete(:Prev)
|
133
|
+
trailer.delete(:XRefStm)
|
134
|
+
end
|
135
|
+
|
136
|
+
rev = Revision.new(@document.wrap(trailer, type: :XXTrailer))
|
137
|
+
@revisions.push(rev)
|
138
|
+
rev
|
139
|
+
end
|
140
|
+
|
141
|
+
# :call-seq:
|
142
|
+
# revisions.delete(index) -> rev or nil
|
143
|
+
# revisions.delete(oid) -> rev or nil
|
144
|
+
#
|
145
|
+
# Deletes a revision from the document, either by index or by specifying the revision object
|
146
|
+
# itself.
|
147
|
+
#
|
148
|
+
# Returns the deleted revision object, or +nil+ if the index was out of range or no matching
|
149
|
+
# revision was found.
|
150
|
+
#
|
151
|
+
# Regarding the index: The oldest revision has index 0 and the current revision the highest
|
152
|
+
# index!
|
153
|
+
def delete(index_or_rev)
|
154
|
+
if @revisions.length == 1
|
155
|
+
raise HexaPDF::Error, "A document must have a least one revision, can't delete last one"
|
156
|
+
elsif index_or_rev.kind_of?(Integer)
|
157
|
+
@revisions.delete_at(index_or_rev)
|
158
|
+
else
|
159
|
+
@revisions.delete(index_or_rev)
|
160
|
+
end
|
161
|
+
end
|
162
|
+
|
163
|
+
# :call-seq:
|
164
|
+
# revisions.merge(range = 0..-1) -> revisions
|
165
|
+
#
|
166
|
+
# Merges the revisions specified by the given range into one. Objects from newer revisions
|
167
|
+
# overwrite those from older ones.
|
168
|
+
def merge(range = 0..-1)
|
169
|
+
@revisions[range].reverse.each_cons(2) do |rev, prev_rev|
|
170
|
+
prev_rev.trailer.value.replace(rev.trailer.value)
|
171
|
+
rev.each do |obj|
|
172
|
+
prev_rev.delete(obj.oid, mark_as_free: false)
|
173
|
+
prev_rev.add(obj)
|
174
|
+
end
|
175
|
+
end
|
176
|
+
_first, *other = *@revisions[range]
|
177
|
+
other.each {|rev| @revisions.delete(rev)}
|
178
|
+
self
|
179
|
+
end
|
180
|
+
|
181
|
+
# :call-seq:
|
182
|
+
# revisions.each {|rev| block } -> revisions
|
183
|
+
# revisions.each -> Enumerator
|
184
|
+
#
|
185
|
+
# Iterates over all revisions from oldest to current one.
|
186
|
+
def each(&block)
|
187
|
+
return to_enum(__method__) unless block_given?
|
188
|
+
@revisions.each(&block)
|
189
|
+
self
|
190
|
+
end
|
191
|
+
|
192
|
+
end
|
193
|
+
|
194
|
+
end
|