file_permissions 0.1.0
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.
- checksums.yaml +7 -0
- data/.coveralls.yml +1 -0
- data/.gitignore +12 -0
- data/.rspec +3 -0
- data/.travis.yml +49 -0
- data/.yardopts +9 -0
- data/Gemfile +25 -0
- data/HISTORY.md +6 -0
- data/LICENSE.md +15 -0
- data/README.md +391 -0
- data/Rakefile +48 -0
- data/file_permissions.gemspec +67 -0
- data/lib/file_permissions.rb +22 -0
- data/lib/god_object/file_permissions.rb +48 -0
- data/lib/god_object/file_permissions/complex_mode.rb +259 -0
- data/lib/god_object/file_permissions/exceptions.rb +33 -0
- data/lib/god_object/file_permissions/helper_mixin.rb +48 -0
- data/lib/god_object/file_permissions/mode.rb +156 -0
- data/lib/god_object/file_permissions/mode_mixin.rb +163 -0
- data/lib/god_object/file_permissions/special_mode.rb +149 -0
- data/lib/god_object/file_permissions/version.rb +30 -0
- data/spec/god_object/posix_mode/complex_mode_spec.rb +345 -0
- data/spec/god_object/posix_mode/mode_spec.rb +396 -0
- data/spec/god_object/posix_mode/special_mode_spec.rb +372 -0
- data/spec/spec_helper.rb +39 -0
- metadata +201 -0
@@ -0,0 +1,30 @@
|
|
1
|
+
# encoding: UTF-8
|
2
|
+
=begin
|
3
|
+
Copyright GodObject Team <dev@godobject.net>, 2012-2016
|
4
|
+
|
5
|
+
This file is part of FilePermissions.
|
6
|
+
|
7
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
8
|
+
purpose with or without fee is hereby granted, provided that the above
|
9
|
+
copyright notice and this permission notice appear in all copies.
|
10
|
+
|
11
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
12
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
13
|
+
FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
14
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
15
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
16
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
17
|
+
PERFORMANCE OF THIS SOFTWARE.
|
18
|
+
=end
|
19
|
+
|
20
|
+
module GodObject
|
21
|
+
module FilePermissions
|
22
|
+
|
23
|
+
# The currently loaded version.
|
24
|
+
#
|
25
|
+
# Using Semantic Versioning (2.0.0) rules
|
26
|
+
# @see http://semver.org/spec/v2.0.0.html
|
27
|
+
VERSION = '0.1.0'.freeze
|
28
|
+
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,345 @@
|
|
1
|
+
# encoding: UTF-8
|
2
|
+
=begin
|
3
|
+
Copyright GodObject Team <dev@godobject.net>, 2012-2016
|
4
|
+
|
5
|
+
This file is part of FilePermissions.
|
6
|
+
|
7
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
8
|
+
purpose with or without fee is hereby granted, provided that the above
|
9
|
+
copyright notice and this permission notice appear in all copies.
|
10
|
+
|
11
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
12
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
13
|
+
FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
14
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
15
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
16
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
17
|
+
PERFORMANCE OF THIS SOFTWARE.
|
18
|
+
=end
|
19
|
+
|
20
|
+
module GodObject
|
21
|
+
module FilePermissions
|
22
|
+
|
23
|
+
describe ComplexMode do
|
24
|
+
describe ".build" do
|
25
|
+
it "should return the same object if a ComplexMode is given" do
|
26
|
+
existing_mode = ComplexMode.new(5)
|
27
|
+
|
28
|
+
mode = ComplexMode.build(existing_mode)
|
29
|
+
|
30
|
+
expect(mode).to equal existing_mode
|
31
|
+
end
|
32
|
+
|
33
|
+
it "should create a new instance if given something else" do
|
34
|
+
argument = [:user_read, :user_write, :group_read, :other_execute, :sticky]
|
35
|
+
|
36
|
+
expect(ComplexMode).to receive(:new).once.with(argument)
|
37
|
+
|
38
|
+
ComplexMode.build(argument)
|
39
|
+
end
|
40
|
+
end
|
41
|
+
|
42
|
+
describe ".new" do
|
43
|
+
it "should handle an octal representation" do
|
44
|
+
complex_mode = ComplexMode.new(00610)
|
45
|
+
expect(complex_mode.user).to eql Mode.parse('rw')
|
46
|
+
expect(complex_mode.group).to eql Mode.parse('x')
|
47
|
+
end
|
48
|
+
|
49
|
+
it "should handle a list of mode components" do
|
50
|
+
complex_mode = ComplexMode.new(
|
51
|
+
:user_read, :user_write, :user_execute, :group_read, :group_execute, :other_execute, :setgid, :sticky)
|
52
|
+
|
53
|
+
expect(complex_mode.user).to eql Mode.parse('rwx')
|
54
|
+
expect(complex_mode.group).to eql Mode.parse('rx')
|
55
|
+
expect(complex_mode.other).to eql Mode.parse('x')
|
56
|
+
expect(complex_mode.special).to eql SpecialMode.parse('-st')
|
57
|
+
end
|
58
|
+
|
59
|
+
it "should handle a Set of mode components" do
|
60
|
+
mode = ComplexMode.new(Set[:user_read, :group_read, :group_execute, :setuid, :sticky])
|
61
|
+
|
62
|
+
expect(mode.user.read?).to eql true
|
63
|
+
expect(mode.user.write?).to eql false
|
64
|
+
expect(mode.user.execute?).to eql false
|
65
|
+
|
66
|
+
expect(mode.group.read?).to eql true
|
67
|
+
expect(mode.group.write?).to eql false
|
68
|
+
expect(mode.group.execute?).to eql true
|
69
|
+
|
70
|
+
expect(mode.other.read?).to eql false
|
71
|
+
expect(mode.other.write?).to eql false
|
72
|
+
expect(mode.other.execute?).to eql false
|
73
|
+
|
74
|
+
expect(mode.special.setuid?).to eql true
|
75
|
+
expect(mode.special.setgid?).to eql false
|
76
|
+
expect(mode.special.sticky?).to eql true
|
77
|
+
end
|
78
|
+
|
79
|
+
it "should complain about invalid input" do
|
80
|
+
expect {
|
81
|
+
ComplexMode.new(:wrong, :user_execute)
|
82
|
+
}.to raise_error(ArgumentError)
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
describe ".from_file" do
|
87
|
+
before(:each) do
|
88
|
+
@test_directory = Pathname.new(Dir.mktmpdir('file_permissions_spec'))
|
89
|
+
end
|
90
|
+
|
91
|
+
after(:each) do
|
92
|
+
@test_directory.rmtree
|
93
|
+
end
|
94
|
+
|
95
|
+
it "should create a complex mode from file" do
|
96
|
+
test_file = (@test_directory + 'test_file')
|
97
|
+
FileUtils.touch(test_file)
|
98
|
+
test_file.chmod(05641)
|
99
|
+
|
100
|
+
complex_mode = ComplexMode.from_file(test_file)
|
101
|
+
expect(complex_mode.to_i).to eql 05641
|
102
|
+
end
|
103
|
+
|
104
|
+
it "should create a complex mode from file given as string" do
|
105
|
+
test_file = (@test_directory + 'test_file')
|
106
|
+
FileUtils.touch(test_file)
|
107
|
+
test_file.chmod(05641)
|
108
|
+
|
109
|
+
complex_mode = ComplexMode.from_file(test_file.to_s)
|
110
|
+
expect(complex_mode.to_i).to eql 05641
|
111
|
+
end
|
112
|
+
|
113
|
+
it "should create a complex mode from the resolved symlink" do
|
114
|
+
test_file = (@test_directory + 'test_file')
|
115
|
+
FileUtils.touch(test_file)
|
116
|
+
test_file.chmod(05641)
|
117
|
+
|
118
|
+
test_link = (@test_directory + 'test_link')
|
119
|
+
test_link.make_symlink(test_file)
|
120
|
+
|
121
|
+
complex_mode = ComplexMode.from_file(test_link)
|
122
|
+
expect(complex_mode.to_i).to eql 05641
|
123
|
+
end
|
124
|
+
|
125
|
+
it "should create a complex mode from the symlink itself if in 'target symlinks' mode" do
|
126
|
+
test_file = (@test_directory + 'test_file')
|
127
|
+
FileUtils.touch(test_file)
|
128
|
+
test_file.chmod(05641)
|
129
|
+
|
130
|
+
test_link = (@test_directory + 'test_link')
|
131
|
+
test_link.make_symlink(test_file)
|
132
|
+
|
133
|
+
complex_mode = ComplexMode.from_file(test_link, :target_symlinks)
|
134
|
+
expect(complex_mode.to_i).to eql 00777
|
135
|
+
end
|
136
|
+
|
137
|
+
it "should complain about an invalid symlink handling" do
|
138
|
+
test_file = (@test_directory + 'test_file')
|
139
|
+
FileUtils.touch(test_file)
|
140
|
+
|
141
|
+
expect {
|
142
|
+
ComplexMode.from_file(test_file, :invalid)
|
143
|
+
}.to raise_error(ArgumentError, "Invalid symlink handling: :invalid")
|
144
|
+
end
|
145
|
+
end
|
146
|
+
|
147
|
+
describe "#user" do
|
148
|
+
it "should return return the user's attributes as mode if setuid is set" do
|
149
|
+
complex_mode = ComplexMode.new(07710)
|
150
|
+
expect(complex_mode.user).to eql Mode.parse('rwx')
|
151
|
+
end
|
152
|
+
|
153
|
+
it "should return return the user's attributes as mode if setuid is unset" do
|
154
|
+
complex_mode = ComplexMode.new(00510)
|
155
|
+
expect(complex_mode.user).to eql Mode.parse('rx')
|
156
|
+
end
|
157
|
+
end
|
158
|
+
|
159
|
+
describe "#group" do
|
160
|
+
it "should return return the group's attributes as mode if setgid is set" do
|
161
|
+
complex_mode = ComplexMode.new(07770)
|
162
|
+
expect(complex_mode.group).to eql Mode.parse('rwx')
|
163
|
+
end
|
164
|
+
|
165
|
+
it "should return return the group's attributes as mode if setgid is unset" do
|
166
|
+
complex_mode = ComplexMode.new(00530)
|
167
|
+
expect(complex_mode.group).to eql Mode.parse('wx')
|
168
|
+
end
|
169
|
+
end
|
170
|
+
|
171
|
+
describe "#assign_to_file" do
|
172
|
+
before(:each) do
|
173
|
+
@test_directory = Pathname.new(Dir.mktmpdir('file_permissions_spec'))
|
174
|
+
end
|
175
|
+
|
176
|
+
after(:each) do
|
177
|
+
@test_directory.rmtree
|
178
|
+
end
|
179
|
+
|
180
|
+
it "should assign a complex mode to file" do
|
181
|
+
test_file = (@test_directory + 'test_file')
|
182
|
+
FileUtils.touch(test_file)
|
183
|
+
|
184
|
+
complex_mode = ComplexMode.new(05641)
|
185
|
+
complex_mode.assign_to_file(test_file)
|
186
|
+
|
187
|
+
expect(test_file.stat.mode & 0b111_111_111_111).to eql 05641
|
188
|
+
end
|
189
|
+
|
190
|
+
it "should assign a complex mode to file" do
|
191
|
+
test_file = (@test_directory + 'test_file')
|
192
|
+
FileUtils.touch(test_file)
|
193
|
+
|
194
|
+
complex_mode = ComplexMode.new(05641)
|
195
|
+
complex_mode.assign_to_file(test_file.to_s)
|
196
|
+
|
197
|
+
expect(test_file.stat.mode & 0b111_111_111_111).to eql 05641
|
198
|
+
end
|
199
|
+
|
200
|
+
it "should assign a complex mode to the resolved symlink" do
|
201
|
+
test_file = (@test_directory + 'test_file')
|
202
|
+
FileUtils.touch(test_file)
|
203
|
+
|
204
|
+
test_link = (@test_directory + 'test_link')
|
205
|
+
test_link.make_symlink(test_file)
|
206
|
+
|
207
|
+
complex_mode = ComplexMode.new(05641)
|
208
|
+
complex_mode.assign_to_file(test_link)
|
209
|
+
|
210
|
+
expect(test_file.stat.mode & 0b111_111_111_111).to eql 05641
|
211
|
+
end
|
212
|
+
|
213
|
+
it "should complain about missing lchmod function in 'target symlinks' mode" do
|
214
|
+
test_file = (@test_directory + 'test_file')
|
215
|
+
FileUtils.touch(test_file)
|
216
|
+
|
217
|
+
test_link = (@test_directory + 'test_link')
|
218
|
+
test_link.make_symlink(test_file)
|
219
|
+
|
220
|
+
complex_mode = ComplexMode.new(05641)
|
221
|
+
|
222
|
+
expect {
|
223
|
+
complex_mode.assign_to_file(test_link, :target_symlinks)
|
224
|
+
}.to raise_error(NotImplementedError, "lchmod function is not available in current OS or Ruby environment")
|
225
|
+
end
|
226
|
+
|
227
|
+
it "should assign a complex mode to the symlink itself in 'target symlinks' mode" do
|
228
|
+
pending "This test is intended for systems where there is a lchmod function. No idea which system that may be. Have fun."
|
229
|
+
|
230
|
+
test_file = (@test_directory + 'test_file')
|
231
|
+
FileUtils.touch(test_file)
|
232
|
+
|
233
|
+
test_link = (@test_directory + 'test_link')
|
234
|
+
test_link.make_symlink(test_file)
|
235
|
+
|
236
|
+
complex_mode = ComplexMode.new(05641)
|
237
|
+
complex_mode.assign_to_file(test_link, :target_symlinks)
|
238
|
+
|
239
|
+
expect(test_file.stat.mode & 0b111_111_111_111).to eql 05641
|
240
|
+
end
|
241
|
+
|
242
|
+
it "should complain about an invalid symlink handling" do
|
243
|
+
test_file = (@test_directory + 'test_file')
|
244
|
+
FileUtils.touch(test_file)
|
245
|
+
|
246
|
+
complex_mode = ComplexMode.new(05641)
|
247
|
+
|
248
|
+
expect {
|
249
|
+
complex_mode.assign_to_file(test_file, :invalid)
|
250
|
+
}.to raise_error(ArgumentError, "Invalid symlink handling: :invalid")
|
251
|
+
end
|
252
|
+
end
|
253
|
+
|
254
|
+
describe "#inspect" do
|
255
|
+
it "should represent the user, group and other read and write flags" do
|
256
|
+
complex_mode = ComplexMode.new(00765)
|
257
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "rwxrw-r-x">'
|
258
|
+
end
|
259
|
+
|
260
|
+
it "should represent the setuid flag if user execute flag is set" do
|
261
|
+
complex_mode = ComplexMode.new(04715)
|
262
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "rws--xr-x">'
|
263
|
+
end
|
264
|
+
|
265
|
+
it "should represent the setuid flag if user execute flag is unset" do
|
266
|
+
complex_mode = ComplexMode.new(04615)
|
267
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "rwS--xr-x">'
|
268
|
+
end
|
269
|
+
|
270
|
+
it "should represent the setgid flag if group execute flag is set" do
|
271
|
+
complex_mode = ComplexMode.new(02615)
|
272
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "rw---sr-x">'
|
273
|
+
end
|
274
|
+
|
275
|
+
it "should represent the setgid flag if group execute flag is unset" do
|
276
|
+
complex_mode = ComplexMode.new(02605)
|
277
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "rw---Sr-x">'
|
278
|
+
end
|
279
|
+
|
280
|
+
it "should represent the sticky flag if other execute flag is set" do
|
281
|
+
complex_mode = ComplexMode.new(01435)
|
282
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "r---wxr-t">'
|
283
|
+
end
|
284
|
+
|
285
|
+
it "should represent the sticky flag if other execute flag is unset" do
|
286
|
+
complex_mode = ComplexMode.new(07004)
|
287
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "--S--Sr-T">'
|
288
|
+
end
|
289
|
+
end
|
290
|
+
|
291
|
+
describe "#inspect" do
|
292
|
+
it "should give a decent string representation for debugging" do
|
293
|
+
complex_mode = ComplexMode.new(07004)
|
294
|
+
expect(complex_mode.inspect).to eql '#<GodObject::FilePermissions::ComplexMode: "--S--Sr-T">'
|
295
|
+
end
|
296
|
+
end
|
297
|
+
|
298
|
+
describe "#to_s" do
|
299
|
+
it "should represent the user, group and other read and write flags" do
|
300
|
+
complex_mode = ComplexMode.new(00765)
|
301
|
+
expect(complex_mode.to_s).to eql 'rwxrw-r-x'
|
302
|
+
end
|
303
|
+
|
304
|
+
it "should represent the setuid flag if user execute flag is set" do
|
305
|
+
complex_mode = ComplexMode.new(04715)
|
306
|
+
expect(complex_mode.to_s).to eql 'rws--xr-x'
|
307
|
+
end
|
308
|
+
|
309
|
+
it "should represent the setuid flag if user execute flag is unset" do
|
310
|
+
complex_mode = ComplexMode.new(04615)
|
311
|
+
expect(complex_mode.to_s).to eql 'rwS--xr-x'
|
312
|
+
end
|
313
|
+
|
314
|
+
it "should represent the setgid flag if group execute flag is set" do
|
315
|
+
complex_mode = ComplexMode.new(02615)
|
316
|
+
expect(complex_mode.to_s).to eql 'rw---sr-x'
|
317
|
+
end
|
318
|
+
|
319
|
+
it "should represent the setgid flag if group execute flag is unset" do
|
320
|
+
complex_mode = ComplexMode.new(02605)
|
321
|
+
expect(complex_mode.to_s).to eql 'rw---Sr-x'
|
322
|
+
end
|
323
|
+
|
324
|
+
it "should represent the sticky flag if other execute flag is set" do
|
325
|
+
complex_mode = ComplexMode.new(01435)
|
326
|
+
expect(complex_mode.to_s).to eql 'r---wxr-t'
|
327
|
+
end
|
328
|
+
|
329
|
+
it "should represent the sticky flag if other execute flag is unset" do
|
330
|
+
complex_mode = ComplexMode.new(07004)
|
331
|
+
expect(complex_mode.to_s).to eql '--S--Sr-T'
|
332
|
+
end
|
333
|
+
end
|
334
|
+
|
335
|
+
describe "#to_i" do
|
336
|
+
it "should return the correct octal representation" do
|
337
|
+
complex_mode = ComplexMode.new(02615)
|
338
|
+
|
339
|
+
expect(complex_mode.to_i).to eql 02615
|
340
|
+
end
|
341
|
+
end
|
342
|
+
end
|
343
|
+
|
344
|
+
end
|
345
|
+
end
|
@@ -0,0 +1,396 @@
|
|
1
|
+
# encoding: UTF-8
|
2
|
+
=begin
|
3
|
+
Copyright GodObject Team <dev@godobject.net>, 2012-2016
|
4
|
+
|
5
|
+
This file is part of FilePermissions.
|
6
|
+
|
7
|
+
Permission to use, copy, modify, and/or distribute this software for any
|
8
|
+
purpose with or without fee is hereby granted, provided that the above
|
9
|
+
copyright notice and this permission notice appear in all copies.
|
10
|
+
|
11
|
+
THE SOFTWARE IS PROVIDED "AS IS" AND THE AUTHOR DISCLAIMS ALL WARRANTIES WITH
|
12
|
+
REGARD TO THIS SOFTWARE INCLUDING ALL IMPLIED WARRANTIES OF MERCHANTABILITY AND
|
13
|
+
FITNESS. IN NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY SPECIAL, DIRECT,
|
14
|
+
INDIRECT, OR CONSEQUENTIAL DAMAGES OR ANY DAMAGES WHATSOEVER RESULTING FROM
|
15
|
+
LOSS OF USE, DATA OR PROFITS, WHETHER IN AN ACTION OF CONTRACT, NEGLIGENCE OR
|
16
|
+
OTHER TORTIOUS ACTION, ARISING OUT OF OR IN CONNECTION WITH THE USE OR
|
17
|
+
PERFORMANCE OF THIS SOFTWARE.
|
18
|
+
=end
|
19
|
+
|
20
|
+
module GodObject
|
21
|
+
module FilePermissions
|
22
|
+
|
23
|
+
describe Mode do
|
24
|
+
describe ".build" do
|
25
|
+
it "should return the same object if a Mode is given" do
|
26
|
+
existing_mode = Mode.new(5)
|
27
|
+
|
28
|
+
mode = Mode.build(existing_mode)
|
29
|
+
|
30
|
+
expect(mode).to equal existing_mode
|
31
|
+
end
|
32
|
+
|
33
|
+
it "should create a new instance through parsing if given a String" do
|
34
|
+
argument = 's-t'
|
35
|
+
|
36
|
+
expect(Mode).to receive(:parse).once.with(argument)
|
37
|
+
|
38
|
+
Mode.build(argument)
|
39
|
+
end
|
40
|
+
|
41
|
+
it "should create a new instance if given something else" do
|
42
|
+
argument = 4
|
43
|
+
|
44
|
+
expect(Mode).to receive(:new).once.with(argument)
|
45
|
+
|
46
|
+
Mode.build(argument)
|
47
|
+
end
|
48
|
+
end
|
49
|
+
|
50
|
+
describe ".parse" do
|
51
|
+
context "given an octal representation" do
|
52
|
+
it "should return an empty set if a 0 is given" do
|
53
|
+
expect(Mode.parse('0')).to eql Mode.new(Set[])
|
54
|
+
end
|
55
|
+
|
56
|
+
it "should return a set including the execute token if a 1 is given" do
|
57
|
+
expect(Mode.parse('1')).to eql Mode.new(Set[:execute])
|
58
|
+
end
|
59
|
+
|
60
|
+
it "should return a set including the write token if a 2 is given" do
|
61
|
+
expect(Mode.parse('2')).to eql Mode.new(Set[:write])
|
62
|
+
end
|
63
|
+
|
64
|
+
it "should return a set including the write and execute tokens if a 3 is given" do
|
65
|
+
expect(Mode.parse('3')).to eql Mode.new(Set[:write, :execute])
|
66
|
+
end
|
67
|
+
|
68
|
+
it "should return a set including the read token if a 4 is given" do
|
69
|
+
expect(Mode.parse('4')).to eql Mode.new(Set[:read])
|
70
|
+
end
|
71
|
+
|
72
|
+
it "should return a set including the read and execute tokens if a 5 is given" do
|
73
|
+
expect(Mode.parse('5')).to eql Mode.new(Set[:read, :execute])
|
74
|
+
end
|
75
|
+
|
76
|
+
it "should return a set including the read and write tokens if a 6 is given" do
|
77
|
+
expect(Mode.parse('6')).to eql Mode.new(Set[:read, :write])
|
78
|
+
end
|
79
|
+
|
80
|
+
it "should return a set including the read, write and execute tokens if a 7 is given" do
|
81
|
+
expect(Mode.parse('7')).to eql Mode.new(Set[:read, :write, :execute])
|
82
|
+
end
|
83
|
+
|
84
|
+
it "should raise an exception if 8 is given" do
|
85
|
+
expect {
|
86
|
+
Mode.parse('8')
|
87
|
+
}.to raise_error(ParserError, 'Invalid format')
|
88
|
+
end
|
89
|
+
end
|
90
|
+
|
91
|
+
context "representation in symbolic mode" do
|
92
|
+
it "should complain about duplicate read symbols" do
|
93
|
+
expect {
|
94
|
+
Mode.parse('r-r')
|
95
|
+
}.to raise_error(ParserError, 'Duplicate digit in: "r-r"')
|
96
|
+
end
|
97
|
+
|
98
|
+
it "should complain about duplicate write symbols" do
|
99
|
+
expect {
|
100
|
+
Mode.parse('www')
|
101
|
+
}.to raise_error(ParserError, 'Duplicate digit in: "www"')
|
102
|
+
end
|
103
|
+
|
104
|
+
it "should complain about duplicate execute symbols" do
|
105
|
+
expect {
|
106
|
+
Mode.parse('xx-')
|
107
|
+
}.to raise_error(ParserError, 'Duplicate digit in: "xx-"')
|
108
|
+
end
|
109
|
+
|
110
|
+
it "should complain about invalid symbols" do
|
111
|
+
expect {
|
112
|
+
Mode.parse('r-a')
|
113
|
+
}.to raise_error(ParserError, 'Invalid format')
|
114
|
+
end
|
115
|
+
|
116
|
+
it "should not complain about duplicate null symbols" do
|
117
|
+
expect {
|
118
|
+
Mode.parse('-r-')
|
119
|
+
}.not_to raise_error
|
120
|
+
end
|
121
|
+
|
122
|
+
it "should parse the read symbol" do
|
123
|
+
result = Mode.parse('r')
|
124
|
+
expect(result).to eql Mode.new(Set[:read])
|
125
|
+
end
|
126
|
+
|
127
|
+
it "should parse the write symbol" do
|
128
|
+
result = Mode.parse('w')
|
129
|
+
expect(result).to eql Mode.new(Set[:write])
|
130
|
+
end
|
131
|
+
|
132
|
+
it "should parse the execute symbol" do
|
133
|
+
result = Mode.parse('x')
|
134
|
+
expect(result).to eql Mode.new(Set[:execute])
|
135
|
+
end
|
136
|
+
end
|
137
|
+
end
|
138
|
+
|
139
|
+
describe ".new" do
|
140
|
+
it "should handle no parameters" do
|
141
|
+
mode = Mode.new
|
142
|
+
expect(mode).to be_a(Mode)
|
143
|
+
expect(mode.read?).to eql false
|
144
|
+
expect(mode.write?).to eql false
|
145
|
+
expect(mode.execute?).to eql false
|
146
|
+
expect(mode.to_i).to eql 0
|
147
|
+
expect(mode.to_s).to eql "---"
|
148
|
+
expect(mode.to_s(:short)).to eql "-"
|
149
|
+
attributes = { read: false, write: false, execute: false }
|
150
|
+
expect(mode.state).to eql attributes
|
151
|
+
end
|
152
|
+
|
153
|
+
it "should handle a Set" do
|
154
|
+
mode = Mode.new(Set[:read, :execute])
|
155
|
+
|
156
|
+
expect(mode.state).to eql(read: true, write: false, execute: true)
|
157
|
+
expect(mode.read?).to eql true
|
158
|
+
expect(mode.write?).to eql false
|
159
|
+
expect(mode.execute?).to eql true
|
160
|
+
end
|
161
|
+
|
162
|
+
it "should handle an array of mode components" do
|
163
|
+
mode = Mode.new(:read, :write, :execute)
|
164
|
+
attributes = { read: true, write: true, execute: true }
|
165
|
+
|
166
|
+
expect(mode.state).to eql attributes
|
167
|
+
end
|
168
|
+
end
|
169
|
+
|
170
|
+
let(:empty_mode) { Mode.parse('---') }
|
171
|
+
let(:x_mode) { Mode.parse('--x') }
|
172
|
+
let(:w_mode) { Mode.parse('-w-') }
|
173
|
+
let(:wx_mode) { Mode.parse('-wx') }
|
174
|
+
let(:r_mode) { Mode.parse('r--') }
|
175
|
+
let(:rx_mode) { Mode.parse('r-x') }
|
176
|
+
let(:rw_mode) { Mode.parse('rw-') }
|
177
|
+
let(:rwx_mode) { Mode.parse('rwx') }
|
178
|
+
|
179
|
+
describe "#==" do
|
180
|
+
it "should be true if compare to itself" do
|
181
|
+
expect(empty_mode).to eq(empty_mode)
|
182
|
+
end
|
183
|
+
|
184
|
+
it "should be false if read attributes differ" do
|
185
|
+
expect(rx_mode).not_to eq(x_mode)
|
186
|
+
end
|
187
|
+
|
188
|
+
it "should be false if write attributes differ" do
|
189
|
+
expect(wx_mode).not_to eq(x_mode)
|
190
|
+
end
|
191
|
+
|
192
|
+
it "should be false if execute attributes differ" do
|
193
|
+
expect(wx_mode).not_to eq(w_mode)
|
194
|
+
end
|
195
|
+
|
196
|
+
it "should be true if compared to a Mode with same attributes" do
|
197
|
+
expect(empty_mode).to eq(Mode.new)
|
198
|
+
end
|
199
|
+
|
200
|
+
it "should be true if compared to an ACL-like with the same entries?" do
|
201
|
+
other = OpenStruct.new(
|
202
|
+
to_i: 6, configuration: OpenStruct.new(digits: [:read, :write, :execute])
|
203
|
+
)
|
204
|
+
|
205
|
+
expect(rw_mode).to eq(other)
|
206
|
+
end
|
207
|
+
end
|
208
|
+
|
209
|
+
describe "#eql?" do
|
210
|
+
it "should be true if compare to itself" do
|
211
|
+
expect(Mode.new).to eql Mode.new
|
212
|
+
end
|
213
|
+
|
214
|
+
it "should be false if attributes are the same but class differs" do
|
215
|
+
other = OpenStruct.new(attribute: { read: false, write: false, execute: false })
|
216
|
+
expect(Mode.new).not_to eql other
|
217
|
+
end
|
218
|
+
end
|
219
|
+
|
220
|
+
describe "#<=>" do
|
221
|
+
it "should return -1 if the compared Mode has a higher octal representation" do
|
222
|
+
expect(w_mode <=> r_mode).to eql -1
|
223
|
+
end
|
224
|
+
|
225
|
+
it "should return 1 if the compared Mode has a lower octal representation" do
|
226
|
+
expect(r_mode <=> w_mode).to eql 1
|
227
|
+
end
|
228
|
+
|
229
|
+
it "should return 0 if the compared Mode has an equal octal representation" do
|
230
|
+
expect(x_mode <=> x_mode).to eql 0
|
231
|
+
end
|
232
|
+
|
233
|
+
it "should return nil if the compared object is incompatible" do
|
234
|
+
expect(w_mode <=> :something).to eql nil
|
235
|
+
end
|
236
|
+
end
|
237
|
+
|
238
|
+
describe "#inspect" do
|
239
|
+
it "should give a decent string representation for debugging" do
|
240
|
+
expect(rwx_mode.inspect).to eq("#<#{subject.class}: \"rwx\">")
|
241
|
+
expect(rw_mode.inspect).to eq("#<#{subject.class}: \"rw-\">")
|
242
|
+
expect(rx_mode.inspect).to eq("#<#{subject.class}: \"r-x\">")
|
243
|
+
expect(wx_mode.inspect).to eq("#<#{subject.class}: \"-wx\">")
|
244
|
+
expect(r_mode.inspect).to eq("#<#{subject.class}: \"r--\">")
|
245
|
+
expect(w_mode.inspect).to eq("#<#{subject.class}: \"-w-\">")
|
246
|
+
expect(x_mode.inspect).to eq("#<#{subject.class}: \"--x\">")
|
247
|
+
expect(empty_mode.inspect).to eq("#<#{subject.class}: \"---\">")
|
248
|
+
end
|
249
|
+
end
|
250
|
+
|
251
|
+
describe "#to_s" do
|
252
|
+
it "should represent attributes as string in long mode" do
|
253
|
+
expect(rwx_mode.to_s(:long)).to eq("rwx")
|
254
|
+
expect(rw_mode.to_s(:long)).to eq("rw-")
|
255
|
+
expect(rx_mode.to_s(:long)).to eq("r-x")
|
256
|
+
expect(wx_mode.to_s(:long)).to eq("-wx")
|
257
|
+
expect(r_mode.to_s(:long)).to eq("r--")
|
258
|
+
expect(w_mode.to_s(:long)).to eq("-w-")
|
259
|
+
expect(x_mode.to_s(:long)).to eq("--x")
|
260
|
+
expect(empty_mode.to_s(:long)).to eq("---")
|
261
|
+
end
|
262
|
+
|
263
|
+
it "should represent attributes as string in short mode" do
|
264
|
+
expect(rwx_mode.to_s(:short)).to eq("rwx")
|
265
|
+
expect(rw_mode.to_s(:short)).to eq("rw")
|
266
|
+
expect(rx_mode.to_s(:short)).to eq("rx")
|
267
|
+
expect(wx_mode.to_s(:short)).to eq("wx")
|
268
|
+
expect(r_mode.to_s(:short)).to eq("r")
|
269
|
+
expect(w_mode.to_s(:short)).to eq("w")
|
270
|
+
expect(x_mode.to_s(:short)).to eq("x")
|
271
|
+
expect(empty_mode.to_s(:short)).to eq("-")
|
272
|
+
end
|
273
|
+
end
|
274
|
+
|
275
|
+
describe "#to_i" do
|
276
|
+
it "should represent no attributes in octal" do
|
277
|
+
expect(empty_mode.to_i).to eql 0
|
278
|
+
end
|
279
|
+
|
280
|
+
it "should represent execute attribute in octal" do
|
281
|
+
expect(x_mode.to_i).to eql 1
|
282
|
+
end
|
283
|
+
|
284
|
+
it "should represent write attribute in octal" do
|
285
|
+
expect(w_mode.to_i).to eql 2
|
286
|
+
end
|
287
|
+
|
288
|
+
it "should represent execute and write attributes in octal" do
|
289
|
+
expect(wx_mode.to_i).to eql 3
|
290
|
+
end
|
291
|
+
|
292
|
+
it "should represent read attribute in octal" do
|
293
|
+
expect(r_mode.to_i).to eql 4
|
294
|
+
end
|
295
|
+
|
296
|
+
it "should represent read and execute attributes in octal" do
|
297
|
+
expect(rx_mode.to_i).to eql 5
|
298
|
+
end
|
299
|
+
|
300
|
+
it "should represent read and write attributes in octal" do
|
301
|
+
expect(rw_mode.to_i).to eql 6
|
302
|
+
end
|
303
|
+
|
304
|
+
it "should represent read, write and execute attributes in octal" do
|
305
|
+
expect(rwx_mode.to_i).to eql 7
|
306
|
+
end
|
307
|
+
end
|
308
|
+
|
309
|
+
describe "#read?" do
|
310
|
+
it "should be true if read attribute is set" do
|
311
|
+
expect(r_mode.read?).to eql true
|
312
|
+
end
|
313
|
+
|
314
|
+
it "should be false if read attribute is not set" do
|
315
|
+
expect(wx_mode.read?).to eql false
|
316
|
+
end
|
317
|
+
end
|
318
|
+
|
319
|
+
describe "#write?" do
|
320
|
+
it "should be true if write attribute is set" do
|
321
|
+
expect(wx_mode.write?).to eql true
|
322
|
+
end
|
323
|
+
|
324
|
+
it "should be false if write attribute is not set" do
|
325
|
+
expect(rx_mode.write?).to eql false
|
326
|
+
end
|
327
|
+
end
|
328
|
+
|
329
|
+
describe "#execute?" do
|
330
|
+
it "should be true if execute attribute is set" do
|
331
|
+
expect(rx_mode.execute?).to eql true
|
332
|
+
end
|
333
|
+
|
334
|
+
it "should be false if execute attribute is not set" do
|
335
|
+
expect(empty_mode.execute?).to eql false
|
336
|
+
end
|
337
|
+
end
|
338
|
+
|
339
|
+
describe "#invert" do
|
340
|
+
it "should create a new Mode with all digits inverted" do
|
341
|
+
result = rx_mode.invert
|
342
|
+
|
343
|
+
expect(result).not_to equal rx_mode
|
344
|
+
expect(result).to eql w_mode
|
345
|
+
end
|
346
|
+
end
|
347
|
+
|
348
|
+
describe "#-" do
|
349
|
+
it "should create a new Mode from the first operand without the digits of the second operand" do
|
350
|
+
result = rwx_mode - w_mode
|
351
|
+
|
352
|
+
expect(result).not_to equal rwx_mode
|
353
|
+
expect(result).not_to equal w_mode
|
354
|
+
expect(result).to eql rx_mode
|
355
|
+
end
|
356
|
+
end
|
357
|
+
|
358
|
+
[:intersection, :&].each do |method_name|
|
359
|
+
describe "##{method_name}" do
|
360
|
+
it "should create a new Mode with only those digits enabled that are enabled in both operands" do
|
361
|
+
result = rw_mode.public_send(method_name, wx_mode)
|
362
|
+
|
363
|
+
expect(result).not_to equal rw_mode
|
364
|
+
expect(result).not_to equal wx_mode
|
365
|
+
expect(result).to eql w_mode
|
366
|
+
end
|
367
|
+
end
|
368
|
+
end
|
369
|
+
|
370
|
+
[:union, :|, :+].each do |method_name|
|
371
|
+
describe "##{method_name}" do
|
372
|
+
it "should create a new Mode with all enabled digits of both operands" do
|
373
|
+
result = r_mode.public_send(method_name, w_mode)
|
374
|
+
|
375
|
+
expect(result).not_to equal r_mode
|
376
|
+
expect(result).not_to equal w_mode
|
377
|
+
expect(result).to eql rw_mode
|
378
|
+
end
|
379
|
+
end
|
380
|
+
end
|
381
|
+
|
382
|
+
[:symmetric_difference, :^].each do |method_name|
|
383
|
+
describe "##{method_name}" do
|
384
|
+
it "should create a new Mode with only those digits enabled that are enabled in only one operand" do
|
385
|
+
result = rwx_mode.public_send(method_name, x_mode)
|
386
|
+
|
387
|
+
expect(result).not_to equal rwx_mode
|
388
|
+
expect(result).not_to equal x_mode
|
389
|
+
expect(result).to eql rw_mode
|
390
|
+
end
|
391
|
+
end
|
392
|
+
end
|
393
|
+
end
|
394
|
+
|
395
|
+
end
|
396
|
+
end
|