omg-activesupport 8.0.0.alpha1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (289) hide show
  1. checksums.yaml +7 -0
  2. data/CHANGELOG.md +86 -0
  3. data/MIT-LICENSE +20 -0
  4. data/README.rdoc +40 -0
  5. data/lib/active_support/actionable_error.rb +50 -0
  6. data/lib/active_support/all.rb +5 -0
  7. data/lib/active_support/array_inquirer.rb +50 -0
  8. data/lib/active_support/backtrace_cleaner.rb +163 -0
  9. data/lib/active_support/benchmark.rb +21 -0
  10. data/lib/active_support/benchmarkable.rb +53 -0
  11. data/lib/active_support/broadcast_logger.rb +251 -0
  12. data/lib/active_support/builder.rb +8 -0
  13. data/lib/active_support/cache/coder.rb +153 -0
  14. data/lib/active_support/cache/entry.rb +134 -0
  15. data/lib/active_support/cache/file_store.rb +244 -0
  16. data/lib/active_support/cache/mem_cache_store.rb +290 -0
  17. data/lib/active_support/cache/memory_store.rb +262 -0
  18. data/lib/active_support/cache/null_store.rb +62 -0
  19. data/lib/active_support/cache/redis_cache_store.rb +492 -0
  20. data/lib/active_support/cache/serializer_with_fallback.rb +152 -0
  21. data/lib/active_support/cache/strategy/local_cache.rb +201 -0
  22. data/lib/active_support/cache/strategy/local_cache_middleware.rb +45 -0
  23. data/lib/active_support/cache.rb +1104 -0
  24. data/lib/active_support/callbacks.rb +944 -0
  25. data/lib/active_support/class_attribute.rb +26 -0
  26. data/lib/active_support/code_generator.rb +79 -0
  27. data/lib/active_support/concern.rb +217 -0
  28. data/lib/active_support/concurrency/load_interlock_aware_monitor.rb +72 -0
  29. data/lib/active_support/concurrency/null_lock.rb +13 -0
  30. data/lib/active_support/concurrency/share_lock.rb +225 -0
  31. data/lib/active_support/configurable.rb +159 -0
  32. data/lib/active_support/configuration_file.rb +60 -0
  33. data/lib/active_support/core_ext/array/access.rb +100 -0
  34. data/lib/active_support/core_ext/array/conversions.rb +213 -0
  35. data/lib/active_support/core_ext/array/extract.rb +21 -0
  36. data/lib/active_support/core_ext/array/extract_options.rb +31 -0
  37. data/lib/active_support/core_ext/array/grouping.rb +109 -0
  38. data/lib/active_support/core_ext/array/inquiry.rb +19 -0
  39. data/lib/active_support/core_ext/array/wrap.rb +48 -0
  40. data/lib/active_support/core_ext/array.rb +9 -0
  41. data/lib/active_support/core_ext/benchmark.rb +13 -0
  42. data/lib/active_support/core_ext/big_decimal/conversions.rb +14 -0
  43. data/lib/active_support/core_ext/big_decimal.rb +3 -0
  44. data/lib/active_support/core_ext/class/attribute.rb +122 -0
  45. data/lib/active_support/core_ext/class/attribute_accessors.rb +6 -0
  46. data/lib/active_support/core_ext/class/subclasses.rb +24 -0
  47. data/lib/active_support/core_ext/class.rb +4 -0
  48. data/lib/active_support/core_ext/date/acts_like.rb +10 -0
  49. data/lib/active_support/core_ext/date/blank.rb +18 -0
  50. data/lib/active_support/core_ext/date/calculations.rb +161 -0
  51. data/lib/active_support/core_ext/date/conversions.rb +98 -0
  52. data/lib/active_support/core_ext/date/zones.rb +8 -0
  53. data/lib/active_support/core_ext/date.rb +7 -0
  54. data/lib/active_support/core_ext/date_and_time/calculations.rb +374 -0
  55. data/lib/active_support/core_ext/date_and_time/compatibility.rb +58 -0
  56. data/lib/active_support/core_ext/date_and_time/zones.rb +40 -0
  57. data/lib/active_support/core_ext/date_time/acts_like.rb +16 -0
  58. data/lib/active_support/core_ext/date_time/blank.rb +18 -0
  59. data/lib/active_support/core_ext/date_time/calculations.rb +215 -0
  60. data/lib/active_support/core_ext/date_time/compatibility.rb +18 -0
  61. data/lib/active_support/core_ext/date_time/conversions.rb +106 -0
  62. data/lib/active_support/core_ext/date_time.rb +7 -0
  63. data/lib/active_support/core_ext/digest/uuid.rb +76 -0
  64. data/lib/active_support/core_ext/digest.rb +3 -0
  65. data/lib/active_support/core_ext/enumerable.rb +267 -0
  66. data/lib/active_support/core_ext/erb/util.rb +201 -0
  67. data/lib/active_support/core_ext/file/atomic.rb +72 -0
  68. data/lib/active_support/core_ext/file.rb +3 -0
  69. data/lib/active_support/core_ext/hash/conversions.rb +262 -0
  70. data/lib/active_support/core_ext/hash/deep_merge.rb +42 -0
  71. data/lib/active_support/core_ext/hash/deep_transform_values.rb +46 -0
  72. data/lib/active_support/core_ext/hash/except.rb +12 -0
  73. data/lib/active_support/core_ext/hash/indifferent_access.rb +24 -0
  74. data/lib/active_support/core_ext/hash/keys.rb +143 -0
  75. data/lib/active_support/core_ext/hash/reverse_merge.rb +25 -0
  76. data/lib/active_support/core_ext/hash/slice.rb +27 -0
  77. data/lib/active_support/core_ext/hash.rb +10 -0
  78. data/lib/active_support/core_ext/integer/inflections.rb +31 -0
  79. data/lib/active_support/core_ext/integer/multiple.rb +12 -0
  80. data/lib/active_support/core_ext/integer/time.rb +22 -0
  81. data/lib/active_support/core_ext/integer.rb +5 -0
  82. data/lib/active_support/core_ext/kernel/concern.rb +14 -0
  83. data/lib/active_support/core_ext/kernel/reporting.rb +45 -0
  84. data/lib/active_support/core_ext/kernel/singleton_class.rb +8 -0
  85. data/lib/active_support/core_ext/kernel.rb +5 -0
  86. data/lib/active_support/core_ext/load_error.rb +9 -0
  87. data/lib/active_support/core_ext/module/aliasing.rb +31 -0
  88. data/lib/active_support/core_ext/module/anonymous.rb +30 -0
  89. data/lib/active_support/core_ext/module/attr_internal.rb +49 -0
  90. data/lib/active_support/core_ext/module/attribute_accessors.rb +214 -0
  91. data/lib/active_support/core_ext/module/attribute_accessors_per_thread.rb +175 -0
  92. data/lib/active_support/core_ext/module/concerning.rb +140 -0
  93. data/lib/active_support/core_ext/module/delegation.rb +225 -0
  94. data/lib/active_support/core_ext/module/deprecation.rb +25 -0
  95. data/lib/active_support/core_ext/module/introspection.rb +62 -0
  96. data/lib/active_support/core_ext/module/redefine_method.rb +40 -0
  97. data/lib/active_support/core_ext/module/remove_method.rb +17 -0
  98. data/lib/active_support/core_ext/module.rb +13 -0
  99. data/lib/active_support/core_ext/name_error.rb +59 -0
  100. data/lib/active_support/core_ext/numeric/bytes.rb +75 -0
  101. data/lib/active_support/core_ext/numeric/conversions.rb +145 -0
  102. data/lib/active_support/core_ext/numeric/time.rb +66 -0
  103. data/lib/active_support/core_ext/numeric.rb +5 -0
  104. data/lib/active_support/core_ext/object/acts_like.rb +45 -0
  105. data/lib/active_support/core_ext/object/blank.rb +199 -0
  106. data/lib/active_support/core_ext/object/conversions.rb +6 -0
  107. data/lib/active_support/core_ext/object/deep_dup.rb +71 -0
  108. data/lib/active_support/core_ext/object/duplicable.rb +69 -0
  109. data/lib/active_support/core_ext/object/inclusion.rb +37 -0
  110. data/lib/active_support/core_ext/object/instance_variables.rb +32 -0
  111. data/lib/active_support/core_ext/object/json.rb +260 -0
  112. data/lib/active_support/core_ext/object/to_param.rb +3 -0
  113. data/lib/active_support/core_ext/object/to_query.rb +87 -0
  114. data/lib/active_support/core_ext/object/try.rb +158 -0
  115. data/lib/active_support/core_ext/object/with.rb +46 -0
  116. data/lib/active_support/core_ext/object/with_options.rb +101 -0
  117. data/lib/active_support/core_ext/object.rb +17 -0
  118. data/lib/active_support/core_ext/pathname/blank.rb +20 -0
  119. data/lib/active_support/core_ext/pathname/existence.rb +23 -0
  120. data/lib/active_support/core_ext/pathname.rb +4 -0
  121. data/lib/active_support/core_ext/range/compare_range.rb +57 -0
  122. data/lib/active_support/core_ext/range/conversions.rb +62 -0
  123. data/lib/active_support/core_ext/range/each.rb +24 -0
  124. data/lib/active_support/core_ext/range/overlap.rb +40 -0
  125. data/lib/active_support/core_ext/range.rb +6 -0
  126. data/lib/active_support/core_ext/regexp.rb +14 -0
  127. data/lib/active_support/core_ext/securerandom.rb +41 -0
  128. data/lib/active_support/core_ext/string/access.rb +95 -0
  129. data/lib/active_support/core_ext/string/behavior.rb +8 -0
  130. data/lib/active_support/core_ext/string/conversions.rb +60 -0
  131. data/lib/active_support/core_ext/string/exclude.rb +13 -0
  132. data/lib/active_support/core_ext/string/filters.rb +151 -0
  133. data/lib/active_support/core_ext/string/indent.rb +45 -0
  134. data/lib/active_support/core_ext/string/inflections.rb +300 -0
  135. data/lib/active_support/core_ext/string/inquiry.rb +16 -0
  136. data/lib/active_support/core_ext/string/multibyte.rb +58 -0
  137. data/lib/active_support/core_ext/string/output_safety.rb +228 -0
  138. data/lib/active_support/core_ext/string/starts_ends_with.rb +6 -0
  139. data/lib/active_support/core_ext/string/strip.rb +27 -0
  140. data/lib/active_support/core_ext/string/zones.rb +16 -0
  141. data/lib/active_support/core_ext/string.rb +15 -0
  142. data/lib/active_support/core_ext/symbol/starts_ends_with.rb +6 -0
  143. data/lib/active_support/core_ext/symbol.rb +3 -0
  144. data/lib/active_support/core_ext/thread/backtrace/location.rb +12 -0
  145. data/lib/active_support/core_ext/time/acts_like.rb +10 -0
  146. data/lib/active_support/core_ext/time/calculations.rb +386 -0
  147. data/lib/active_support/core_ext/time/compatibility.rb +32 -0
  148. data/lib/active_support/core_ext/time/conversions.rb +75 -0
  149. data/lib/active_support/core_ext/time/zones.rb +97 -0
  150. data/lib/active_support/core_ext/time.rb +7 -0
  151. data/lib/active_support/core_ext.rb +5 -0
  152. data/lib/active_support/current_attributes/test_helper.rb +13 -0
  153. data/lib/active_support/current_attributes.rb +233 -0
  154. data/lib/active_support/deep_mergeable.rb +53 -0
  155. data/lib/active_support/delegation.rb +202 -0
  156. data/lib/active_support/dependencies/autoload.rb +72 -0
  157. data/lib/active_support/dependencies/interlock.rb +49 -0
  158. data/lib/active_support/dependencies/require_dependency.rb +28 -0
  159. data/lib/active_support/dependencies.rb +98 -0
  160. data/lib/active_support/deprecation/behaviors.rb +148 -0
  161. data/lib/active_support/deprecation/constant_accessor.rb +74 -0
  162. data/lib/active_support/deprecation/deprecators.rb +104 -0
  163. data/lib/active_support/deprecation/disallowed.rb +54 -0
  164. data/lib/active_support/deprecation/method_wrappers.rb +68 -0
  165. data/lib/active_support/deprecation/proxy_wrappers.rb +189 -0
  166. data/lib/active_support/deprecation/reporting.rb +179 -0
  167. data/lib/active_support/deprecation.rb +81 -0
  168. data/lib/active_support/deprecator.rb +7 -0
  169. data/lib/active_support/descendants_tracker.rb +112 -0
  170. data/lib/active_support/digest.rb +22 -0
  171. data/lib/active_support/duration/iso8601_parser.rb +123 -0
  172. data/lib/active_support/duration/iso8601_serializer.rb +64 -0
  173. data/lib/active_support/duration.rb +520 -0
  174. data/lib/active_support/encrypted_configuration.rb +126 -0
  175. data/lib/active_support/encrypted_file.rb +133 -0
  176. data/lib/active_support/environment_inquirer.rb +40 -0
  177. data/lib/active_support/error_reporter/test_helper.rb +15 -0
  178. data/lib/active_support/error_reporter.rb +265 -0
  179. data/lib/active_support/evented_file_update_checker.rb +182 -0
  180. data/lib/active_support/execution_context/test_helper.rb +13 -0
  181. data/lib/active_support/execution_context.rb +53 -0
  182. data/lib/active_support/execution_wrapper.rb +150 -0
  183. data/lib/active_support/executor/test_helper.rb +7 -0
  184. data/lib/active_support/executor.rb +8 -0
  185. data/lib/active_support/file_update_checker.rb +164 -0
  186. data/lib/active_support/fork_tracker.rb +43 -0
  187. data/lib/active_support/gem_version.rb +17 -0
  188. data/lib/active_support/gzip.rb +40 -0
  189. data/lib/active_support/hash_with_indifferent_access.rb +445 -0
  190. data/lib/active_support/html_safe_translation.rb +56 -0
  191. data/lib/active_support/i18n.rb +17 -0
  192. data/lib/active_support/i18n_railtie.rb +138 -0
  193. data/lib/active_support/inflections.rb +72 -0
  194. data/lib/active_support/inflector/inflections.rb +273 -0
  195. data/lib/active_support/inflector/methods.rb +387 -0
  196. data/lib/active_support/inflector/transliterate.rb +149 -0
  197. data/lib/active_support/inflector.rb +9 -0
  198. data/lib/active_support/isolated_execution_state.rb +75 -0
  199. data/lib/active_support/json/decoding.rb +76 -0
  200. data/lib/active_support/json/encoding.rb +120 -0
  201. data/lib/active_support/json.rb +4 -0
  202. data/lib/active_support/key_generator.rb +66 -0
  203. data/lib/active_support/lazy_load_hooks.rb +107 -0
  204. data/lib/active_support/locale/en.rb +33 -0
  205. data/lib/active_support/locale/en.yml +141 -0
  206. data/lib/active_support/log_subscriber/test_helper.rb +106 -0
  207. data/lib/active_support/log_subscriber.rb +192 -0
  208. data/lib/active_support/logger.rb +55 -0
  209. data/lib/active_support/logger_silence.rb +21 -0
  210. data/lib/active_support/logger_thread_safe_level.rb +47 -0
  211. data/lib/active_support/message_encryptor.rb +374 -0
  212. data/lib/active_support/message_encryptors.rb +141 -0
  213. data/lib/active_support/message_pack/cache_serializer.rb +23 -0
  214. data/lib/active_support/message_pack/extensions.rb +305 -0
  215. data/lib/active_support/message_pack/serializer.rb +63 -0
  216. data/lib/active_support/message_pack.rb +50 -0
  217. data/lib/active_support/message_verifier.rb +368 -0
  218. data/lib/active_support/message_verifiers.rb +135 -0
  219. data/lib/active_support/messages/codec.rb +65 -0
  220. data/lib/active_support/messages/metadata.rb +146 -0
  221. data/lib/active_support/messages/rotation_configuration.rb +23 -0
  222. data/lib/active_support/messages/rotation_coordinator.rb +93 -0
  223. data/lib/active_support/messages/rotator.rb +59 -0
  224. data/lib/active_support/messages/serializer_with_fallback.rb +158 -0
  225. data/lib/active_support/multibyte/chars.rb +178 -0
  226. data/lib/active_support/multibyte/unicode.rb +42 -0
  227. data/lib/active_support/multibyte.rb +23 -0
  228. data/lib/active_support/notifications/fanout.rb +446 -0
  229. data/lib/active_support/notifications/instrumenter.rb +240 -0
  230. data/lib/active_support/notifications.rb +281 -0
  231. data/lib/active_support/number_helper/number_converter.rb +190 -0
  232. data/lib/active_support/number_helper/number_to_currency_converter.rb +46 -0
  233. data/lib/active_support/number_helper/number_to_delimited_converter.rb +30 -0
  234. data/lib/active_support/number_helper/number_to_human_converter.rb +69 -0
  235. data/lib/active_support/number_helper/number_to_human_size_converter.rb +60 -0
  236. data/lib/active_support/number_helper/number_to_percentage_converter.rb +16 -0
  237. data/lib/active_support/number_helper/number_to_phone_converter.rb +60 -0
  238. data/lib/active_support/number_helper/number_to_rounded_converter.rb +59 -0
  239. data/lib/active_support/number_helper/rounding_helper.rb +46 -0
  240. data/lib/active_support/number_helper.rb +479 -0
  241. data/lib/active_support/option_merger.rb +38 -0
  242. data/lib/active_support/ordered_hash.rb +50 -0
  243. data/lib/active_support/ordered_options.rb +147 -0
  244. data/lib/active_support/parameter_filter.rb +157 -0
  245. data/lib/active_support/proxy_object.rb +20 -0
  246. data/lib/active_support/rails.rb +26 -0
  247. data/lib/active_support/railtie.rb +161 -0
  248. data/lib/active_support/reloader.rb +138 -0
  249. data/lib/active_support/rescuable.rb +176 -0
  250. data/lib/active_support/secure_compare_rotator.rb +58 -0
  251. data/lib/active_support/security_utils.rb +38 -0
  252. data/lib/active_support/string_inquirer.rb +35 -0
  253. data/lib/active_support/subscriber.rb +146 -0
  254. data/lib/active_support/syntax_error_proxy.rb +60 -0
  255. data/lib/active_support/tagged_logging.rb +152 -0
  256. data/lib/active_support/test_case.rb +304 -0
  257. data/lib/active_support/testing/assertions.rb +332 -0
  258. data/lib/active_support/testing/autorun.rb +5 -0
  259. data/lib/active_support/testing/constant_lookup.rb +51 -0
  260. data/lib/active_support/testing/constant_stubbing.rb +54 -0
  261. data/lib/active_support/testing/declarative.rb +28 -0
  262. data/lib/active_support/testing/deprecation.rb +82 -0
  263. data/lib/active_support/testing/error_reporter_assertions.rb +107 -0
  264. data/lib/active_support/testing/file_fixtures.rb +38 -0
  265. data/lib/active_support/testing/isolation.rb +121 -0
  266. data/lib/active_support/testing/method_call_assertions.rb +69 -0
  267. data/lib/active_support/testing/parallelization/server.rb +85 -0
  268. data/lib/active_support/testing/parallelization/worker.rb +103 -0
  269. data/lib/active_support/testing/parallelization.rb +55 -0
  270. data/lib/active_support/testing/parallelize_executor.rb +81 -0
  271. data/lib/active_support/testing/setup_and_teardown.rb +57 -0
  272. data/lib/active_support/testing/stream.rb +41 -0
  273. data/lib/active_support/testing/strict_warnings.rb +43 -0
  274. data/lib/active_support/testing/tagged_logging.rb +27 -0
  275. data/lib/active_support/testing/tests_without_assertions.rb +19 -0
  276. data/lib/active_support/testing/time_helpers.rb +269 -0
  277. data/lib/active_support/time.rb +20 -0
  278. data/lib/active_support/time_with_zone.rb +609 -0
  279. data/lib/active_support/values/time_zone.rb +614 -0
  280. data/lib/active_support/version.rb +10 -0
  281. data/lib/active_support/xml_mini/jdom.rb +175 -0
  282. data/lib/active_support/xml_mini/libxml.rb +80 -0
  283. data/lib/active_support/xml_mini/libxmlsax.rb +83 -0
  284. data/lib/active_support/xml_mini/nokogiri.rb +83 -0
  285. data/lib/active_support/xml_mini/nokogirisax.rb +86 -0
  286. data/lib/active_support/xml_mini/rexml.rb +137 -0
  287. data/lib/active_support/xml_mini.rb +211 -0
  288. data/lib/active_support.rb +144 -0
  289. metadata +526 -0
