contrast-agent 4.6.0 → 4.9.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (190) hide show
  1. checksums.yaml +4 -4
  2. data/.gitignore +6 -1
  3. data/.gitmodules +1 -1
  4. data/.simplecov +1 -0
  5. data/Rakefile +1 -2
  6. data/ext/build_funchook.rb +3 -3
  7. data/ext/extconf_common.rb +1 -5
  8. data/lib/contrast.rb +24 -14
  9. data/lib/contrast/agent/assess.rb +1 -1
  10. data/lib/contrast/agent/assess/contrast_event.rb +1 -4
  11. data/lib/contrast/agent/assess/contrast_object.rb +2 -2
  12. data/lib/contrast/agent/assess/events/event_factory.rb +2 -1
  13. data/lib/contrast/agent/assess/finalizers/hash.rb +2 -4
  14. data/lib/contrast/agent/assess/policy/dynamic_source_factory.rb +6 -3
  15. data/lib/contrast/agent/assess/policy/patcher.rb +16 -21
  16. data/lib/contrast/agent/assess/policy/policy.rb +1 -1
  17. data/lib/contrast/agent/assess/policy/policy_node.rb +25 -33
  18. data/lib/contrast/agent/assess/policy/policy_scanner.rb +3 -5
  19. data/lib/contrast/agent/assess/policy/preshift.rb +7 -5
  20. data/lib/contrast/agent/assess/policy/propagation_method.rb +10 -19
  21. data/lib/contrast/agent/assess/policy/propagation_node.rb +19 -8
  22. data/lib/contrast/agent/assess/policy/propagator.rb +1 -0
  23. data/lib/contrast/agent/assess/policy/propagator/center.rb +2 -1
  24. data/lib/contrast/agent/assess/policy/propagator/database_write.rb +3 -6
  25. data/lib/contrast/agent/assess/policy/propagator/insert.rb +3 -1
  26. data/lib/contrast/agent/assess/policy/propagator/match_data.rb +2 -1
  27. data/lib/contrast/agent/assess/policy/propagator/rack_protection.rb +73 -0
  28. data/lib/contrast/agent/assess/policy/propagator/select.rb +2 -12
  29. data/lib/contrast/agent/assess/policy/propagator/split.rb +12 -13
  30. data/lib/contrast/agent/assess/policy/propagator/substitution.rb +3 -10
  31. data/lib/contrast/agent/assess/policy/propagator/trim.rb +3 -15
  32. data/lib/contrast/agent/assess/policy/rewriter_patch.rb +13 -10
  33. data/lib/contrast/agent/assess/policy/source_method.rb +12 -12
  34. data/lib/contrast/agent/assess/policy/source_validation/source_validation.rb +1 -3
  35. data/lib/contrast/agent/assess/policy/trigger/reflected_xss.rb +5 -1
  36. data/lib/contrast/agent/assess/policy/trigger/xpath.rb +0 -3
  37. data/lib/contrast/agent/assess/policy/trigger_method.rb +8 -18
  38. data/lib/contrast/agent/assess/policy/trigger_node.rb +3 -2
  39. data/lib/contrast/agent/assess/policy/trigger_validation/redos_validator.rb +4 -3
  40. data/lib/contrast/agent/assess/policy/trigger_validation/ssrf_validator.rb +1 -2
  41. data/lib/contrast/agent/assess/policy/trigger_validation/xss_validator.rb +1 -8
  42. data/lib/contrast/agent/assess/property/evented.rb +8 -5
  43. data/lib/contrast/agent/assess/rule/provider/hardcoded_key.rb +11 -5
  44. data/lib/contrast/agent/assess/rule/provider/hardcoded_password.rb +4 -1
  45. data/lib/contrast/agent/assess/rule/provider/hardcoded_value_rule.rb +7 -9
  46. data/lib/contrast/agent/at_exit_hook.rb +3 -3
  47. data/lib/contrast/agent/class_reopener.rb +9 -6
  48. data/lib/contrast/agent/disable_reaction.rb +4 -7
  49. data/lib/contrast/agent/exclusion_matcher.rb +7 -14
  50. data/lib/contrast/agent/inventory/dependencies.rb +2 -0
  51. data/lib/contrast/agent/inventory/dependency_analysis.rb +2 -6
  52. data/lib/contrast/agent/inventory/dependency_usage_analysis.rb +3 -5
  53. data/lib/contrast/agent/inventory/policy/datastores.rb +3 -4
  54. data/lib/contrast/agent/inventory/policy/policy.rb +1 -1
  55. data/lib/contrast/agent/middleware.rb +17 -18
  56. data/lib/contrast/agent/module_data.rb +3 -3
  57. data/lib/contrast/agent/patching/policy/after_load_patch.rb +3 -3
  58. data/lib/contrast/agent/patching/policy/after_load_patcher.rb +9 -9
  59. data/lib/contrast/agent/patching/policy/method_policy.rb +6 -2
  60. data/lib/contrast/agent/patching/policy/module_policy.rb +14 -7
  61. data/lib/contrast/agent/patching/policy/patch.rb +20 -25
  62. data/lib/contrast/agent/patching/policy/patch_status.rb +6 -7
  63. data/lib/contrast/agent/patching/policy/patcher.rb +21 -18
  64. data/lib/contrast/agent/patching/policy/policy.rb +2 -4
  65. data/lib/contrast/agent/patching/policy/policy_node.rb +16 -7
  66. data/lib/contrast/agent/patching/policy/trigger_node.rb +21 -8
  67. data/lib/contrast/agent/protect/policy/applies_command_injection_rule.rb +1 -1
  68. data/lib/contrast/agent/protect/policy/applies_deserialization_rule.rb +1 -1
  69. data/lib/contrast/agent/protect/policy/applies_no_sqli_rule.rb +1 -1
  70. data/lib/contrast/agent/protect/policy/applies_path_traversal_rule.rb +2 -3
  71. data/lib/contrast/agent/protect/policy/applies_sqli_rule.rb +1 -1
  72. data/lib/contrast/agent/protect/policy/applies_xxe_rule.rb +5 -9
  73. data/lib/contrast/agent/protect/policy/policy.rb +1 -1
  74. data/lib/contrast/agent/protect/policy/rule_applicator.rb +7 -9
  75. data/lib/contrast/agent/protect/rule/base.rb +20 -23
  76. data/lib/contrast/agent/protect/rule/base_service.rb +9 -5
  77. data/lib/contrast/agent/protect/rule/cmd_injection.rb +18 -23
  78. data/lib/contrast/agent/protect/rule/deserialization.rb +6 -13
  79. data/lib/contrast/agent/protect/rule/http_method_tampering.rb +3 -14
  80. data/lib/contrast/agent/protect/rule/no_sqli.rb +6 -2
  81. data/lib/contrast/agent/protect/rule/no_sqli/mongo_no_sql_scanner.rb +1 -3
  82. data/lib/contrast/agent/protect/rule/path_traversal.rb +6 -10
  83. data/lib/contrast/agent/protect/rule/sqli.rb +1 -1
  84. data/lib/contrast/agent/protect/rule/unsafe_file_upload.rb +1 -1
  85. data/lib/contrast/agent/protect/rule/xss.rb +1 -1
  86. data/lib/contrast/agent/protect/rule/xxe.rb +5 -12
  87. data/lib/contrast/agent/protect/rule/xxe/entity_wrapper.rb +1 -2
  88. data/lib/contrast/agent/reaction_processor.rb +13 -13
  89. data/lib/contrast/agent/request.rb +27 -26
  90. data/lib/contrast/agent/request_context.rb +17 -22
  91. data/lib/contrast/agent/request_handler.rb +5 -3
  92. data/lib/contrast/agent/response.rb +2 -3
  93. data/lib/contrast/agent/rewriter.rb +9 -6
  94. data/lib/contrast/agent/rule_set.rb +5 -4
  95. data/lib/contrast/agent/service_heartbeat.rb +4 -6
  96. data/lib/contrast/agent/static_analysis.rb +6 -5
  97. data/lib/contrast/agent/thread.rb +2 -4
  98. data/lib/contrast/agent/thread_watcher.rb +3 -4
  99. data/lib/contrast/agent/tracepoint_hook.rb +5 -5
  100. data/lib/contrast/agent/version.rb +1 -1
  101. data/lib/contrast/api/communication/messaging_queue.rb +4 -5
  102. data/lib/contrast/api/communication/response_processor.rb +11 -13
  103. data/lib/contrast/api/communication/service_lifecycle.rb +9 -6
  104. data/lib/contrast/api/communication/socket_client.rb +22 -31
  105. data/lib/contrast/api/communication/speedracer.rb +8 -13
  106. data/lib/contrast/api/decorators/address.rb +2 -3
  107. data/lib/contrast/api/decorators/agent_startup.rb +7 -9
  108. data/lib/contrast/api/decorators/application_startup.rb +12 -10
  109. data/lib/contrast/api/decorators/application_update.rb +0 -4
  110. data/lib/contrast/api/decorators/http_request.rb +3 -7
  111. data/lib/contrast/api/decorators/instrumentation_mode.rb +3 -5
  112. data/lib/contrast/api/decorators/library.rb +8 -6
  113. data/lib/contrast/api/decorators/message.rb +9 -9
  114. data/lib/contrast/api/decorators/trace_event.rb +3 -1
  115. data/lib/contrast/api/decorators/trace_event_object.rb +3 -6
  116. data/lib/contrast/api/decorators/trace_taint_range_tags.rb +1 -6
  117. data/lib/contrast/components/agent.rb +17 -17
  118. data/lib/contrast/components/app_context.rb +11 -15
  119. data/lib/contrast/components/assess.rb +16 -16
  120. data/lib/contrast/components/base.rb +40 -0
  121. data/lib/contrast/components/config.rb +2 -3
  122. data/lib/contrast/components/contrast_service.rb +12 -18
  123. data/lib/contrast/components/heap_dump.rb +5 -4
  124. data/lib/contrast/components/inventory.rb +2 -7
  125. data/lib/contrast/components/logger.rb +1 -2
  126. data/lib/contrast/components/protect.rb +10 -13
  127. data/lib/contrast/components/sampling.rb +13 -7
  128. data/lib/contrast/components/scope.rb +0 -4
  129. data/lib/contrast/components/settings.rb +5 -7
  130. data/lib/contrast/config/assess_rules_configuration.rb +1 -3
  131. data/lib/contrast/config/base_configuration.rb +4 -5
  132. data/lib/contrast/config/exception_configuration.rb +1 -5
  133. data/lib/contrast/config/heap_dump_configuration.rb +12 -6
  134. data/lib/contrast/config/logger_configuration.rb +1 -5
  135. data/lib/contrast/configuration.rb +6 -18
  136. data/lib/contrast/extension/assess/array.rb +3 -10
  137. data/lib/contrast/extension/assess/erb.rb +1 -7
  138. data/lib/contrast/extension/assess/eval_trigger.rb +4 -9
  139. data/lib/contrast/extension/assess/exec_trigger.rb +3 -9
  140. data/lib/contrast/extension/assess/fiber.rb +8 -17
  141. data/lib/contrast/extension/assess/hash.rb +3 -3
  142. data/lib/contrast/extension/assess/kernel.rb +4 -13
  143. data/lib/contrast/extension/assess/marshal.rb +6 -10
  144. data/lib/contrast/extension/assess/regexp.rb +6 -10
  145. data/lib/contrast/extension/assess/string.rb +8 -6
  146. data/lib/contrast/extension/kernel.rb +2 -2
  147. data/lib/contrast/extension/protect/kernel.rb +0 -5
  148. data/lib/contrast/framework/manager.rb +3 -5
  149. data/lib/contrast/framework/rack/patch/session_cookie.rb +11 -24
  150. data/lib/contrast/framework/rack/patch/support.rb +6 -4
  151. data/lib/contrast/framework/rails/patch/assess_configuration.rb +12 -9
  152. data/lib/contrast/framework/rails/patch/support.rb +41 -35
  153. data/lib/contrast/framework/rails/railtie.rb +34 -0
  154. data/lib/contrast/framework/rails/rewrite/action_controller_railties_helper_inherited.rb +4 -1
  155. data/lib/contrast/framework/rails/rewrite/active_record_attribute_methods_read.rb +2 -0
  156. data/lib/contrast/framework/rails/rewrite/active_record_named.rb +5 -4
  157. data/lib/contrast/framework/rails/rewrite/active_record_time_zone_inherited.rb +2 -0
  158. data/lib/contrast/framework/rails/support.rb +2 -2
  159. data/lib/contrast/framework/sinatra/support.rb +3 -1
  160. data/lib/contrast/funchook/funchook.rb +5 -8
  161. data/lib/contrast/logger/application.rb +13 -15
  162. data/lib/contrast/logger/format.rb +2 -5
  163. data/lib/contrast/logger/log.rb +26 -9
  164. data/lib/contrast/logger/request.rb +1 -6
  165. data/lib/contrast/security_exception.rb +1 -1
  166. data/lib/contrast/tasks/config.rb +0 -1
  167. data/lib/contrast/tasks/service.rb +6 -7
  168. data/lib/contrast/utils/assess/sampling_util.rb +2 -3
  169. data/lib/contrast/utils/assess/tracking_util.rb +3 -6
  170. data/lib/contrast/utils/class_util.rb +0 -8
  171. data/lib/contrast/utils/hash_digest.rb +2 -5
  172. data/lib/contrast/utils/heap_dump_util.rb +5 -3
  173. data/lib/contrast/utils/invalid_configuration_util.rb +4 -3
  174. data/lib/contrast/utils/inventory_util.rb +2 -3
  175. data/lib/contrast/utils/io_util.rb +3 -5
  176. data/lib/contrast/utils/job_servers_running.rb +13 -7
  177. data/lib/contrast/utils/os.rb +4 -4
  178. data/lib/contrast/utils/ruby_ast_rewriter.rb +2 -1
  179. data/lib/contrast/utils/string_utils.rb +2 -3
  180. data/lib/contrast/utils/tag_util.rb +25 -19
  181. data/resources/assess/policy.json +55 -0
  182. data/ruby-agent.gemspec +17 -16
  183. data/service_executables/VERSION +1 -1
  184. data/service_executables/linux/contrast-service +0 -0
  185. data/service_executables/mac/contrast-service +0 -0
  186. data/sonar-project.properties +9 -0
  187. metadata +61 -46
  188. data/lib/contrast/agent/railtie.rb +0 -31
  189. data/lib/contrast/components/interface.rb +0 -195
  190. data/lib/contrast/delegators/input_analysis.rb +0 -12
