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.
@@ -5,8 +5,7 @@ require 'sleeping_king_studios/tools'
5
5
  module SleepingKingStudios::Tools
6
6
  # Tools for working with an application or working environment.
7
7
  class CoreTools < Base
8
- # Exception class used when deprecated code is called and the deprecation
9
- # strategy is 'raise'.
8
+ # Exception raised when deprecated code is called with strategy 'raise'.
10
9
  class DeprecationError < StandardError; end
11
10
 
12
11
  class << self
@@ -16,9 +15,9 @@ module SleepingKingStudios::Tools
16
15
  :require_each
17
16
  end
18
17
 
19
- # @param deprecation_caller_depth [Integer] The number of backtrace lines to
18
+ # @param deprecation_caller_depth [Integer] the number of backtrace lines to
20
19
  # display when outputting a deprecation warning.
21
- # @param deprecation_strategy [String] The name of the strategy used when
20
+ # @param deprecation_strategy [String] the name of the strategy used when
22
21
  # deprecated code is called. Must be 'ignore', 'raise', or 'warn'.
23
22
  def initialize(
24
23
  deprecation_caller_depth: nil,
@@ -40,27 +39,62 @@ module SleepingKingStudios::Tools
40
39
  # @return [String] the current deprecation strategy.
41
40
  attr_reader :deprecation_strategy
42
41
 
42
+ # Prints a deprecation warning or raises an exception.
43
+ #
44
+ # The behavior of this method depends on the configured deprecation
45
+ # strategy, which can be passed to the constructor or configured using the
46
+ # DEPRECATION_STRATEGY environment variable.
47
+ #
48
+ # - If the strategy is 'warn' (the default), the formatted message is passed
49
+ # to Kernel.warn, which prints the message to $stderr.
50
+ # - If the strategy is 'raise', raises a DeprecationError with the message.
51
+ # - If the strategy is 'ignore', this method does nothing.
52
+ #
53
+ # @example
54
+ # CoreTools.deprecate 'ObjectTools#old_method'
55
+ # #=> prints to stderr:
56
+ # #
57
+ # # [WARNING] ObjectTools#old_method is deprecated.
58
+ # # called from /path/to/file.rb:4: in something_or_other
59
+ #
60
+ # @example With An Additional Message
61
+ # CoreTools.deprecate 'ObjectTools#old_method',
62
+ # 'Use #new_method instead.'
63
+ # #=> prints to stderr:
64
+ # #
65
+ # # [WARNING] ObjectTools#old_method is deprecated. Use #new_method instead.
66
+ # # called from /path/to/file.rb:4: in something_or_other
67
+ #
68
+ # @example With A Format String
69
+ # CoreTools.deprecate 'ObjectTools#old_method',
70
+ # '0.1.0',
71
+ # format: '%s was deprecated in version %s.'
72
+ # #=> prints to stderr:
73
+ # #
74
+ # # ObjectTools#old_method was deprecated in version 0.1.0.
75
+ # # called from /path/to/file.rb:4: in something_or_other
76
+ #
43
77
  # @overload deprecate(name, message: nil)
44
- # Prints a deprecation warning.
78
+ # Prints a deprecation warning or raises an exception.
45
79
  #
46
- # @param name [String] The name of the object, method, or feature that
80
+ # @param name [String] the name of the object, method, or feature that
47
81
  # has been deprecated.
48
- # @param message [String] An optional message to print after the formatted
82
+ # @param message [String] an optional message to print after the formatted
49
83
  # string. Defaults to nil.
50
84
  #
51
85
  # @overload deprecate(*args, format:, message: nil)
52
- # Prints a deprecation warning with the specified format.
86
+ # Prints a deprecation warning with the specified format or raises.
53
87
  #
54
- # @param args [Array] The arguments to pass into the format string.
55
- # @param format [String] The format string.
56
- # @param message [String] An optional message to print after the formatted
88
+ # @param args [Array] the arguments to pass into the format string.
89
+ # @param format [String] the format string.
90
+ # @param message [String] an optional message to print after the formatted
57
91
  # string. Defaults to nil.
58
92
  def deprecate(*args, format: nil, message: nil)
59
93
  send(
60
94
  :"deprecate_as_#{deprecation_strategy}",
61
95
  *args,
62
- format: format,
63
- message: message
96
+ format:,
97
+ message:
64
98
  )
65
99
  end
66
100
 
@@ -22,13 +22,28 @@ module SleepingKingStudios::Tools
22
22
  :symbolize_keys
23
23
  end
24
24
 
25
- # Returns a copy of the hash with the keys converted to strings.
25
+ # Returns a deep copy of the hash with the keys converted to strings.
26
26
  #
27
- # @param [Hash] hsh The hash to convert.
27
+ # @param hsh [Hash] the hash to convert.
28
28
  #
29
- # @return [Hash] The converted copy of the hash.
29
+ # @return [Hash] the converted copy of the hash.
30
+ #
31
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
32
+ #
33
+ # @example
34
+ # hsh = { :one => 1, :two => 2, :three => 3 }
35
+ # cpy = HashTools.convert_keys_to_strings(hsh)
36
+ # #=> { 'one' => 1, 'two' => 2, 'three' => 3 }
37
+ # hsh
38
+ # #=> { :one => 1, :two => 2, :three => 3 }
39
+ #
40
+ # hsh = { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
41
+ # cpy = HashTools.convert_keys_to_strings(hsh)
42
+ # #=> { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
43
+ # hsh
44
+ # #=> { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
30
45
  def convert_keys_to_strings(hsh)
31
- require_hash! hsh
46
+ require_hash!(hsh)
32
47
 
33
48
  hsh.each.with_object({}) do |(key, value), cpy|
34
49
  sym = key.to_s
@@ -38,13 +53,28 @@ module SleepingKingStudios::Tools
38
53
  end
39
54
  alias stringify_keys convert_keys_to_strings
40
55
 
41
- # Returns a copy of the hash with the keys converted to symbols.
56
+ # Returns a deep copy of the hash with the keys converted to symbols.
57
+ #
58
+ # @param hsh [Hash] the hash to convert.
42
59
  #
43
- # @param [Hash] hsh The hash to convert.
60
+ # @return [Hash] the converted copy of the hash.
44
61
  #
45
- # @return [Hash] The converted copy of the hash.
62
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
63
+ #
64
+ # @example
65
+ # hsh = { 'one' => 1, 'two' => 2, 'three' => 3 }
66
+ # cpy = HashTools.convert_keys_to_symbols(hsh)
67
+ # #=> { :one => 1, :two => 2, :three => 3 }
68
+ # hsh
69
+ # #=> { 'one' => 1, 'two' => 2, 'three' => 3 }
70
+ #
71
+ # hsh = { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
72
+ # cpy = HashTools.convert_keys_to_strings(hsh)
73
+ # #=> { :odd => { :one => 1, :three => 3 }, :even => { :two => 2, :four => 4 } }
74
+ # hsh
75
+ # #=> { 'odd' => { 'one' => 1, 'three' => 3 }, 'even' => { 'two' => 2, 'four' => 4 } }
46
76
  def convert_keys_to_symbols(hsh)
47
- require_hash! hsh
77
+ require_hash!(hsh)
48
78
 
49
79
  hsh.each.with_object({}) do |(key, value), cpy|
50
80
  sym = key.to_s.intern
@@ -54,26 +84,53 @@ module SleepingKingStudios::Tools
54
84
  end
55
85
  alias symbolize_keys convert_keys_to_symbols
56
86
 
57
- # Creates a deep copy of the object by returning a new Hash with deep
58
- # copies of each key and value.
87
+ # Creates a deep copy of the Hash.
88
+ #
89
+ # @param hsh [Hash<Object>] the hash to copy.
59
90
  #
60
- # @param [Hash<Object>] hsh The hash to copy.
91
+ # @return [Hash] the copy of the hash.
61
92
  #
62
- # @return [Hash] The copy of the hash.
93
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
94
+ #
95
+ # @example
96
+ # hsh = { :one => 'one', :two => 'two', :three => 'three' }
97
+ # cpy = HashTools.deep_dup hsh
98
+ #
99
+ # cpy.update :four => 'four'
100
+ # #=> { :one => 'one', :two => 'two', :three => 'three', :four => 'four' }
101
+ # hsh
102
+ # #=> { :one => 'one', :two => 'two', :three => 'three' }
103
+ #
104
+ # cpy[:one].sub!(/on/, 'vu'); cpy
105
+ # #=> { :one => 'vun', :two => 'two', :three => 'three', :four => 'four' }
106
+ # hsh
107
+ # #=> { :one => 'one', :two => 'two', :three => 'three' }
63
108
  def deep_dup(hsh)
64
- require_hash! hsh
109
+ require_hash!(hsh)
65
110
 
66
111
  hsh.each.with_object({}) do |(key, value), copy|
67
112
  copy[ObjectTools.deep_dup key] = ObjectTools.deep_dup(value)
68
113
  end
69
114
  end
70
115
 
71
- # Freezes the hash and performs a deep freeze on each hash key and
72
- # value.
116
+ # Freezes the hash and performs a deep freeze on each hash key and value.
117
+ #
118
+ # @param hsh [Hash] the hash to freeze.
119
+ #
120
+ # @return [Hash] the frozen hash.
121
+ #
122
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
123
+ #
124
+ # @example
125
+ # hsh = { :one => 'one', :two => 'two', :three => 'three' }
126
+ # HashTools.deep_freeze hsh
73
127
  #
74
- # @param [Hash] hsh The object to freeze.
128
+ # hsh.frozen?
129
+ # #=> true
130
+ # hsh[:one].frozen?
131
+ # #=> true
75
132
  def deep_freeze(hsh)
76
- require_hash! hsh
133
+ require_hash!(hsh)
77
134
 
78
135
  hsh.freeze
79
136
 
@@ -83,12 +140,25 @@ module SleepingKingStudios::Tools
83
140
  end
84
141
  end
85
142
 
86
- # Generates a Binding object with an Object as the receiver and the hash
87
- # key-value pairs set as local variables.
143
+ # Generates a Binding with the hash values as local variables.
88
144
  #
89
- # @return [Binding] The binding object.
145
+ # @return [Binding] the binding object.
146
+ #
147
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
148
+ #
149
+ # @example
150
+ # hsh = { :one => 'one', :two => 'two', :three => 'three' }
151
+ # binding = HashTools.generate_binding(hsh)
152
+ # #=> Binding
153
+ #
154
+ # binding.local_variable_defined?(:one)
155
+ # #=> true
156
+ # binding.local_variable_get(:one)
157
+ # #=> 'one'
158
+ # binding.eval('one')
159
+ # #=> 'one'
90
160
  def generate_binding(hsh)
91
- require_hash! hsh
161
+ require_hash!(hsh)
92
162
 
93
163
  CoreTools.empty_binding.tap do |binding|
94
164
  hsh.each do |key, value|
@@ -99,27 +169,57 @@ module SleepingKingStudios::Tools
99
169
 
100
170
  # Returns true if the object is or appears to be a Hash.
101
171
  #
102
- # @param hsh [Object] The object to test.
172
+ # This method checks for the method signatures of the object. A Hash-like
173
+ # method will define all of the the #[], #count, #each, #each_key, and
174
+ # #each_value methods.
175
+ #
176
+ # @param obj [Object] the object to test.
103
177
  #
104
- # @return [Boolean] True if the object is a Hash, otherwise false.
105
- def hash?(hsh)
106
- return true if hsh.is_a?(Hash)
178
+ # @return [Boolean] true if the object is a Hash, otherwise false.
179
+ #
180
+ # @example
181
+ # HashTools.hash?(nil)
182
+ # #=> false
183
+ # HashTools.hash?([])
184
+ # #=> false
185
+ # HashTools.hash?({})
186
+ # #=> true
187
+ def hash?(obj)
188
+ return true if obj.is_a?(Hash)
107
189
 
108
190
  HASH_METHODS.each do |method_name|
109
- return false unless hsh.respond_to?(method_name)
191
+ return false unless obj.respond_to?(method_name)
110
192
  end
111
193
 
112
194
  true
113
195
  end
114
196
 
115
- # Returns true if the hash is immutable, i.e. if the hash is frozen and each
116
- # hash key and hash value are immutable.
197
+ # Returns true if the hash is immutable.
198
+ #
199
+ # A hash is considered immutable if the hash itself is frozen and each
200
+ # key and value in the hash is immutable.
201
+ #
202
+ # @param hsh [Hash] the hash to test.
117
203
  #
118
- # @param hsh [Hash] The hash to test.
204
+ # @return [Boolean] true if the hash is immutable, otherwise false.
119
205
  #
120
- # @return [Boolean] True if the hash is immutable, otherwise false.
206
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
207
+ #
208
+ # @see HashTools#mutable?
209
+ #
210
+ # @see ObjectTools#immutable?
211
+ #
212
+ # @example
213
+ # HashTools.immutable?({ :id => 0, :title => 'The Ramayana' })
214
+ # #=> false
215
+ #
216
+ # HashTools.immutable?({ :id => 0, :title => +'The Ramayana' }.freeze)
217
+ # #=> false
218
+ #
219
+ # HashTools.immutable?({ :id => 0, :title => 'The Ramayana' }.freeze)
220
+ # #=> true
121
221
  def immutable?(hsh)
122
- require_hash! hsh
222
+ require_hash!(hsh)
123
223
 
124
224
  return false unless hsh.frozen?
125
225
 
@@ -132,13 +232,15 @@ module SleepingKingStudios::Tools
132
232
  true
133
233
  end
134
234
 
135
- # Returns true if the hash is mutable.
235
+ # Returns true if the hash or any of its contents are mutable.
236
+ #
237
+ # @param hsh [Hash] the hash to test.
136
238
  #
137
- # @param hsh [Array] The hash to test.
239
+ # @return [Boolean] true if the hash is mutable, otherwise false.
138
240
  #
139
- # @return [Boolean] True if the hash is mutable, otherwise false.
241
+ # @raise [ArgumentError] if the first argument is not an Hash-like object.
140
242
  #
141
- # @see #immutable?
243
+ # @see HashTools#immutable?
142
244
  def mutable?(hsh)
143
245
  !immutable?(hsh)
144
246
  end
@@ -58,8 +58,16 @@ module SleepingKingStudios::Tools
58
58
  :romanize
59
59
  end
60
60
 
61
- # Returns the number of digits in the given integer when represented in the
62
- # specified base. Ignores minus sign for negative numbers.
61
+ # Returns the number of digits in the given integer and base.
62
+ #
63
+ # This method ignores minus sign for negative numbers. You can use the :base
64
+ # parameter to configure the numeric base for the string representation.
65
+ #
66
+ # @param integer [Integer] the integer to analyze.
67
+ # @param base [Integer] the numeric base to represent the integer in.
68
+ # Defaults to 10.
69
+ #
70
+ # @return [Integer] the number of digits.
63
71
  #
64
72
  # @example With a positive number.
65
73
  # IntegerTools.count_digits(31)
@@ -76,18 +84,18 @@ module SleepingKingStudios::Tools
76
84
  # @example With a hexadecimal number.
77
85
  # IntegerTools.count_digits(16724838, :base => 16)
78
86
  # #=> 6
79
- #
80
- # @param [Integer] integer The integer to analyze.
81
- # @param [Integer] base The numeric base to represent the integer in.
82
- # Defaults to 10.
83
- #
84
- # @return [Integer] The number of digits.
85
87
  def count_digits(integer, base: 10)
86
- digits(integer.abs, base: base).count
88
+ digits(integer.abs, base:).count
87
89
  end
88
90
 
89
- # Decomposes the given integer into its digits when represented in the
90
- # given base.
91
+ # Decomposes the given integer into its digits in the given base.
92
+ #
93
+ # @param integer [Integer] the integer to decompose.
94
+ # @param base [Integer] the numeric base to represent the integer in.
95
+ # Defaults to 10.
96
+ #
97
+ # @return [Array<String>] the digits of the decomposed integer,
98
+ # represented as a bigendian array of strings.
91
99
  #
92
100
  # @example With a number in base 10.
93
101
  # IntegerTools.digits(15926)
@@ -100,39 +108,47 @@ module SleepingKingStudios::Tools
100
108
  # @example With a hexadecimal number.
101
109
  # IntegerTools.digits(16724838)
102
110
  # #=> ['f', 'f', '3', '3', '6', '6']
103
- #
104
- # @param [Integer] integer The integer to decompose.
105
- # @param [Integer] base The numeric base to represent the integer in.
106
- # Defaults to 10.
107
- #
108
- # @return [Array<String>] The digits of the decomposed integer,
109
- # represented as a bigendian array of strings.
110
111
  def digits(integer, base: 10)
111
112
  integer.to_s(base).chars
112
113
  end
113
114
 
114
115
  # Returns true if the object is an Integer.
115
116
  #
116
- # @param int [Object] The object to test.
117
+ # @param obj [Object] the object to test.
118
+ #
119
+ # @return [Boolean] true if the object is an Integer, otherwise false.
117
120
  #
118
- # @return [Boolean] True if the object is an Integer, otherwise false.
119
- def integer?(int)
120
- int.is_a?(Integer)
121
+ # @example
122
+ # IntegerTools.integer?(nil)
123
+ # #=> false
124
+ # IntegerTools.integer?([])
125
+ # #=> false
126
+ # IntegerTools.integer?({})
127
+ # #=> false
128
+ # IntegerTools.integer?(1)
129
+ # #=> true
130
+ def integer?(obj)
131
+ obj.is_a?(Integer)
121
132
  end
122
133
 
123
- # Returns the singular or the plural value, depending on the provided
124
- # item count.
134
+ # Returns the singular or the plural value, depending on the provided count.
125
135
  #
126
- # @example
127
- # "There are four #{StringTools.pluralize 4, 'light', 'lights'}!"
128
- # #=> 'There are four lights!'
136
+ # If the plural form is not given, passes the singular form to
137
+ # StringTools#pluralize.
129
138
  #
130
- # @param [Integer] count The number of items.
131
- # @param [String] single The singular form of the word or phrase.
132
- # @param [String] plural The plural form of the word or phrase.
139
+ # @param count [Integer] the number of items.
140
+ # @param single [String] the singular form of the word or phrase.
141
+ # @param plural [String] the plural form of the word or phrase.
133
142
  #
134
143
  # @return [String] The single form if count == 1; otherwise the plural
135
144
  # form.
145
+ #
146
+ # @example
147
+ # IntegerTools.pluralize 4, 'light'
148
+ # #=> 'lights'
149
+ #
150
+ # IntegerTools.pluralize 3, 'cow', 'kine'
151
+ # #=> 'kine'
136
152
  def pluralize(count, single, plural = nil)
137
153
  plural ||= StringTools.pluralize(single)
138
154
 
@@ -141,6 +157,15 @@ module SleepingKingStudios::Tools
141
157
 
142
158
  # Represents an integer between 1 and 4999 (inclusive) as a Roman numeral.
143
159
  #
160
+ # @param integer [Integer] the integer to convert.
161
+ # @param additive [Boolean] if true, then uses only additive Roman numerals
162
+ # (e.g. four will be converted to IIII instead of IV, and nine will be
163
+ # converted to VIIII instead of IX). Defaults to false.
164
+ #
165
+ # @return [String] the representation of the integer as a Roman numeral.
166
+ #
167
+ # @raise [RangeError] if the integer is less than 1 or greater than 4999.
168
+ #
144
169
  # @example
145
170
  # IntegerTools.romanize(4) #=> 'IV'
146
171
  #
@@ -149,15 +174,6 @@ module SleepingKingStudios::Tools
149
174
  #
150
175
  # @example
151
176
  # IntegerTools.romanize(499) #=> 'CDXCIX'
152
- #
153
- # @param [Integer] integer The integer to convert.
154
- # @param [Boolean] additive If true, then uses only additive Roman numerals
155
- # (e.g. four will be converted to IIII instead of IV, and nine will be
156
- # converted to VIIII instead of IX). Defaults to false.
157
- #
158
- # @return [String] The representation of the integer as a Roman numeral.
159
- #
160
- # @raise [RangeError] If the integer is less than 1 or greater than 4999.
161
177
  def romanize(integer, additive: false)
162
178
  check_romanize_range(integer)
163
179
 
@@ -165,7 +181,7 @@ module SleepingKingStudios::Tools
165
181
  .reverse
166
182
  .map
167
183
  .with_index do |digit, index|
168
- romanize_digit(additive: additive, digit: digit.to_i, tens: index)
184
+ romanize_digit(additive:, digit: digit.to_i, tens: index)
169
185
  end
170
186
  .reverse
171
187
  .join