filerenamer 0.0.2 → 0.0.3
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.
- data/CHANGES +7 -1
- data/VERSION +1 -1
- data/bin/classify1stchar +8 -6
- data/bin/rennum +6 -5
- data/bin/renpad +48 -0
- data/bin/renpar +6 -5
- data/bin/renreg +5 -5
- data/bin/rensub +5 -6
- data/filerenamer.gemspec +34 -7
- data/lib/filerenamer.rb +4 -0
- data/lib/filerenamer/commander.rb +262 -0
- data/lib/filerenamer/optionparser.rb +35 -0
- data/test/classify1stchar/123 +0 -0
- data/test/classify1stchar/abc +0 -0
- data/test/classify1stchar/def +0 -0
- data/test/rennum/a +0 -0
- data/test/rennum/b +0 -0
- data/test/rennum/c +0 -0
- data/test/rennum/d +0 -0
- data/test/renpad/1 +0 -0
- data/test/renpad/10 +0 -0
- data/test/renpad/100 +0 -0
- data/test/renpad/100b +0 -0
- data/test/renpad/10b +0 -0
- data/test/renpad/1b +0 -0
- data/test/renpad/a1 +0 -0
- data/test/renpad/a10 +0 -0
- data/test/renpad/a100 +0 -0
- data/test/renpad/a100b +0 -0
- data/test/renpad/a10b +0 -0
- data/test/renpad/a1b +0 -0
- data/test/renreg/a(b) +0 -0
- data/test/renreg//357/274/221 +0 -0
- data/test/renreg//357/275/201 +0 -0
- data/test/rensub/a0 +0 -0
- data/test/rensub/a1 +0 -0
- data/test/rensub/b0 +0 -0
- data/test/rensub/b1 +0 -0
- data/test/{test_filerenamer.rb → test_commander.rb} +40 -76
- data/test/test_optionparser.rb +53 -0
- metadata +47 -19
- data/lib/filerenamer/filerenamer.rb +0 -252
- data/lib/filerenamer/filerenameroptionparser.rb +0 -36
- data/test/test_filerenameroptionparser.rb +0 -53
@@ -1,252 +0,0 @@
|
|
1
|
-
#! /usr/bin/env ruby
|
2
|
-
# coding: utf-8
|
3
|
-
|
4
|
-
require "fileutils"
|
5
|
-
require "optparse"
|
6
|
-
require "filerenamer/filerenameroptionparser.rb"
|
7
|
-
|
8
|
-
require "rubygems"
|
9
|
-
gem "builtinextension"
|
10
|
-
require "string_escape_zsh.rb"
|
11
|
-
require "string_width.rb"
|
12
|
-
|
13
|
-
class FileRenamerNoRenameRuleError < Exception; end
|
14
|
-
class FileRenamerOptionError < Exception; end
|
15
|
-
|
16
|
-
# ファイル群の名前変更を一括して行うためのクラス。
|
17
|
-
#
|
18
|
-
# MEMO:
|
19
|
-
# 一度 temporary directory にリネームして格納し、
|
20
|
-
# これをカレントディレクトリからのパスに移動する形を取ると、
|
21
|
-
# ディレクトリをまたぐリネームが非常に面倒なことになる。
|
22
|
-
# - 他のリネームに依存する場合に順序問題。
|
23
|
-
# - 相互依存のデッドロック
|
24
|
-
# gem などで提供される temporary directory は
|
25
|
-
# 基本的に 抜けたときに削除される筈なので、
|
26
|
-
# 途中まで変換してから interrupt されたときに
|
27
|
-
# 中にあるファイルごと消される可能性がありそう。
|
28
|
-
# そのため、メソッド内で自分で temporary directory を作成する。
|
29
|
-
#
|
30
|
-
# 没案
|
31
|
-
# rename_rule メソッドをデフォルトで例外にしておいて、
|
32
|
-
# コマンドスクリプト側で定義するという方法は、
|
33
|
-
# メソッドの引数で ArgumentError のもとになり、
|
34
|
-
# 自由度が少なくなる。
|
35
|
-
# たとえば old_str, new_str を渡したい時と、
|
36
|
-
# 更新時刻でリネームしたいときでは必要な引数の数が異なる。
|
37
|
-
class FileRenamer
|
38
|
-
|
39
|
-
#:yes と :no が両方 true ならば例外。
|
40
|
-
#:copy, :move, :hardlink, :symlink のうち、1つ以下が true。
|
41
|
-
#全て nil ならば :move が true になる。
|
42
|
-
#:quiet が true ならば自動的に :yes が立てられる。
|
43
|
-
#:quiet が true で :no も true ならば例外。
|
44
|
-
def initialize(options)
|
45
|
-
@options = options
|
46
|
-
|
47
|
-
if (@options[:yes] && @options[:no])
|
48
|
-
raise FileRenamerOptionError,
|
49
|
-
"Conflict options: --yes and --no."
|
50
|
-
end
|
51
|
-
|
52
|
-
fileProcessModes = []
|
53
|
-
[:move, :copy, :hardlink, :symlink].each do |mode|
|
54
|
-
fileProcessModes << mode if @options[mode]
|
55
|
-
end
|
56
|
-
# 1つもなければ :move に。
|
57
|
-
@options[:move] = true if fileProcessModes == []
|
58
|
-
# 2つ以上あれば例外。
|
59
|
-
if fileProcessModes.size > 1
|
60
|
-
raise FileRenamerOptionError,
|
61
|
-
"File process modes duplicate: #{fileProcessModes.join(", ")}"
|
62
|
-
end
|
63
|
-
@command = "mv" if @options[:move]
|
64
|
-
@command = "cp -r" if @options[:copy]
|
65
|
-
@command = "ln" if @options[:hardlink]
|
66
|
-
@command = "ln -s" if @options[:symlink]
|
67
|
-
|
68
|
-
# :quiet ならば自動的に :yes
|
69
|
-
@options[:yes] = true if @options[:quiet]
|
70
|
-
|
71
|
-
# :quiet と同時に :no なら例外
|
72
|
-
if @options[:quiet] && @options[:no]
|
73
|
-
raise FileRenamerOptionError,
|
74
|
-
"Conflict options: --quiet and --no"
|
75
|
-
end
|
76
|
-
end
|
77
|
-
|
78
|
-
# 変更される名前のリストを表示し、ユーザの指示に従って実行する。
|
79
|
-
# ユーザの入力は [y|Y] で始まる文字列ならば実行、
|
80
|
-
# それ以外なら実行しない。
|
81
|
-
# files は対象のファイル名のリスト。
|
82
|
-
# 新しい名前の生成方法をブロックで渡す。
|
83
|
-
# ブロックがなければ例外 FileRenamerNoRenameRuleError
|
84
|
-
def execute(files, &block)
|
85
|
-
files = Dir::glob("*").sort if files.empty?
|
86
|
-
|
87
|
-
new_names = make_new_names(files, &block)
|
88
|
-
|
89
|
-
ok_files, ng_files = check_new_names(new_names)
|
90
|
-
|
91
|
-
unless @options[:quiet]
|
92
|
-
if ok_files.size > 0
|
93
|
-
puts "Enable files:"
|
94
|
-
max_width = ok_files.keys.max_by{|file| file.width}.width
|
95
|
-
ok_files.keys.sort.each do |old|
|
96
|
-
printf(" %s %s%s %s\n",
|
97
|
-
@command, old, " " * (max_width - old.width), ok_files[old]
|
98
|
-
)
|
99
|
-
end
|
100
|
-
end
|
101
|
-
|
102
|
-
if ng_files.size > 0
|
103
|
-
puts "Unable files:"
|
104
|
-
max_width = ng_files.keys.max_by{|file| file.width}.width
|
105
|
-
ng_files.keys.sort.each do |old|
|
106
|
-
printf(" %s %s%s %s\n",
|
107
|
-
@command, old, " " * (max_width - old.width), ng_files[old]
|
108
|
-
)
|
109
|
-
end
|
110
|
-
puts
|
111
|
-
end
|
112
|
-
end
|
113
|
-
|
114
|
-
if ok_files.empty?
|
115
|
-
puts "Done. No executable files." unless @options[:quiet]
|
116
|
-
return
|
117
|
-
elsif @options[:no]
|
118
|
-
puts "Execute? no"
|
119
|
-
return
|
120
|
-
end
|
121
|
-
|
122
|
-
if @options[:yes]
|
123
|
-
puts "Execute? yes" unless @options[:quiet]
|
124
|
-
elsif (! ask_yes?)
|
125
|
-
return
|
126
|
-
end
|
127
|
-
ok_files.each do |old, new|
|
128
|
-
run(old, new)
|
129
|
-
end
|
130
|
-
end
|
131
|
-
|
132
|
-
private
|
133
|
-
|
134
|
-
def make_new_names(files, &block)
|
135
|
-
results = {}
|
136
|
-
files.each { |i| results[i] = yield i }
|
137
|
-
return results
|
138
|
-
end
|
139
|
-
|
140
|
-
# 新しい名前リストを受け取り、問題なくリネームできるものと
|
141
|
-
# できないものを選別する。
|
142
|
-
# OK, NG の順に2つの配列を返す。
|
143
|
-
#
|
144
|
-
# 判断基準は、以下の AND 条件。
|
145
|
-
# - 新しい名前が古い名前リストに含まれないこと。
|
146
|
-
# - 新しい名前が新しい名前リストに1つ(自分)のみしかないこと。
|
147
|
-
# - 新しい名前のファイルが既に存在しないこと。
|
148
|
-
#
|
149
|
-
# 没案
|
150
|
-
# 新しい名前が変換されない古い名前であれば、
|
151
|
-
# これも NG リストに追加する。(表示用)
|
152
|
-
# このルールにすると、変換に順序依存が生じる可能性があり、
|
153
|
-
# temporary directory を経由する必要ができる。
|
154
|
-
# その結果、リネーム過程で深い temporary directory を
|
155
|
-
# 作成したり削除したりしなければならなくなる。
|
156
|
-
def check_new_names(old_new_hash)
|
157
|
-
ok_files = {}
|
158
|
-
ng_files = {}
|
159
|
-
|
160
|
-
old_new_hash.each do |old, new|
|
161
|
-
# 変化がない場合はスルー
|
162
|
-
next if old == new
|
163
|
-
|
164
|
-
# 新しい名前に nil が定義されていてもスルー
|
165
|
-
next if new == nil
|
166
|
-
|
167
|
-
# 既に存在していれば、既に存在するファイルごと NG に登録。
|
168
|
-
if FileTest.exist?(new)
|
169
|
-
ng_files[old] = new
|
170
|
-
ng_files[new] = new
|
171
|
-
next
|
172
|
-
end
|
173
|
-
|
174
|
-
# new が複数 重複していれば、 NG。
|
175
|
-
if (old_new_hash.values.select{|name| name == new}.size > 1)
|
176
|
-
ng_files[old] = new
|
177
|
-
next
|
178
|
-
end
|
179
|
-
|
180
|
-
# new が old に含まれていれば、NG。
|
181
|
-
if (old_new_hash.keys.include?(new))
|
182
|
-
ng_files[old] = new
|
183
|
-
next
|
184
|
-
end
|
185
|
-
|
186
|
-
# 上記以外の場合ならば OK
|
187
|
-
|
188
|
-
ok_files[old] = new
|
189
|
-
end
|
190
|
-
return ok_files, ng_files
|
191
|
-
end
|
192
|
-
|
193
|
-
# ユーザに判断を求める。
|
194
|
-
# 標準入力から文字列を受け取り、
|
195
|
-
# [y|Y] から始まる文字列なら true を、
|
196
|
-
# それ以外なら false を返す。
|
197
|
-
def ask_yes?
|
198
|
-
puts "Execute?"
|
199
|
-
str = $stdin.gets
|
200
|
-
if /^y/i =~ str
|
201
|
-
return true
|
202
|
-
else
|
203
|
-
return false
|
204
|
-
end
|
205
|
-
end
|
206
|
-
|
207
|
-
# コマンドを表示しつつ実行。
|
208
|
-
# 既にファイルチェックされているはずなので、チェックはしない。
|
209
|
-
# 少なくとも当面は。
|
210
|
-
# ディレクトリが必要ならここで作成。
|
211
|
-
# ディレクトリが不要になればここで削除。
|
212
|
-
# 本来ここでない方が分かり易い気もするのだが、ここに追加するのが一番簡単なので。
|
213
|
-
def run(old, new)
|
214
|
-
# 変換先のディレクトリがなければ生成
|
215
|
-
#pp old, new; exit
|
216
|
-
paths(new).each do |directory|
|
217
|
-
unless FileTest.exist?(directory)
|
218
|
-
puts " make directory: #{directory}" unless @options[:quiet]
|
219
|
-
FileUtils.mkdir_p(directory)
|
220
|
-
end
|
221
|
-
end
|
222
|
-
|
223
|
-
# 変換の実行
|
224
|
-
command = " #{@command} #{old.escape_zsh} #{new.escape_zsh}"
|
225
|
-
puts command unless @options[:quiet]
|
226
|
-
system(command)
|
227
|
-
|
228
|
-
# 変換元のディレクトリが空になっていれば削除
|
229
|
-
dirs = paths(old)
|
230
|
-
dirs.reverse.each do |directory|
|
231
|
-
if Dir.entries(directory).size == 2 # . と .. のみ
|
232
|
-
puts " remove directory: #{directory}" unless @options[:quiet]
|
233
|
-
Dir.rmdir(directory)
|
234
|
-
end
|
235
|
-
end
|
236
|
-
end
|
237
|
-
|
238
|
-
# パスに含まれるディレクトリ名を、階層ごとに切り分けたソート済配列にして返す。
|
239
|
-
# 最後の要素は含まない。
|
240
|
-
# e.g. foo/bar/baz.txt => ["foo", "foo/bar"]
|
241
|
-
# e.g. /foo/bar/baz.txt => ["/foo", "foo/bar"]
|
242
|
-
def paths(path)
|
243
|
-
dirs = path.split("/")
|
244
|
-
results = []
|
245
|
-
(dirs.size - 1).times do |i|
|
246
|
-
results << dirs[0..i].join("/")
|
247
|
-
end
|
248
|
-
results.delete_at(0) if results[0] == ""
|
249
|
-
return results
|
250
|
-
end
|
251
|
-
end
|
252
|
-
|
@@ -1,36 +0,0 @@
|
|
1
|
-
#! /usr/bin/env ruby
|
2
|
-
# coding: utf-8
|
3
|
-
|
4
|
-
require "optparse"
|
5
|
-
|
6
|
-
# FileRenamer 用のオプション解析器。
|
7
|
-
# ユーザはこれに任意のオプションを追加できる。
|
8
|
-
#
|
9
|
-
# 通常の OptionParser と異なり、インスタンス変数 @options に情報を格納する。
|
10
|
-
# @options は attr_reader に指定されているので、外部から読み取れる。
|
11
|
-
# @options は Hash であるので、ユーザが on で追加するブロックで
|
12
|
-
# そのまま鍵と値を追加できる。
|
13
|
-
# また、@options に追加せずに自前で何か適当な処理をしても良い。
|
14
|
-
class FileRenamerOptionParser < OptionParser
|
15
|
-
|
16
|
-
attr_reader :options
|
17
|
-
|
18
|
-
def initialize
|
19
|
-
@options = {}
|
20
|
-
super
|
21
|
-
on("-y", "--yes" , "Yes for all questions."){@options[:yes] = true}
|
22
|
-
on("-n", "--no" , "No for all questions."){@options[:no] = true}
|
23
|
-
on("-c", "--copy" , "Copy mode." ){@options[:copy] = true}
|
24
|
-
on("-m", "--move" , "Move mode.(default)" ){@options[:move] = true}
|
25
|
-
on("-h", "--hardlink", "Hardlink mode." ){@options[:hardlink]= true}
|
26
|
-
on("-s", "--symlink" , "Symlink mode." ){@options[:symlink] = true}
|
27
|
-
on("-q", "--quiet" , "Quiet mode. Forced non-interactive."){
|
28
|
-
@options[:quiet] = true
|
29
|
-
#このオプションが設定されているときは強制的に --yes として扱われる。
|
30
|
-
#non_interactive_mode になる。
|
31
|
-
}
|
32
|
-
end
|
33
|
-
|
34
|
-
end
|
35
|
-
|
36
|
-
|
@@ -1,53 +0,0 @@
|
|
1
|
-
#! /usr/bin/env ruby
|
2
|
-
# coding: utf-8
|
3
|
-
|
4
|
-
require "test/unit"
|
5
|
-
require "filerenamer/filerenameroptionparser.rb"
|
6
|
-
|
7
|
-
class TC_FileRenamerOptionParser < Test::Unit::TestCase
|
8
|
-
#def setup
|
9
|
-
# @frop00 = FileRenamerOptionParser.new
|
10
|
-
#end
|
11
|
-
|
12
|
-
def test_sanity_check
|
13
|
-
frop01 = FileRenamerOptionParser.new
|
14
|
-
ary = %w(-y --copy)
|
15
|
-
frop01.parse(ary)
|
16
|
-
assert_equal( { :yes => true, :copy => true }, frop01.options)
|
17
|
-
|
18
|
-
# このクラスではオプションの整合性チェックは行わない。
|
19
|
-
frop01 = FileRenamerOptionParser.new
|
20
|
-
ary = %w(-y -n)
|
21
|
-
frop01.parse(ary)
|
22
|
-
assert_equal( { :yes => true, :no => true }, frop01.options)
|
23
|
-
|
24
|
-
frop01 = FileRenamerOptionParser.new
|
25
|
-
ary = %w(-y -n -c -m -h -s -q)
|
26
|
-
frop01.parse(ary)
|
27
|
-
corrects = {
|
28
|
-
:yes => true,
|
29
|
-
:no => true,
|
30
|
-
:copy => true,
|
31
|
-
:move => true,
|
32
|
-
:hardlink => true,
|
33
|
-
:symlink => true,
|
34
|
-
:quiet => true
|
35
|
-
}
|
36
|
-
assert_equal(corrects, frop01.options)
|
37
|
-
|
38
|
-
frop01 = FileRenamerOptionParser.new
|
39
|
-
ary = %w(--yes --no --copy --move --hardlink --symlink --quiet)
|
40
|
-
frop01.parse(ary)
|
41
|
-
corrects = {
|
42
|
-
:yes => true,
|
43
|
-
:no => true,
|
44
|
-
:copy => true,
|
45
|
-
:move => true,
|
46
|
-
:hardlink => true,
|
47
|
-
:symlink => true,
|
48
|
-
:quiet => true
|
49
|
-
}
|
50
|
-
assert_equal(corrects, frop01.options)
|
51
|
-
end
|
52
|
-
end
|
53
|
-
|