@@ -9,14 +9,6 @@ module Contrast
9
9
  # Utility methods for exploring the complete space of Objects
10
10
  class ClassUtil
11
11
  class << self
12
- # Given a module, return all of its descendants
13
- #
14
- # @param mod [Module] the module whose descendants you want to find.
15
- # @return [Array<Module>] those Modules that inherit from the given.
16
- def descendants mod
17
- ObjectSpace.each_object(mod).to_a
18
- end
19
-
20
12
  # some classes have had things prepended to them, like Marshal in Rails
21
13
  # 5 and higher. Their ActiveSupport::MarshalWithAutoloading will break
22
14
  # our alias patching approach, as will any other prepend on something
@@ -15,10 +15,7 @@ module Contrast
15
15
  include Digest::Instance
16
16
 
17
17
  CONTENT_LENGTH_HEADER = 'Content-Length'
18
- CRYPTO_RULES = %w[
19
- crypto-bad-ciphers
20
- crypto-bad-mac
21
- ].cs__freeze
18
+ CRYPTO_RULES = %w[crypto-bad-ciphers crypto-bad-mac].cs__freeze
22
19
  CONFIG_PATH_KEY = 'path'
23
20
  CONFIG_SESSION_ID_KEY = 'sessionId'
24
21
  CLASS_SOURCE_KEY = 'source'
