hexapdf 0.28.0 → 0.31.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/CHANGELOG.md +86 -10
- data/examples/024-digital-signatures.rb +23 -0
- data/lib/hexapdf/cli/command.rb +16 -1
- data/lib/hexapdf/cli/info.rb +9 -1
- data/lib/hexapdf/cli/inspect.rb +2 -2
- data/lib/hexapdf/composer.rb +76 -28
- data/lib/hexapdf/configuration.rb +29 -16
- data/lib/hexapdf/dictionary_fields.rb +13 -4
- data/lib/hexapdf/digital_signature/cms_handler.rb +137 -0
- data/lib/hexapdf/digital_signature/handler.rb +138 -0
- data/lib/hexapdf/digital_signature/pkcs1_handler.rb +96 -0
- data/lib/hexapdf/{type → digital_signature}/signature.rb +3 -8
- data/lib/hexapdf/digital_signature/signatures.rb +210 -0
- data/lib/hexapdf/digital_signature/signing/default_handler.rb +317 -0
- data/lib/hexapdf/digital_signature/signing/signed_data_creator.rb +308 -0
- data/lib/hexapdf/digital_signature/signing/timestamp_handler.rb +148 -0
- data/lib/hexapdf/digital_signature/signing.rb +101 -0
- data/lib/hexapdf/{type/signature → digital_signature}/verification_result.rb +37 -41
- data/lib/hexapdf/digital_signature.rb +56 -0
- data/lib/hexapdf/document/pages.rb +31 -18
- data/lib/hexapdf/document.rb +29 -15
- data/lib/hexapdf/encryption/standard_security_handler.rb +4 -3
- data/lib/hexapdf/filter/flate_decode.rb +20 -8
- data/lib/hexapdf/layout/page_style.rb +144 -0
- data/lib/hexapdf/layout.rb +1 -0
- data/lib/hexapdf/task/optimize.rb +8 -6
- data/lib/hexapdf/type/font_simple.rb +14 -2
- data/lib/hexapdf/type/object_stream.rb +7 -2
- data/lib/hexapdf/type/outline.rb +1 -1
- data/lib/hexapdf/type/outline_item.rb +1 -1
- data/lib/hexapdf/type/page.rb +29 -8
- data/lib/hexapdf/type/xref_stream.rb +11 -4
- data/lib/hexapdf/type.rb +0 -1
- data/lib/hexapdf/version.rb +1 -1
- data/lib/hexapdf/writer.rb +1 -1
- data/test/hexapdf/{type/signature → digital_signature}/common.rb +31 -3
- data/test/hexapdf/digital_signature/signing/test_default_handler.rb +162 -0
- data/test/hexapdf/digital_signature/signing/test_signed_data_creator.rb +225 -0
- data/test/hexapdf/digital_signature/signing/test_timestamp_handler.rb +88 -0
- data/test/hexapdf/{type/signature/test_adbe_pkcs7_detached.rb → digital_signature/test_cms_handler.rb} +7 -7
- data/test/hexapdf/{type/signature → digital_signature}/test_handler.rb +4 -4
- data/test/hexapdf/{type/signature/test_adbe_x509_rsa_sha1.rb → digital_signature/test_pkcs1_handler.rb} +3 -3
- data/test/hexapdf/{type → digital_signature}/test_signature.rb +7 -7
- data/test/hexapdf/digital_signature/test_signatures.rb +137 -0
- data/test/hexapdf/digital_signature/test_signing.rb +53 -0
- data/test/hexapdf/{type/signature → digital_signature}/test_verification_result.rb +7 -7
- data/test/hexapdf/document/test_pages.rb +25 -0
- data/test/hexapdf/encryption/test_standard_security_handler.rb +2 -2
- data/test/hexapdf/filter/test_flate_decode.rb +19 -5
- data/test/hexapdf/layout/test_page_style.rb +70 -0
- data/test/hexapdf/task/test_optimize.rb +11 -9
- data/test/hexapdf/test_composer.rb +35 -10
- data/test/hexapdf/test_dictionary_fields.rb +9 -3
- data/test/hexapdf/test_document.rb +1 -1
- data/test/hexapdf/test_writer.rb +8 -8
- data/test/hexapdf/type/test_font_simple.rb +18 -6
- data/test/hexapdf/type/test_object_stream.rb +16 -7
- data/test/hexapdf/type/test_outline.rb +3 -1
- data/test/hexapdf/type/test_outline_item.rb +3 -1
- data/test/hexapdf/type/test_page.rb +42 -11
- data/test/hexapdf/type/test_xref_stream.rb +6 -1
- metadata +27 -15
- data/lib/hexapdf/document/signatures.rb +0 -546
- data/lib/hexapdf/type/signature/adbe_pkcs7_detached.rb +0 -135
- data/lib/hexapdf/type/signature/adbe_x509_rsa_sha1.rb +0 -95
- data/lib/hexapdf/type/signature/handler.rb +0 -140
- data/test/hexapdf/document/test_signatures.rb +0 -352
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: 7a32d8f7558ea14ae5dc40849c690b79f5ed4a797cffb85253e7fd6d00c54736
|
4
|
+
data.tar.gz: 92713168ee64efc59f86ff1d4f8989054ab02653ccaf84795aed3fea02a4811d
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 43b05643d9a59e9656a464eda221fa28e04bb4fe8ba1f97d0e31e434fbce64ffb9ed551689ce68957c8b4a7b82fbc85ecd91c1eac6d177b0109d4b9b5d275a6c
|
7
|
+
data.tar.gz: 780e7051327b0463e800ca4ea3b9f1ac1c3723d960154a2ac27c9e9bed867d0a257c48c20762cd0947f8638dc1f72b598a33ff76e1f0d1d2b0f1ff843497eb29
|
data/CHANGELOG.md
CHANGED
@@ -1,3 +1,79 @@
|
|
1
|
+
## 0.31.0 - 2023-02-22
|
2
|
+
|
3
|
+
### Added
|
4
|
+
|
5
|
+
* [HexaPDF::Layout::PageStyle] for collecting all styling information for pages
|
6
|
+
* [HexaPDF::Composer#page_style] for configuring different page styles
|
7
|
+
|
8
|
+
### Changed
|
9
|
+
|
10
|
+
* **Breaking change**: [HexaPDF::Composer] uses page styles underneath
|
11
|
+
* **Breaking change**: Configuration options `filter.flate_compression` and
|
12
|
+
`filter.flate_memory` are changed to `filter.flate.compression` and
|
13
|
+
`filter.flate.memory`
|
14
|
+
* **Breaking change**: [HexaPDF::Document#wrap] handles cross-reference and
|
15
|
+
object stream specially to avoid problems with invalid PDFs
|
16
|
+
* [HexaPDF::Composer::new] to allow skipping the initial page creation
|
17
|
+
* CLI command `hexapdf info --check` to process streams to reveal stream errors
|
18
|
+
* CLI commands to output the name of created PDF files in verbose mode
|
19
|
+
|
20
|
+
### Fixed
|
21
|
+
|
22
|
+
* Validation of document outline items in case the first or last item got
|
23
|
+
deleted
|
24
|
+
* `HexaPDF::Type::Page#perform_validation` to set a /MediaBox for invalid pages
|
25
|
+
that don't have one
|
26
|
+
|
27
|
+
|
28
|
+
## 0.30.0 - 2023-02-13
|
29
|
+
|
30
|
+
### Added
|
31
|
+
|
32
|
+
* [HexaPDF::Document::Pages#create] for creating a page object without adding it
|
33
|
+
to the page tree
|
34
|
+
|
35
|
+
### Changed
|
36
|
+
|
37
|
+
* `HexaPDF::Type::FontSimple#perform_validation` to correct /Widths fields in
|
38
|
+
case it has an invalid number of entries
|
39
|
+
|
40
|
+
### Fixed
|
41
|
+
|
42
|
+
* [HexaPDF::DictionaryFields::DateConverter] to handle invalid months, day,
|
43
|
+
hour, minute and second values
|
44
|
+
|
45
|
+
|
46
|
+
## 0.29.0 - 2023-01-30
|
47
|
+
|
48
|
+
### Added
|
49
|
+
|
50
|
+
* [HexaPDF::DigitalSignature::Signing::SignedDataCreator] for creating custom
|
51
|
+
CMS signed data objects
|
52
|
+
|
53
|
+
### Changed
|
54
|
+
|
55
|
+
* **Breaking change**: Refactored digital signature support and moved all
|
56
|
+
related code under the [HexaPDF::DigitalSignature] module
|
57
|
+
* **Breaking change**: New external signing mode without the need for creating
|
58
|
+
the PKCS#7/CMS signed data object for
|
59
|
+
[HexaPDF::DigitalSignature::Signing::DefaultHandler]
|
60
|
+
* **Breaking change**: Use value :pades instead of :etsi for
|
61
|
+
[HexaPDF::DigitalSignature::Signing::DefaultHandler#signature_type]
|
62
|
+
* [HexaPDF::DigitalSignature::Signing::DefaultHandler] to allow creating PAdES
|
63
|
+
level B-B and B-T signatures
|
64
|
+
* [HexaPDF::DigitalSignature::Signing::DefaultHandler] to allow specifying the
|
65
|
+
used digest algorithm
|
66
|
+
* [HexaPDF::DigitalSignature::Signing::DefaultHandler] to allow specifying a
|
67
|
+
timestamp handler for including a timestamp token in the signature
|
68
|
+
* Moved setting of signature entries /Filter, /SubFilter and /M fields to the
|
69
|
+
signing handlers
|
70
|
+
|
71
|
+
### Fixed
|
72
|
+
|
73
|
+
* [HexaPDF::DictionaryFields::DateConverter] to handle invalid timezone hour and
|
74
|
+
minute values
|
75
|
+
|
76
|
+
|
1
77
|
## 0.28.0 - 2022-12-30
|
2
78
|
|
3
79
|
### Added
|
@@ -61,30 +137,30 @@
|
|
61
137
|
### Added
|
62
138
|
|
63
139
|
* Support for timestamp signatures through the
|
64
|
-
|
140
|
+
`HexaPDF::Document::Signatures::TimestampHandler`
|
65
141
|
* [HexaPDF::Document::Destinations#resolve] for resolving destination values
|
66
142
|
* [HexaPDF::Document::Destinations::Destination#value] to return the destination
|
67
143
|
array
|
68
144
|
* Support for verifying document timestamp signatures
|
69
|
-
*
|
145
|
+
* `HexaPDF::Document::Signatures::DefaultHandler#signature_size` to support
|
70
146
|
setting custom signature sizes
|
71
|
-
*
|
147
|
+
* `HexaPDF::Document::Signatures::DefaultHandler#external_signing` to support
|
72
148
|
signing via custom mechanisms
|
73
|
-
*
|
149
|
+
* `HexaPDF::Document::Signatures::embed_signature` to enable asynchronous
|
74
150
|
external signing
|
75
151
|
|
76
152
|
### Changed
|
77
153
|
|
78
154
|
* **Breaking change**: The crop box is now used instead of the media box in most
|
79
155
|
cases to be in line with the specification
|
80
|
-
*
|
156
|
+
* `HexaPDF::Document::Signatures::DefaultHandler` to allow setting the used
|
81
157
|
signature method
|
82
|
-
* **Breaking change**:
|
158
|
+
* **Breaking change**: `HexaPDF::Document::Signatures::DefaultHandler#sign`
|
83
159
|
needs to accept the IO object and the byte range instead of just the data
|
84
160
|
* **Breaking change**: Enhanced support for outline items with new methods
|
85
161
|
`#level` and `#destination_page` as well as changes to `#add` and `#each_item`
|
86
162
|
* **Breaking change**: Removed `#filter_name` and `#sub_filter_name` from
|
87
|
-
|
163
|
+
`HexaPDF::Document::Signatures::DefaultHandler`
|
88
164
|
* `HexaPDF::Type::Resources#perform_validation` to not add a default procedure
|
89
165
|
set since this feature is deprecated
|
90
166
|
|
@@ -101,7 +177,7 @@
|
|
101
177
|
* [HexaPDF::Type::OutlineItem] to always be an indirect object
|
102
178
|
* `HexaPDF::Tokenizer#parse_number` to handle references correctly in all cases
|
103
179
|
* [HexaPDF::Type::Page#rotate] to correctly flatten all page boxes
|
104
|
-
*
|
180
|
+
* `HexaPDF::Document::Signatures#add` to raise an error if the reserved space
|
105
181
|
for the signature is not enough
|
106
182
|
* `HexaPDF::Type::AcroForm::Form#perform_validation` to fix broken /Parent
|
107
183
|
entries and to remove invalid objects from the field hierarchy
|
@@ -276,7 +352,7 @@
|
|
276
352
|
moved node doesn't change
|
277
353
|
* [HexaPDF::Type::PageTreeNode#move_page] to use the correct target position
|
278
354
|
when the moved node is before the target position
|
279
|
-
*
|
355
|
+
* `HexaPDF::Document::Signatures#add` to work in case the signature object is
|
280
356
|
the last object written
|
281
357
|
* CLI command `hexapdf inspect` to show correct byte range of the last revision
|
282
358
|
* [HexaPDF::Writer#write_incremental] to only use a cross-reference stream if a
|
@@ -285,7 +361,7 @@
|
|
285
361
|
disabled
|
286
362
|
* [HexaPDF::Font::Encoding::GlyphList] to use binary reading to avoid problems
|
287
363
|
on Windows
|
288
|
-
*
|
364
|
+
* `HexaPDF::Document::Signatures#add` to use binary writing to avoid problems on
|
289
365
|
Windows
|
290
366
|
|
291
367
|
|
@@ -0,0 +1,23 @@
|
|
1
|
+
# # Images
|
2
|
+
#
|
3
|
+
# This example shows how to embed images into a PDF document, directly on a
|
4
|
+
# page's canvas and through the high-level [HexaPDF::Composer].
|
5
|
+
#
|
6
|
+
# Usage:
|
7
|
+
# : `ruby digital-signatures.rb`
|
8
|
+
#
|
9
|
+
|
10
|
+
require 'hexapdf'
|
11
|
+
require HexaPDF.data_dir + '/cert/demo_cert.rb'
|
12
|
+
|
13
|
+
doc = if ARGV[0]
|
14
|
+
HexaPDF::Document.open(ARGV[0])
|
15
|
+
else
|
16
|
+
HexaPDF::Document.new.pages.add.document
|
17
|
+
end
|
18
|
+
doc.sign("digital-signatures.pdf",
|
19
|
+
reason: 'Some reason',
|
20
|
+
certificate: HexaPDF.demo_cert.cert,
|
21
|
+
key: HexaPDF.demo_cert.key,
|
22
|
+
certificate_chain: [HexaPDF.demo_cert.sub_ca,
|
23
|
+
HexaPDF.demo_cert.root_ca])
|
data/lib/hexapdf/cli/command.rb
CHANGED
@@ -101,6 +101,17 @@ module HexaPDF
|
|
101
101
|
def pdf_options(password)
|
102
102
|
hash = {decryption_opts: {password: password}, config: {}}
|
103
103
|
HexaPDF::GlobalConfiguration['filter.predictor.strict'] = command_parser.strict
|
104
|
+
HexaPDF::GlobalConfiguration['filter.flate.on_error'] =
|
105
|
+
if command_parser.strict
|
106
|
+
proc { true }
|
107
|
+
else
|
108
|
+
proc do |_, error|
|
109
|
+
if command_parser.verbosity_info?
|
110
|
+
$stderr.puts "Ignoring error in flate encoded stream: #{error}"
|
111
|
+
end
|
112
|
+
false
|
113
|
+
end
|
114
|
+
end
|
104
115
|
hash[:config]['parser.try_xref_reconstruction'] = !command_parser.strict
|
105
116
|
hash[:config]['parser.on_correctable_error'] =
|
106
117
|
if command_parser.strict
|
@@ -120,6 +131,7 @@ module HexaPDF
|
|
120
131
|
# Writes the document to the given file or does nothing if +out_file+ is +nil+.
|
121
132
|
def write_document(doc, out_file, incremental: false)
|
122
133
|
if out_file
|
134
|
+
doc.trailer.update_id
|
123
135
|
doc.validate(auto_correct: true) do |msg, correctable, object|
|
124
136
|
if command_parser.strict && !correctable
|
125
137
|
raise "Validation error for object (#{object.oid},#{object.gen}): #{msg}"
|
@@ -128,6 +140,9 @@ module HexaPDF
|
|
128
140
|
"for object (#{object.oid},#{object.gen}): #{msg}"
|
129
141
|
end
|
130
142
|
end
|
143
|
+
if command_parser.verbosity_info?
|
144
|
+
puts "Creating output document #{out_file}"
|
145
|
+
end
|
131
146
|
doc.write(out_file, validate: false, incremental: incremental)
|
132
147
|
end
|
133
148
|
end
|
@@ -331,7 +346,7 @@ module HexaPDF
|
|
331
346
|
rotation = ROTATE_MAP[$4]
|
332
347
|
start_nr.step(to: end_nr, by: step) {|n| arr << [n, rotation] }
|
333
348
|
else
|
334
|
-
raise OptionParser::InvalidArgument, "invalid page range format: #{str}"
|
349
|
+
raise OptionParser::InvalidArgument, "invalid page range format: #{str.inspect}"
|
335
350
|
end
|
336
351
|
end
|
337
352
|
end
|
data/lib/hexapdf/cli/info.rb
CHANGED
@@ -106,6 +106,13 @@ module HexaPDF
|
|
106
106
|
doc.each(only_loaded: false) do |obj|
|
107
107
|
indirect_object = obj
|
108
108
|
obj.validate(auto_correct: true, &validation_block)
|
109
|
+
if obj.data.stream
|
110
|
+
begin
|
111
|
+
obj.stream
|
112
|
+
rescue StandardError
|
113
|
+
puts "ERROR: Stream of object (#{obj.oid},#{obj.gen}) invalid: #{$!.message}"
|
114
|
+
end
|
115
|
+
end
|
109
116
|
end
|
110
117
|
end
|
111
118
|
|
@@ -177,7 +184,8 @@ module HexaPDF
|
|
177
184
|
def pdf_options(password)
|
178
185
|
if @check_file
|
179
186
|
options = {decryption_opts: {password: password}, config: {}}
|
180
|
-
HexaPDF::GlobalConfiguration['filter.predictor.strict'] =
|
187
|
+
HexaPDF::GlobalConfiguration['filter.predictor.strict'] = true
|
188
|
+
HexaPDF::GlobalConfiguration['filter.flate.on_error'] = proc { true }
|
181
189
|
options[:config]['parser.try_xref_reconstruction'] = true
|
182
190
|
options[:config]['parser.on_correctable_error'] = lambda do |_, msg, pos|
|
183
191
|
puts "WARNING: Parse error at position #{pos}: #{msg}"
|
data/lib/hexapdf/cli/inspect.rb
CHANGED
@@ -166,8 +166,8 @@ module HexaPDF
|
|
166
166
|
when 'p', 'pages'
|
167
167
|
begin
|
168
168
|
pages = parse_pages_specification(data.shift || '1-e', @doc.pages.count)
|
169
|
-
rescue StandardError
|
170
|
-
$stderr.puts("Error:
|
169
|
+
rescue StandardError => e
|
170
|
+
$stderr.puts("Error: #{e}")
|
171
171
|
next
|
172
172
|
end
|
173
173
|
page_list = @doc.pages.to_a
|
data/lib/hexapdf/composer.rb
CHANGED
@@ -106,6 +106,14 @@ module HexaPDF
|
|
106
106
|
|
107
107
|
# Creates a new Composer object and optionally yields it to the given block.
|
108
108
|
#
|
109
|
+
# skip_page_creation::
|
110
|
+
# If this argument is +true+ (the default), the arguments +page_size+, +page_orientation+
|
111
|
+
# and +margin+ are used to create a page style with the name :default and an initial page is
|
112
|
+
# created as well.
|
113
|
+
#
|
114
|
+
# Otherwise, i.e. when this argument is +false+, no initial page or default page style is
|
115
|
+
# created. This has to be done manually using the #page_style and #new_page methods.
|
116
|
+
#
|
109
117
|
# page_size::
|
110
118
|
# Can be any valid predefined page size (see Type::Page::PAPER_SIZE) or an array [llx, lly,
|
111
119
|
# urx, ury] specifying a custom page size.
|
@@ -120,36 +128,53 @@ module HexaPDF
|
|
120
128
|
# Example:
|
121
129
|
#
|
122
130
|
# composer = HexaPDF::Composer.new # uses the default values
|
131
|
+
#
|
123
132
|
# HexaPDF::Composer.new(page_size: :Letter, margin: 72) do |composer|
|
124
133
|
# #...
|
125
134
|
# end
|
126
|
-
|
135
|
+
#
|
136
|
+
# HexaPDF::Composer.new(skip_page_creation: true) do |composer|
|
137
|
+
# page_template = lambda {|canvas, style| style.create_frame(canvas.context, 36) }
|
138
|
+
# page_style(:default, template: page_template)
|
139
|
+
# new_page
|
140
|
+
# # ...
|
141
|
+
# end
|
142
|
+
def initialize(skip_page_creation: false, page_size: :A4, page_orientation: :portrait,
|
143
|
+
margin: 36) #:yields: composer
|
127
144
|
@document = HexaPDF::Document.new
|
128
|
-
@
|
129
|
-
@
|
130
|
-
|
131
|
-
|
132
|
-
|
145
|
+
@page_styles = {}
|
146
|
+
@page_style = :default
|
147
|
+
unless skip_page_creation
|
148
|
+
page_style(:default, page_size: page_size, orientation: page_orientation) do |canvas, style|
|
149
|
+
style.frame = style.create_frame(canvas.context, margin)
|
150
|
+
end
|
151
|
+
new_page
|
152
|
+
end
|
133
153
|
yield(self) if block_given?
|
134
154
|
end
|
135
155
|
|
136
156
|
# Creates a new page, making it the current one.
|
137
157
|
#
|
138
|
-
#
|
139
|
-
# the
|
158
|
+
# The page style to use for the new page can be set via the +style+ argument. If not provided,
|
159
|
+
# the currently set page style is used.
|
160
|
+
#
|
161
|
+
# The used page style determines the page style that should be used for the following new pages.
|
162
|
+
# If this information is not provided, the used page style is used again.
|
140
163
|
#
|
141
164
|
# Examples:
|
142
165
|
#
|
143
|
-
# composer.
|
144
|
-
# composer.
|
145
|
-
|
146
|
-
|
147
|
-
|
148
|
-
|
149
|
-
|
150
|
-
|
166
|
+
# composer.page_style(:cover, page_size: :A4).next_style = :content
|
167
|
+
# composer.page_style(:content, page_size: :A4)
|
168
|
+
# composer.new_page(:cover) # uses the :cover style, set next style to :content
|
169
|
+
# composer.new_page # uses the :content style, next style again :content
|
170
|
+
def new_page(style = @page_style)
|
171
|
+
page_style = @page_styles.fetch(style) do |key|
|
172
|
+
raise ArgumentError, "Page style #{key} has not been defined"
|
173
|
+
end
|
174
|
+
@page = @document.pages.add(page_style.create_page(@document))
|
151
175
|
@canvas = @page.canvas
|
152
|
-
|
176
|
+
@frame = page_style.frame
|
177
|
+
@page_style = page_style.next_style || style
|
153
178
|
end
|
154
179
|
|
155
180
|
# The x-position of the cursor inside the current frame.
|
@@ -190,6 +215,40 @@ module HexaPDF
|
|
190
215
|
@document.layout.style(name, base: base, **properties)
|
191
216
|
end
|
192
217
|
|
218
|
+
# :call-seq:
|
219
|
+
# composer.page_style(name) -> page_style
|
220
|
+
# composer.page_style(name, **attributes, &template_block) -> page_style
|
221
|
+
#
|
222
|
+
# Creates and/or returns the page style +name+.
|
223
|
+
#
|
224
|
+
# If no attributes are given, the page style +name+ is returned. In case it does not exist,
|
225
|
+
# +nil+ is returned.
|
226
|
+
#
|
227
|
+
# If one or more page style attributes are given, a new HexaPDF::Layout::PageStyle object with
|
228
|
+
# those attribute values is created, stored under +name+ and returned. If a block is provided,
|
229
|
+
# it is used to define the page template.
|
230
|
+
#
|
231
|
+
# Example:
|
232
|
+
#
|
233
|
+
# composer.page_style(:default)
|
234
|
+
# composer.page_style(:cover, page_size: :A4) do |canvas, style|
|
235
|
+
# page_box = canvas.context.box
|
236
|
+
# canvas.fill_color("fd0") do
|
237
|
+
# canvas.rectangle(0, 0, page_box.width, page_box.height).
|
238
|
+
# fill
|
239
|
+
# end
|
240
|
+
# style.frame = style.create_frame(canvas.context, 36)
|
241
|
+
# end
|
242
|
+
#
|
243
|
+
# See: HexaPDF::Layout::PageStyle
|
244
|
+
def page_style(name, **attributes, &block)
|
245
|
+
if attributes.empty?
|
246
|
+
@page_styles[name]
|
247
|
+
else
|
248
|
+
@page_styles[name] = HexaPDF::Layout::PageStyle.new(**attributes, &block)
|
249
|
+
end
|
250
|
+
end
|
251
|
+
|
193
252
|
# Draws the given text at the current position into the current frame.
|
194
253
|
#
|
195
254
|
# The text will be positioned at the current position if possible. Otherwise the next best
|
@@ -325,17 +384,6 @@ module HexaPDF
|
|
325
384
|
stamp
|
326
385
|
end
|
327
386
|
|
328
|
-
private
|
329
|
-
|
330
|
-
# Creates the frame into which boxes are layed out when a new page is created.
|
331
|
-
def create_frame
|
332
|
-
media_box = @page.box
|
333
|
-
@frame = Layout::Frame.new(media_box.left + @margin.left,
|
334
|
-
media_box.bottom + @margin.bottom,
|
335
|
-
media_box.width - @margin.left - @margin.right,
|
336
|
-
media_box.height - @margin.bottom - @margin.top)
|
337
|
-
end
|
338
|
-
|
339
387
|
end
|
340
388
|
|
341
389
|
end
|
@@ -350,6 +350,9 @@ module HexaPDF
|
|
350
350
|
# The media box that is used for new pages that don't define a media box. Default value is
|
351
351
|
# A4. See HexaPDF::Type::Page::PAPER_SIZE for a list of predefined paper sizes.
|
352
352
|
#
|
353
|
+
# This configuration option (together with 'page.default_media_orientation') is also used when
|
354
|
+
# validating pages and a page without a media box is found.
|
355
|
+
#
|
353
356
|
# The value can either be a rectangle defining the paper size or a Symbol referencing one of
|
354
357
|
# the predefined paper sizes.
|
355
358
|
#
|
@@ -393,8 +396,8 @@ module HexaPDF
|
|
393
396
|
#
|
394
397
|
# signature.sub_filter_map::
|
395
398
|
# A mapping from a PDF name (a Symbol) to a signature handler class (see
|
396
|
-
# HexaPDF::
|
397
|
-
# constant to such a class.
|
399
|
+
# HexaPDF::DigitalSignature::Handler). If the value is a String, it should contain the name of
|
400
|
+
# a constant to such a class.
|
398
401
|
#
|
399
402
|
# The sub filter map is used for mapping specific signature algorithms to handler classes. The
|
400
403
|
# filter value of a signature dictionary is ignored since we only support the standard
|
@@ -486,14 +489,14 @@ module HexaPDF
|
|
486
489
|
link: 'HexaPDF::Layout::Style::LinkLayer',
|
487
490
|
},
|
488
491
|
'signature.signing_handler' => {
|
489
|
-
default: 'HexaPDF::
|
490
|
-
timestamp: 'HexaPDF::
|
492
|
+
default: 'HexaPDF::DigitalSignature::Signing::DefaultHandler',
|
493
|
+
timestamp: 'HexaPDF::DigitalSignature::Signing::TimestampHandler',
|
491
494
|
},
|
492
495
|
'signature.sub_filter_map' => {
|
493
|
-
'adbe.x509.rsa_sha1': 'HexaPDF::
|
494
|
-
'adbe.pkcs7.detached': 'HexaPDF::
|
495
|
-
'ETSI.CAdES.detached': 'HexaPDF::
|
496
|
-
'ETSI.RFC3161': 'HexaPDF::
|
496
|
+
'adbe.x509.rsa_sha1': 'HexaPDF::DigitalSignature::PKCS1Handler',
|
497
|
+
'adbe.pkcs7.detached': 'HexaPDF::DigitalSignature::CMSHandler',
|
498
|
+
'ETSI.CAdES.detached': 'HexaPDF::DigitalSignature::CMSHandler',
|
499
|
+
'ETSI.RFC3161': 'HexaPDF::DigitalSignature::CMSHandler',
|
497
500
|
},
|
498
501
|
'task.map' => {
|
499
502
|
optimize: 'HexaPDF::Task::Optimize',
|
@@ -511,11 +514,20 @@ module HexaPDF
|
|
511
514
|
#
|
512
515
|
# See PDF1.7 s8.6
|
513
516
|
#
|
514
|
-
# filter.
|
517
|
+
# filter.flate.compression::
|
515
518
|
# Specifies the compression level that should be used with the FlateDecode filter. The level
|
516
519
|
# can range from 0 (no compression), 1 (best speed) to 9 (best compression, default).
|
517
520
|
#
|
518
|
-
# filter.
|
521
|
+
# filter.flate.on_error::
|
522
|
+
# Callback hook when a potentially recoverable Zlib error occurs in the FlateDecode filter.
|
523
|
+
#
|
524
|
+
# The value needs to be an object that responds to \#call(stream, error) where stream is the
|
525
|
+
# Zlib stream object and error is the thrown error. The method needs to return +true+ if an
|
526
|
+
# error should be raised.
|
527
|
+
#
|
528
|
+
# The default implementation prevents errors from being raised.
|
529
|
+
#
|
530
|
+
# filter.flate.memory::
|
519
531
|
# Specifies the memory level that should be used with the FlateDecode filter. The level can
|
520
532
|
# range from 1 (minimum memory usage; slow, reduces compression) to 9 (maximum memory usage).
|
521
533
|
#
|
@@ -543,8 +555,9 @@ module HexaPDF
|
|
543
555
|
# This mapping is used to provide automatic wrapping of objects in the HexaPDF::Document#wrap
|
544
556
|
# method.
|
545
557
|
GlobalConfiguration =
|
546
|
-
Configuration.new('filter.
|
547
|
-
'filter.
|
558
|
+
Configuration.new('filter.flate.compression' => 9,
|
559
|
+
'filter.flate.on_error' => proc { false },
|
560
|
+
'filter.flate.memory' => 6,
|
548
561
|
'filter.predictor.strict' => false,
|
549
562
|
'color_space.map' => {
|
550
563
|
DeviceRGB: 'HexaPDF::Content::ColorSpace::DeviceRGB',
|
@@ -583,10 +596,10 @@ module HexaPDF
|
|
583
596
|
SigFieldLock: 'HexaPDF::Type::AcroForm::SignatureField::LockDictionary',
|
584
597
|
SV: 'HexaPDF::Type::AcroForm::SignatureField::SeedValueDictionary',
|
585
598
|
SVCert: 'HexaPDF::Type::AcroForm::SignatureField::CertificateSeedValueDictionary',
|
586
|
-
Sig: 'HexaPDF::
|
587
|
-
DocTimeStamp: 'HexaPDF::
|
588
|
-
SigRef: 'HexaPDF::
|
589
|
-
TransformParams: 'HexaPDF::
|
599
|
+
Sig: 'HexaPDF::DigitalSignature::Signature',
|
600
|
+
DocTimeStamp: 'HexaPDF::DigitalSignature::Signature',
|
601
|
+
SigRef: 'HexaPDF::DigitalSignature::Signature::SignatureReference',
|
602
|
+
TransformParams: 'HexaPDF::DigitalSignature::Signature::TransformParams',
|
590
603
|
Outlines: 'HexaPDF::Type::Outline',
|
591
604
|
XXOutlineItem: 'HexaPDF::Type::OutlineItem',
|
592
605
|
PageLabel: 'HexaPDF::Type::PageLabel',
|
@@ -293,16 +293,25 @@ module HexaPDF
|
|
293
293
|
end
|
294
294
|
|
295
295
|
# :nodoc:
|
296
|
-
DATE_RE = /\AD:(\d{4})(\d\d)?(\d\d)?(\d\d)?(\d\d)?(\d\d)?([Z+-])?(?:(\d
|
296
|
+
DATE_RE = /\AD:(\d{4})(\d\d)?(\d\d)?(\d\d)?(\d\d)?(\d\d)?([Z+-])?(?:(\d+)(?:'|'(\d+)'?|\z)?)?\z/n
|
297
297
|
|
298
298
|
# Checks if the given object is a string and converts into a Time object if possible.
|
299
299
|
# Otherwise returns +nil+.
|
300
300
|
def self.convert(str, _type, _document)
|
301
301
|
return unless str.kind_of?(String) && (m = str.match(DATE_RE))
|
302
302
|
|
303
|
-
utc_offset =
|
304
|
-
|
305
|
-
|
303
|
+
utc_offset = if m[7].nil? || m[7] == 'Z'
|
304
|
+
0
|
305
|
+
else
|
306
|
+
(m[7] == '-' ? -1 : 1) * (m[8].to_i * 3600 + m[9].to_i * 60).clamp(0, 86399)
|
307
|
+
end
|
308
|
+
begin
|
309
|
+
Time.new(m[1].to_i, (m[2] ? m[2].to_i : 1), (m[3] ? m[3].to_i : 1),
|
310
|
+
m[4].to_i, m[5].to_i, m[6].to_i, utc_offset)
|
311
|
+
rescue ArgumentError
|
312
|
+
Time.new(m[1].to_i, m[2].to_i.clamp(1, 12), m[3].to_i.clamp(1, 31),
|
313
|
+
m[4].to_i.clamp(0, 23), m[5].to_i.clamp(0, 59), m[6].to_i.clamp(0, 59), utc_offset)
|
314
|
+
end
|
306
315
|
end
|
307
316
|
|
308
317
|
end
|
@@ -0,0 +1,137 @@
|
|
1
|
+
# -*- encoding: utf-8; frozen_string_literal: true -*-
|
2
|
+
#
|
3
|
+
#--
|
4
|
+
# This file is part of HexaPDF.
|
5
|
+
#
|
6
|
+
# HexaPDF - A Versatile PDF Creation and Manipulation Library For Ruby
|
7
|
+
# Copyright (C) 2014-2022 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
|
+
# If the GNU Affero General Public License doesn't fit your need,
|
34
|
+
# commercial licenses are available at <https://gettalong.at/hexapdf/>.
|
35
|
+
#++
|
36
|
+
|
37
|
+
require 'openssl'
|
38
|
+
require 'hexapdf/digital_signature/handler'
|
39
|
+
|
40
|
+
module HexaPDF
|
41
|
+
module DigitalSignature
|
42
|
+
|
43
|
+
# The signature handler for PKCS#7 a.k.a. CMS signatures. Those include, for example, the
|
44
|
+
# adbe.pkcs7.detached sub-filter.
|
45
|
+
#
|
46
|
+
# See: PDF1.7/2.0 s12.8.3.3
|
47
|
+
class CMSHandler < Handler
|
48
|
+
|
49
|
+
# Creates a new signature handler for the given signature dictionary.
|
50
|
+
def initialize(signature_dict)
|
51
|
+
super
|
52
|
+
@pkcs7 = OpenSSL::PKCS7.new(signature_dict.contents)
|
53
|
+
end
|
54
|
+
|
55
|
+
# Returns the common name of the signer.
|
56
|
+
def signer_name
|
57
|
+
signer_certificate.subject.to_a.assoc("CN")&.[](1) || super
|
58
|
+
end
|
59
|
+
|
60
|
+
# Returns the time of signing.
|
61
|
+
def signing_time
|
62
|
+
signer_info.signed_time rescue super
|
63
|
+
end
|
64
|
+
|
65
|
+
# Returns the certificate chain.
|
66
|
+
def certificate_chain
|
67
|
+
@pkcs7.certificates
|
68
|
+
end
|
69
|
+
|
70
|
+
# Returns the signer certificate (an instance of OpenSSL::X509::Certificate).
|
71
|
+
def signer_certificate
|
72
|
+
info = signer_info
|
73
|
+
certificate_chain.find {|cert| cert.issuer == info.issuer && cert.serial == info.serial }
|
74
|
+
end
|
75
|
+
|
76
|
+
# Returns the signer information object (an instance of OpenSSL::PKCS7::SignerInfo).
|
77
|
+
def signer_info
|
78
|
+
@pkcs7.signers.first
|
79
|
+
end
|
80
|
+
|
81
|
+
# Verifies the signature using the provided OpenSSL::X509::Store object.
|
82
|
+
def verify(store, allow_self_signed: false)
|
83
|
+
result = super
|
84
|
+
|
85
|
+
signer_info = self.signer_info
|
86
|
+
signer_certificate = self.signer_certificate
|
87
|
+
certificate_chain = self.certificate_chain
|
88
|
+
|
89
|
+
if certificate_chain.empty?
|
90
|
+
result.log(:error, "No certificates found in signature")
|
91
|
+
return result
|
92
|
+
end
|
93
|
+
|
94
|
+
if @pkcs7.signers.size != 1
|
95
|
+
result.log(:error, "Exactly one signer needed, found #{@pkcs7.signers.size}")
|
96
|
+
end
|
97
|
+
|
98
|
+
unless signer_certificate
|
99
|
+
result.log(:error, "Signer serial=#{signer_info.serial} issuer=#{signer_info.issuer} " \
|
100
|
+
"not found in certificates stored in PKCS7 object")
|
101
|
+
return result
|
102
|
+
end
|
103
|
+
|
104
|
+
key_usage = signer_certificate.extensions.find {|ext| ext.oid == 'keyUsage' }
|
105
|
+
unless key_usage && key_usage.value.split(', ').include?("Digital Signature")
|
106
|
+
result.log(:error, "Certificate key usage is missing 'Digital Signature'")
|
107
|
+
end
|
108
|
+
|
109
|
+
if signature_dict.signature_type == 'ETSI.RFC3161'
|
110
|
+
# Getting the needed values is not directly supported by Ruby OpenSSL
|
111
|
+
p7 = OpenSSL::ASN1.decode(signature_dict.contents.sub(/\x00*\z/, ''))
|
112
|
+
signed_data = p7.value[1].value[0]
|
113
|
+
content_info = signed_data.value[2]
|
114
|
+
content = OpenSSL::ASN1.decode(content_info.value[1].value[0].value)
|
115
|
+
digest_algorithm = content.value[2].value[0].value[0].value
|
116
|
+
original_hash = content.value[2].value[1].value
|
117
|
+
recomputed_hash = OpenSSL::Digest.digest(digest_algorithm, signature_dict.signed_data)
|
118
|
+
hash_valid = (original_hash == recomputed_hash)
|
119
|
+
else
|
120
|
+
data = signature_dict.signed_data
|
121
|
+
hash_valid = true # hash will be checked by @pkcs7.verify
|
122
|
+
end
|
123
|
+
if hash_valid && @pkcs7.verify(certificate_chain, store, data,
|
124
|
+
OpenSSL::PKCS7::DETACHED | OpenSSL::PKCS7::BINARY)
|
125
|
+
result.log(:info, "Signature valid")
|
126
|
+
else
|
127
|
+
result.log(:error, "Signature verification failed")
|
128
|
+
end
|
129
|
+
|
130
|
+
result
|
131
|
+
end
|
132
|
+
|
133
|
+
end
|
134
|
+
|
135
|
+
end
|
136
|
+
end
|
137
|
+
|