rack 3.0.18 → 3.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.

Potentially problematic release.


This version of rack might be problematic. Click here for more details.

data/lib/rack/response.rb CHANGED
@@ -25,19 +25,11 @@ module Rack
25
25
  self.new(body, status, headers)
26
26
  end
27
27
 
28
- CHUNKED = 'chunked'
29
28
  STATUS_WITH_NO_ENTITY_BODY = Utils::STATUS_WITH_NO_ENTITY_BODY
30
29
 
31
30
  attr_accessor :length, :status, :body
32
31
  attr_reader :headers
33
32
 
34
- # Deprecated, use headers instead.
35
- def header
36
- warn 'Rack::Response#header is deprecated and will be removed in Rack 3.1, use #headers instead', uplevel: 1
37
-
38
- headers
39
- end
40
-
41
33
  # Initialize the response object with the specified +body+, +status+
42
34
  # and +headers+.
43
35
  #
@@ -62,7 +54,7 @@ module Rack
62
54
  @status = status.to_i
63
55
 
64
56
  unless headers.is_a?(Hash)
65
- warn "Providing non-hash headers to Rack::Response is deprecated and will be removed in Rack 3.1", uplevel: 1
57
+ raise ArgumentError, "Headers must be a Hash!"
66
58
  end
67
59
 
68
60
  @headers = Headers.new
@@ -97,16 +89,12 @@ module Rack
97
89
  self.status = status
98
90
  self.location = target
99
91
  end
100
-
101
- def chunked?
102
- CHUNKED == get_header(TRANSFER_ENCODING)
103
- end
104
-
92
+
105
93
  def no_entity_body?
106
94
  # The response body is an enumerable body and it is not allowed to have an entity body.
107
95
  @body.respond_to?(:each) && STATUS_WITH_NO_ENTITY_BODY[@status]
108
96
  end
109
-
97
+
110
98
  # Generate a response array consistent with the requirements of the SPEC.
111
99
  # @return [Array] a 3-tuple suitable of `[status, headers, body]`
112
100
  # which is suitable to be returned from the middleware `#call(env)` method.
@@ -117,6 +105,10 @@ module Rack
117
105
  close
118
106
  return [@status, @headers, []]
119
107
  else
108
+ if @length && @length > 0
109
+ set_header CONTENT_LENGTH, @length.to_s
110
+ end
111
+
120
112
  if block_given?
121
113
  @block = block
122
114
  return [@status, @headers, self]
@@ -320,20 +312,26 @@ module Rack
320
312
 
321
313
  protected
322
314
 
315
+ # Convert the body of this response into an internally buffered Array if possible.
316
+ #
317
+ # `@buffered` is a ternary value which indicates whether the body is buffered. It can be:
318
+ # * `nil` - The body has not been buffered yet.
319
+ # * `true` - The body is buffered as an Array instance.
320
+ # * `false` - The body is not buffered and cannot be buffered.
321
+ #
322
+ # @return [Boolean] whether the body is buffered as an Array instance.
323
323
  def buffered_body!
324
324
  if @buffered.nil?
325
325
  if @body.is_a?(Array)
326
326
  # The user supplied body was an array:
327
327
  @body = @body.compact
328
- @body.each do |part|
329
- @length += part.to_s.bytesize
330
- end
331
-
328
+ @length = @body.sum{|part| part.bytesize}
332
329
  @buffered = true
333
330
  elsif @body.respond_to?(:each)
334
331
  # Turn the user supplied body into a buffered array:
335
332
  body = @body
336
333
  @body = Array.new
334
+ @length = 0
337
335
 
338
336
  body.each do |part|
339
337
  @writer.call(part.to_s)
@@ -341,8 +339,10 @@ module Rack
341
339
 
342
340
  body.close if body.respond_to?(:close)
343
341
 
342
+ # We have converted the body into an Array:
344
343
  @buffered = true
345
344
  else
345
+ # We don't know how to buffer the user-supplied body:
346
346
  @buffered = false
347
347
  end
348
348
  end
@@ -351,12 +351,10 @@ module Rack
351
351
  end
352
352
 
353
353
  def append(chunk)
354
+ chunk = chunk.dup unless chunk.frozen?
354
355
  @body << chunk
355
356
 