@@ -111,7 +108,7 @@ module Contrast
111
108
  events.each do |event|
112
109
  event.event_sources.each do |source|
113
110
  update(source.type)
114
- update(source.name)
111
+ update(source.name) # rubocop:disable Security/Module/Name -- API attribute.
115
112
  end
116
113
  end
117
114
  end
@@ -3,14 +3,16 @@
3
3
 
4
4
  require 'objspace'
5
5
  require 'singleton'
6
- require 'contrast/components/interface'
6
+ require 'contrast/components/heap_dump'
7
+ require 'contrast/components/logger'
7
8
 
8
9
  module Contrast
9
10
  module Utils
10
11
  # Implementation of a heap dump util to automate generation
11
12
  class HeapDumpUtil < Contrast::Agent::WorkerThread
12
- include Contrast::Components::Interface
13
- access_component :heap_dump, :logging
13
+ extend Contrast::Components::Logger::InstanceMethods
14
+ include Contrast::Components::Logger::InstanceMethods
15
+ extend Contrast::Components::HeapDump::InstanceMethods
14
16
 
15
17
  LOG_ERROR_DUMPS = 'Unable to generate heap dumps'
16
18
  FILE_WRITE_FLAGS = 'w'
@@ -2,15 +2,16 @@
2
2
  # frozen_string_literal: true
