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.
@@ -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
-