-
Notifications
You must be signed in to change notification settings - Fork 32
/
claide_spec.rb
315 lines (259 loc) · 10 KB
/
claide_spec.rb
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
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
require 'bacon'
require 'mocha-on-bacon'
$:.unshift File.expand_path('../../lib', __FILE__)
require 'claide'
def should_raise_help(error_message)
error = nil
begin
yield
rescue CLAide::Help => e
error = e
end
error.should.not == nil
error.error_message.should == error_message
end
module CLAide
describe ARGV do
it "converts objects into strings while parsing" do
flag = stub(:to_s => '--flag')
arg = stub(:to_s => 'ARG')
ARGV.new([flag, arg]).remainder.should == %w{ --flag ARG }
end
it "only removes one entry when calling shift_argument" do
argv = ARGV.new(%w{ ARG ARG })
argv.shift_argument
argv.remainder.should == %w{ ARG }
end
before do
@argv = ARGV.new(%w{ --flag --option=VALUE ARG1 ARG2 --no-other-flag })
end
it "returns the options as a hash" do
@argv.options.should == {
'flag' => true,
'other-flag' => false,
'option' => 'VALUE'
}
end
it "returns the arguments" do
@argv.arguments.should == %w{ ARG1 ARG2 }
end
it "returns a flag and deletes it" do
@argv.flag?('flag').should == true
@argv.flag?('other-flag').should == false
@argv.flag?('option').should == nil
@argv.remainder.should == %w{ --option=VALUE ARG1 ARG2 }
end
it "returns a default value if a flag does not exist" do
@argv.flag?('option', true).should == true
@argv.flag?('option', false).should == false
end
it "returns an option and deletes it" do
@argv.option('flag').should == nil
@argv.option('other-flag').should == nil
@argv.option('option').should == 'VALUE'
@argv.remainder.should == %w{ --flag ARG1 ARG2 --no-other-flag }
end
it "returns a default value if an option does not exist" do
@argv.option('flag', 'value').should == 'value'
end
it "returns the first argument and deletes it" do
@argv.shift_argument.should == 'ARG1'
@argv.remainder.should == %w{ --flag --option=VALUE ARG2 --no-other-flag }
end
it "returns and deletes all arguments" do
@argv.arguments!.should == %w{ ARG1 ARG2 }
@argv.remainder.should == %w{ --flag --option=VALUE --no-other-flag }
end
end
end
module Fixture
class Error < StandardError
include CLAide::InformativeError
end
class Command < CLAide::Command
self.command = 'bin'
class SpecFile < Command
self.abstract_command = true
self.description = 'Manage spec files.'
class Lint < SpecFile
self.summary = 'Checks the validity of a spec file.'
self.arguments = '[NAME]'
def self.options
[['--only-errors', 'Skip warnings']].concat(super)
end
class Repo < Lint
self.summary = 'Checks the validity of ALL specs in a repo.'
end
end
class Create < SpecFile
self.summary = 'Creates a spec file stub.'
self.description = <<-DESC
Creates a spec file called NAME
and populates it with defaults.
DESC
self.arguments = '[NAME]'
attr_reader :spec
def initialize(argv)
@spec = argv.shift_argument
super
end
end
end
end
end
module CLAide
describe Command do
it "registers the subcommand classes" do
Fixture::Command.subcommands.map(&:command).should == %w{ spec-file }
Fixture::Command::SpecFile.subcommands.map(&:command).should == %w{ lint create }
Fixture::Command::SpecFile::Create.subcommands.map(&:command).should == []
Fixture::Command::SpecFile::Lint.subcommands.map(&:command).should == %w{ repo }
end
it "tries to match a subclass for each of the subcommands" do
#Fixture::Command.parse(%w{ spec-file }).should.be.instance_of Fixture::Command::SpecFile
Fixture::Command.parse(%w{ spec-file --verbose lint }).should.be.instance_of Fixture::Command::SpecFile::Lint
#Fixture::Command.parse(%w{ spec-file lint --help repo }).should.be.instance_of Fixture::Command::SpecFile::Lint::Repo
end
# TODO might be more the task of the application?
#it "raises a Help exception when run without any subcommands" do
#lambda { Fixture::Command.run([]) }.should.raise Command::Help
#end
end
describe Command, "validation" do
it "does not raise if one of the subcommands consumes arguments" do
subcommand = Fixture::Command.parse(%w{ spec-file create AFNetworking })
subcommand.spec.should == 'AFNetworking'
end
it "raises a Help exception when created with an invalid subcommand" do
should_raise_help 'Unknown arguments: unknown' do
Fixture::Command.parse(%w{ unknown }).validate!
end
should_raise_help 'Unknown arguments: unknown' do
Fixture::Command.parse(%w{ spec-file unknown }).validate!
end
end
it "raises a Help exception (without error message) when called on an abstract command" do
should_raise_help nil do
Fixture::Command.parse(%w{ spec-file }).validate!
end
end
end
describe Command, "default options" do
it "raises a Help exception, without error message" do
should_raise_help nil do
Fixture::Command.parse(%w{ --help }).validate!
end
end
it "sets the verbose flag" do
command = Fixture::Command.parse([])
command.should.not.be.verbose
command = Fixture::Command.parse(%w{ --verbose })
command.should.be.verbose
end
end
describe Command, "when running" do
before do
Fixture::Command.stubs(:puts)
Fixture::Command.stubs(:exit)
end
it "does not print the backtrace of a InformativeError exception by default" do
expected = Help.new(Fixture::Command.parse([])).message
Fixture::Command.expects(:puts).with(expected)
Fixture::Command.run(%w{ --help })
end
it "does print the backtrace of an exception, that includes InformativeError, if set to verbose" do
error = Fixture::Error.new
Fixture::Command.any_instance.stubs(:validate!).raises(error)
error.stubs(:message).returns('the message')
error.stubs(:backtrace).returns(['the', 'backtrace'])
printed = states('printed').starts_as(:nothing)
Fixture::Command.expects(:puts).with('the message').when(printed.is(:nothing)).then(printed.is(:message))
Fixture::Command.expects(:puts).with('the', 'backtrace').when(printed.is(:message)).then(printed.is(:done))
Fixture::Command.run(%w{ --verbose })
end
it "exits with a failure status when an exception that includes InformativeError occurs" do
Fixture::Command.expects(:exit).with(1)
Fixture::Command.any_instance.stubs(:validate!).raises(Fixture::Error.new)
Fixture::Command.run([])
end
it "exits with a failure status when a Help exception occurs that has an error message" do
Fixture::Command.expects(:exit).with(1)
Fixture::Command.run(%w{ unknown })
end
it "exits with a success status when a Help exception occurs that has *no* error message" do
Fixture::Command.expects(:exit).with(0)
Fixture::Command.run(%w{ --help })
end
#it "exits with a failure status when any other type of exception occurs" do
#Fixture::Command.expects(:exit).with(1)
#Fixture::Command.any_instance.stubs(:validate!).raises(ArgumentError.new)
#Fixture::Command.run([])
#end
end
describe Command, "banner formatting in general" do
it "returns a 'usage' description based on the command's description" do
Fixture::Command::SpecFile::Create.parse([]).formatted_usage_description.should == <<-USAGE.rstrip
$ bin spec-file create [NAME]
Creates a spec file called NAME
and populates it with defaults.
USAGE
end
it "returns a 'usage' description based on the command's summary, if no description is present" do
Fixture::Command::SpecFile::Lint::Repo.parse([]).formatted_usage_description.should == <<-USAGE.rstrip
$ bin spec-file lint repo
Checks the validity of ALL specs in a repo.
USAGE
end
it "returns summaries of the subcommands of a command, sorted by name" do
Fixture::Command::SpecFile.parse([]).formatted_subcommand_summaries.should == <<-COMMANDS.rstrip
* create Creates a spec file stub.
* lint Checks the validity of a spec file.
COMMANDS
end
it "returns the options, for all ancestor commands, aligned so they're all aligned with the largest option name" do
Fixture::Command::SpecFile.parse([]).formatted_options_description.should == <<-OPTIONS.rstrip
--verbose Show more debugging information
--help Show help banner of specified command
OPTIONS
Fixture::Command::SpecFile::Lint::Repo.parse([]).formatted_options_description.should == <<-OPTIONS.rstrip
--only-errors Skip warnings
--verbose Show more debugging information
--help Show help banner of specified command
OPTIONS
end
end
describe Command, "complete banner formatting" do
it "does not include a 'usage' banner for an abstract command" do
command = Fixture::Command::SpecFile.parse([])
command.formatted_banner.should == <<-BANNER.rstrip
Manage spec files.
Commands:
#{command.formatted_subcommand_summaries}
Options:
#{command.formatted_options_description}
BANNER
end
it "shows a banner that is a combination of the summary/description, commands, and options" do
command = Fixture::Command::SpecFile::Create.parse([])
command.formatted_banner.should == <<-BANNER.rstrip
Usage:
#{command.formatted_usage_description}
Options:
#{command.formatted_options_description}
BANNER
end
end
describe Help, "formatting for a command" do
it "shows just the banner if no error message is specified" do
command = Fixture::Command.parse([])
Help.new(command).message.should == command.formatted_banner
end
it "shows the specified error message before the rest of the banner" do
command = Fixture::Command.parse([])
Help.new(command, "Unable to process, captain.").message.should == <<-BANNER.rstrip
[!] Unable to process, captain.
#{command.formatted_banner}
BANNER
end
end
end