3
3
 
4
4
  require 'contrast/agent/assess/policy/trigger_method'
5
- require 'contrast/components/interface'
5
+ require 'contrast/components/logger'
6
+ require 'contrast/components/scope'
6
7
 
7
8
  module Contrast
8
9
  module Utils
9
10
  # This utility allows us to report invalid configurations detected in
10
11
  # customer applications, as determined by Configuration Rules at runtime.
11
12
  module InvalidConfigurationUtil
12
- include Contrast::Components::Interface
13
- access_component :analysis, :app_context, :logging, :scope
13
+ include Contrast::Components::Logger::InstanceMethods
14
+ include Contrast::Components::Scope::InstanceMethods
14
15
 
15
16
  CS__PATH = 'path'
16
17
  CS__SESSION_ID = 'sessionId'
@@ -3,14 +3,13 @@
3
3
 
4
4
  require 'contrast/utils/timer'
5
5
  require 'contrast/utils/object_share'
6
- require 'contrast/components/interface'
6
+ require 'contrast/components/logger'
7
7
 
8
8
  module Contrast
9
9
  module Utils
10
10
  # Utilities for getting inventory information from the application
11
11
  class InventoryUtil
12
- include Contrast::Components::Interface
13
- access_component :logging
12
+ extend Contrast::Components::Logger::InstanceMethods
14
13
 
