minitar 0.5.4 → 0.6
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/Code-of-Conduct.md +74 -0
- data/Contributing.md +84 -0
- data/History.md +107 -0
- data/Licence.md +15 -0
- data/Manifest.txt +24 -0
- data/README.rdoc +81 -0
- data/Rakefile +46 -107
- data/docs/bsdl.txt +19 -0
- data/docs/ruby.txt +56 -0
- data/lib/archive-tar-minitar.rb +3 -0
- data/lib/archive/tar/minitar.rb +197 -889
- data/lib/archive/tar/minitar/input.rb +212 -0
- data/lib/archive/tar/minitar/output.rb +69 -0
- data/lib/archive/tar/minitar/posix_header.rb +259 -0
- data/lib/archive/tar/minitar/reader.rb +237 -0
- data/lib/archive/tar/minitar/writer.rb +297 -0
- data/lib/minitar.rb +12 -0
- data/test/minitest_helper.rb +11 -0
- data/test/support/tar_test_helpers.rb +119 -0
- data/test/test_tar_header.rb +74 -0
- data/test/test_tar_input.rb +167 -0
- data/test/test_tar_output.rb +53 -0
- data/test/test_tar_reader.rb +148 -0
- data/test/test_tar_writer.rb +190 -0
- metadata +232 -43
- data/ChangeLog +0 -17
- data/Install +0 -6
- data/README +0 -68
- data/bin/minitar +0 -27
- data/lib/archive/tar/minitar/command.rb +0 -814
- data/tests/tc_tar.rb +0 -629
- data/tests/testall.rb +0 -10
data/ChangeLog
DELETED
@@ -1,17 +0,0 @@
|
|
1
|
-
Revision history for Ruby library Archive::Tar::Minitar. Unless explicitly
|
2
|
-
noted otherwise, all changes are produced by Austin Ziegler
|
3
|
-
<austin@rubyforge.org>.
|
4
|
-
|
5
|
-
== 0.5.3
|
6
|
-
* Fix for long names (http://rubyforge.org/tracker/index.php?func=detail&aid=22628&group_id=84&atid=407)
|
7
|
-
|
8
|
-
== 0.5.2
|
9
|
-
* Fixed a Ruby 1.9 compatibility error.
|
10
|
-
|
11
|
-
== 0.5.1
|
12
|
-
* Fixed a variable name error.
|
13
|
-
|
14
|
-
== Archive::Tar::Minitar 0.5.0
|
15
|
-
* Initial release. Does files and directories. Command does create, extract,
|
16
|
-
* and list.
|
17
|
-
|
data/Install
DELETED
data/README
DELETED
@@ -1,68 +0,0 @@
|
|
1
|
-
Archive::Tar::Minitar README
|
2
|
-
============================
|
3
|
-
Archive::Tar::Minitar is a pure-Ruby library and command-line utility that
|
4
|
-
provides the ability to deal with POSIX tar(1) archive files. The
|
5
|
-
implementation is based heavily on Mauricio Ferna'ndez's implementation in
|
6
|
-
rpa-base, but has been reorganised to promote reuse in other projects.
|
7
|
-
Antoine Toulme forked the original project on rubyforge to place it on github, under
|
8
|
-
http://www.github.com/atoulme/minitar
|
9
|
-
|
10
|
-
This release is version 0.5.2, offering a Ruby 1.9 compatibility bugfix over
|
11
|
-
version 0.5.1. The library can only handle files and directories at this
|
12
|
-
point. A future version will be expanded to handle symbolic links and hard
|
13
|
-
links in a portable manner. The command line utility, minitar, can only create
|
14
|
-
archives, extract from archives, and list archive contents.
|
15
|
-
|
16
|
-
Using this library is easy. The simplest case is:
|
17
|
-
|
18
|
-
require 'zlib'
|
19
|
-
require 'archive/tar/minitar'
|
20
|
-
include Archive::Tar
|
21
|
-
|
22
|
-
# Packs everything that matches Find.find('tests')
|
23
|
-
File.open('test.tar', 'wb') { |tar| Minitar.pack('tests', tar) }
|
24
|
-
# Unpacks 'test.tar' to 'x', creating 'x' if necessary.
|
25
|
-
Minitar.unpack('test.tar', 'x')
|
26
|
-
|
27
|
-
A gzipped tar can be written with:
|
28
|
-
|
29
|
-
tgz = Zlib::GzipWriter.new(File.open('test.tgz', 'wb'))
|
30
|
-
# Warning: tgz will be closed!
|
31
|
-
Minitar.pack('tests', tgz)
|
32
|
-
|
33
|
-
tgz = Zlib::GzipReader.new(File.open('test.tgz', 'rb'))
|
34
|
-
# Warning: tgz will be closed!
|
35
|
-
Minitar.unpack(tgz, 'x')
|
36
|
-
|
37
|
-
As the case above shows, one need not write to a file. However, it will
|
38
|
-
sometimes require that one dive a little deeper into the API, as in the case
|
39
|
-
of StringIO objects. Note that I'm not providing a block with Minitar::Output,
|
40
|
-
as Minitar::Output#close automatically closes both the Output object and the
|
41
|
-
wrapped data stream object.
|
42
|
-
|
43
|
-
begin
|
44
|
-
sgz = Zlib::GzipWriter.new(StringIO.new(""))
|
45
|
-
tar = Output.new(sgz)
|
46
|
-
Find.find('tests') do |entry|
|
47
|
-
Minitar.pack_file(entry, tar)
|
48
|
-
end
|
49
|
-
ensure
|
50
|
-
# Closes both tar and sgz.
|
51
|
-
tar.close
|
52
|
-
end
|
53
|
-
|
54
|
-
Copyright
|
55
|
-
=========
|
56
|
-
# Copyright 2004 Mauricio Julio Ferna'ndez Pradier and Austin Ziegler
|
57
|
-
#
|
58
|
-
# This program is based on and incorporates parts of RPA::Package from
|
59
|
-
# rpa-base (lib/rpa/package.rb and lib/rpa/util.rb) by Mauricio and has been
|
60
|
-
# adapted to be more generic by Austin.
|
61
|
-
#
|
62
|
-
# 'minitar' contains an adaptation of Ruby/ProgressBar by Satoru
|
63
|
-
# Takabayashi <satoru@namazu.org>, copyright 2001 - 2004.
|
64
|
-
#
|
65
|
-
# This program is free software. It may be redistributed and/or modified
|
66
|
-
# under the terms of the GPL version 2 (or later) or Ruby's licence.
|
67
|
-
#
|
68
|
-
# $Id$
|
data/bin/minitar
DELETED
@@ -1,27 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
#--
|
3
|
-
# Archive::Tar::Minitar 0.5.2
|
4
|
-
# Copyright 2004 Mauricio Julio Ferna'ndez Pradier and Austin Ziegler
|
5
|
-
#
|
6
|
-
# This program is based on and incorporates parts of RPA::Package from
|
7
|
-
# rpa-base (lib/rpa/package.rb and lib/rpa/util.rb) by Mauricio and has been
|
8
|
-
# adapted to be more generic by Austin.
|
9
|
-
#
|
10
|
-
# It is licensed under the GNU General Public Licence or Ruby's licence.
|
11
|
-
#
|
12
|
-
# $Id$
|
13
|
-
#++
|
14
|
-
|
15
|
-
# 1) Try to load Archive::Tar::Minitar from the gem.
|
16
|
-
# 2) Try to load Archive::Tar::Minitar from $LOAD_PATH.
|
17
|
-
begin
|
18
|
-
require 'rubygems'
|
19
|
-
require_gem 'archive-tar-minitar', '= 0.5.2'
|
20
|
-
rescue LoadError
|
21
|
-
nil
|
22
|
-
end
|
23
|
-
|
24
|
-
require 'archive/tar/minitar'
|
25
|
-
require 'archive/tar/minitar/command'
|
26
|
-
|
27
|
-
exit Archive::Tar::Minitar::Command.run(ARGV)
|
@@ -1,814 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
#--
|
3
|
-
# Archive::Tar::Baby 0.5.2
|
4
|
-
# Copyright 2004 Mauricio Julio Ferna'ndez Pradier and Austin Ziegler
|
5
|
-
# This is free software with ABSOLUTELY NO WARRANTY.
|
6
|
-
#
|
7
|
-
# This program is based on and incorporates parts of RPA::Package from
|
8
|
-
# rpa-base (lib/rpa/package.rb and lib/rpa/util.rb) by Mauricio and has been
|
9
|
-
# adapted to be more generic by Austin.
|
10
|
-
#
|
11
|
-
# This file contains an adaptation of Ruby/ProgressBar by Satoru
|
12
|
-
# Takabayashi <satoru@namazu.org>, copyright 2001 - 2004.
|
13
|
-
#
|
14
|
-
# It is licensed under the GNU General Public Licence or Ruby's licence.
|
15
|
-
#
|
16
|
-
# $Id$
|
17
|
-
#++
|
18
|
-
|
19
|
-
require 'zlib'
|
20
|
-
|
21
|
-
# TODO: add
|
22
|
-
# TODO: delete ???
|
23
|
-
|
24
|
-
require 'optparse'
|
25
|
-
require 'ostruct'
|
26
|
-
require 'fileutils'
|
27
|
-
|
28
|
-
module Archive::Tar::Minitar::Command
|
29
|
-
class ProgressBar
|
30
|
-
VERSION = "0.8"
|
31
|
-
|
32
|
-
attr_accessor :total
|
33
|
-
attr_accessor :title
|
34
|
-
|
35
|
-
def initialize (title, total, out = STDERR)
|
36
|
-
@title = title
|
37
|
-
@total = total
|
38
|
-
@out = out
|
39
|
-
@bar_width = 80
|
40
|
-
@bar_mark = "o"
|
41
|
-
@current = 0
|
42
|
-
@previous = 0
|
43
|
-
@is_finished = false
|
44
|
-
@start_time = Time.now
|
45
|
-
@previous_time = @start_time
|
46
|
-
@title_width = 14
|
47
|
-
@format = "%-#{@title_width}s %3d%% %s %s"
|
48
|
-
@format_arguments = [:title, :percentage, :bar, :stat]
|
49
|
-
show
|
50
|
-
end
|
51
|
-
|
52
|
-
private
|
53
|
-
def convert_bytes (bytes)
|
54
|
-
if bytes < 1024
|
55
|
-
sprintf("%6dB", bytes)
|
56
|
-
elsif bytes < 1024 * 1000 # 1000kb
|
57
|
-
sprintf("%5.1fKB", bytes.to_f / 1024)
|
58
|
-
elsif bytes < 1024 * 1024 * 1000 # 1000mb
|
59
|
-
sprintf("%5.1fMB", bytes.to_f / 1024 / 1024)
|
60
|
-
else
|
61
|
-
sprintf("%5.1fGB", bytes.to_f / 1024 / 1024 / 1024)
|
62
|
-
end
|
63
|
-
end
|
64
|
-
|
65
|
-
def transfer_rate
|
66
|
-
bytes_per_second = @current.to_f / (Time.now - @start_time)
|
67
|
-
sprintf("%s/s", convert_bytes(bytes_per_second))
|
68
|
-
end
|
69
|
-
|
70
|
-
def bytes
|
71
|
-
convert_bytes(@current)
|
72
|
-
end
|
73
|
-
|
74
|
-
def format_time (t)
|
75
|
-
t = t.to_i
|
76
|
-
sec = t % 60
|
77
|
-
min = (t / 60) % 60
|
78
|
-
hour = t / 3600
|
79
|
-
sprintf("%02d:%02d:%02d", hour, min, sec);
|
80
|
-
end
|
81
|
-
|
82
|
-
# ETA stands for Estimated Time of Arrival.
|
83
|
-
def eta
|
84
|
-
if @current == 0
|
85
|
-
"ETA: --:--:--"
|
86
|
-
else
|
87
|
-
elapsed = Time.now - @start_time
|
88
|
-
eta = elapsed * @total / @current - elapsed;
|
89
|
-
sprintf("ETA: %s", format_time(eta))
|
90
|
-
end
|
91
|
-
end
|
92
|
-
|
93
|
-
def elapsed
|
94
|
-
elapsed = Time.now - @start_time
|
95
|
-
sprintf("Time: %s", format_time(elapsed))
|
96
|
-
end
|
97
|
-
|
98
|
-
def stat
|
99
|
-
if @is_finished then elapsed else eta end
|
100
|
-
end
|
101
|
-
|
102
|
-
def stat_for_file_transfer
|
103
|
-
if @is_finished then
|
104
|
-
sprintf("%s %s %s", bytes, transfer_rate, elapsed)
|
105
|
-
else
|
106
|
-
sprintf("%s %s %s", bytes, transfer_rate, eta)
|
107
|
-
end
|
108
|
-
end
|
109
|
-
|
110
|
-
def eol
|
111
|
-
if @is_finished then "\n" else "\r" end
|
112
|
-
end
|
113
|
-
|
114
|
-
def bar
|
115
|
-
len = percentage * @bar_width / 100
|
116
|
-
sprintf("|%s%s|", @bar_mark * len, " " * (@bar_width - len))
|
117
|
-
end
|
118
|
-
|
119
|
-
def percentage(value = nil)
|
120
|
-
if @total.zero?
|
121
|
-
100
|
122
|
-
else
|
123
|
-
(value || @current) * 100 / @total
|
124
|
-
end
|
125
|
-
end
|
126
|
-
|
127
|
-
def title
|
128
|
-
@title[0,(@title_width - 1)] + ":"
|
129
|
-
end
|
130
|
-
|
131
|
-
def get_width
|
132
|
-
# FIXME: I don't know how portable it is.
|
133
|
-
default_width = 80
|
134
|
-
# begin
|
135
|
-
# tiocgwinsz = 0x5413
|
136
|
-
# data = [0, 0, 0, 0].pack("SSSS")
|
137
|
-
# if @out.ioctl(tiocgwinsz, data) >= 0 then
|
138
|
-
# rows, cols, xpixels, ypixels = data.unpack("SSSS")
|
139
|
-
# if cols >= 0 then cols else default_width end
|
140
|
-
# else
|
141
|
-
# default_width
|
142
|
-
# end
|
143
|
-
# rescue Exception
|
144
|
-
# default_width
|
145
|
-
# end
|
146
|
-
end
|
147
|
-
|
148
|
-
def show
|
149
|
-
arguments = @format_arguments.map {|method| send(method) }
|
150
|
-
line = sprintf(@format, *arguments)
|
151
|
-
|
152
|
-
width = get_width
|
153
|
-
if line.length == width - 1
|
154
|
-
@out.print(line + eol)
|
155
|
-
elsif line.length >= width
|
156
|
-
@bar_width = [@bar_width - (line.length - width + 1), 0].max
|
157
|
-
if @bar_width == 0 then @out.print(line + eol) else show end
|
158
|
-
else # line.length < width - 1
|
159
|
-
@bar_width += width - line.length + 1
|
160
|
-
show
|
161
|
-
end
|
162
|
-
@previous_time = Time.now
|
163
|
-
end
|
164
|
-
|
165
|
-
def show_progress
|
166
|
-
if @total.zero?
|
167
|
-
cur_percentage = 100
|
168
|
-
prev_percentage = 0
|
169
|
-
else
|
170
|
-
cur_percentage = (@current * 100 / @total).to_i
|
171
|
-
prev_percentage = (@previous * 100 / @total).to_i
|
172
|
-
end
|
173
|
-
|
174
|
-
if cur_percentage > prev_percentage ||
|
175
|
-
Time.now - @previous_time >= 1 ||
|
176
|
-
@is_finished
|
177
|
-
show
|
178
|
-
end
|
179
|
-
end
|
180
|
-
|
181
|
-
public
|
182
|
-
def file_transfer_mode
|
183
|
-
@format_arguments = [:title, :percentage, :bar, :stat_for_file_transfer]
|
184
|
-
end
|
185
|
-
|
186
|
-
def format= (format)
|
187
|
-
@format = format
|
188
|
-
end
|
189
|
-
|
190
|
-
def format_arguments= (arguments)
|
191
|
-
@format_arguments = arguments
|
192
|
-
end
|
193
|
-
|
194
|
-
def finish
|
195
|
-
@current = @total
|
196
|
-
@is_finished = true
|
197
|
-
show_progress
|
198
|
-
end
|
199
|
-
|
200
|
-
def halt
|
201
|
-
@is_finished = true
|
202
|
-
show_progress
|
203
|
-
end
|
204
|
-
|
205
|
-
def set (count)
|
206
|
-
if count < 0 || count > @total
|
207
|
-
raise "invalid count: #{count} (total: #{@total})"
|
208
|
-
end
|
209
|
-
@current = count
|
210
|
-
show_progress
|
211
|
-
@previous = @current
|
212
|
-
end
|
213
|
-
|
214
|
-
def inc (step = 1)
|
215
|
-
@current += step
|
216
|
-
@current = @total if @current > @total
|
217
|
-
show_progress
|
218
|
-
@previous = @current
|
219
|
-
end
|
220
|
-
|
221
|
-
def inspect
|
222
|
-
"(ProgressBar: #{@current}/#{@total})"
|
223
|
-
end
|
224
|
-
end
|
225
|
-
|
226
|
-
class CommandPattern
|
227
|
-
class AbstractCommandError < Exception; end
|
228
|
-
class UnknownCommandError < RuntimeError; end
|
229
|
-
class CommandAlreadyExists < RuntimeError; end
|
230
|
-
|
231
|
-
class << self
|
232
|
-
def add(command)
|
233
|
-
command = command.new if command.kind_of?(Class)
|
234
|
-
|
235
|
-
@commands ||= {}
|
236
|
-
if @commands.has_key?(command.name)
|
237
|
-
raise CommandAlreadyExists
|
238
|
-
else
|
239
|
-
@commands[command.name] = command
|
240
|
-
end
|
241
|
-
|
242
|
-
if command.respond_to?(:altname)
|
243
|
-
unless @commands.has_key?(command.altname)
|
244
|
-
@commands[command.altname] = command
|
245
|
-
end
|
246
|
-
end
|
247
|
-
end
|
248
|
-
|
249
|
-
def <<(command)
|
250
|
-
add(command)
|
251
|
-
end
|
252
|
-
|
253
|
-
attr_accessor :default
|
254
|
-
def default=(command) #:nodoc:
|
255
|
-
if command.kind_of?(CommandPattern)
|
256
|
-
@default = command
|
257
|
-
elsif command.kind_of?(Class)
|
258
|
-
@default = command.new
|
259
|
-
elsif @commands.has_key?(command)
|
260
|
-
@default = @commands[command]
|
261
|
-
else
|
262
|
-
raise UnknownCommandError
|
263
|
-
end
|
264
|
-
end
|
265
|
-
|
266
|
-
def command?(command)
|
267
|
-
@commands.has_key?(command)
|
268
|
-
end
|
269
|
-
|
270
|
-
def command(command)
|
271
|
-
if command?(command)
|
272
|
-
@commands[command]
|
273
|
-
else
|
274
|
-
@default
|
275
|
-
end
|
276
|
-
end
|
277
|
-
|
278
|
-
def [](cmd)
|
279
|
-
self.command(cmd)
|
280
|
-
end
|
281
|
-
|
282
|
-
def default_ioe(ioe = {})
|
283
|
-
ioe[:input] ||= $stdin
|
284
|
-
ioe[:output] ||= $stdout
|
285
|
-
ioe[:error] ||= $stderr
|
286
|
-
ioe
|
287
|
-
end
|
288
|
-
end
|
289
|
-
|
290
|
-
def [](args, opts = {}, ioe = {})
|
291
|
-
call(args, opts, ioe)
|
292
|
-
end
|
293
|
-
|
294
|
-
def name
|
295
|
-
raise AbstractCommandError
|
296
|
-
end
|
297
|
-
|
298
|
-
def call(args, opts = {}, ioe = {})
|
299
|
-
raise AbstractCommandError
|
300
|
-
end
|
301
|
-
|
302
|
-
def help
|
303
|
-
raise AbstractCommandError
|
304
|
-
end
|
305
|
-
end
|
306
|
-
|
307
|
-
class CommandHelp < CommandPattern
|
308
|
-
def name
|
309
|
-
"help"
|
310
|
-
end
|
311
|
-
|
312
|
-
def call(args, opts = {}, ioe = {})
|
313
|
-
ioe = CommandPattern.default_ioe(ioe)
|
314
|
-
|
315
|
-
help_on = args.shift
|
316
|
-
|
317
|
-
if CommandPattern.command?(help_on)
|
318
|
-
ioe[:output] << CommandPattern[help_on].help
|
319
|
-
elsif help_on == "commands"
|
320
|
-
ioe[:output] << <<-EOH
|
321
|
-
The commands known to minitar are:
|
322
|
-
|
323
|
-
minitar create Creates a new tarfile.
|
324
|
-
minitar extract Extracts files from a tarfile.
|
325
|
-
minitar list Lists files in the tarfile.
|
326
|
-
|
327
|
-
All commands accept the options --verbose and --progress, which are
|
328
|
-
mutually exclusive. In "minitar list", --progress means the same as
|
329
|
-
--verbose.
|
330
|
-
|
331
|
-
--verbose, -V Performs the requested command verbosely.
|
332
|
-
--progress, -P Shows a progress bar, if appropriate, for the action
|
333
|
-
being performed.
|
334
|
-
|
335
|
-
EOH
|
336
|
-
else
|
337
|
-
ioe[:output] << "Unknown command: #{help_on}\n" unless help_on.nil? or help_on.empty?
|
338
|
-
ioe[:output] << self.help
|
339
|
-
end
|
340
|
-
|
341
|
-
0
|
342
|
-
end
|
343
|
-
|
344
|
-
def help
|
345
|
-
help = <<-EOH
|
346
|
-
This is a basic help message containing pointers to more information on
|
347
|
-
how to use this command-line tool. Try:
|
348
|
-
|
349
|
-
minitar help commands list all 'minitar' commands
|
350
|
-
minitar help <COMMAND> show help on <COMMAND>
|
351
|
-
(e.g., 'minitar help create')
|
352
|
-
EOH
|
353
|
-
end
|
354
|
-
# minitar add Adds a file to an existing tarfile.
|
355
|
-
# minitar delete Deletes a file from an existing tarfile.
|
356
|
-
end
|
357
|
-
|
358
|
-
class CommandCreate < CommandPattern
|
359
|
-
def name
|
360
|
-
"create"
|
361
|
-
end
|
362
|
-
|
363
|
-
def altname
|
364
|
-
"cr"
|
365
|
-
end
|
366
|
-
|
367
|
-
def call(args, opts = {}, ioe = {})
|
368
|
-
argv = []
|
369
|
-
|
370
|
-
while (arg = args.shift)
|
371
|
-
case arg
|
372
|
-
when '--compress', '-z'
|
373
|
-
opts[:compress] = true
|
374
|
-
else
|
375
|
-
argv << arg
|
376
|
-
end
|
377
|
-
end
|
378
|
-
|
379
|
-
if argv.size < 2
|
380
|
-
ioe[:output] << "Not enough arguments.\n\n"
|
381
|
-
CommandPattern["help"][["create"]]
|
382
|
-
return 255
|
383
|
-
end
|
384
|
-
|
385
|
-
output = argv.shift
|
386
|
-
if '-' == output
|
387
|
-
opts[:name] = "STDOUT"
|
388
|
-
output = ioe[:output]
|
389
|
-
opts[:output] = ioe[:error]
|
390
|
-
else
|
391
|
-
opts[:name] = output
|
392
|
-
output = File.open(output, "wb")
|
393
|
-
opts[:output] = ioe[:output]
|
394
|
-
end
|
395
|
-
|
396
|
-
if opts[:name] =~ /\.tar\.gz$|\.tgz$/ or opts[:compress]
|
397
|
-
output = Zlib::GzipWriter.new(output)
|
398
|
-
end
|
399
|
-
|
400
|
-
files = []
|
401
|
-
if argv.include?("--")
|
402
|
-
# Read stdin for the list of files.
|
403
|
-
files = ""
|
404
|
-
files << ioe[:input].read while not ioe[:input].eof?
|
405
|
-
files = files.split(/\r\n|\n|\r/)
|
406
|
-
args.delete("--")
|
407
|
-
end
|
408
|
-
|
409
|
-
files << argv.to_a
|
410
|
-
files.flatten!
|
411
|
-
|
412
|
-
if opts[:verbose]
|
413
|
-
watcher = lambda do |action, name, stats|
|
414
|
-
opts[:output] << "#{name}\n" if action == :dir or action == :file_done
|
415
|
-
end
|
416
|
-
finisher = lambda { opts[:output] << "\n" }
|
417
|
-
elsif opts[:progress]
|
418
|
-
progress = ProgressBar.new(opts[:name], 1)
|
419
|
-
watcher = lambda do |action, name, stats|
|
420
|
-
case action
|
421
|
-
when :file_start, :dir
|
422
|
-
progress.title = File.basename(name)
|
423
|
-
if action == :dir
|
424
|
-
progress.total += 1
|
425
|
-
progress.inc
|
426
|
-
else
|
427
|
-
progress.total += stats[:size]
|
428
|
-
end
|
429
|
-
when :file_progress
|
430
|
-
progress.inc(stats[:currinc])
|
431
|
-
end
|
432
|
-
end
|
433
|
-
finisher = lambda do
|
434
|
-
progress.title = opts[:name]
|
435
|
-
progress.finish
|
436
|
-
end
|
437
|
-
else
|
438
|
-
watcher = nil
|
439
|
-
finisher = lambda { }
|
440
|
-
end
|
441
|
-
|
442
|
-
Archive::Tar::Minitar.pack(files, output, &watcher)
|
443
|
-
finisher.call
|
444
|
-
0
|
445
|
-
ensure
|
446
|
-
output.close if output and not output.closed?
|
447
|
-
end
|
448
|
-
|
449
|
-
def help
|
450
|
-
help = <<-EOH
|
451
|
-
minitar create [OPTIONS] <tarfile|-> <file|directory|-->+
|
452
|
-
|
453
|
-
Creates a new tarfile. If the tarfile is named .tar.gz or .tgz, then it
|
454
|
-
will be compressed automatically. If the tarfile is "-", then it will be
|
455
|
-
output to standard output (stdout) so that minitar may be piped.
|
456
|
-
|
457
|
-
The files or directories that will be packed into the tarfile are
|
458
|
-
specified after the name of the tarfile itself. Directories will be
|
459
|
-
processed recursively. If the token "--" is found in the list of files
|
460
|
-
to be packed, additional filenames will be read from standard input
|
461
|
-
(stdin). If any file is not found, the packaging will be halted.
|
462
|
-
|
463
|
-
create Options:
|
464
|
-
--compress, -z Compresses the tarfile with gzip.
|
465
|
-
|
466
|
-
EOH
|
467
|
-
end
|
468
|
-
end
|
469
|
-
|
470
|
-
class CommandExtract < CommandPattern
|
471
|
-
def name
|
472
|
-
"extract"
|
473
|
-
end
|
474
|
-
|
475
|
-
def altname
|
476
|
-
"ex"
|
477
|
-
end
|
478
|
-
|
479
|
-
def call(args, opts = {}, ioe = {})
|
480
|
-
argv = []
|
481
|
-
output = nil
|
482
|
-
dest = "."
|
483
|
-
files = []
|
484
|
-
|
485
|
-
while (arg = args.shift)
|
486
|
-
case arg
|
487
|
-
when '--uncompress', '-z'
|
488
|
-
opts[:uncompress] = true
|
489
|
-
when '--pipe'
|
490
|
-
opts[:output] = ioe[:error]
|
491
|
-
output = ioe[:output]
|
492
|
-
when '--output', '-o'
|
493
|
-
dest = args.shift
|
494
|
-
else
|
495
|
-
argv << arg
|
496
|
-
end
|
497
|
-
end
|
498
|
-
|
499
|
-
if argv.size < 1
|
500
|
-
ioe[:output] << "Not enough arguments.\n\n"
|
501
|
-
CommandPattern["help"][["extract"]]
|
502
|
-
return 255
|
503
|
-
end
|
504
|
-
|
505
|
-
input = argv.shift
|
506
|
-
if '-' == input
|
507
|
-
opts[:name] = "STDIN"
|
508
|
-
input = ioe[:input]
|
509
|
-
else
|
510
|
-
opts[:name] = input
|
511
|
-
input = File.open(input, "rb")
|
512
|
-
end
|
513
|
-
|
514
|
-
if opts[:name] =~ /\.tar\.gz$|\.tgz$/ or opts[:uncompress]
|
515
|
-
input = Zlib::GzipReader.new(input)
|
516
|
-
end
|
517
|
-
|
518
|
-
files << argv.to_a
|
519
|
-
files.flatten!
|
520
|
-
|
521
|
-
if opts[:verbose]
|
522
|
-
watcher = lambda do |action, name, stats|
|
523
|
-
opts[:output] << "#{name}\n" if action == :dir or action == :file_done
|
524
|
-
end
|
525
|
-
finisher = lambda { opts[:output] << "\n" }
|
526
|
-
elsif opts[:progress]
|
527
|
-
progress = ProgressBar.new(opts[:name], 1)
|
528
|
-
watcher = lambda do |action, name, stats|
|
529
|
-
case action
|
530
|
-
when :file_start, :dir
|
531
|
-
progress.title = File.basename(name)
|
532
|
-
if action == :dir
|
533
|
-
progress.total += 1
|
534
|
-
progress.inc
|
535
|
-
else
|
536
|
-
progress.total += stats[:entry].size
|
537
|
-
end
|
538
|
-
when :file_progress
|
539
|
-
progress.inc(stats[:currinc])
|
540
|
-
end
|
541
|
-
end
|
542
|
-
finisher = lambda do
|
543
|
-
progress.title = opts[:name]
|
544
|
-
progress.finish
|
545
|
-
end
|
546
|
-
else
|
547
|
-
watcher = nil
|
548
|
-
finisher = lambda { }
|
549
|
-
end
|
550
|
-
|
551
|
-
if output.nil?
|
552
|
-
Archive::Tar::Minitar.unpack(input, dest, files, &watcher)
|
553
|
-
finisher.call
|
554
|
-
else
|
555
|
-
Archive::Tar::Minitar::Input.open(input) do |inp|
|
556
|
-
inp.each do |entry|
|
557
|
-
stats = {
|
558
|
-
:mode => entry.mode,
|
559
|
-
:mtime => entry.mtime,
|
560
|
-
:size => entry.size,
|
561
|
-
:gid => entry.gid,
|
562
|
-
:uid => entry.uid,
|
563
|
-
:current => 0,
|
564
|
-
:currinc => 0,
|
565
|
-
:entry => entry
|
566
|
-
}
|
567
|
-
|
568
|
-
if files.empty? or files.include?(entry.full_name)
|
569
|
-
if entry.directory?
|
570
|
-
puts "Directory: #{entry.full_name}"
|
571
|
-
watcher[:dir, dest, stats] unless watcher.nil?
|
572
|
-
else
|
573
|
-
puts "File: #{entry.full_name}"
|
574
|
-
watcher[:file_start, destfile, stats] unless watcher.nil?
|
575
|
-
loop do
|
576
|
-
data = entry.read(4096)
|
577
|
-
break unless data
|
578
|
-
stats[:currinc] = output.write(data)
|
579
|
-
stats[:current] += stats[:currinc]
|
580
|
-
|
581
|
-
watcher[:file_progress, name, stats] unless watcher.nil?
|
582
|
-
end
|
583
|
-
watcher[:file_done, name, stats] unless watcher.nil?
|
584
|
-
end
|
585
|
-
end
|
586
|
-
end
|
587
|
-
end
|
588
|
-
end
|
589
|
-
|
590
|
-
0
|
591
|
-
end
|
592
|
-
|
593
|
-
def help
|
594
|
-
help = <<-EOH
|
595
|
-
minitar extract [OPTIONS] <tarfile|-> [<file>+]
|
596
|
-
|
597
|
-
Extracts files from an existing tarfile. If the tarfile is named .tar.gz
|
598
|
-
or .tgz, then it will be uncompressed automatically. If the tarfile is
|
599
|
-
"-", then it will be read from standard input (stdin) so that minitar
|
600
|
-
may be piped.
|
601
|
-
|
602
|
-
The files or directories that will be extracted from the tarfile are
|
603
|
-
specified after the name of the tarfile itself. Directories will be
|
604
|
-
processed recursively. Files must be specified in full. A file
|
605
|
-
"foo/bar/baz.txt" cannot simply be specified by specifying "baz.txt".
|
606
|
-
Any file not found will simply be skipped and an error will be reported.
|
607
|
-
|
608
|
-
extract Options:
|
609
|
-
--uncompress, -z Uncompresses the tarfile with gzip.
|
610
|
-
--pipe Emits the extracted files to STDOUT for piping.
|
611
|
-
--output, -o Extracts the files to the specified directory.
|
612
|
-
|
613
|
-
EOH
|
614
|
-
end
|
615
|
-
end
|
616
|
-
|
617
|
-
class CommandList < CommandPattern
|
618
|
-
def name
|
619
|
-
"list"
|
620
|
-
end
|
621
|
-
|
622
|
-
def altname
|
623
|
-
"ls"
|
624
|
-
end
|
625
|
-
|
626
|
-
def modestr(mode)
|
627
|
-
s = "---"
|
628
|
-
s[0] = ?r if (mode & 4) == 4
|
629
|
-
s[1] = ?w if (mode & 2) == 2
|
630
|
-
s[2] = ?x if (mode & 1) == 1
|
631
|
-
s
|
632
|
-
end
|
633
|
-
|
634
|
-
def call(args, opts = {}, ioe = {})
|
635
|
-
argv = []
|
636
|
-
output = nil
|
637
|
-
dest = "."
|
638
|
-
files = []
|
639
|
-
opts[:field] = "name"
|
640
|
-
|
641
|
-
while (arg = args.shift)
|
642
|
-
case arg
|
643
|
-
when '--sort', '-S'
|
644
|
-
opts[:sort] = true
|
645
|
-
opts[:field] = args.shift
|
646
|
-
when '--reverse', '-R'
|
647
|
-
opts[:reverse] = true
|
648
|
-
opts[:sort] = true
|
649
|
-
when '--uncompress', '-z'
|
650
|
-
opts[:uncompress] = true
|
651
|
-
when '-l'
|
652
|
-
opts[:verbose] = true
|
653
|
-
else
|
654
|
-
argv << arg
|
655
|
-
end
|
656
|
-
end
|
657
|
-
|
658
|
-
if argv.size < 1
|
659
|
-
ioe[:output] << "Not enough arguments.\n\n"
|
660
|
-
CommandPattern["help"][["list"]]
|
661
|
-
return 255
|
662
|
-
end
|
663
|
-
|
664
|
-
input = argv.shift
|
665
|
-
if '-' == input
|
666
|
-
opts[:name] = "STDIN"
|
667
|
-
input = ioe[:input]
|
668
|
-
else
|
669
|
-
opts[:name] = input
|
670
|
-
input = File.open(input, "rb")
|
671
|
-
end
|
672
|
-
|
673
|
-
if opts[:name] =~ /\.tar\.gz$|\.tgz$/ or opts[:uncompress]
|
674
|
-
input = Zlib::GzipReader.new(input)
|
675
|
-
end
|
676
|
-
|
677
|
-
files << argv.to_a
|
678
|
-
files.flatten!
|
679
|
-
|
680
|
-
if opts[:verbose] or opts[:progress]
|
681
|
-
format = "%10s %4d %8s %8s %8d %12s %s"
|
682
|
-
datefmt = "%b %d %Y"
|
683
|
-
timefmt = "%b %d %H:%M"
|
684
|
-
fields = %w(permissions inodes user group size date fullname)
|
685
|
-
else
|
686
|
-
format = "%s"
|
687
|
-
fields = %w(fullname)
|
688
|
-
end
|
689
|
-
|
690
|
-
opts[:field] = opts[:field].intern
|
691
|
-
opts[:field] = :full_name if opts[:field] == :name
|
692
|
-
|
693
|
-
output = []
|
694
|
-
|
695
|
-
Archive::Tar::Minitar::Input.open(input) do |inp|
|
696
|
-
today = Time.now
|
697
|
-
oneyear = Time.mktime(today.year - 1, today.month, today.day)
|
698
|
-
inp.each do |entry|
|
699
|
-
value = format % fields.map do |ff|
|
700
|
-
case ff
|
701
|
-
when "permissions"
|
702
|
-
s = entry.directory? ? "d" : "-"
|
703
|
-
s << modestr(entry.mode / 0100)
|
704
|
-
s << modestr(entry.mode / 0010)
|
705
|
-
s << modestr(entry.mode)
|
706
|
-
when "inodes"
|
707
|
-
entry.size / 512
|
708
|
-
when "user"
|
709
|
-
entry.uname || entry.uid || 0
|
710
|
-
when "group"
|
711
|
-
entry.gname || entry.gid || 0
|
712
|
-
when "size"
|
713
|
-
entry.size
|
714
|
-
when "date"
|
715
|
-
if Time.at(entry.mtime) > (oneyear)
|
716
|
-
Time.at(entry.mtime).strftime(timefmt)
|
717
|
-
else
|
718
|
-
Time.at(entry.mtime).strftime(datefmt)
|
719
|
-
end
|
720
|
-
when "fullname"
|
721
|
-
entry.full_name
|
722
|
-
end
|
723
|
-
end
|
724
|
-
|
725
|
-
if opts[:sort]
|
726
|
-
output << [entry.send(opts[:field]), value]
|
727
|
-
else
|
728
|
-
ioe[:output] << value << "\n"
|
729
|
-
end
|
730
|
-
|
731
|
-
end
|
732
|
-
end
|
733
|
-
|
734
|
-
if opts[:sort]
|
735
|
-
output = output.sort { |a, b| a[0] <=> b[0] }
|
736
|
-
if opts[:reverse]
|
737
|
-
output.reverse_each { |oo| ioe[:output] << oo[1] << "\n" }
|
738
|
-
else
|
739
|
-
output.each { |oo| ioe[:output] << oo[1] << "\n" }
|
740
|
-
end
|
741
|
-
end
|
742
|
-
|
743
|
-
0
|
744
|
-
end
|
745
|
-
|
746
|
-
def help
|
747
|
-
help = <<-EOH
|
748
|
-
minitar list [OPTIONS] <tarfile|-> [<file>+]
|
749
|
-
|
750
|
-
Lists files in an existing tarfile. If the tarfile is named .tar.gz or
|
751
|
-
.tgz, then it will be uncompressed automatically. If the tarfile is "-",
|
752
|
-
then it will be read from standard input (stdin) so that minitar may be
|
753
|
-
piped.
|
754
|
-
|
755
|
-
If --verbose or --progress is specified, then the file list will be
|
756
|
-
similar to that produced by the Unix command "ls -l".
|
757
|
-
|
758
|
-
list Options:
|
759
|
-
--uncompress, -z Uncompresses the tarfile with gzip.
|
760
|
-
--sort [<FIELD>], -S Sorts the list of files by the specified
|
761
|
-
field. The sort defaults to the filename.
|
762
|
-
--reverse, -R Reverses the sort.
|
763
|
-
-l Lists the files in detail.
|
764
|
-
|
765
|
-
Sort Fields:
|
766
|
-
name, mtime, size
|
767
|
-
|
768
|
-
EOH
|
769
|
-
end
|
770
|
-
end
|
771
|
-
|
772
|
-
CommandPattern << CommandHelp
|
773
|
-
CommandPattern << CommandCreate
|
774
|
-
CommandPattern << CommandExtract
|
775
|
-
CommandPattern << CommandList
|
776
|
-
# CommandPattern << CommandAdd
|
777
|
-
# CommandPattern << CommandDelete
|
778
|
-
|
779
|
-
def self.run(argv, input = $stdin, output = $stdout, error = $stderr)
|
780
|
-
ioe = {
|
781
|
-
:input => input,
|
782
|
-
:output => output,
|
783
|
-
:error => error,
|
784
|
-
}
|
785
|
-
opts = { }
|
786
|
-
|
787
|
-
if argv.include?("--version")
|
788
|
-
output << <<-EOB
|
789
|
-
minitar #{Archive::Tar::Minitar::VERSION}
|
790
|
-
Copyright 2004 Mauricio Julio Ferna'ndez Pradier and Austin Ziegler
|
791
|
-
This is free software with ABSOLUTELY NO WARRANTY.
|
792
|
-
|
793
|
-
see http://rubyforge.org/projects/ruwiki for more information
|
794
|
-
EOB
|
795
|
-
end
|
796
|
-
|
797
|
-
if argv.include?("--verbose") or argv.include?("-V")
|
798
|
-
opts[:verbose] = true
|
799
|
-
argv.delete("--verbose")
|
800
|
-
argv.delete("-V")
|
801
|
-
end
|
802
|
-
|
803
|
-
if argv.include?("--progress") or argv.include?("-P")
|
804
|
-
opts[:progress] = true
|
805
|
-
opts[:verbose] = false
|
806
|
-
argv.delete("--progress")
|
807
|
-
argv.delete("-P")
|
808
|
-
end
|
809
|
-
|
810
|
-
command = CommandPattern[(argv.shift or "").downcase]
|
811
|
-
command ||= CommandPattern["help"]
|
812
|
-
return command[argv, opts, ioe]
|
813
|
-
end
|
814
|
-
end
|