@@ -0,0 +1,192 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/core_ext/module/attribute_accessors"
4
+ require "active_support/core_ext/class/attribute"
5
+ require "active_support/core_ext/enumerable"
6
+ require "active_support/subscriber"
7
+ require "active_support/deprecation/proxy_wrappers"
8
+
9
+ module ActiveSupport
10
+ # = Active Support Log \Subscriber
11
+ #
12
+ # +ActiveSupport::LogSubscriber+ is an object set to consume
13
+ # ActiveSupport::Notifications with the sole purpose of logging them.
14
+ # The log subscriber dispatches notifications to a registered object based
15
+ # on its given namespace.
16
+ #
17
+ # An example would be Active Record log subscriber responsible for logging
18
+ # queries:
19
+ #
20
+ # module ActiveRecord
21
+ # class LogSubscriber < ActiveSupport::LogSubscriber
22
+ # attach_to :active_record
23
+ #
24
+ # def sql(event)
25
+ # info "#{event.payload[:name]} (#{event.duration}) #{event.payload[:sql]}"
26
+ # end
27
+ # end
28
+ # end
29
+ #
30
+ # ActiveRecord::LogSubscriber.logger must be set as well, but it is assigned
31
+ # automatically in a \Rails environment.
32
+ #
33
+ # After configured, whenever a <tt>"sql.active_record"</tt> notification is
34
+ # published, it will properly dispatch the event
35
+ # (ActiveSupport::Notifications::Event) to the +sql+ method.
36
+ #
37
+ # Being an ActiveSupport::Notifications consumer,
38
+ # +ActiveSupport::LogSubscriber+ exposes a simple interface to check if
39
+ # instrumented code raises an exception. It is common to log a different
40
+ # message in case of an error, and this can be achieved by extending
41
+ # the previous example:
42
+ #
43
+ # module ActiveRecord
44
+ # class LogSubscriber < ActiveSupport::LogSubscriber
45
+ # def sql(event)
46
+ # exception = event.payload[:exception]
47
+ #
48
+ # if exception
49
+ # exception_object = event.payload[:exception_object]
50
+ #
51
+ # error "[ERROR] #{event.payload[:name]}: #{exception.join(', ')} " \
52
+ # "(#{exception_object.backtrace.first})"
53
+ # else
54
+ # # standard logger code
55
+ # end
56
+ # end
57
+ # end
58
+ # end
59
+ #
60
+ # +ActiveSupport::LogSubscriber+ also has some helpers to deal with
61
+ # logging. For example, ActiveSupport::LogSubscriber.flush_all! will ensure
62
+ # that all logs are flushed, and it is called in Rails::Rack::Logger after a
63
+ # request finishes.
64
+ class LogSubscriber < Subscriber
65
+ # ANSI sequence modes
66
+ MODES = {
67
+ clear: 0,
68
+ bold: 1,
69
+ italic: 3,
70
+ underline: 4,
71
+ }
72
+
73
+ # ANSI sequence colors
74
+ BLACK = "\e[30m"
75
+ RED = "\e[31m"
76
+ GREEN = "\e[32m"
77
+ YELLOW = "\e[33m"
78
+ BLUE = "\e[34m"
79
+ MAGENTA = "\e[35m"
80
+ CYAN = "\e[36m"
81
+ WHITE = "\e[37m"
82
+
83
+ mattr_accessor :colorize_logging, default: true
84
+ class_attribute :log_levels, instance_accessor: false, default: {} # :nodoc:
85
+
86
+ LEVEL_CHECKS = {
87
+ debug: -> (logger) { !logger.debug? },
88
+ info: -> (logger) { !logger.info? },
89
+ error: -> (logger) { !logger.error? },
90
+ }
91
+
92
+ class << self
93
+ def logger
94
+ @logger ||= if defined?(Rails) && Rails.respond_to?(:logger)
95
+ Rails.logger
96
+ end
97
+ end
98
+
99
+ def attach_to(...) # :nodoc:
100
+ result = super
101
+ set_event_levels
102
+ result
103
+ end
104
+
105
+ attr_writer :logger
106
+
107
+ def log_subscribers
108
+ subscribers
109
+ end
110
+
111
+ # Flush all log_subscribers' logger.
112
+ def flush_all!
113
+ logger.flush if logger.respond_to?(:flush)
114
+ end
115
+
116
+ private
117
+ def fetch_public_methods(subscriber, inherit_all)
118
+ subscriber.public_methods(inherit_all) - LogSubscriber.public_instance_methods(true)
119
+ end
120
+
121
+ def set_event_levels
122
+ if subscriber
123
+ subscriber.event_levels = log_levels.transform_keys { |k| "#{k}.#{namespace}" }
124
+ end
125
+ end
126
+
127
+ def subscribe_log_level(method, level)
128
+ self.log_levels = log_levels.merge(method => LEVEL_CHECKS.fetch(level))
129
+ set_event_levels
130
+ end
131
+ end
132
+
133
+ def initialize
134
+ super
135
+ @event_levels = {}
136
+ end
137
+
138
+ def logger
139
+ LogSubscriber.logger
140
+ end
141
+
142
+ def silenced?(event)
143
+ logger.nil? || @event_levels[event]&.call(logger)
144
+ end
145
+
146
+ def call(event)
147
+ super if logger
148
+ rescue => e
149
+ log_exception(event.name, e)
150
+ end
151
+
152
+ def publish_event(event)
153
+ super if logger
154
+ rescue => e
155
+ log_exception(event.name, e)
156
+ end
157
+
158
+ attr_writer :event_levels # :nodoc:
159
+
160
+ private
161
+ %w(info debug warn error fatal unknown).each do |level|
162
+ class_eval <<-METHOD, __FILE__, __LINE__ + 1
163
+ def #{level}(progname = nil, &block)
164
+ logger.#{level}(progname, &block) if logger
165
+ end
166
+ METHOD
167
+ end
168
+
169
+ # Set color by using a symbol or one of the defined constants. Set modes
170
+ # by specifying bold, italic, or underline options. Inspired by Highline,
171
+ # this method will automatically clear formatting at the end of the returned String.
172
+ def color(text, color, mode_options = {}) # :doc:
173
+ return text unless colorize_logging
174
+ color = self.class.const_get(color.upcase) if color.is_a?(Symbol)
175
+ mode = mode_from(mode_options)
176
+ clear = "\e[#{MODES[:clear]}m"
177
+ "#{mode}#{color}#{text}#{clear}"
178
+ end
179
+
180
+ def mode_from(options)
181
+ modes = MODES.values_at(*options.compact_blank.keys)
182
+
183
+ "\e[#{modes.join(";")}m" if modes.any?
184
+ end
185
+
186
+ def log_exception(name, e)
187
+ if logger
188
+ logger.error "Could not log #{name.inspect} event. #{e.class}: #{e.message} #{e.backtrace}"
189
+ end
190
+ end
191
+ end
192
+ end
@@ -0,0 +1,55 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/logger_silence"
4
+ require "active_support/logger_thread_safe_level"
5
+ require "logger"
6
+
7
+ module ActiveSupport
8
+ class Logger < ::Logger
9
+ include LoggerSilence
10
+
11
+ # Returns true if the logger destination matches one of the sources
12
+ #
13
+ # logger = Logger.new(STDOUT)
14
+ # ActiveSupport::Logger.logger_outputs_to?(logger, STDOUT)
15
+ # # => true
16
+ #
17
+ # logger = Logger.new('/var/log/rails.log')
18
+ # ActiveSupport::Logger.logger_outputs_to?(logger, '/var/log/rails.log')
19
+ # # => true
20
+ def self.logger_outputs_to?(logger, *sources)
21
+ loggers = if logger.is_a?(BroadcastLogger)
22
+ logger.broadcasts
23
+ else
24
+ [logger]
25
+ end
26
+
27
+ logdevs = loggers.map { |logger| logger.instance_variable_get(:@logdev) }
28
+ logger_sources = logdevs.filter_map { |logdev| logdev.try(:filename) || logdev.try(:dev) }
29
+
30
+ normalize_sources(sources).intersect?(normalize_sources(logger_sources))
31
+ end
32
+
33
+ def initialize(*args, **kwargs)
34
+ super
35
+ @formatter ||= SimpleFormatter.new
36
+ end
37
+
38
+ # Simple formatter which only displays the message.
39
+ class SimpleFormatter < ::Logger::Formatter
40
+ # This method is invoked when a log event occurs
41
+ def call(severity, timestamp, progname, msg)
42
+ "#{String === msg ? msg : msg.inspect}\n"
43
+ end
44
+ end
45
+
46
+ private
47
+ def self.normalize_sources(sources)
48
+ sources.map do |source|
49
+ source = source.path if source.respond_to?(:path)
50
+ source = File.realpath(source) if source.is_a?(String) && File.exist?(source)
51
+ source
52
+ end
53
+ end
54
+ end
55
+ end
@@ -0,0 +1,21 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/concern"
4
+ require "active_support/core_ext/module/attribute_accessors"
5
+ require "active_support/logger_thread_safe_level"
6
+
7
+ module ActiveSupport
8
+ module LoggerSilence
9
+ extend ActiveSupport::Concern
10
+
11
+ included do
12
+ cattr_accessor :silencer, default: true
13
+ include ActiveSupport::LoggerThreadSafeLevel
14
+ end
15
+
16
+ # Silences the logger for the duration of the block.
17
+ def silence(severity = Logger::ERROR)
18
+ silencer ? log_at(severity) { yield self } : yield(self)
19
+ end
20
+ end
21
+ end
@@ -0,0 +1,47 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "active_support/concern"
4
+ require "logger"
5
+
6
+ module ActiveSupport
7
+ module LoggerThreadSafeLevel # :nodoc:
8
+ extend ActiveSupport::Concern
9
+
10
+ def local_level
11
+ IsolatedExecutionState[local_level_key]
12
+ end
13
+
14
+ def local_level=(level)
15
+ case level
16
+ when Integer
17
+ when Symbol
18
+ level = Logger::Severity.const_get(level.to_s.upcase)
19
+ when nil
20
+ else
21
+ raise ArgumentError, "Invalid log level: #{level.inspect}"
22
+ end
23
+ if level.nil?
24
+ IsolatedExecutionState.delete(local_level_key)
25
+ else
26
+ IsolatedExecutionState[local_level_key] = level
27
+ end
28
+ end
29
+
30
+ def level
31
+ local_level || super
32
+ end
33
+
34
+ # Change the thread-local level for the duration of the given block.
35
+ def log_at(level)
36
+ old_local_level, self.local_level = local_level, level
37
+ yield
38
+ ensure
39
+ self.local_level = old_local_level
40
+ end
41
+
42
+ private
43
+ def local_level_key
44
+ @local_level_key ||= :"logger_thread_safe_level_#{object_id}"
45
+ end
46
+ end
47
+ end
@@ -0,0 +1,374 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "openssl"
4
+ require "base64"
5
+ require "active_support/core_ext/module/attribute_accessors"
6
+ require "active_support/messages/codec"
7
+ require "active_support/messages/rotator"
8
+ require "active_support/message_verifier"
9
+
10
+ module ActiveSupport
11
+ # = Active Support Message Encryptor
12
+ #
13
+ # MessageEncryptor is a simple way to encrypt values which get stored
14
+ # somewhere you don't trust.
15
+ #
16
+ # The cipher text and initialization vector are base64 encoded and returned
17
+ # to you.
18
+ #
19
+ # This can be used in situations similar to the MessageVerifier, but
20
+ # where you don't want users to be able to determine the value of the payload.
21
+ #
22
+ # len = ActiveSupport::MessageEncryptor.key_len
23
+ # salt = SecureRandom.random_bytes(len)
24
+ # key = ActiveSupport::KeyGenerator.new('password').generate_key(salt, len) # => "\x89\xE0\x156\xAC..."
25
+ # crypt = ActiveSupport::MessageEncryptor.new(key) # => #<ActiveSupport::MessageEncryptor ...>
26
+ # encrypted_data = crypt.encrypt_and_sign('my secret data') # => "NlFBTTMwOUV5UlA1QlNEN2xkY2d6eThYWWh..."
27
+ # crypt.decrypt_and_verify(encrypted_data) # => "my secret data"
28
+ #
29
+ # The +decrypt_and_verify+ method will raise an
30
+ # +ActiveSupport::MessageEncryptor::InvalidMessage+ exception if the data
31
+ # provided cannot be decrypted or verified.
32
+ #
33
+ # crypt.decrypt_and_verify('not encrypted data') # => ActiveSupport::MessageEncryptor::InvalidMessage
34
+ #
35
+ # === Confining messages to a specific purpose
36
+ #
37
+ # By default any message can be used throughout your app. But they can also be
38
+ # confined to a specific +:purpose+.
39
+ #
40
+ # token = crypt.encrypt_and_sign("this is the chair", purpose: :login)
41
+ #
42
+ # Then that same purpose must be passed when verifying to get the data back out:
43
+ #
44
+ # crypt.decrypt_and_verify(token, purpose: :login) # => "this is the chair"
45
+ # crypt.decrypt_and_verify(token, purpose: :shipping) # => nil
46
+ # crypt.decrypt_and_verify(token) # => nil
47
+ #
48
+ # Likewise, if a message has no purpose it won't be returned when verifying with
49
+ # a specific purpose.
50
+ #
51
+ # token = crypt.encrypt_and_sign("the conversation is lively")
52
+ # crypt.decrypt_and_verify(token, purpose: :scare_tactics) # => nil
53
+ # crypt.decrypt_and_verify(token) # => "the conversation is lively"
54
+ #
55
+ # === Making messages expire
56
+ #
57
+ # By default messages last forever and verifying one year from now will still
58
+ # return the original value. But messages can be set to expire at a given
59
+ # time with +:expires_in+ or +:expires_at+.
60
+ #
61
+ # crypt.encrypt_and_sign(parcel, expires_in: 1.month)
62
+ # crypt.encrypt_and_sign(doowad, expires_at: Time.now.end_of_year)
63
+ #
64
+ # Then the messages can be verified and returned up to the expire time.
65
+ # Thereafter, verifying returns +nil+.
66
+ #
67
+ # === Rotating keys
68
+ #
69
+ # MessageEncryptor also supports rotating out old configurations by falling
70
+ # back to a stack of encryptors. Call +rotate+ to build and add an encryptor
71
+ # so +decrypt_and_verify+ will also try the fallback.
72
+ #
73
+ # By default any rotated encryptors use the values of the primary
74
+ # encryptor unless specified otherwise.
75
+ #
76
+ # You'd give your encryptor the new defaults:
77
+ #
78
+ # crypt = ActiveSupport::MessageEncryptor.new(@secret, cipher: "aes-256-gcm")
79
+ #
80
+ # Then gradually rotate the old values out by adding them as fallbacks. Any message
81
+ # generated with the old values will then work until the rotation is removed.
82
+ #
83
+ # crypt.rotate old_secret # Fallback to an old secret instead of @secret.
84
+ # crypt.rotate cipher: "aes-256-cbc" # Fallback to an old cipher instead of aes-256-gcm.
85
+ #
86
+ # Though if both the secret and the cipher was changed at the same time,
87
+ # the above should be combined into:
88
+ #
89
+ # crypt.rotate old_secret, cipher: "aes-256-cbc"
90
+ class MessageEncryptor < Messages::Codec
91
+ prepend Messages::Rotator
92
+
93
+ cattr_accessor :use_authenticated_message_encryption, instance_accessor: false, default: false
94
+
95
+ class << self
96
+ def default_cipher # :nodoc:
97
+ if use_authenticated_message_encryption
98
+ "aes-256-gcm"
99
+ else
100
+ "aes-256-cbc"
101
+ end
102
+ end
103
+ end
104
+
105
+ module NullSerializer # :nodoc:
106
+ def self.load(value)
107
+ value
108
+ end
109
+
110
+ def self.dump(value)
111
+ value
112
+ end
113
+ end
114
+
115
+ class InvalidMessage < StandardError; end
116
+ OpenSSLCipherError = OpenSSL::Cipher::CipherError
117
+
118
+ AUTH_TAG_LENGTH = 16 # :nodoc:
119
+ SEPARATOR = "--" # :nodoc:
120
+
121
+ # Initialize a new MessageEncryptor. +secret+ must be at least as long as
122
+ # the cipher key size. For the default 'aes-256-gcm' cipher, this is 256
123
+ # bits. If you are using a user-entered secret, you can generate a suitable
124
+ # key by using ActiveSupport::KeyGenerator or a similar key
125
+ # derivation function.
126
+ #
127
+ # The first additional parameter is used as the signature key for
128
+ # MessageVerifier. This allows you to specify keys to encrypt and sign
129
+ # data. Ignored when using an AEAD cipher like 'aes-256-gcm'.
130
+ #
131
+ # ActiveSupport::MessageEncryptor.new('secret', 'signature_secret')
132
+ #
133
+ # ==== Options
134
+ #
135
+ # [+:cipher+]
136
+ # Cipher to use. Can be any cipher returned by +OpenSSL::Cipher.ciphers+.
137
+ # Default is 'aes-256-gcm'.
138
+ #
139
+ # [+:digest+]
140
+ # Digest used for signing. Ignored when using an AEAD cipher like
141
+ # 'aes-256-gcm'.
142
+ #
143
+ # [+:serializer+]
144
+ # The serializer used to serialize message data. You can specify any
145
+ # object that responds to +dump+ and +load+, or you can choose from
146
+ # several preconfigured serializers: +:marshal+, +:json_allow_marshal+,
147
+ # +:json+, +:message_pack_allow_marshal+, +:message_pack+.
148
+ #
149
+ # The preconfigured serializers include a fallback mechanism to support
150
+ # multiple deserialization formats. For example, the +:marshal+ serializer
151
+ # will serialize using +Marshal+, but can deserialize using +Marshal+,
152
+ # ActiveSupport::JSON, or ActiveSupport::MessagePack. This makes it easy
153
+ # to migrate between serializers.
154
+ #
155
+ # The +:marshal+, +:json_allow_marshal+, and +:message_pack_allow_marshal+
156
+ # serializers support deserializing using +Marshal+, but the others do
157
+ # not. Beware that +Marshal+ is a potential vector for deserialization
158
+ # attacks in cases where a message signing secret has been leaked. <em>If
159
+ # possible, choose a serializer that does not support +Marshal+.</em>
160
+ #
161
+ # The +:message_pack+ and +:message_pack_allow_marshal+ serializers use
162
+ # ActiveSupport::MessagePack, which can roundtrip some Ruby types that are
163
+ # not supported by JSON, and may provide improved performance. However,
164
+ # these require the +msgpack+ gem.
165
+ #
166
+ # When using \Rails, the default depends on +config.active_support.message_serializer+.
167
+ # Otherwise, the default is +:marshal+.
168
+ #
169
+ # [+:url_safe+]
170
+ # By default, MessageEncryptor generates RFC 4648 compliant strings
171
+ # which are not URL-safe. In other words, they can contain "+" and "/".
172
+ # If you want to generate URL-safe strings (in compliance with "Base 64
173
+ # Encoding with URL and Filename Safe Alphabet" in RFC 4648), you can
174
+ # pass +true+.
175
+ #
176
+ # [+:force_legacy_metadata_serializer+]
177
+ # Whether to use the legacy metadata serializer, which serializes the
178
+ # message first, then wraps it in an envelope which is also serialized. This
179
+ # was the default in \Rails 7.0 and below.
180
+ #
181
+ # If you don't pass a truthy value, the default is set using
182
+ # +config.active_support.use_message_serializer_for_metadata+.
183
+ def initialize(secret, sign_secret = nil, **options)
184
+ super(**options)
185
+ @secret = secret
186
+ @cipher = options[:cipher] || self.class.default_cipher
187
+ @aead_mode = new_cipher.authenticated?
188
+ @verifier = if !@aead_mode
189
+ MessageVerifier.new(sign_secret || secret, **options, serializer: NullSerializer)
190
+ end
191
+ end
192
+
193
+ # Encrypt and sign a message. We need to sign the message in order to avoid
194
+ # padding attacks. Reference: https://www.limited-entropy.com/padding-oracle-attacks/.
195
+ #
196
+ # ==== Options
197
+ #
198
+ # [+:expires_at+]
199
+ # The datetime at which the message expires. After this datetime,
200
+ # verification of the message will fail.
201
+ #
202
+ # message = encryptor.encrypt_and_sign("hello", expires_at: Time.now.tomorrow)
203
+ # encryptor.decrypt_and_verify(message) # => "hello"
204
+ # # 24 hours later...
205
+ # encryptor.decrypt_and_verify(message) # => nil
206
+ #
207
+ # [+:expires_in+]
208
+ # The duration for which the message is valid. After this duration has
209
+ # elapsed, verification of the message will fail.
210
+ #
211
+ # message = encryptor.encrypt_and_sign("hello", expires_in: 24.hours)
212
+ # encryptor.decrypt_and_verify(message) # => "hello"
213
+ # # 24 hours later...
214
+ # encryptor.decrypt_and_verify(message) # => nil
215
+ #
216
+ # [+:purpose+]
217
+ # The purpose of the message. If specified, the same purpose must be
218
+ # specified when verifying the message; otherwise, verification will fail.
219
+ # (See #decrypt_and_verify.)
220
+ def encrypt_and_sign(value, **options)
221
+ create_message(value, **options)
222
+ end
223
+
224
+ # Decrypt and verify a message. We need to verify the message in order to
225
+ # avoid padding attacks. Reference: https://www.limited-entropy.com/padding-oracle-attacks/.
226
+ #
227
+ # ==== Options
228
+ #
229
+ # [+:purpose+]
230
+ # The purpose that the message was generated with. If the purpose does not
231
+ # match, +decrypt_and_verify+ will return +nil+.
232
+ #
233
+ # message = encryptor.encrypt_and_sign("hello", purpose: "greeting")
234
+ # encryptor.decrypt_and_verify(message, purpose: "greeting") # => "hello"
235
+ # encryptor.decrypt_and_verify(message) # => nil
236
+ #
237
+ # message = encryptor.encrypt_and_sign("bye")
238
+ # encryptor.decrypt_and_verify(message) # => "bye"
239
+ # encryptor.decrypt_and_verify(message, purpose: "greeting") # => nil
240
+ #
241
+ def decrypt_and_verify(message, **options)
242
+ catch_and_raise :invalid_message_format, as: InvalidMessage do
243
+ catch_and_raise :invalid_message_serialization, as: InvalidMessage do
244
+ catch_and_ignore :invalid_message_content do
245
+ read_message(message, **options)
246
+ end
247
+ end
248
+ end
249
+ end
250
+
251
+ # Given a cipher, returns the key length of the cipher to help generate the key of desired size
252
+ def self.key_len(cipher = default_cipher)
253
+ OpenSSL::Cipher.new(cipher).key_len
254
+ end
255
+
256
+ def create_message(value, **options) # :nodoc:
257
+ sign(encrypt(serialize_with_metadata(value, **options)))
258
+ end
259
+
260
+ def read_message(message, **options) # :nodoc:
261
+ deserialize_with_metadata(decrypt(verify(message)), **options)
262
+ end
263
+
264
+ def inspect # :nodoc:
265
+ "#<#{self.class.name}:#{'%#016x' % (object_id << 1)}>"
266
+ end
267
+
268
+ private
269
+ def sign(data)
270
+ @verifier ? @verifier.create_message(data) : data
271
+ end
272
+
273
+ def verify(data)
274
+ @verifier ? @verifier.read_message(data) : data
275
+ end
276
+
277
+ def encrypt(data)
278
+ cipher = new_cipher
279
+ cipher.encrypt
280
+ cipher.key = @secret
281
+
282
+ # Rely on OpenSSL for the initialization vector
283
+ iv = cipher.random_iv
284
+ cipher.auth_data = "" if aead_mode?
285
+
286
+ encrypted_data = cipher.update(data)
287
+ encrypted_data << cipher.final
288
+
289
+ parts = [encrypted_data, iv]
290
+ parts << cipher.auth_tag(AUTH_TAG_LENGTH) if aead_mode?
291
+
292
+ join_parts(parts)
293
+ end
294
+
295
+ def decrypt(encrypted_message)
296
+ cipher = new_cipher
297
+ encrypted_data, iv, auth_tag = extract_parts(encrypted_message)
298
+
299
+ # Currently the OpenSSL bindings do not raise an error if auth_tag is
300
+ # truncated, which would allow an attacker to easily forge it. See
301
+ # https://github.com/ruby/openssl/issues/63
302
+ if aead_mode? && auth_tag.bytesize != AUTH_TAG_LENGTH
303
+ throw :invalid_message_format, "truncated auth_tag"
304
+ end
305
+
306
+ cipher.decrypt
307
+ cipher.key = @secret
308
+ cipher.iv = iv
309
+ if aead_mode?
310
+ cipher.auth_tag = auth_tag
311
+ cipher.auth_data = ""
312
+ end
313
+
314
+ decrypted_data = cipher.update(encrypted_data)
315
+ decrypted_data << cipher.final
316
+ rescue OpenSSLCipherError => error
317
+ throw :invalid_message_format, error
318
+ end
319
+
320
+ def length_after_encode(length_before_encode)
321
+ if @url_safe
322
+ (4 * length_before_encode / 3.0).ceil # length without padding
323
+ else
324
+ 4 * (length_before_encode / 3.0).ceil # length with padding
325
+ end
326
+ end
327
+
328
+ def length_of_encoded_iv
329
+ @length_of_encoded_iv ||= length_after_encode(new_cipher.iv_len)
330
+ end
331
+
332
+ def length_of_encoded_auth_tag
333
+ @length_of_encoded_auth_tag ||= length_after_encode(AUTH_TAG_LENGTH)
334
+ end
335
+
336
+ def join_parts(parts)
337
+ parts.map! { |part| encode(part) }.join(SEPARATOR)
338
+ end
339
+
340
+ def extract_part(encrypted_message, rindex, length)
341
+ index = rindex - length
342
+
343
+ if encrypted_message[index - SEPARATOR.length, SEPARATOR.length] == SEPARATOR
344
+ encrypted_message[index, length]
345
+ else
346
+ throw :invalid_message_format, "missing separator"
347
+ end
348
+ end
349
+
350
+ def extract_parts(encrypted_message)
351
+ parts = []
352
+ rindex = encrypted_message.length
353
+
354
+ if aead_mode?
355
+ parts << extract_part(encrypted_message, rindex, length_of_encoded_auth_tag)
356
+ rindex -= SEPARATOR.length + length_of_encoded_auth_tag
357
+ end
358
+
359
+ parts << extract_part(encrypted_message, rindex, length_of_encoded_iv)
360
+ rindex -= SEPARATOR.length + length_of_encoded_iv
361
+
362
+ parts << encrypted_message[0, rindex]
363
+
364
+ parts.reverse!.map! { |part| decode(part) }
365
+ end
366
+
367
+ def new_cipher
368
+ OpenSSL::Cipher.new(@cipher)
369
+ end
370
+
371
+ attr_reader :aead_mode
372
+ alias :aead_mode? :aead_mode
373
+ end
374
+ end