15
14
  # TeamServer only accepts certain values for ArchitectureComponents.
16
15
  # DO NOT CHANGE THIS!
@@ -1,15 +1,13 @@
1
1
  # Copyright (c) 2021 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
2
  # frozen_string_literal: true
3
3
 
4
- require 'contrast/components/interface'
4
+ require 'contrast/components/logger'
5
5
 
6
6
  module Contrast
7
7
  module Utils
8
8
  # Util for information about an IO
9
9
  class IOUtil
10
- include Contrast::Components::Interface
11
-
12
- access_component :logging
10
+ extend Contrast::Components::Logger::InstanceMethods
13
11
 
14
12
  # We're only going to call rewind on things that we believe are safe to
15
13
  # call it on. This method white lists those cases and returns false in
@@ -20,7 +18,7 @@ module Contrast
20
18
 
21
19
  should_rewind_io?(potential_io)
22
20
  rescue StandardError => e
23
- logger.debug('Encountered an issue determining if rewindable', e, module: potential_io.cs__class.name)
21
+ logger.debug('Encountered an issue determining if rewindable', e, module: potential_io.cs__class.cs__name)
24
22
  false
25
23
  end
26
24
 
@@ -1,13 +1,14 @@
1
1
  # Copyright (c) 2021 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
2
  # frozen_string_literal: true
3
3
 
4
+ require 'contrast/components/logger'
5
+
4
6
  module Contrast
5
7
  module Utils
6
8
  # A module that detects whether any job servers attached to
7
9
  # the application are running
8
10
  module JobServersRunning
9
- include Contrast::Components::Interface
10
- access_component :app_context, :logging
11
+ extend Contrast::Components::Logger::InstanceMethods
11
12
 
12
13
  class << self
13
14
  def job_servers_running?
@@ -24,12 +25,17 @@ module Contrast
24
25
  end
25
26
 
26
27
  def rake_running?
27
- return unless defined?(Rake) &&
28
- Rake.cs__respond_to?(:application) &&
29
- Rake.application.cs__respond_to?(:top_level_tasks)
28
+ unless defined?(Rake) &&
29
+ Rake.cs__respond_to?(:application) &&
30
+ Rake.application.cs__respond_to?(:top_level_tasks)
31
+
32
+ return
33
+ end
30
34
 
31
- disabled_rake_tasks = APP_CONTEXT.disabled_agent_rake_tasks
32
- has_disabled_task = Rake.application.top_level_tasks.any? { |top_level_task| disabled_rake_tasks.include?(top_level_task) }
35
+ disabled_rake_tasks = ::Contrast::APP_CONTEXT.disabled_agent_rake_tasks
36
+ has_disabled_task = Rake.application.top_level_tasks.any? do |top_level_task|
37
+ disabled_rake_tasks.include?(top_level_task)
38
+ end
33
39
  return false unless has_disabled_task
34
40
 
35
41
  logger.trace('Detected startup within Rake task')
@@ -1,7 +1,7 @@
1
1
  # Copyright (c) 2021 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
2
  # frozen_string_literal: true
3
3
 
4
- require 'contrast/components/interface'
4
+ require 'contrast/components/scope'
5
5
 
6
6
  module Contrast
7
7
  module Utils
@@ -9,8 +9,7 @@ module Contrast
9
9
  # which will not change at runtime, such as the operating system, the
10
10
  # Utility memozies to avoid multiple lookups.
11
11
  module OS
12
- include Contrast::Components::Interface
13
- access_component :scope
12
+ extend Contrast::Components::Scope::InstanceMethods
14
13
 
15
14
  class << self
16
15
  def running?
@@ -27,7 +26,8 @@ module Contrast
27
26
  # returns an array of zombie process PIDs as strings; empty array if there are none
28
27
  def zombie_pids
29
28
  with_contrast_scope do
30
- zombie_pid_list = `ps aux | grep contrast-servic[e] | grep Z | awk '{print $2}'` # retrieve pid of service processes
29
+ # retrieve pid of service processes
30
+ zombie_pid_list = `ps aux | grep contrast-servic[e] | grep Z | awk '{print $2}'`
31
31
  zombie_pid_list.split("\n")
32
32
  end
33
33
  end
