sleeping_king_studios-tools 1.1.1 → 1.2.0.rc.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.
data/README.md CHANGED
@@ -1,10 +1,17 @@
1
- # SleepingKingStudios::Tools [![Build Status](https://travis-ci.org/sleepingkingstudios/sleeping_king_studios-tools.svg?branch=master)](https://travis-ci.org/sleepingkingstudios/sleeping_king_studios-tools)
1
+ # SleepingKingStudios::Tools
2
2
 
3
3
  A library of utility services and concerns to expand the functionality of core classes without polluting the global namespace.
4
4
 
5
+ <blockquote>
6
+ Read The
7
+ <a href="https://www.sleepingkingstudios.com/sleeping_king_studios-tools" target="_blank">
8
+ Documentation
9
+ </a>
10
+ </blockquote>
11
+
5
12
  ## About
6
13
 
7
- SleepingKingStudios::Tools is tested against MRI Ruby 2.7 through 3.2.
14
+ SleepingKingStudios::Tools is tested against MRI Ruby 3.1 through 3.4.
8
15
 
9
16
  ### Documentation
10
17
 
@@ -23,1179 +30,3 @@ The canonical repository for this gem is on [GitHub](https://github.com/sleeping
23
30
  ### Code of Conduct
24
31
 
25
32
  Please note that the `SleepingKingStudios::Tools` project is released with a [Contributor Code of Conduct](https://github.com/sleepingkingstudios/sleeping_king_studios-tools/blob/master/CODE_OF_CONDUCT.md). By contributing to this project, you agree to abide by its terms.
26
-
27
- ## Tools
28
-
29
- ### Toolbelt
30
-
31
- The tools can be accessed in a convenient form using the Toolbelt class.
32
-
33
- ```ruby
34
- require 'sleeping_king_studios/tools'
35
-
36
- tools = ::SleepingKingStudios::Tools::Toolbelt.instance
37
-
38
- tools.ary.humanize_list 'one', 'two', 'three'
39
- #=> calls ArrayTools#humanize_list
40
-
41
- tools.core.deprecate 'my_method'
42
- #=> calls CoreTools#deprecate
43
-
44
- tools.str.underscore 'MyModuleName'
45
- #=> calls StringTools#underscore
46
- ```
47
-
48
- ### Array Tools
49
-
50
- require 'sleeping_king_studios/tools/array_tools'
51
-
52
- Tools for working with array-like enumerable objects.
53
-
54
- #### `#array?`
55
-
56
- Returns true if the object is or appears to be an Array.
57
-
58
- ArrayTools.array?(nil)
59
- #=> false
60
- ArrayTools.array?([])
61
- #=> true
62
- ArrayTools.array?({})
63
- #=> false
64
-
65
- #### `#bisect`
66
-
67
- Separates the array into two arrays, the first containing all items in the original array that matches the provided block, and the second containing all items in the original array that do not match the provided block.
68
-
69
- original = [0, 1, 2, 3, 4, 5, 6, 7, 8, 9]
70
- selected, rejected = ArrayTools.bisect(original) { |item| item.even? }
71
- selected
72
- #=> [0, 2, 4, 6, 8]
73
- rejected
74
- #=> [1, 3, 5, 7, 9]
75
-
76
- #### `#count_values`
77
-
78
- Counts the number of times each value appears in the array, or if a block is given, calls the block with each item and counts the number of times each result appears.
79
-
80
- ArrayTools.count_values([1, 1, 1, 2, 2, 3])
81
- #=> { 1 => 3, 2 => 2, 3 => 1 }
82
-
83
- ArrayTools.count_values([1, 1, 1, 2, 2, 3]) { |i| i ** 2 }
84
- #=> { 1 => 3, 4 => 2, 9 => 1 }
85
-
86
- ArrayTools.count_values([1, 1, 1, 2, 2, 3], &:even?)
87
- #=> { false => 4, true => 2 }
88
-
89
- #### `#deep_dup`
90
-
91
- Creates a deep copy of the object by returning a new Array with deep copies of each array item. See also ObjectTools#deep_dup[#label-Object+Tools].
92
-
93
- ary = ['one', 'two', 'three']
94
- cpy = ArrayTools.deep_dup ary
95
-
96
- cpy << 'four'
97
- #=> ['one', 'two', 'three', 'four']
98
- ary
99
- #=> ['one', 'two', 'three']
100
-
101
- cpy.first.sub!(/on/, 'vu'); cpy
102
- #=> ['vun', 'two', 'three', 'four']
103
- ary
104
- #=> ['one', 'two', 'three']
105
-
106
- #### `#deep_freeze`
107
-
108
- Freezes the array and performs a deep freeze on each array item. See also ObjectTools#deep_freeze[#label-Object+Tools].
109
-
110
- ary = ['one', 'two', 'three']
111
- ArrayTools.deep_freeze ary
112
-
113
- ary.frozen?
114
- #=> true
115
- ary.first.frozen?
116
- #=> true
117
-
118
- #### `#humanize_list`
119
-
120
- Accepts a list of values and returns a human-readable string of the values, with the format based on the number of items.
121
-
122
- # With One Item
123
- ArrayTools.humanize_list(['spam'])
124
- #=> 'spam'
125
-
126
- # With Two Items
127
- ArrayTools.humanize_list(['spam', 'eggs'])
128
- #=> 'spam and eggs'
129
-
130
- # With Three Or More Items
131
- ArrayTools.humanize_list(['spam', 'eggs', 'bacon', 'spam'])
132
- #=> 'spam, eggs, bacon, and spam'
133
-
134
- # With A Block
135
- ArrayTools.humanize_list(['bronze', 'silver', 'gold'], { |str| str.capitalize })
136
- #=> 'Bronze, Silver, and Gold'
137
-
138
- #### `#immutable?`
139
-
140
- Returns true if the array is immutable, i.e. the array is frozen and each array item is immutable.
141
-
142
- ArrayTools.immutable?([1, 2, 3])
143
- #=> false
144
-
145
- ArrayTools.immutable?([1, 2, 3].freeze)
146
- #=> true
147
-
148
- ArrayTools.immutable?(['ichi', 'ni', 'san'])
149
- #=> false
150
-
151
- ArrayTools.immutable?(['ichi', 'ni', 'san'].freeze)
152
- #=> false
153
-
154
- ArrayTools.immutable?(['ichi'.freeze, 'ni'.freeze, 'san'.freeze].freeze)
155
- #=> true
156
-
157
- #### `#mutable?`
158
-
159
- Returns true if the array is mutable (see `#immutable?`, above).
160
-
161
- #### `#splice`
162
-
163
- Accepts an array, a start value, a number of items to delete, and zero or more items to insert at that index. Deletes the specified number of items, then inserts the given items at the index and returns the array of deleted items.
164
-
165
- # Deleting items from an Array
166
- values = %w(katana wakizashi tachi daito shoto)
167
- ArrayTools.splice values, 1, 2
168
- #=> ['wakizashi', 'tachi']
169
- values
170
- #=> ['katana', 'daito', 'shoto']
171
-
172
- # Inserting items into an Array
173
- values = %w(longsword broadsword claymore)
174
- ArrayTools.splice values, 1, 0, 'zweihänder'
175
- #=> []
176
- values
177
- #=> ['longsword', 'zweihänder', 'broadsword', 'claymore']
178
-
179
- # Inserting and deleting items
180
- values = %w(shortbow longbow crossbow)
181
- ArrayTools.splice values, 2, 1, 'arbalest', 'chu-ko-nu'
182
- #=> ['crossbow']
183
- values
184
- #=> ['shortbow', 'longbow', 'arbalest', 'chu-ko-nu']
185
-
186
- ### Assertions
187
-
188
- Tools for validating the current application state.
189
-
190
- #### `#assert`
191
-
192
- Raises an exception unless the given block returns a truthy value.
193
-
194
- ```ruby
195
- Assertions.assert { true == false }
196
- #=> raises an AssertionError with message 'block returned a falsy value'
197
-
198
- Assertions.assert { true == true }
199
- #=> does not raise an exception
200
- ```
201
-
202
- It accepts the following options:
203
-
204
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
205
- - `message`: The error message to display.
206
-
207
- #### `#assert_boolean`
208
-
209
- Raises an exception unless the given value is either `true` or `false`.
210
-
211
- ```ruby
212
- Assertions.assert_boolean(nil)
213
- #=> raises an AssertionError with message 'value must be true or false'
214
-
215
- Assertions.assert_boolean(Object.new)
216
- #=> raises an AssertionError with message 'value must be true or false'
217
-
218
- Assertions.assert_boolean(false)
219
- #=> does not raise an exception
220
-
221
- Assertions.assert_boolean(true)
222
- #=> does not raise an exception
223
- ```
224
-
225
- It accepts the following options:
226
-
227
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
228
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
229
- - `message`: The error message to display.
230
-
231
- #### `#assert_class`
232
-
233
- Raises an exception unless the given value is a Class.
234
-
235
- ```ruby
236
- Assertions.assert_class(Object.new)
237
- #=> raises an AssertionError with message 'value is not a class'
238
-
239
- Assertions.assert_class(String)
240
- #=> does not raise an exception
241
- ```
242
-
243
- It accepts the following options:
244
-
245
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
246
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
247
- - `message`: The error message to display.
248
-
249
- #### `#assert_instance_of`
250
-
251
- Raises an exception unless the given value is an instance of the expected Class or Module.
252
-
253
- ```ruby
254
- Assertions.assert_instance_of(:foo, expected: String)
255
- #=> raises an AssertionError with message 'value is not an instance of String'
256
-
257
- Assertions.assert_instance_of('foo', expected: String)
258
- #=> does not raise an exception
259
- ```
260
-
261
- It accepts the following options:
262
-
263
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
264
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
265
- - `message`: The error message to display.
266
- - `optional`: If true, accepts `nil` values as well as instances of the Class or Module.
267
-
268
- #### `#assert_matches`
269
-
270
- Raises an exception unless the given value matches the expected value using case equality (`#===`).
271
-
272
-
273
- ```ruby
274
- Assertions.assert_matches('bar', expected: /foo/)
275
- #=> raises an AssertionError with message 'value does not match the pattern /foo/'
276
-
277
- Assertions.assert_matches('foo', expected: /foo/)
278
- #=> does not raise an exception
279
- ```
280
-
281
- It accepts the following options:
282
-
283
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
284
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
285
- - `message`: The error message to display.
286
- - `optional`: If true, accepts `nil` values as well as matching values.
287
-
288
- #### `#assert_name`
289
-
290
- Raises an exception unless the given value is non-empty a String or Symbol.
291
-
292
- ```ruby
293
- Assertions.assert_name(nil)
294
- #=> raises an AssertionError with message "value can't be blank"
295
-
296
- Assertions.assert_name(Object.new)
297
- #=> raises an AssertionError with message 'value is not a String or a Symbol'
298
-
299
- Assertions.assert_name('')
300
- #=> raises an AssertionError with message "value can't be blank"
301
-
302
- Assertions.assert_name('foo')
303
- #=> does not raise an exception
304
-
305
- Assertions.assert_name(:bar)
306
- #=> does not raise an exception
307
- ```
308
-
309
- It accepts the following options:
310
-
311
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
312
- - `error_class:` The class of exception to raise. Defaults to `SleepingKingStudios::Tools::Assertions::AssertionError`.
313
- - `message`: The error message to display.
314
- - `optional`: If true, accepts `nil` values as well as valid Symbols and Strings.
315
-
316
- #### `#validate`
317
-
318
- Raises an `ArgumentError` unless the given block returns a truthy value.
319
-
320
- ```ruby
321
- Assertions.validate { true == false }
322
- #=> raises an ArgumentError with message 'block returned a falsy value'
323
-
324
- Assertions.validate { true == true }
325
- #=> does not raise an exception
326
- ```
327
-
328
- It accepts the following options:
329
-
330
- - `message`: The error message to display.
331
-
332
- #### `#validate_boolean`
333
-
334
- Raises an exception unless the given value is either `true` or `false`.
335
-
336
- ```ruby
337
- Assertions.validate_boolean(nil)
338
- #=> raises an ArgumentError with message 'value must be true or false'
339
-
340
- Assertions.validate_boolean(Object.new)
341
- #=> raises an ArgumentError with message 'value must be true or false'
342
-
343
- Assertions.validate_boolean(false)
344
- #=> does not raise an exception
345
-
346
- Assertions.validate_boolean(true)
347
- #=> does not raise an exception
348
- ```
349
-
350
- It accepts the following options:
351
-
352
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
353
- - `message`: The error message to display.
354
-
355
- #### `#validate_class`
356
-
357
- Raises an `ArgumentError` unless the given value is a Class.
358
-
359
- ```ruby
360
- Assertions.validate_class(Object.new)
361
- #=> raises an ArgumentError with message 'value is not a class'
362
-
363
- Assertions.validate_class(String)
364
- #=> does not raise an exception
365
- ```
366
-
367
- It accepts the following options:
368
-
369
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
370
- - `message`: The error message to display.
371
-
372
- #### `#validate_instance_of`
373
-
374
- Raises an `ArgumentError` unless the given value is an instance of the expected Class or Module.
375
-
376
- ```ruby
377
- Assertions.validate_instance_of(:foo, expected: String)
378
- #=> raises an AssertionError with message 'value is not an instance of String'
379
-
380
- Assertions.validate_instance_of('foo', expected: String)
381
- #=> does not raise an exception
382
- ```
383
-
384
- It accepts the following options:
385
-
386
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
387
- - `message`: The error message to display.
388
- - `optional`: If true, accepts `nil` values as well as instances of the Class or Module.
389
-
390
- #### `#validate_matches`
391
-
392
- Raises an `ArgumentError` unless the given value matches the expected value using case equality (`#===`).
393
-
394
-
395
- ```ruby
396
- Assertions.validate_matches('bar', expected: /foo/)
397
- #=> raises an ArgumentError with message 'value does not match the pattern /foo/'
398
-
399
- Assertions.validate_matches('foo', expected: /foo/)
400
- #=> does not raise an exception
401
- ```
402
-
403
- It accepts the following options:
404
-
405
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
406
- - `message`: The error message to display.
407
- - `optional`: If true, accepts `nil` values as well as matching values.
408
-
409
- #### `#validate_name`
410
-
411
- Raises an `ArgumentError` unless the given value is non-empty a String or Symbol.
412
-
413
- ```ruby
414
- Assertions.validate_name(nil)
415
- #=> raises an ArgumentError with message "value can't be blank"
416
-
417
- Assertions.validate_name(Object.new)
418
- #=> raises an AssertionError with message 'value is not a String or a Symbol'
419
-
420
- Assertions.validate_name('')
421
- #=> raises an ArgumentError with message "value can't be blank"
422
-
423
- Assertions.validate_name('foo')
424
- #=> does not raise an exception
425
-
426
- Assertions.validate_name(:bar)
427
- #=> does not raise an exception
428
- ```
429
-
430
- It accepts the following options:
431
-
432
- - `as:` A short descriptor of the given value. Defaults to `"value"`.
433
- - `message`: The error message to display.
434
- - `optional`: If true, accepts `nil` values as well as valid Symbols and Strings.
435
-
436
- ### Core Tools
437
-
438
- Tools for working with an application or working environment.
439
-
440
- #### `#deprecate`
441
-
442
- Prints a deprecation warning.
443
-
444
- ```ruby
445
- CoreTools.deprecate 'ObjectTools#old_method'
446
- #=> prints to stderr:
447
- #
448
- # [WARNING] ObjectTools#old_method is deprecated.
449
- # called from /path/to/file.rb:4: in something_or_other
450
- ```
451
-
452
- You can also specify an additional message to display:
453
-
454
- ```ruby
455
- CoreTools.deprecate 'ObjectTools#old_method',
456
- 'Use #new_method instead.'
457
- #=> prints to stderr:
458
- #
459
- # [WARNING] ObjectTools#old_method is deprecated. Use #new_method instead.
460
- # called from /path/to/file.rb:4: in something_or_other
461
- ```
462
-
463
- You can specify a custom format for the deprecation message:
464
-
465
- ```ruby
466
- CoreTools.deprecate 'ObjectTools#old_method',
467
- '0.1.0',
468
- format: '%s was deprecated in version %s.'
469
- #=> prints to stderr:
470
- #
471
- # ObjectTools#old_method was deprecated in version 0.1.0.
472
- # called from /path/to/file.rb:4: in something_or_other
473
- ```
474
-
475
- By default, `#deprecate` will print the last 3 lines of the caller, excluding
476
- any lines from `Forwardable` and from `SleepingKingStudios::Tools` itself. To
477
- print a different number of lines, pass a custom `deprecation_caller_depth` parameter to `CoreTools.new` or set the `DEPRECATION_CALLER_DEPTH` environment variable.
478
-
479
- #### `#empty_binding`
480
-
481
- Generates an empty Binding object. Note that this binding object still has
482
- access to Object methods and constants - it is **not** eval-safe.
483
-
484
- CoreTools.empty_binding
485
- #=> Binding
486
-
487
- #### `#require_each`
488
-
489
- Takes a file pattern or a list of file names and requires each file.
490
-
491
- CoreTools.require_each '/path/to/one', '/path/to/two', '/path/to/three'
492
- #=> Requires each file.
493
-
494
- CoreTools.require_each '/path/to/directory/**/*.rb'
495
- #=> Requires each file matching the pattern.
496
-
497
- ### Hash Tools
498
-
499
- Tools for working with array-like enumerable objects.
500
-
501
- #### `#convert_keys_to_strings`
502
-
503
- Creates a copy of the hash with the keys converted to strings, including keys of nested hashes and hashes inside nested arrays.
504
-
505
- hsh = { :one => 1, :two => 2, :three => 3 }
506
- cpy = HashTools.convert_keys_to_strings(hsh)
507
- #=> { 'one' => 1, 'two' => 2, 'three' => 3 }
508
- hsh
509
- #=> { :one => 1, :two => 2, :three => 3 }
510
-
511
- hsh = { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
512
- cpy = HashTools.convert_keys_to_strings(hsh)
513
- #=> { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
514
- hsh
515
- #=> { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
516
-
517
- #### `#convert_keys_to_symbols`
518
-
519
- Creates a copy of the hash with the keys converted to symbols, including keys of nested hashes and hashes inside nested arrays.
520
-
521
- hsh = { 'one' => 1, 'two' => 2, 'three' => 3 }
522
- cpy = HashTools.convert_keys_to_strings(hsh)
523
- #=> { :one => 1, :two => 2, :three => 3 }
524
- hsh
525
- #=> { 'one' => 1, 'two' => 2, 'three' => 3 }
526
-
527
- hsh = { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
528
- cpy = HashTools.convert_keys_to_strings(hsh)
529
- #=> { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
530
- hsh
531
- #=> { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
532
-
533
- #### `#deep_dup`
534
-
535
- Creates a deep copy of the object by returning a new Hash with deep copies of each key and value. See also ObjectTools#deep_dup[#label-Object+Tools].
536
-
537
- hsh = { :one => 'one', :two => 'two', :three => 'three' }
538
- cpy = HashTools.deep_dup hsh
539
-
540
- cpy.update :four => 'four'
541
- #=> { :one => 'one', :two => 'two', :three => 'three', :four => 'four' }
542
- hsh
543
- #=> { :one => 'one', :two => 'two', :three => 'three' }
544
-
545
- cpy[:one].sub!(/on/, 'vu'); cpy
546
- #=> { :one => 'vun', :two => 'two', :three => 'three', :four => 'four' }
547
- hsh
548
- #=> { :one => 'one', :two => 'two', :three => 'three' }
549
-
550
- #### `#deep_freeze`
551
-
552
- Freezes the hash and performs a deep freeze on each hash key and value.
553
-
554
- hsh = { :one => 'one', :two => 'two', :three => 'three' }
555
- HashTools.deep_freeze hsh
556
-
557
- hsh.frozen?
558
- #=> true
559
- hsh[:one].frozen?
560
- #=> true
561
-
562
- #### `#generate_binding`
563
-
564
- Generates a Binding object, with the hash converted to local variables in the
565
- binding.
566
-
567
- hsh = { :one => 'one', :two => 'two', :three => 'three' }
568
- binding = HashTools.generate_binding(hsh)
569
- #=> Binding
570
-
571
- binding.local_variable_defined?(:one)
572
- #=> true
573
- binding.local_variable_get(:one)
574
- #=> 'one'
575
- binding.eval('one')
576
- #=> 'one'
577
-
578
- #### `#hash?`
579
-
580
- Returns true if the object is or appears to be a Hash.
581
-
582
- HashTools.hash?(nil)
583
- #=> false
584
- HashTools.hash?([])
585
- #=> false
586
- HashTools.hash?({})
587
- #=> true
588
-
589
- #### `#immutable?`
590
-
591
- Returns true if the hash is immutable, i.e. if the hash is frozen and each hash key and hash value are immutable.
592
-
593
- HashTools.immutable?({ :id => 0, :title => 'The Ramayana' })
594
- #=> false
595
-
596
- HashTools.immutable?({ :id => 0, :title => 'The Ramayana' }.freeze)
597
- #=> false
598
-
599
- HashTools.immutable?({ :id => 0, :title => 'The Ramayana'.freeze }.freeze)
600
- #=> true
601
-
602
- #### `#mutable?`
603
-
604
- Returns true if the hash is mutable (see `#immutable?`, above).
605
-
606
- #### `#stringify_keys`
607
-
608
- See HashTools#convert_keys_to_strings[#label-Hash+Tools]
609
-
610
- #### `#symbolize_keys`
611
-
612
- See HashTools#convert_keys_to_symbols[#label-Hash+Tools]
613
-
614
- ### Integer Tools
615
-
616
- Tools for working with integers.
617
-
618
- #### `#count_digits`
619
-
620
- Returns the number of digits in the given integer when represented in the specified base. Ignores minus sign for negative numbers.
621
-
622
- # With a positive number.
623
- IntegerTools.count_digits(31)
624
- #=> 2
625
-
626
- # With a negative number.
627
- IntegerTools.count_digits(-141)
628
- #=> 3
629
-
630
- # With a binary number.
631
- IntegerTools.count_digits(189, :base => 2)
632
- #=> 8
633
-
634
- # With a hexadecimal number.
635
- IntegerTools.count_digits(16724838, :base => 16)
636
- #=> 6
637
-
638
- #### `#digits`
639
-
640
- Decomposes the given integer into its digits when represented in the given base.
641
-
642
- # With a number in base 10.
643
- IntegerTools.digits(15926)
644
- #=> ['1', '5', '9', '2', '6']
645
-
646
- # With a binary number.
647
- IntegerTools.digits(189, :base => 2)
648
- #=> ['1', '0', '1', '1', '1', '1', '0', '1']
649
-
650
- # With a hexadecimal number.
651
- IntegerTools.digits(16724838)
652
- #=> ['f', 'f', '3', '3', '6', '6']
653
-
654
- #### `#integer?`
655
-
656
- Returns true if the object is an Integer.
657
-
658
- IntegerTools.integer?(nil)
659
- #=> false
660
- IntegerTools.integer?([])
661
- #=> false
662
- IntegerTools.integer?({})
663
- #=> false
664
- IntegerTools.integer?(1)
665
- #=> true
666
-
667
- #### `#pluralize`
668
-
669
- Returns the singular or the plural value, depending on the provided item count. Can be given an explicit plural argument, or will delegate to StringTools#pluralize.
670
-
671
- IntegerTools.pluralize 4, 'light'
672
- #=> 'lights'
673
-
674
- IntegerTools.pluralize 3, 'cow', 'kine'
675
- #=> 'kine'
676
-
677
- #### `#romanize`
678
-
679
- Represents an integer between 1 and 4999 (inclusive) as a Roman numeral.
680
-
681
- IntegerTools.romanize(499)
682
- #=> 'CDXCIX'
683
-
684
- ### Object Tools
685
-
686
- require 'sleeping_king_studios/tools/object_tools'
687
-
688
- Low-level tools for working with objects.
689
-
690
- #### `#apply`
691
-
692
- Takes a proc or lambda and invokes it with the given object as receiver, with any additional arguments or block provided.
693
-
694
- my_object = double('object', :to_s => 'A mock object')
695
- my_proc = ->() { puts %{#{self.to_s} says "Greetings, programs!"} }
696
-
697
- ObjectTools.apply my_object, my_proc
698
- #=> Writes 'A mock object says "Greetings, programs!"' to STDOUT.
699
-
700
- #### `#deep_dup`
701
-
702
- Creates a deep copy of the object. If the object is an Array, returns a new Array with deep copies of each array item (see ArrayTools#deep_dup[#label-Array+Tools]). If the object is a Hash, returns a new Hash with deep copies of each hash key and value (see HashTools#deep_dup[#label-Hash+Tools]). Otherwise, returns Object#dup.
703
-
704
- data = {
705
- :songs = [
706
- {
707
- :name => 'Welcome to the Jungle',
708
- :artist => "Guns N' Roses",
709
- :album => 'Appetite for Destruction'
710
- },
711
- {
712
- :name => 'Hells Bells',
713
- :artist => 'AC/DC',
714
- :album => 'Back in Black'
715
- },
716
- {
717
- :name => "Knockin' on Heaven's Door",
718
- :artist => 'Bob Dylan',
719
- :album => 'Pat Garrett & Billy The Kid'
720
- }
721
- ]
722
- }
723
-
724
- copy = ObjectTools.deep_dup data
725
-
726
- copy[:songs] << { :name => 'Sympathy for the Devil', :artist => 'The Rolling Stones', :album => 'Beggars Banquet' }
727
- data[:songs].count
728
- #=> 3
729
-
730
- copy[:songs][1][:name] = 'Shoot to Thrill'
731
- data[:songs][1]
732
- #=> { :name => 'Hells Bells', :artist => 'AC/DC', :album => 'Back in Black' }
733
-
734
- #### `#deep_freeze`
735
-
736
- Performs a deep freeze of the object. If the object is an Array, freezes the array and performs a deep freeze on each array item (see ArrayTools#deep_dup[#label-Array+Tools]). If the object is a hash, freezes the hash and performs a deep freeze on each hash key and value (see HashTools#deep_dup[#label-Hash+Tools]). Otherwise, calls Object#freeze unless the object is already immutable.
737
-
738
- data = {
739
- :songs = [
740
- {
741
- :name => 'Welcome to the Jungle',
742
- :artist => "Guns N' Roses",
743
- :album => 'Appetite for Destruction'
744
- },
745
- {
746
- :name => 'Hells Bells',
747
- :artist => 'AC/DC',
748
- :album => 'Back in Black'
749
- },
750
- {
751
- :name => "Knockin' on Heaven's Door",
752
- :artist => 'Bob Dylan',
753
- :album => 'Pat Garrett & Billy The Kid'
754
- }
755
- ]
756
- }
757
- ObjectTools.deep_freeze(data)
758
-
759
- data.frozen?
760
- #=> true
761
- data[:songs].frozen?
762
- #=> true
763
- data[:songs][0].frozen?
764
- #=> true
765
- data[:songs][0].name.frozen?
766
- #=> true
767
-
768
- #### `#dig`
769
-
770
- Accesses deeply nested attributes by calling the first named method on the given object, and each subsequent method on the result of the previous method call. If the object does not respond to the method name, nil is returned instead of calling the method.
771
-
772
- ObjectTools.dig my_object, :first_method, :second_method, :third_method
773
- #=> my_object.first_method.second_method.third_method
774
-
775
- #### `#eigenclass`, `#metaclass`
776
-
777
- Returns the object's eigenclass.
778
-
779
- ObjectTools.eigenclass my_object
780
- #=> Shortcut for class << self; self; end.
781
-
782
- #### `#immutable?`
783
-
784
- Returns true if the object is immutable. Values of nil, false, and true are always immutable, as are instances of Numeric and Symbol. Arrays are immutable if the array is frozen and each array item is immutable. Hashes are immutable if the hash is frozen and each hash key and hash value are immutable. Otherwise, objects are immutable if they are frozen.
785
-
786
- ObjectTools.immutable?(nil)
787
- #=> true
788
-
789
- ObjectTools.immutable?(false)
790
- #=> true
791
-
792
- ObjectTools.immutable?(0)
793
- #=> true
794
-
795
- ObjectTools.immutable?(:hello)
796
- #=> true
797
-
798
- ObjectTools.immutable?("Greetings, programs!")
799
- #=> false
800
-
801
- ObjectTools.immutable?([1, 2, 3])
802
- #=> false
803
-
804
- ObjectTools.immutable?([1, 2, 3].freeze)
805
- #=> false
806
-
807
- #### `#mutable?`
808
-
809
- Returns true if the object is mutable (see `#immutable?`, above).
810
-
811
- #### `#object?`
812
-
813
- Returns true if the object is an Object.
814
-
815
- ObjectTools.object?(nil)
816
- #=> true
817
- ObjectTools.object?([])
818
- #=> true
819
- ObjectTools.object?({})
820
- #=> true
821
- ObjectTools.object?(1)
822
- #=> true
823
- ObjectTools.object?(BasicObject.new)
824
- #=> false
825
-
826
- #### `#try`
827
-
828
- As #send, but returns nil if the object does not respond to the method.
829
-
830
- ObjectTools.try(%w(ichi ni san), :count)
831
- #=> 3
832
- ObjectTools.try(nil, :count)
833
- #=> nil
834
-
835
- ### String Tools
836
-
837
- require 'sleeping_king_studios/tools/string_tools'
838
-
839
- Tools for working with strings.
840
-
841
- #### `#camelize`
842
-
843
- Converts a lowercase, underscore separated string to a mixed-case string expression, as per ActiveSupport::Inflector#camelize.
844
-
845
- StringTools#camelize 'valhalla'
846
- #=> 'Valhalla'
847
-
848
- StringTools#camelize 'muspelheimr_and_niflheimr'
849
- #=> 'MuspelheimrAndNiflheimr'
850
-
851
- #### `#chain`
852
-
853
- Performs multiple string tools operations in sequence, starting with the given string and passing the result of each operation to the next.
854
-
855
- # Equivalent to `StringTools.underscore(StringTools.pluralize str)`.
856
- StringTools#chain 'ArchivedPeriodical', :underscore, :pluralize
857
- # => 'archived_periodicals'
858
-
859
- Adds the specified number of spaces to the start of each line of the string. Defaults to 2 spaces.
860
-
861
- string = 'The Hobbit'
862
- StringTools.indent(string)
863
- #=> ' The Hobbit'
864
-
865
- titles = [
866
- "The Fellowship of the Ring",
867
- "The Two Towers",
868
- "The Return of the King"
869
- ]
870
- string = titles.join "\n"
871
- StringTools.indent(string, 4)
872
- #=> " The Fellowship of the Ring\n"\
873
- " The Two Towers\n"\
874
- " The Return of the King"
875
-
876
- #### `#map_lines`
877
-
878
- Yields each line of the string to the provided block and combines the results into a new multiline string.
879
-
880
- string = 'The Hobbit'
881
- StringTools.map_lines(string) { |line| " #{line}" }
882
- #=> '- The Hobbit'
883
-
884
- titles = [
885
- "The Fellowship of the Ring",
886
- "The Two Towers",
887
- "The Return of the King"
888
- ]
889
- string = titles.join "\n"
890
- StringTools.map_lines(string) { |line, index| "#{index}. #{line}" }
891
- #=> "0. The Fellowship of the Ring\n"\
892
- "1. The Two Towers\n"\
893
- "2. The Return of the King"
894
-
895
- #### `#plural?`
896
-
897
- Returns true if the word is in plural form, and returns false otherwise. A word is in plural form if and only if calling `#pluralize` (see below) on the word returns the word without modification.
898
-
899
- StringTools.plural? 'light'
900
- #=> false
901
-
902
- StringTools.plural? 'lights'
903
- #=> true
904
-
905
- #### `#pluralize`
906
-
907
- Takes a word in singular form and returns the plural form, based on the defined rules and known irregular/uncountable words.
908
-
909
- First, checks if the word is known to be uncountable (see #define_uncountable_word). Then, checks if the word is known to be irregular (see #define_irregular_word). Finally, iterates through the defined plural rules from most recently defined to first defined (see #define_plural_rule).
910
-
911
- StringTools.pluralize 'light'
912
- #=> 'lights'
913
-
914
- **Important Note:** The defined rules and exceptions are deliberately basic. Each application is responsible for defining its own pluralization rules using this framework.
915
-
916
- Additional rules can be defined using the following methods:
917
-
918
- # Define a plural rule.
919
- StringTools.define_plural_rule(/lf$/, 'lves')
920
- StringTools.pluralize 'elf'
921
- #=> 'elves'
922
-
923
- # Define an irregular word.
924
- StringTools.define_irregular_word('goose', 'geese')
925
- StringTools.pluralize 'goose'
926
- #=> 'geese'
927
-
928
- # Define an uncountable word.
929
- StringTools.define_uncountable_word('series')
930
- StringTools.pluralize 'series'
931
- # => 'series'
932
-
933
- #### `#singular?`
934
-
935
- Returns true if the word is in singular form, and returns false otherwise. A word is in singular form if and only if calling `#singularize` (see below) on the word returns the word without modification.
936
-
937
- StringTools.singular? 'light'
938
- #=> true
939
-
940
- StringTools.singular? 'lights'
941
- #=> false
942
-
943
- #### `#singularize`
944
-
945
- Takes a word in plural form and returns the singular form, based on the defined rules and known irregular/uncountable words.
946
-
947
- StringTools.singularize 'lights'
948
- #=> 'light'
949
-
950
- `StringTools#singularize` uses the same rules for irregular and uncountable words as `#pluralize`. Additional rules can be defined using the following method:
951
-
952
- StringTools.define_singular_rule(/lves$/, 'lf')
953
- StringTools.singularize 'elves'
954
- #=> 'elf'
955
-
956
- #### `#string?`
957
-
958
- Returns true if the object is a String.
959
-
960
- StringTools.string?(nil)
961
- #=> false
962
- StringTools.string?([])
963
- #=> false
964
- StringTools.string?('Greetings, programs!')
965
- #=> true
966
- StringTools.string?(:greetings_starfighter)
967
- #=> false
968
-
969
- #### `#underscore`
970
-
971
- Converts a mixed-case string expression to a lowercase, underscore separated string, as per ActiveSupport::Inflector#underscore.
972
-
973
- StringTools#underscore 'Bifrost'
974
- #=> 'bifrost'
975
-
976
- StringTools#underscore 'FenrisWolf'
977
- #=> 'fenris_wolf'
978
-
979
- ## Toolbox
980
-
981
- Common objects or patterns that are useful across projects but are larger than or do not fit the functional paradigm of the tools.* pattern.
982
-
983
- ### ConstantMap
984
-
985
- require 'sleeping_king_studios/tools/toolbox/constant_map'
986
-
987
- Provides an enumerable interface for defining a group of constants.
988
-
989
- UserRoles = ConstantMap.new(
990
- {
991
- GUEST: 'guest',
992
- USER: 'user',
993
- ADMIN: 'admin'
994
- }'
995
- )
996
-
997
- UserRoles::GUEST
998
- #=> 'guest'
999
-
1000
- UserRoles.user
1001
- #=> 'user'
1002
-
1003
- UserRoles.all
1004
- #=> { :GUEST => 'guest', :USER => 'user', :ADMIN => 'admin' }
1005
-
1006
- ConstantMap includes `Enumerable`, with `#each` yielding the name and value of each defined constant. It also defines the following additional methods:
1007
-
1008
- #### `#each_key`
1009
-
1010
- Yields each defined constant name, similar to `Hash#each_key`.
1011
-
1012
- #### `#each_value`
1013
-
1014
- Yields each defined constant value, similar to `Hash#each_value`.
1015
-
1016
- #### `#keys`
1017
-
1018
- Returns an array containing the names of the defined constants, similar to `Hash#keys`.
1019
-
1020
- #### `#to_h`
1021
-
1022
- Also `#all`. Returns a Hash representation of the constants.
1023
-
1024
- #### `#values`
1025
-
1026
- Returns an array containing the values of the defined constants, similar to `Hash#values`.
1027
-
1028
- ### Mixin
1029
-
1030
- require 'sleeping_king_studios/tools/toolbox/mixin'
1031
-
1032
- Implements module-based inheritance for both instance- and class-level methods, similar to the (in)famous ActiveSupport::Concern. When a Mixin is included into a class, the class will be extended with any methods defined in the special ClassMethods module, even if the Mixin is being included indirectly via one or more intermediary Mixins.
1033
-
1034
- Widget = Struct.new(:widget_type)
1035
-
1036
- module Widgets
1037
- extend SleepingKingStudios::Tools::Toolbox::Mixin
1038
-
1039
- module ClassMethods
1040
- def widget_types
1041
- %w(gadget doohickey thingamabob)
1042
- end # class method widget_types
1043
- end # module
1044
-
1045
- def widget? widget_type
1046
- self.class.widget_types.include?(widget_type)
1047
- end # method widget?
1048
- end # module
1049
-
1050
- module WidgetBuilding
1051
- extend SleepingKingStudios::Tools::Toolbox::Mixin
1052
-
1053
- include Widgets
1054
-
1055
- def build_widget widget_type
1056
- raise ArgumentError, 'not a widget', caller unless widget?(widget_type)
1057
-
1058
- Widget.new(widget_type)
1059
- end # method build_widget
1060
- end # module
1061
-
1062
- class WidgetFactory
1063
- include WidgetBuilding
1064
- end # class
1065
-
1066
- factory = WidgetFactory.new
1067
-
1068
- factory.build_widget('gadget')
1069
- #=> Widget
1070
-
1071
- WidgetFactory.widget_types
1072
- #=> ['gadget', 'doohickey', 'thingamabob']
1073
-
1074
- ### Semantic Version
1075
-
1076
- require 'sleeping_king_studios/tools/toolbox/semantic_version'
1077
-
1078
- Module mixin for using semantic versioning (see http://semver.org) with helper methods for generating strict and gem-compatible version strings.
1079
-
1080
- module Version
1081
- extend SleepingKingStudios::Tools::Toolbox::SemanticVersion
1082
-
1083
- MAJOR = 3
1084
- MINOR = 1
1085
- PATCH = 4
1086
- PRERELEASE = 'beta'
1087
- BUILD = 1
1088
- end # module
1089
-
1090
- GEM_VERSION = Version.to_gem_version
1091
- #=> '3.1.4.beta.1'
1092
-
1093
- VERSION = Version.to_version
1094
- #=> '3.1.4-beta+1'
1095
-
1096
- #### `#to_gem_version`
1097
-
1098
- Concatenates the MAJOR, MINOR, and PATCH constant values with PRERELEASE and BUILD (if available) to generate a modified semantic version string compatible with Rubygems. The major, minor, patch, prerelease, and build values (if available) are separated by dots `.`.
1099
-
1100
- #### `#to_version`
1101
-
1102
- Concatenates the MAJOR, MINOR, and PATCH constant values with PRERELEASE and BUILD (if available) to generate a semantic version string. The major, minor, and patch values are separated by dots `.`, then the prerelease (if available) preceded by a hyphen `-`, and the build (if available) preceded by a plus sign `+`.
1103
-
1104
- ### Subclass
1105
-
1106
- ```ruby
1107
- require 'sleeping_king_studios/tools/toolbox/subclass'
1108
- ```
1109
-
1110
- The `Subclass` module provides a mechanism for performing partial application (or "currying") of constructor parameters. This is useful for defining subclasses that inject pre-defined values into the constructor.
1111
-
1112
- Let's consider an example. We'll start by defining a base class.
1113
-
1114
- ```ruby
1115
- class Query
1116
- extend SleepingKingStudios::Tools::Toolbox::Subclass
1117
-
1118
- def initialize(entity_class, **options)
1119
- @entity_class = entity_class
1120
- @options = options
1121
- end
1122
-
1123
- attr_reader :entity_class
1124
-
1125
- attr_reader :options
1126
-
1127
- def call
1128
- # Querying logic here.
1129
- end
1130
- end
1131
-
1132
- query = Query.new(Book, limit: 10)
1133
- query.entity_class
1134
- #=> Book
1135
- query.options
1136
- #=> { limit: 10 }
1137
- ```
1138
-
1139
- Our `Query` class is used to perform queries on some data source - a relational table, an API, or some in-memory data structure. To perform a query, we need to know what data to request. This is represented by the `:entity_class` argument. Additionally, we can pass arbitrary `:options` as keywords.
1140
-
1141
- ```ruby
1142
- BooksQuery = Query.subclass(Book)
1143
-
1144
- query = BooksQuery.new(order: :title)
1145
- query.entity_class
1146
- #=> Book
1147
- query.options
1148
- #=> { order: :title }
1149
- ```
1150
-
1151
- By calling `.subclass` on our `Query` class, we are defining a new subclass of `Query` that injects the given parameters and partially applies them to the constructor. In this case, we are injecting the `Book` class into our query.
1152
-
1153
- We can also use `.subclass` to partially apply keywords or a block.
1154
-
1155
- ```ruby
1156
- RecentItemsQuery = Query.subclass(order: { created_at: :desc })
1157
-
1158
- query = RecentItemsQuery.new(Book)
1159
- query.entity_class
1160
- #=> Book
1161
- query.options
1162
- #=> { order: { created_at: :desc } }
1163
- ```
1164
-
1165
- When you call the subclass's constructor with additional parameters, they are applied in addition to the configured values (if any).
1166
-
1167
- - Any arguments passed to `.new` are added *after* the configured arguments.
1168
- - Any keywords passed to `.new` are merged into the configured keywords, with the values passed to `.new` taking precedence.
1169
- - A block passed to `.new` will take precedence over a configured block.
1170
-
1171
- ```ruby
1172
- class Character
1173
- extend SleepingKingStudios::Tools::Toolbox::Subclass
1174
-
1175
- def initialize(*traits, **stats, &special)
1176
- @traits = traits
1177
- @stats = stats
1178
- @special = special
1179
- end
1180
-
1181
- attr_reader :special
1182
-
1183
- attr_reader :stats
1184
-
1185
- attr_reader :traits
1186
- end
1187
-
1188
- Bard = Character.subclass(:musical, performance: 5, persuasion: 10) do
1189
- 'The bard sings a cheerful tune.'
1190
- end
1191
-
1192
- aoife = Bard.new(:sorcerous, magic: 5, performance: 10) do
1193
- 'Aoife drops her lute!'
1194
- end
1195
- aoife.traits
1196
- #=> [:musical, :sorcerous]
1197
- aoife.stats
1198
- #=> { magic: 5, performance: 10, persuasion: 10 }
1199
- aoife.special.call
1200
- #=> 'Aoife drops her lute!'
1201
- ```