File: output.rb

package info (click to toggle)
ruby-minitar 1.1.0-1
  • links: PTS, VCS
  • area: main
  • in suites: forky, sid
  • size: 364 kB
  • sloc: ruby: 2,602; makefile: 11
file content (89 lines) | stat: -rw-r--r-- 3,071 bytes parent folder | download
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
# frozen_string_literal: true

require "minitar/writer"

class Minitar
  # Wraps a Minitar::Writer with convenience methods and wrapped stream management. If the
  # stream provided to Output does not support random access, only Writer#add_file_simple
  # and Writer#mkdir are guaranteed to work.
  #
  # === Security Notice
  #
  # Constructing a Minitar::Output will use Kernel.open if the provided output is not
  # a readable stream object. Using an untrusted value for output may allow a malicious
  # user to execute arbitrary system commands. It is the caller's responsibility to ensure
  # that the output value is safe.
  #
  # * {CWE-073}[https://cwe.mitre.org/data/definitions/73.html]
  # * {CWE-078}[https://cwe.mitre.org/data/definitions/78.html]
  # * {CWE-088}[https://cwe.mitre.org/data/definitions/88.html]
  #
  # This notice applies to Minitar::Output.open, Minitar::Output.tar, and
  # Minitar::Output.new.
  class Output
    # With no associated block, +Output.open+ is a synonym for +Output.new+.
    #
    # If a block is given, the new Output will be yielded to the block as an argument and
    # the Output object will automatically be closed when the block terminates (this also
    # closes the wrapped stream object). The return value will be the value of the block.
    #
    # :call-seq:
    #    Minitar::Output.open(io) -> output
    #    Minitar::Output.open(io) { |output| block } -> obj
    def self.open(output)
      stream = new(output)
      return stream unless block_given?

      # This exception context must remain, otherwise the stream closes on open
      # even if a block is not given.
      begin
        yield stream
      ensure
        stream.close
      end
    end

    # Output.tar is a wrapper for Output.open that yields the owned tar object instead of
    # the Output object. If a block is not provided, an enumerator will be created with
    # the same behaviour.
    #
    # :call-seq:
    #    Minitar::Output.tar(io) -> enumerator
    #    Minitar::Output.tar(io) { |tar| block } -> obj
    def self.tar(output)
      return to_enum(__method__, output) unless block_given?

      Output.open(output) do |stream|
        yield stream.tar
      end
    end

    # Creates a new Output object. If +output+ is a stream object that responds to #write,
    # then it will simply be wrapped. Otherwise, one will be created and opened using
    # Kernel#open. When Output#close is called, the stream object wrapped will be closed.
    #
    # :call-seq:
    #    Minitar::Output.new(io) -> output
    #    Minitar::Output.new(path) -> output
    def initialize(output)
      @io = if output.respond_to?(:write)
        output
      else
        ::Kernel.open(output, "wb")
      end
      @tar = Minitar::Writer.new(@io)
    end

    # Returns the Writer object for direct access.
    attr_reader :tar

    # Returns false if the wrapped data stream is open.
    def closed? = @io.closed?

    # Closes the Writer object and the wrapped data stream.
    def close
      @tar.close
      @io.close
    end
  end
end