@@ -1,8 +1,9 @@
1
1
  # Copyright (c) 2021 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
2
  # frozen_string_literal: true
3
3
 
4
- require 'parser/current'
4
+ return unless RUBY_VERSION < '2.6.0' # TODO: RUBY-714 remove guard w/ EOL of 2.5
5
5
 
6
+ require 'parser/current'
6
7
  module Contrast
7
8
  module Utils
8
9
  # This utility allows us to parse and rewrite the AST in Ruby 2.5,
@@ -1,14 +1,13 @@
1
1
  # Copyright (c) 2021 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
2
  # frozen_string_literal: true
3
3
 
4
- require 'contrast/components/interface'
4
+ require 'contrast/components/logger'
5
5
 
6
6
  module Contrast
7
7
  module Utils
8
8
  # Utilities for encoding and normalizing strings
9
9
  class StringUtils
10
- include Contrast::Components::Interface
11
- access_component :logging
10
+ include Contrast::Components::Logger::InstanceMethods
12
11
 
13
12
  UTF8 = 'utf-8'
14
13
  HTTP_PREFIX = 'HTTP_'
@@ -7,8 +7,9 @@ module Contrast
7
7
  class TagUtil
8
8
  class << self
9
9
  # Determine if the given array of tags is covered by the other
10
- # remaining_ranges: the tags left that haven't been covered by those given
11
- # ranges: the tags that are covering the first
10
+ #
11
+ # @param remaining_ranges [Array<Contrast::Agent::Assess::Tag>] the tags left that haven't been covered by those given
12
+ # @param ranges Array<Contrast::Agent::Assess::Tag> the tags that are covering the first
12
13
  def covered? remaining_ranges, ranges
13
14
  return true unless remaining_ranges&.any?
14
15
 
@@ -74,14 +75,14 @@ module Contrast
74
75
 
75
76
  # Given a collection of tags, merge any tags that are continuous
76
77
  #
77
- # If tags is a hash, it should be in the format label => [tags]
78
- # The array of tags will each be merged
78
+ # If tags is a hash, it should be in the format label => [tags]. The array of tags will each be merged
79
+ # If tags is an array in the format [tags], the array will be merged
79
80
  #
80
- # If tags is an array in the format [tags], the array will be
81
- # merged
81
+ # The original object is returned, although setters should not be necessary since tags is a collection in
82
+ # either case
82
83
  #
83
- # The original object is returned, although setters should not be
84
- # necessary since tags is a collection in either case
84
+ # @param tags [Hash{String => Array<Contrast::Agent::Assess::Tag>}, Array<Contrast::Agent::Assess::Tag>]
85
+ # @return [Hash{String => Array<Contrast::Agent::Assess::Tag>}, Array<Contrast::Agent::Assess::Tag>]
85
86
  def merge_tags tags
86
87
  if tags.is_a?(Hash)
87
88
  tags.each_value { |value| smallerize(value) }
@@ -90,6 +91,12 @@ module Contrast
90
91
  end
91
92
  end
92
93
 
94
+ # Merge the given set of tags such that any overlap combines. For any tag which extends beyond the size of the
95
+ # target_object, the end will be updated to the target_object's length.
96
+ #
97
+ # @param target_object [Object] the thing to which the tags apply
98
+ # @param tags [Hash{String => Array<Contrast::Agent::Assess::Tag>}, Array<Contrast::Agent::Assess::Tag>]
99
+ # @return [Hash{String => Array<Contrast::Agent::Assess::Tag>}, Array<Contrast::Agent::Assess::Tag>]
93
100
  def size_aware_merge target_object, tags
94
101
  max_size = target_object.to_s.length
95
102
  tags = merge_tags(tags)
@@ -100,14 +107,12 @@ module Contrast
100
107
 
101
108
  private
102
109
 
103
- # Add one new element to the given array
110
+ # Add one new element to the given array. The addition is done such that the new entry is inserted so that the
111
+ # range they cover is in order. Any overlapping ranges are merged before returning.
104
112
  #
105
- # The addition is done such that the new entry(ies)
106
- # are inserted so that the range they cover is in order
107
- # Any overlapping ranges are merged before returning
108
- #
109
- # arr: the array to which the element is added
110
- # new_element: the element to be added to the array
113
+ # @param arr [Array<Contrast::Agent::Assess::Tag>]
114
+ # @param new_element []Contrast::Agent::Assess::Tag]
115
+ # @return [Array<Contrast::Agent::Assess::Tag>]
111
116
  def single_ordered_merge arr, new_element
112
117
  idx = 0
113
118
  arr.each do |existing|