356
- unless chunked?
357
- @length += chunk.bytesize
358
- set_header(CONTENT_LENGTH, @length.to_s)
359
- end
357
+ @length += chunk.bytesize
360
358
 
361
359
  return chunk
362
360
  end
data/lib/rack/sendfile.rb CHANGED
@@ -138,7 +138,7 @@ module Rack
138
138
  end
139
139
  when '', nil
140
140
  else
141
- env[RACK_ERRORS].puts "Unknown x-sendfile variation: #{type.inspect}"
141
+ env[RACK_ERRORS].puts "Unknown x-sendfile variation: '#{type}'.\n"
142
142
  end
143
143
  end
144
144
  response
@@ -1,6 +1,5 @@
1
1
  # frozen_string_literal: true
2
2
 
3
- require 'ostruct'
4
3
  require 'erb'
5
4
 
6
5
  require_relative 'constants'
@@ -19,6 +18,11 @@ module Rack
19
18
  class ShowExceptions
20
19
  CONTEXT = 7
21
20
 
21
+ Frame = Struct.new(:filename, :lineno, :function,
22
+ :pre_context_lineno, :pre_context,
23
+ :context_line, :post_context_lineno,
24
+ :post_context)
25
+
22
26
  def initialize(app)
23
27
  @app = app
24
28
  end
@@ -79,7 +83,7 @@ module Rack
79
83
  # This double assignment is to prevent an "unused variable" warning.
80
84
  # Yes, it is dumb, but I don't like Ruby yelling at me.