@@ -122,10 +127,11 @@ module Contrast
122
127
  arr.insert(idx, new_element)
123
128
  end
124
129
 
125
- # Given an arry of tags, merge any that overlap
126
- # The tag that was higher up is removed from the
127
- # list of tags.
128
- # ranges like [0-3][3-6]-6-9] that should become [0-9]
130
+ # Given an arry of tags, merge any that overlap. The tag that was higher up is removed from the list of tags.
131
+ # ranges like [0-3][3-6]-6-9] become [0-9]
132
+ #
133
+ # @param tags [Array<Contrast::Agent::Assess::Tag>]
134
+ # @return [Array<Contrast::Agent::Assess::Tag>]
129
135
  def smallerize tags
130
136
  smallered = []
131
137
  curr = nil
@@ -1026,6 +1026,61 @@
1026
1026
  "action": "CUSTOM",
1027
1027
  "patch_class": "ERBPropagator",
1028
1028
  "patch_method": "result_tagger"
1029
+ },
1030
+ {
1031
+ "class_name": "ActionView::Helpers::SanitizeHelper",
1032
+ "method_name": "sanitize",
1033
+ "method_visibility": "public",
1034
+ "instance_method": true,
1035
+ "source": "P0",
1036
+ "target": "R",
1037
+ "action": "REMOVE",
1038
+ "tags":["HTML_ENCODED"],
1039
+ "untags":["HTML_DECODED"]
1040
+ },
1041
+ {
1042
+ "class_name": "ActionView::Helpers::SanitizeHelper",
1043
+ "method_name": "strip_tags",
1044
+ "method_visibility": "public",
1045
+ "instance_method": true,
1046
+ "source": "P0",
1047
+ "target": "R",
1048
+ "action": "REMOVE",
1049
+ "tags":["HTML_ENCODED"],
1050
+ "untags":["HTML_DECODED"]
1051
+ },
1052
+ {
1053
+ "class_name": "Rack::Protection::EscapedParams",
1054
+ "method_name": "escape_string",
1055
+ "method_visibility": "public",
1056
+ "instance_method": true,
1057
+ "source": "P0",
1058
+ "target": "R",
1059
+ "action": "CUSTOM",
1060
+ "patch_class": "Contrast::Agent::Assess::Policy::Propagator::RackProtection",
1061
+ "patch_method": "escaped_params"
1062
+ },
1063
+ {
1064
+ "class_name": "Rails::Html::FullSanitizer",
1065
+ "method_name": "sanitize",
1066
+ "method_visibility": "public",
1067
+ "instance_method": true,
1068
+ "source": "P0",
1069
+ "target": "R",
1070
+ "action": "REMOVE",
1071
+ "tags":["HTML_ENCODED"],
1072
+ "untags":["HTML_DECODED"]
1073
+ },
1074
+ {
1075
+ "class_name": "Rails::Html::SafeListSanitizer",
1076
+ "method_name": "sanitize",
1077
+ "method_visibility": "public",
1078
+ "instance_method": true,
1079
+ "source": "P0",
1080
+ "target": "R",
1081
+ "action": "REMOVE",
1082
+ "tags":["HTML_ENCODED"],
1083
+ "untags":["HTML_DECODED"]
1029
1084
  }
1030
1085
  ],
1031
1086
  "rules":[
data/ruby-agent.gemspec CHANGED
@@ -22,7 +22,7 @@ end
22
22
  def self.add_dev_dependencies spec
23
23
  add_builders(spec)
24
24
  add_debuggers(spec)
25
- add_linters(spec)
25
+ add_linters(spec) # if RUBY_VERSION >= '2.6.0' # TODO: RUBY-714 remove guard w/ EOL of 2.5
26
26
  add_specs(spec)
27
27
  end
28
28
 
@@ -41,15 +41,17 @@ end
41
41
 
42
42
  # Dependencies used for framework testing.
43
43
  def self.add_frameworks spec
44
+ spec.add_development_dependency 'rack-protection', '>= 2'
44
45
  spec.add_development_dependency 'rails', '6.0.3.5'
45
46
  spec.add_development_dependency 'sinatra', '>= 2'
46
47
  end
47
48
 
48
49
  # Dependencies used for linting prior to commit.
49
50
  def self.add_linters spec
50
- spec.add_development_dependency 'debride'
51
- spec.add_development_dependency 'fasterer'
52
- spec.add_development_dependency 'flay'
51
+ spec.add_development_dependency 'debride', '1.8.2'
52
+ spec.add_development_dependency 'fasterer', '0.9.0'
53
+ spec.add_development_dependency 'flay', '2.12.1'
54
+ # spec.add_development_dependency 'steep', '0.44.1' # TODO: RUBY-714 uncomment w/ EOL of 2.5
53
55
  add_rubocop(spec)
54
56
  end
55
57
 
@@ -68,29 +70,29 @@ def self.add_specs spec
68
70
  spec.add_development_dependency 'rspec-benchmark'
69
71
  spec.add_development_dependency 'rspec_junit_formatter', '0.3.0'
70
72
  spec.add_development_dependency 'rspec-rails', '5.0'
73
+ spec.add_development_dependency 'warning'
71
74
  spec.add_development_dependency 'tzinfo-data' # Alpine rspec-rails requirement.
72
75
  end
73
76
 
74
77
  def self.add_coverage spec
75
- spec.add_development_dependency 'codecov'
76
- spec.add_development_dependency 'simplecov', '0.20.0'
78
+ spec.add_development_dependency 'simplecov', '0.21.2'
77
79
  end
78
80
 
79
81
  # Dependencies used to run all of our Rubocop during the linting phase.
80
82
  def self.add_rubocop spec
81
- spec.add_development_dependency 'rubocop', '1.6.1'
82
- spec.add_development_dependency 'rubocop-performance', '1.9.1'
83
+ spec.add_development_dependency 'rubocop', '1.13.0'
84
+ spec.add_development_dependency 'rubocop-performance', '1.11.0'
83
85
  spec.add_development_dependency 'rubocop-rails', '2.9.1'
84
86
  spec.add_development_dependency 'rubocop-rake', '0.5.1'
85
- spec.add_development_dependency 'rubocop-rspec', '2.1.0'
87
+ spec.add_development_dependency 'rubocop-rspec', '2.2.0'
86
88
  end
87
89
 
88
90
  # Dependencies not mocked out during RSpec that we test real code of, beyond just frameworks.
89
91
  def self.add_tested_gems spec
90
92
  spec.add_development_dependency 'async'
91
93
  spec.add_development_dependency 'execjs'
94
+ spec.add_development_dependency 'rhino'
92
95
  spec.add_development_dependency 'sqlite3'
93
- spec.add_development_dependency 'therubyracer'
94
96
  spec.add_development_dependency 'tilt'
95
97
  spec.add_development_dependency 'xpath'
96
98
  end
@@ -102,7 +104,7 @@ end
102
104
  # corresponding update to the fake gem server data in TeamServer.
103
105
  def self.add_dependencies spec
104
106
  spec.add_dependency 'ougai', '~> 1.8'
105
- spec.add_dependency 'parser', '~> 2.6' # TODO: RUBY-714 remove w/ EOL of 2.5
107
+ spec.add_dependency 'parser', '>= 2.6' # if RUBY_VERSION < '2.6.0' # TODO: RUBY-714 remove guard w/ EOL of 2.5
106
108
  spec.add_dependency 'protobuf', '~> 3.10'
107
109
  spec.add_dependency 'rack', '~> 2.0'
108
110
  end
@@ -113,7 +115,7 @@ def self.add_files spec
113
115
  # Directories used for testing:
114
116
  f.match(%r{^(spec|test)/}) ||
115
117
  # Directories used in pipelines
116
- f.match(%r{^(\.github|bin|internal_resources|vendor)/}) ||
118
+ f.match(%r{^(\.github|bin|internal_resources|sig|vendor)/}) ||
117
119
  # Configuration and other files that don't belong to one directory
118
120
  f.match(/(Dockerfile)/) ||
119
121
  f.match(/(.*\.csv)/) ||
@@ -154,9 +156,7 @@ Gem::Specification.new do |spec|
154
156
  spec.name = 'contrast-agent'
155
157
  spec.version = Contrast::Agent::VERSION
156
158
 
157
- spec.email = %w[
158
- ruby@contrastsecurity.com
159
- ]
159
+ spec.email = %w[ruby@contrastsecurity.com]
160
160
 
161
161
  spec.summary = 'Contrast Security\'s agent for rack-based applications.'
162
162
  spec.description = 'This gem instantiates a Rack middleware for rack-based ' \
@@ -173,7 +173,8 @@ Gem::Specification.new do |spec|
173
173
  spec.require_paths = ['lib']
174
174
 
175
175
  unless File.exist?(File.join(Dir.pwd, 'contrast_security.yaml'))
176
- spec.post_install_message = 'To generate the required contrast_security.yaml file you can run: bundle exec rake contrast:config:create'
176
+ spec.post_install_message = 'To generate the required contrast_security.yaml file you can run: '\
177
+ 'bundle exec rake contrast:config:create'
177
178
  end
178
179
 
179
180
  add_authors(spec)