81
85
  frames = frames = exception.backtrace.map { |line|
82
- frame = OpenStruct.new
86
+ frame = Frame.new
83
87
  if line =~ /(.*?):(\d+)(:in `(.*)')?/
84
88
  frame.filename = $1
85
89
  frame.lineno = $2.to_i
data/lib/rack/static.rb CHANGED
@@ -124,9 +124,8 @@ module Rack
124
124
 
125
125
  def call(env)
126
126
  path = env[PATH_INFO]
127
- actual_path = Utils.clean_path_info(Utils.unescape_path(path))
128
127
 
129
- if can_serve(actual_path)
128
+ if can_serve(path)
130
129
  if overwrite_file_path(path)
131
130
  env[PATH_INFO] = (add_index_root?(path) ? path + @index : @urls[path])
132
131
  elsif @gzip && env['HTTP_ACCEPT_ENCODING'] && /\bgzip\b/.match?(env['HTTP_ACCEPT_ENCODING'])
data/lib/rack/utils.rb CHANGED
@@ -6,6 +6,7 @@ require 'fileutils'
6
6
  require 'set'
7
7
  require 'tempfile'
8
8
  require 'time'
9
+ require 'cgi/escape'
9
10
 
10
11
  require_relative 'query_parser'
11
12
  require_relative 'mime'
@@ -85,15 +86,6 @@ module Rack
85
86
  self.default_query_parser = self.default_query_parser.new_depth_limit(v)
86
87
  end
87
88
 
88
- def self.key_space_limit
89
- warn("`Rack::Utils.key_space_limit` is deprecated as this value no longer has an effect. It will be removed in Rack 3.1", uplevel: 1)
90
- 65536
91
- end
92
-
93
- def self.key_space_limit=(v)
94
- warn("`Rack::Utils.key_space_limit=` is deprecated and no longer has an effect. It will be removed in Rack 3.1", uplevel: 1)
95
- end
96
-
97
89
  if defined?(Process::CLOCK_MONOTONIC)
98
90
  def clock_time
99
91
  Process.clock_gettime(Process::CLOCK_MONOTONIC)
@@ -184,21 +176,8 @@ module Rack
184
176
  matches&.first
185
177
  end
186
178
 
187
- ESCAPE_HTML = {
188
- "&" => "&amp;",
189
- "<" => "&lt;",
190
- ">" => "&gt;",
191
- "'" => "&#x27;",
192
- '"' => "&quot;",
193
- "/" => "&#x2F;"
194
- }
195
-
196
- ESCAPE_HTML_PATTERN = Regexp.union(*ESCAPE_HTML.keys)
197
-
198
179
  # Escape ampersands, brackets and quotes to their HTML/XML entities.
199
- def escape_html(string)
200
- string.to_s.gsub(ESCAPE_HTML_PATTERN){|c| ESCAPE_HTML[c] }
201
- end
180
+ define_method(:escape_html, CGI.method(:escapeHTML))
202
181
 
203
182
  def select_best_encoding(available_encodings, accept_encoding)
204
183
  # http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html
@@ -252,21 +231,6 @@ module Rack
252
231
  end
253
232
  end
254
233
 
255
- def add_cookie_to_header(header, key, value)
256
- warn("add_cookie_to_header is deprecated and will be removed in Rack 3.1", uplevel: 1)
257
-
258
- case header
259
- when nil, ''
260
- return set_cookie_header(key, value)
261
- when String
262
- [header, set_cookie_header(key, value)]
263
- when Array
264
- header + [set_cookie_header(key, value)]
265
- else
266
- raise ArgumentError, "Unrecognized cookie header value. Expected String, Array, or nil, got #{header.inspect}"
267
- end
268
- end
269
-
270
234
  # :call-seq:
271
235
  # parse_cookies(env) -> hash
272
236
  #
@@ -280,6 +244,20 @@ module Rack
280
244
  parse_cookies_header env[HTTP_COOKIE]
281
245
  end
282
246
 
247
+ # A valid cookie key according to RFC2616.
248
+ # A <cookie-name> can be any US-ASCII characters, except control characters, spaces, or tabs. It also must not contain a separator character like the following: ( ) < > @ , ; : \ " / [ ] ? = { }.
249
+ VALID_COOKIE_KEY = /\A[!#$%&'*+\-\.\^_`|~0-9a-zA-Z]+\z/.freeze
250
+ private_constant :VALID_COOKIE_KEY
251
+
252
+ private def escape_cookie_key(key)
253
+ if key =~ VALID_COOKIE_KEY
254
+ key
255
+ else
256
+ warn "Cookie key #{key.inspect} is not valid according to RFC2616; it will be escaped. This behaviour is deprecated and will be removed in a future version of Rack.", uplevel: 2
257
+ escape(key)
258
+ end
259
+ end
260
+
283
261
  # :call-seq:
284
262
  # set_cookie_header(key, value) -> encoded string
285
263
  #
@@ -306,7 +284,7 @@ module Rack
306
284
  def set_cookie_header(key, value)
307
285
  case value
308
286
  when Hash
309
- key = escape(key) unless value[:escape_key] == false
287
+ key = escape_cookie_key(key) unless value[:escape_key] == false
310
288
  domain = "; domain=#{value[:domain]}" if value[:domain]
311
289
  path = "; path=#{value[:path]}" if value[:path]
312
290
  max_age = "; max-age=#{value[:max_age]}" if value[:max_age]
@@ -318,23 +296,24 @@ module Rack
318
296
  when false, nil
319
297
  nil
320
298
  when :none, 'None', :None
321
- '; SameSite=None'
299
+ '; samesite=none'
322
300
  when :lax, 'Lax', :Lax
323
- '; SameSite=Lax'
301
+ '; samesite=lax'
324
302
  when true, :strict, 'Strict', :Strict
325
- '; SameSite=Strict'
303
+ '; samesite=strict'
326
304
  else
327
- raise ArgumentError, "Invalid SameSite value: #{value[:same_site].inspect}"
305
+ raise ArgumentError, "Invalid :same_site value: #{value[:same_site].inspect}"
328
306
  end
307
+ partitioned = "; partitioned" if value[:partitioned]
329
308
  value = value[:value]
330
309
  else
331
- key = escape(key)
310
+ key = escape_cookie_key(key)
332
311
  end
333
312
 
334
313
  value = [value] unless Array === value
335
314
 
336
315
  return "#{key}=#{value.map { |v| escape v }.join('&')}#{domain}" \
337
- "#{path}#{max_age}#{expires}#{secure}#{httponly}#{same_site}"
316
+ "#{path}#{max_age}#{expires}#{secure}#{httponly}#{same_site}#{partitioned}"
338
317
  end
339
318
 
340
319
  # :call-seq:
@@ -375,24 +354,12 @@ module Rack
375
354
  set_cookie_header(key, value.merge(max_age: '0', expires: Time.at(0), value: ''))
376
355
  end
377
356
 
378
- def make_delete_cookie_header(header, key, value)
379
- warn("make_delete_cookie_header is deprecated and will be removed in Rack 3.1, use delete_set_cookie_header! instead", uplevel: 1)
380
-
381
- delete_set_cookie_header!(header, key, value)
382
- end
383
-
384
357
  def delete_cookie_header!(headers, key, value = {})
385
358
  headers[SET_COOKIE] = delete_set_cookie_header!(headers[SET_COOKIE], key, value)
386
359
 
387
360
  return nil
388
361
  end
389
362
 
390
- def add_remove_cookie_to_header(header, key, value = {})
391
- warn("add_remove_cookie_to_header is deprecated and will be removed in Rack 3.1, use delete_set_cookie_header! instead", uplevel: 1)
392
-
393
- delete_set_cookie_header!(header, key, value)
394
- end
395
-
396
363
  # :call-seq:
397
364
  # delete_set_cookie_header!(header, key, value = {}) -> header value
398
365
  #
@@ -435,6 +402,8 @@ module Rack
435
402
 
436
403
  def get_byte_ranges(http_range, size)
437
404
  # See <http://www.w3.org/Protocols/rfc2616/rfc2616-sec14.html#sec14.35>
405
+ # Ignore Range when file size is 0 to avoid a 416 error.
406
+ return nil if size.zero?
438
407
  return nil unless http_range && http_range =~ /bytes=([^;]+)/
439
408
  ranges = []
440
409
  $1.split(/,\s*/).each do |range_spec|
@@ -517,39 +486,12 @@ module Rack
517
486
  end
518
487
  end
519
488
 
520
- # A wrapper around Headers
521
- # header when set.
522
- #
523
- # @api private
524
- class HeaderHash < Hash # :nodoc:
525
- def self.[](headers)
526
- warn "Rack::Utils::HeaderHash is deprecated and will be removed in Rack 3.1, switch to Rack::Headers", uplevel: 1
527
- if headers.is_a?(Headers) && !headers.frozen?
528
- return headers
529
- end
530
-
531
- new_headers = Headers.new
532
- headers.each{|k,v| new_headers[k] = v}
533
- new_headers
534
- end
535
-
536
- def self.new(hash = {})
537
- warn "Rack::Utils::HeaderHash is deprecated and will be removed in Rack 3.1, switch to Rack::Headers", uplevel: 1
538
- headers = Headers.new
539
- hash.each{|k,v| headers[k] = v}
540
- headers
541
- end
542
-
543
- def self.allocate
544
- raise TypeError, "cannot allocate HeaderHash"
545
- end
546
- end
547
-
548
489
  # Every standard HTTP code mapped to the appropriate message.
549
490
  # Generated with:
550
- # curl -s https://www.iana.org/assignments/http-status-codes/http-status-codes-1.csv | \
551
- # ruby -ne 'm = /^(\d{3}),(?!Unassigned|\(Unused\))([^,]+)/.match($_) and \
552
- # puts "#{m[1]} => \x27#{m[2].strip}\x27,"'
491
+ # curl -s https://www.iana.org/assignments/http-status-codes/http-status-codes-1.csv \
492
+ # | ruby -rcsv -e "puts CSV.parse(STDIN, headers: true) \
493
+ # .reject {|v| v['Description'] == 'Unassigned' or v['Description'].include? '(' } \
494
+ # .map {|v| %Q/#{v['Value']} => '#{v['Description']}'/ }.join(','+?\n)"
553
495
  HTTP_STATUS_CODES = {
554
496
  100 => 'Continue',
555
497
  101 => 'Switching Protocols',
@@ -571,7 +513,6 @@ module Rack
571
513
  303 => 'See Other',
572
514
  304 => 'Not Modified',
573
515
  305 => 'Use Proxy',
574
- 306 => '(Unused)',
575
516
  307 => 'Temporary Redirect',
576
517
  308 => 'Permanent Redirect',
577
518
  400 => 'Bad Request',
@@ -587,13 +528,13 @@ module Rack
587
528
  410 => 'Gone',
588
529
  411 => 'Length Required',
589
530
  412 => 'Precondition Failed',
590
- 413 => 'Payload Too Large',
531
+ 413 => 'Content Too Large',
591
532
  414 => 'URI Too Long',
592
533
  415 => 'Unsupported Media Type',
593
534
  416 => 'Range Not Satisfiable',
594
535
  417 => 'Expectation Failed',
595
536
  421 => 'Misdirected Request',
596
- 422 => 'Unprocessable Entity',
537
+ 422 => 'Unprocessable Content',
597
538
  423 => 'Locked',
598
539
  424 => 'Failed Dependency',
599
540
  425 => 'Too Early',
@@ -601,7 +542,7 @@ module Rack
601
542
  428 => 'Precondition Required',
602
543
  429 => 'Too Many Requests',
603
544
  431 => 'Request Header Fields Too Large',
604
- 451 => 'Unavailable for Legal Reasons',
545
+ 451 => 'Unavailable For Legal Reasons',
605
546
  500 => 'Internal Server Error',
606
547
  501 => 'Not Implemented',
607
548
  502 => 'Bad Gateway',
@@ -611,8 +552,6 @@ module Rack
611
552
  506 => 'Variant Also Negotiates',
612
553
  507 => 'Insufficient Storage',
613
554
  508 => 'Loop Detected',
614
- 509 => 'Bandwidth Limit Exceeded',
615
- 510 => 'Not Extended',
616
555
  511 => 'Network Authentication Required'
617
556
  }
618
557
 
@@ -620,12 +559,34 @@ module Rack
620
559
  STATUS_WITH_NO_ENTITY_BODY = Hash[((100..199).to_a << 204 << 304).product([true])]
621
560
 
622
561
  SYMBOL_TO_STATUS_CODE = Hash[*HTTP_STATUS_CODES.map { |code, message|
623
- [message.downcase.gsub(/\s|-|'/, '_').to_sym, code]
562
+ [message.downcase.gsub(/\s|-/, '_').to_sym, code]
624
563
  }.flatten]
625
564
 
565
+ OBSOLETE_SYMBOLS_TO_STATUS_CODES = {
566
+ payload_too_large: 413,
567
+ unprocessable_entity: 422,
568
+ bandwidth_limit_exceeded: 509,
569
+ not_extended: 510
570
+ }.freeze
571
+ private_constant :OBSOLETE_SYMBOLS_TO_STATUS_CODES
572
+
573
+ OBSOLETE_SYMBOL_MAPPINGS = {
574
+ payload_too_large: :content_too_large,
575
+ unprocessable_entity: :unprocessable_content
576
+ }.freeze
577
+ private_constant :OBSOLETE_SYMBOL_MAPPINGS
578
+
626
579
  def status_code(status)
627
580
  if status.is_a?(Symbol)
628
- SYMBOL_TO_STATUS_CODE.fetch(status) { raise ArgumentError, "Unrecognized status code #{status.inspect}" }
581
+ SYMBOL_TO_STATUS_CODE.fetch(status) do
582
+ fallback_code = OBSOLETE_SYMBOLS_TO_STATUS_CODES.fetch(status) { raise ArgumentError, "Unrecognized status code #{status.inspect}" }
583
+ message = "Status code #{status.inspect} is deprecated and will be removed in a future version of Rack."
584
+ if canonical_symbol = OBSOLETE_SYMBOL_MAPPINGS[status]
585
+ message = "#{message} Please use #{canonical_symbol.inspect} instead."
586
+ end
587
+ warn message, uplevel: 1
588
+ fallback_code
589
+ end
629
590
  else
630
591
  status.to_i
631
592
  end
data/lib/rack/version.rb CHANGED
@@ -12,20 +12,7 @@
12
12
  # so it should be enough just to <tt>require 'rack'</tt> in your code.
13
13
 
14
14
  module Rack
15
- # The Rack protocol version number implemented.
16
- VERSION = [1, 3].freeze
17
- deprecate_constant :VERSION
18
-
19
- VERSION_STRING = "1.3".freeze
20
- deprecate_constant :VERSION_STRING
21
-
22
- # The Rack protocol version number implemented.
23
- def self.version
24
- warn "Rack.version is deprecated and will be removed in Rack 3.1!", uplevel: 1
25
- VERSION
26
- end
27
-
28
- RELEASE = "3.0.18"
15
+ RELEASE = "3.1.0"
29
16
 
30
17
  # Return the Rack release as a dotted string.
31
18
  def self.release
data/lib/rack.rb CHANGED
@@ -15,10 +15,10 @@ require_relative 'rack/version'
15
15
  require_relative 'rack/constants'
16
16
 
17
17
  module Rack
18
+ autoload :BadRequest, "rack/bad_request"
18
19
  autoload :BodyProxy, "rack/body_proxy"
19
20
  autoload :Builder, "rack/builder"
20
21
  autoload :Cascade, "rack/cascade"
21
- autoload :Chunked, "rack/chunked"
22
22
  autoload :CommonLogger, "rack/common_logger"
23
23
  autoload :ConditionalGet, "rack/conditional_get"
24
24
  autoload :Config, "rack/config"
@@ -28,7 +28,6 @@ module Rack
28
28
  autoload :Directory, "rack/directory"
29
29
  autoload :ETag, "rack/etag"
30
30
  autoload :Events, "rack/events"
31
- autoload :File, "rack/file"
32
31
  autoload :Files, "rack/files"
33
32
  autoload :ForwardRequest, "rack/recursive"
34
33
  autoload :Head, "rack/head"
@@ -60,7 +59,6 @@ module Rack
60
59
 
61
60
  module Auth
62
61
  autoload :Basic, "rack/auth/basic"
63
- autoload :Digest, "rack/auth/digest"
64
62
  autoload :AbstractHandler, "rack/auth/abstract/handler"
65
63
  autoload :AbstractRequest, "rack/auth/abstract/request"
66
64
  end
metadata CHANGED
@@ -1,13 +1,14 @@
1
1
  --- !ruby/object:Gem::Specification
2
2
  name: rack
3
3
  version: !ruby/object:Gem::Version
4
- version: 3.0.18
4
+ version: 3.1.0
5
5
  platform: ruby
6
6
  authors:
7
7
  - Leah Neukirchen
8
+ autorequire:
8
9
  bindir: bin
9
10
  cert_chain: []
10
- date: 2025-05-22 00:00:00.000000000 Z
11
+ date: 2024-06-11 00:00:00.000000000 Z
11
12
  dependencies:
12
13
  - !ruby/object:Gem::Dependency
13
14
  name: minitest
@@ -88,15 +89,10 @@ files:
88
89
  - lib/rack/auth/abstract/handler.rb
89
90
  - lib/rack/auth/abstract/request.rb
90
91
  - lib/rack/auth/basic.rb
91
- - lib/rack/auth/digest.rb
92
- - lib/rack/auth/digest/md5.rb
93
- - lib/rack/auth/digest/nonce.rb
94
- - lib/rack/auth/digest/params.rb
95
- - lib/rack/auth/digest/request.rb
92
+ - lib/rack/bad_request.rb
96
93
  - lib/rack/body_proxy.rb
97
94
  - lib/rack/builder.rb
98
95
  - lib/rack/cascade.rb
99
- - lib/rack/chunked.rb
100
96
  - lib/rack/common_logger.rb
101
97
  - lib/rack/conditional_get.rb
102
98
  - lib/rack/config.rb
@@ -107,7 +103,6 @@ files:
107
103
  - lib/rack/directory.rb
108
104
  - lib/rack/etag.rb
109
105
  - lib/rack/events.rb
110
- - lib/rack/file.rb
111
106
  - lib/rack/files.rb
112
107
  - lib/rack/head.rb
113
108
  - lib/rack/headers.rb
@@ -148,6 +143,7 @@ metadata:
148
143
  changelog_uri: https://github.com/rack/rack/blob/main/CHANGELOG.md
149
144
  documentation_uri: https://rubydoc.info/github/rack/rack
150
145
  source_code_uri: https://github.com/rack/rack
146
+ post_install_message:
151
147
  rdoc_options: []
152
148
  require_paths:
153
149
  - lib
@@ -162,7 +158,8 @@ required_rubygems_version: !ruby/object:Gem::Requirement
162
158
  - !ruby/object:Gem::Version
163
159
  version: '0'
164
160
  requirements: []
165
- rubygems_version: 3.6.2
161
+ rubygems_version: 3.5.9
162
+ signing_key:
166
163
  specification_version: 4
167
164
  summary: A modular Ruby webserver interface.
168
165
  test_files: []
@@ -1 +0,0 @@
1
- require_relative '../digest'
@@ -1 +0,0 @@
1
- require_relative '../digest'
@@ -1 +0,0 @@
1
- require_relative '../digest'
@@ -1 +0,0 @@
1
- require_relative '../digest'