contrast-agent 3.8.4 → 6.9.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (676) hide show
  1. checksums.yaml +4 -4
  2. data/.dockerignore +0 -1
  3. data/.flayignore +1 -0
  4. data/.gitignore +8 -5
  5. data/.gitmodules +0 -3
  6. data/.rspec +0 -1
  7. data/.rspec_parallel +6 -0
  8. data/.simplecov +6 -2
  9. data/Gemfile +1 -1
  10. data/LICENSE.txt +1 -1
  11. data/Rakefile +5 -2
  12. data/ext/build_funchook.rb +27 -11
  13. data/ext/cs__assess_array/cs__assess_array.c +45 -7
  14. data/ext/cs__assess_array/cs__assess_array.h +5 -1
  15. data/ext/cs__assess_array/extconf.rb +3 -0
  16. data/ext/cs__assess_basic_object/cs__assess_basic_object.c +39 -17
  17. data/ext/cs__assess_basic_object/cs__assess_basic_object.h +2 -1
  18. data/ext/cs__assess_basic_object/extconf.rb +3 -0
  19. data/ext/cs__assess_fiber_track/cs__assess_fiber_track.c +9 -13
  20. data/ext/cs__assess_fiber_track/cs__assess_fiber_track.h +3 -4
  21. data/ext/cs__assess_fiber_track/extconf.rb +3 -0
  22. data/ext/cs__assess_hash/cs__assess_hash.c +46 -21
  23. data/ext/cs__assess_hash/cs__assess_hash.h +5 -6
  24. data/ext/cs__assess_hash/extconf.rb +3 -0
  25. data/ext/cs__assess_kernel/cs__assess_kernel.c +29 -15
  26. data/ext/cs__assess_kernel/cs__assess_kernel.h +3 -0
  27. data/ext/cs__assess_kernel/extconf.rb +3 -0
  28. data/ext/cs__assess_marshal_module/cs__assess_marshal_module.c +55 -23
  29. data/ext/cs__assess_marshal_module/cs__assess_marshal_module.h +6 -3
  30. data/ext/cs__assess_marshal_module/extconf.rb +3 -0
  31. data/ext/cs__assess_module/cs__assess_module.c +82 -23
  32. data/ext/cs__assess_module/cs__assess_module.h +10 -0
  33. data/ext/cs__assess_module/extconf.rb +3 -0
  34. data/ext/cs__assess_regexp/cs__assess_regexp.c +28 -9
  35. data/ext/cs__assess_regexp/cs__assess_regexp.h +3 -0
  36. data/ext/cs__assess_regexp/extconf.rb +3 -0
  37. data/ext/cs__assess_string/cs__assess_string.c +53 -21
  38. data/ext/cs__assess_string/cs__assess_string.h +7 -1
  39. data/ext/cs__assess_string/extconf.rb +3 -0
  40. data/ext/cs__assess_string_interpolation/cs__assess_string_interpolation.c +39 -0
  41. data/ext/cs__assess_string_interpolation/cs__assess_string_interpolation.h +13 -0
  42. data/ext/cs__assess_string_interpolation/extconf.rb +5 -0
  43. data/ext/cs__assess_test/cs__assess_test.h +9 -0
  44. data/ext/cs__assess_test/cs__assess_tests.c +22 -0
  45. data/ext/cs__assess_test/extconf.rb +5 -0
  46. data/ext/cs__assess_yield_track/cs__assess_yield_track.c +30 -0
  47. data/ext/cs__assess_yield_track/cs__assess_yield_track.h +11 -0
  48. data/ext/cs__assess_yield_track/extconf.rb +5 -0
  49. data/ext/cs__common/cs__common.c +246 -10
  50. data/ext/cs__common/cs__common.h +71 -2
  51. data/ext/cs__common/extconf.rb +3 -16
  52. data/ext/cs__contrast_patch/cs__contrast_patch.c +255 -155
  53. data/ext/cs__contrast_patch/cs__contrast_patch.h +13 -14
  54. data/ext/cs__contrast_patch/extconf.rb +3 -0
  55. data/ext/cs__os_information/cs__os_information.c +34 -0
  56. data/ext/cs__os_information/cs__os_information.h +7 -0
  57. data/ext/cs__os_information/extconf.rb +5 -0
  58. data/ext/cs__scope/cs__scope.c +755 -55
  59. data/ext/cs__scope/cs__scope.h +75 -20
  60. data/ext/cs__scope/extconf.rb +3 -0
  61. data/ext/cs__tests/cs__tests.c +12 -0
  62. data/ext/cs__tests/cs__tests.h +3 -0
  63. data/ext/cs__tests/extconf.rb +5 -0
  64. data/ext/extconf_common.rb +4 -34
  65. data/lib/contrast/agent/assess/contrast_object.rb +54 -0
  66. data/lib/contrast/agent/assess/events/event_data.rb +30 -0
  67. data/lib/contrast/agent/assess/finalizers/freeze.rb +15 -0
  68. data/lib/contrast/agent/assess/finalizers/hash.rb +107 -0
  69. data/lib/contrast/agent/assess/policy/dynamic_source_factory.rb +58 -36
  70. data/lib/contrast/agent/assess/policy/patcher.rb +13 -48
  71. data/lib/contrast/agent/assess/policy/policy.rb +20 -37
  72. data/lib/contrast/agent/assess/policy/policy_node.rb +96 -200
  73. data/lib/contrast/agent/assess/policy/policy_node_utils.rb +50 -0
  74. data/lib/contrast/agent/assess/policy/policy_scanner.rb +15 -12
  75. data/lib/contrast/agent/assess/policy/preshift.rb +50 -19
  76. data/lib/contrast/agent/assess/policy/propagation_method.rb +200 -192
  77. data/lib/contrast/agent/assess/policy/propagation_node.rb +49 -41
  78. data/lib/contrast/agent/assess/policy/propagator/append.rb +32 -15
  79. data/lib/contrast/agent/assess/policy/propagator/base.rb +5 -3
  80. data/lib/contrast/agent/assess/policy/propagator/buffer.rb +119 -0
  81. data/lib/contrast/agent/assess/policy/propagator/center.rb +12 -8
  82. data/lib/contrast/agent/assess/policy/propagator/custom.rb +7 -3
  83. data/lib/contrast/agent/assess/policy/propagator/database_write.rb +34 -25
  84. data/lib/contrast/agent/assess/policy/propagator/insert.rb +15 -11
  85. data/lib/contrast/agent/assess/policy/propagator/keep.rb +23 -6
  86. data/lib/contrast/agent/assess/policy/propagator/match_data.rb +118 -0
  87. data/lib/contrast/agent/assess/policy/propagator/next.rb +7 -6
  88. data/lib/contrast/agent/assess/policy/propagator/prepend.rb +13 -6
  89. data/lib/contrast/agent/assess/policy/propagator/rack_protection.rb +73 -0
  90. data/lib/contrast/agent/assess/policy/propagator/remove.rb +53 -41
  91. data/lib/contrast/agent/assess/policy/propagator/replace.rb +5 -3
  92. data/lib/contrast/agent/assess/policy/propagator/reverse.rb +7 -6
  93. data/lib/contrast/agent/assess/policy/propagator/select.rb +45 -36
  94. data/lib/contrast/agent/assess/policy/propagator/splat.rb +44 -21
  95. data/lib/contrast/agent/assess/policy/propagator/split.rb +176 -22
  96. data/lib/contrast/agent/assess/policy/propagator/substitution.rb +7 -132
  97. data/lib/contrast/agent/assess/policy/propagator/substitution_utils.rb +190 -0
  98. data/lib/contrast/agent/assess/policy/propagator/trim.rb +74 -52
  99. data/lib/contrast/agent/assess/policy/propagator.rb +21 -18
  100. data/lib/contrast/agent/assess/policy/source_method.rb +176 -177
  101. data/lib/contrast/agent/assess/policy/source_node.rb +3 -17
  102. data/lib/contrast/agent/assess/policy/source_validation/cross_site_validator.rb +32 -0
  103. data/lib/contrast/agent/assess/policy/source_validation/source_validation.rb +34 -0
  104. data/lib/contrast/agent/assess/policy/trigger/reflected_xss.rb +102 -0
  105. data/lib/contrast/agent/assess/policy/trigger/xpath.rb +57 -0
  106. data/lib/contrast/agent/assess/policy/trigger_method.rb +160 -173
  107. data/lib/contrast/agent/assess/policy/trigger_node.rb +162 -39
  108. data/lib/contrast/agent/assess/policy/trigger_validation/redos_validator.rb +60 -0
  109. data/lib/contrast/agent/assess/policy/trigger_validation/ssrf_validator.rb +8 -38
  110. data/lib/contrast/agent/assess/policy/trigger_validation/trigger_validation.rb +22 -7
  111. data/lib/contrast/agent/assess/policy/trigger_validation/xss_validator.rb +6 -15
  112. data/lib/contrast/agent/assess/properties.rb +15 -354
  113. data/lib/contrast/agent/assess/property/evented.rb +58 -0
  114. data/lib/contrast/agent/assess/property/tagged.rb +246 -0
  115. data/lib/contrast/agent/assess/property/updated.rb +131 -0
  116. data/lib/contrast/agent/assess/rule/provider/hardcoded_key.rb +58 -19
  117. data/lib/contrast/agent/assess/rule/provider/hardcoded_password.rb +22 -17
  118. data/lib/contrast/agent/assess/rule/provider/hardcoded_value_rule.rb +93 -81
  119. data/lib/contrast/agent/assess/rule/provider.rb +4 -4
  120. data/lib/contrast/agent/assess/rule/response/auto_complete_rule.rb +69 -0
  121. data/lib/contrast/agent/assess/rule/response/base_rule.rb +121 -0
  122. data/lib/contrast/agent/assess/rule/response/body_rule.rb +107 -0
  123. data/lib/contrast/agent/assess/rule/response/cache_control_header_rule.rb +195 -0
  124. data/lib/contrast/agent/assess/rule/response/click_jacking_header_rule.rb +26 -0
  125. data/lib/contrast/agent/assess/rule/response/csp_header_insecure_rule.rb +100 -0
  126. data/lib/contrast/agent/assess/rule/response/csp_header_missing_rule.rb +26 -0
  127. data/lib/contrast/agent/assess/rule/response/framework/rails_support.rb +34 -0
  128. data/lib/contrast/agent/assess/rule/response/header_rule.rb +70 -0
  129. data/lib/contrast/agent/assess/rule/response/hsts_header_rule.rb +36 -0
  130. data/lib/contrast/agent/assess/rule/response/parameters_pollution_rule.rb +61 -0
  131. data/lib/contrast/agent/assess/rule/response/x_content_type_header_rule.rb +26 -0
  132. data/lib/contrast/agent/assess/rule/response/x_xss_protection_header_rule.rb +34 -0
  133. data/lib/contrast/agent/assess/tag.rb +84 -41
  134. data/lib/contrast/agent/assess/tracker.rb +70 -0
  135. data/lib/contrast/agent/assess.rb +7 -29
  136. data/lib/contrast/agent/at_exit_hook.rb +28 -17
  137. data/lib/contrast/agent/deadzone/policy/deadzone_node.rb +11 -6
  138. data/lib/contrast/agent/deadzone/policy/policy.rb +11 -7
  139. data/lib/contrast/agent/disable_reaction.rb +6 -10
  140. data/lib/contrast/agent/excluder.rb +224 -0
  141. data/lib/contrast/agent/exclusion_matcher.rb +40 -74
  142. data/lib/contrast/agent/inventory/database_config.rb +174 -0
  143. data/lib/contrast/agent/inventory/dependencies.rb +52 -0
  144. data/lib/contrast/agent/inventory/dependency_analysis.rb +34 -0
  145. data/lib/contrast/agent/inventory/dependency_usage_analysis.rb +120 -0
  146. data/lib/contrast/agent/inventory/policy/datastores.rb +51 -0
  147. data/lib/contrast/agent/inventory/policy/policy.rb +5 -5
  148. data/lib/contrast/agent/inventory/policy/trigger_node.rb +2 -2
  149. data/lib/contrast/agent/inventory.rb +14 -0
  150. data/lib/contrast/agent/middleware.rb +146 -299
  151. data/lib/contrast/agent/module_data.rb +5 -4
  152. data/lib/contrast/agent/patching/policy/after_load_patch.rb +54 -7
  153. data/lib/contrast/agent/patching/policy/after_load_patcher.rb +103 -27
  154. data/lib/contrast/agent/patching/policy/method_policy.rb +53 -64
  155. data/lib/contrast/agent/patching/policy/method_policy_extend.rb +113 -0
  156. data/lib/contrast/agent/patching/policy/module_policy.rb +27 -47
  157. data/lib/contrast/agent/patching/policy/patch.rb +147 -241
  158. data/lib/contrast/agent/patching/policy/patch_status.rb +21 -45
  159. data/lib/contrast/agent/patching/policy/patcher.rb +126 -161
  160. data/lib/contrast/agent/patching/policy/policy.rb +66 -57
  161. data/lib/contrast/agent/patching/policy/policy_node.rb +63 -32
  162. data/lib/contrast/agent/patching/policy/trigger_node.rb +32 -15
  163. data/lib/contrast/agent/protect/exploitable_collection.rb +38 -0
  164. data/lib/contrast/agent/protect/input_analyzer/input_analyzer.rb +170 -0
  165. data/lib/contrast/agent/protect/input_analyzer/worth_watching_analyzer.rb +116 -0
  166. data/lib/contrast/agent/protect/policy/applies_command_injection_rule.rb +65 -0
  167. data/lib/contrast/agent/protect/policy/applies_deserialization_rule.rb +97 -0
  168. data/lib/contrast/agent/protect/policy/applies_no_sqli_rule.rb +69 -0
  169. data/lib/contrast/agent/protect/policy/applies_path_traversal_rule.rb +138 -0
  170. data/lib/contrast/agent/protect/policy/applies_sqli_rule.rb +55 -0
  171. data/lib/contrast/agent/protect/policy/applies_xxe_rule.rb +125 -0
  172. data/lib/contrast/agent/protect/policy/policy.rb +10 -10
  173. data/lib/contrast/agent/protect/policy/rule_applicator.rb +102 -0
  174. data/lib/contrast/agent/protect/policy/trigger_node.rb +2 -2
  175. data/lib/contrast/agent/protect/rule/base.rb +205 -95
  176. data/lib/contrast/agent/protect/rule/base_service.rb +73 -14
  177. data/lib/contrast/agent/protect/rule/bot_blocker/bot_blocker_input_classification.rb +98 -0
  178. data/lib/contrast/agent/protect/rule/bot_blocker.rb +81 -0
  179. data/lib/contrast/agent/protect/rule/cmd_injection.rb +53 -123
  180. data/lib/contrast/agent/protect/rule/cmdi/cmdi_backdoors.rb +132 -0
  181. data/lib/contrast/agent/protect/rule/cmdi/cmdi_base_rule.rb +169 -0
  182. data/lib/contrast/agent/protect/rule/cmdi/cmdi_chained_command.rb +64 -0
  183. data/lib/contrast/agent/protect/rule/cmdi/cmdi_dangerous_path.rb +63 -0
  184. data/lib/contrast/agent/protect/rule/cmdi/cmdi_input_classification.rb +27 -0
  185. data/lib/contrast/agent/protect/rule/default_scanner.rb +69 -25
  186. data/lib/contrast/agent/protect/rule/deserialization.rb +32 -48
  187. data/lib/contrast/agent/protect/rule/http_method_tampering/http_method_tampering_input_classification.rb +96 -0
  188. data/lib/contrast/agent/protect/rule/http_method_tampering.rb +65 -62
  189. data/lib/contrast/agent/protect/rule/no_sqli/mongo_no_sql_scanner.rb +2 -3
  190. data/lib/contrast/agent/protect/rule/no_sqli/no_sqli_input_classification.rb +226 -0
  191. data/lib/contrast/agent/protect/rule/no_sqli.rb +47 -53
  192. data/lib/contrast/agent/protect/rule/path_traversal/path_traversal_input_classification.rb +61 -0
  193. data/lib/contrast/agent/protect/rule/path_traversal/path_traversal_semantic_security_bypass.rb +114 -0
  194. data/lib/contrast/agent/protect/rule/path_traversal.rb +57 -26
  195. data/lib/contrast/agent/protect/rule/sql_sample_builder.rb +155 -0
  196. data/lib/contrast/agent/protect/rule/sqli/default_sql_scanner.rb +1 -1
  197. data/lib/contrast/agent/protect/rule/sqli/mysql_sql_scanner.rb +1 -1
  198. data/lib/contrast/agent/protect/rule/sqli/postgres_sql_scanner.rb +2 -2
  199. data/lib/contrast/agent/protect/rule/sqli/sqli_base_rule.rb +37 -0
  200. data/lib/contrast/agent/protect/rule/sqli/sqli_input_classification.rb +28 -0
  201. data/lib/contrast/agent/protect/rule/sqli/sqli_semantic/sqli_dangerous_functions.rb +67 -0
  202. data/lib/contrast/agent/protect/rule/sqli/sqlite_sql_scanner.rb +1 -1
  203. data/lib/contrast/agent/protect/rule/sqli.rb +78 -62
  204. data/lib/contrast/agent/protect/rule/unsafe_file_upload/unsafe_file_upload_input_classification.rb +58 -0
  205. data/lib/contrast/agent/protect/rule/unsafe_file_upload.rb +19 -2
  206. data/lib/contrast/agent/protect/rule/xss/reflected_xss_input_classification.rb +58 -0
  207. data/lib/contrast/agent/protect/rule/xss.rb +20 -2
  208. data/lib/contrast/agent/protect/rule/xxe/entity_wrapper.rb +25 -21
  209. data/lib/contrast/agent/protect/rule/xxe.rb +69 -39
  210. data/lib/contrast/agent/protect/rule.rb +22 -25
  211. data/lib/contrast/agent/reporting/attack_result/attack_result.rb +71 -0
  212. data/lib/contrast/agent/reporting/attack_result/rasp_rule_sample.rb +86 -0
  213. data/lib/contrast/agent/reporting/attack_result/response_type.rb +29 -0
  214. data/lib/contrast/agent/reporting/attack_result/user_input.rb +98 -0
  215. data/lib/contrast/agent/reporting/details/bot_blocker_details.rb +29 -0
  216. data/lib/contrast/agent/reporting/details/cmd_injection_details.rb +30 -0
  217. data/lib/contrast/agent/reporting/details/details.rb +18 -0
  218. data/lib/contrast/agent/reporting/details/http_method_tempering_details.rb +27 -0
  219. data/lib/contrast/agent/reporting/details/ip_denylist_details.rb +35 -0
  220. data/lib/contrast/agent/reporting/details/no_sqli_details.rb +36 -0
  221. data/lib/contrast/agent/reporting/details/path_traversal_details.rb +24 -0
  222. data/lib/contrast/agent/reporting/details/path_traversal_semantic_analysis_details.rb +32 -0
  223. data/lib/contrast/agent/reporting/details/protect_rule_details.rb +17 -0
  224. data/lib/contrast/agent/reporting/details/sqli_dangerous_functions.rb +22 -0
  225. data/lib/contrast/agent/reporting/details/sqli_details.rb +36 -0
  226. data/lib/contrast/agent/reporting/details/untrusted_deserialization_details.rb +27 -0
  227. data/lib/contrast/agent/reporting/details/virtual_patch_details.rb +30 -0
  228. data/lib/contrast/agent/reporting/details/xss_details.rb +33 -0
  229. data/lib/contrast/agent/reporting/details/xss_match.rb +30 -0
  230. data/lib/contrast/agent/reporting/details/xxe_details.rb +36 -0
  231. data/lib/contrast/agent/reporting/details/xxe_match.rb +25 -0
  232. data/lib/contrast/agent/reporting/details/xxe_wrapper.rb +25 -0
  233. data/lib/contrast/agent/reporting/input_analysis/details/bot_blocker_details.rb +27 -0
  234. data/lib/contrast/agent/reporting/input_analysis/details/protect_rule_details.rb +15 -0
  235. data/lib/contrast/agent/reporting/input_analysis/input_analysis.rb +43 -0
  236. data/lib/contrast/agent/reporting/input_analysis/input_analysis_result.rb +129 -0
  237. data/lib/contrast/agent/reporting/input_analysis/input_type.rb +44 -0
  238. data/lib/contrast/agent/reporting/input_analysis/score_level.rb +21 -0
  239. data/lib/contrast/agent/reporting/masker/masker.rb +258 -0
  240. data/lib/contrast/agent/reporting/masker/masker_utils.rb +33 -0
  241. data/lib/contrast/agent/reporting/report.rb +31 -0
  242. data/lib/contrast/agent/reporting/reporter.rb +165 -0
  243. data/lib/contrast/agent/reporting/reporter_heartbeat.rb +47 -0
  244. data/lib/contrast/agent/reporting/reporting_events/agent_startup.rb +34 -0
  245. data/lib/contrast/agent/reporting/reporting_events/application_activity.rb +120 -0
  246. data/lib/contrast/agent/reporting/reporting_events/application_defend_activity.rb +85 -0
  247. data/lib/contrast/agent/reporting/reporting_events/application_defend_attack_activity.rb +65 -0
  248. data/lib/contrast/agent/reporting/reporting_events/application_defend_attack_sample.rb +102 -0
  249. data/lib/contrast/agent/reporting/reporting_events/application_defend_attack_sample_activity.rb +68 -0
  250. data/lib/contrast/agent/reporting/reporting_events/application_defend_attack_sample_stack.rb +22 -0
  251. data/lib/contrast/agent/reporting/reporting_events/application_defend_attacker_activity.rb +62 -0
  252. data/lib/contrast/agent/reporting/reporting_events/application_inventory.rb +42 -0
  253. data/lib/contrast/agent/reporting/reporting_events/application_inventory_activity.rb +57 -0
  254. data/lib/contrast/agent/reporting/reporting_events/application_reporting_event.rb +27 -0
  255. data/lib/contrast/agent/reporting/reporting_events/application_startup.rb +44 -0
  256. data/lib/contrast/agent/reporting/reporting_events/application_startup_instrumentation.rb +27 -0
  257. data/lib/contrast/agent/reporting/reporting_events/application_update.rb +56 -0
  258. data/lib/contrast/agent/reporting/reporting_events/architecture_component.rb +72 -0
  259. data/lib/contrast/agent/reporting/reporting_events/discovered_route.rb +126 -0
  260. data/lib/contrast/agent/reporting/reporting_events/finding.rb +210 -0
  261. data/lib/contrast/agent/reporting/reporting_events/finding_event.rb +449 -0
  262. data/lib/contrast/agent/reporting/reporting_events/finding_event_object.rb +104 -0
  263. data/lib/contrast/agent/reporting/reporting_events/finding_event_parent_object.rb +49 -0
  264. data/lib/contrast/agent/reporting/reporting_events/finding_event_property.rb +51 -0
  265. data/lib/contrast/agent/reporting/reporting_events/finding_event_signature.rb +106 -0
  266. data/lib/contrast/agent/reporting/reporting_events/finding_event_source.rb +74 -0
  267. data/lib/contrast/agent/reporting/reporting_events/finding_event_stack.rb +68 -0
  268. data/lib/contrast/agent/reporting/reporting_events/finding_event_taint_range.rb +71 -0
  269. data/lib/contrast/agent/reporting/reporting_events/finding_event_taint_range_tags.rb +105 -0
  270. data/lib/contrast/agent/reporting/reporting_events/finding_request.rb +134 -0
  271. data/lib/contrast/agent/reporting/reporting_events/library_discovery.rb +89 -0
  272. data/lib/contrast/agent/reporting/reporting_events/library_usage_observation.rb +48 -0
  273. data/lib/contrast/agent/reporting/reporting_events/observed_library_usage.rb +45 -0
  274. data/lib/contrast/agent/reporting/reporting_events/observed_route.rb +89 -0
  275. data/lib/contrast/agent/reporting/reporting_events/poll.rb +23 -0
  276. data/lib/contrast/agent/reporting/reporting_events/preflight.rb +41 -0
  277. data/lib/contrast/agent/reporting/reporting_events/preflight_message.rb +74 -0
  278. data/lib/contrast/agent/reporting/reporting_events/reporting_event.rb +66 -0
  279. data/lib/contrast/agent/reporting/reporting_events/route_coverage.rb +89 -0
  280. data/lib/contrast/agent/reporting/reporting_events/route_discovery.rb +63 -0
  281. data/lib/contrast/agent/reporting/reporting_events/route_discovery_observation.rb +53 -0
  282. data/lib/contrast/agent/reporting/reporting_events/server_reporting_event.rb +35 -0
  283. data/lib/contrast/agent/reporting/reporting_events/server_settings.rb +40 -0
  284. data/lib/contrast/agent/reporting/reporting_utilities/audit.rb +130 -0
  285. data/lib/contrast/agent/reporting/reporting_utilities/build_preflight.rb +35 -0
  286. data/lib/contrast/agent/reporting/reporting_utilities/endpoints.rb +176 -0
  287. data/lib/contrast/agent/reporting/reporting_utilities/headers.rb +54 -0
  288. data/lib/contrast/agent/reporting/reporting_utilities/reporter_client.rb +143 -0
  289. data/lib/contrast/agent/reporting/reporting_utilities/reporter_client_utils.rb +144 -0
  290. data/lib/contrast/agent/reporting/reporting_utilities/reporting_storage.rb +66 -0
  291. data/lib/contrast/agent/reporting/reporting_utilities/response.rb +98 -0
  292. data/lib/contrast/agent/reporting/reporting_utilities/response_extractor.rb +176 -0
  293. data/lib/contrast/agent/reporting/reporting_utilities/response_handler.rb +117 -0
  294. data/lib/contrast/agent/reporting/reporting_utilities/response_handler_mode.rb +63 -0
  295. data/lib/contrast/agent/reporting/reporting_utilities/response_handler_utils.rb +342 -0
  296. data/lib/contrast/agent/reporting/server_settings_worker.rb +44 -0
  297. data/lib/contrast/agent/reporting/settings/application_settings.rb +61 -0
  298. data/lib/contrast/agent/reporting/settings/assess.rb +45 -0
  299. data/lib/contrast/agent/reporting/settings/assess_server_feature.rb +114 -0
  300. data/lib/contrast/agent/reporting/settings/bot_blocker.rb +68 -0
  301. data/lib/contrast/agent/reporting/settings/code_exclusion.rb +32 -0
  302. data/lib/contrast/agent/reporting/settings/exclusion_base.rb +51 -0
  303. data/lib/contrast/agent/reporting/settings/exclusions.rb +106 -0
  304. data/lib/contrast/agent/reporting/settings/helpers.rb +63 -0
  305. data/lib/contrast/agent/reporting/settings/input_exclusion.rb +43 -0
  306. data/lib/contrast/agent/reporting/settings/ip_filter.rb +35 -0
  307. data/lib/contrast/agent/reporting/settings/keyword.rb +74 -0
  308. data/lib/contrast/agent/reporting/settings/log_enhancer.rb +65 -0
  309. data/lib/contrast/agent/reporting/settings/protect.rb +106 -0
  310. data/lib/contrast/agent/reporting/settings/protect_server_feature.rb +227 -0
  311. data/lib/contrast/agent/reporting/settings/reaction.rb +39 -0
  312. data/lib/contrast/agent/reporting/settings/rule_definition.rb +66 -0
  313. data/lib/contrast/agent/reporting/settings/sampling.rb +46 -0
  314. data/lib/contrast/agent/reporting/settings/sanitizer.rb +38 -0
  315. data/lib/contrast/agent/reporting/settings/security_logger.rb +77 -0
  316. data/lib/contrast/agent/reporting/settings/sensitive_data_masking.rb +118 -0
  317. data/lib/contrast/agent/reporting/settings/sensitive_data_masking_rule.rb +65 -0
  318. data/lib/contrast/agent/reporting/settings/server_features.rb +95 -0
  319. data/lib/contrast/agent/reporting/settings/syslog.rb +205 -0
  320. data/lib/contrast/agent/reporting/settings/url_exclusion.rb +42 -0
  321. data/lib/contrast/agent/reporting/settings/validator.rb +17 -0
  322. data/lib/contrast/agent/request.rb +107 -411
  323. data/lib/contrast/agent/request_context.rb +78 -162
  324. data/lib/contrast/agent/request_context_extend.rb +85 -0
  325. data/lib/contrast/agent/request_handler.rb +41 -0
  326. data/lib/contrast/agent/response.rb +37 -165
  327. data/lib/contrast/agent/rule_set.rb +52 -0
  328. data/lib/contrast/agent/scope.rb +142 -20
  329. data/lib/contrast/agent/static_analysis.rb +51 -0
  330. data/lib/contrast/agent/telemetry/base.rb +155 -0
  331. data/lib/contrast/agent/telemetry/events/event.rb +35 -0
  332. data/lib/contrast/agent/telemetry/events/exceptions/obfuscate.rb +119 -0
  333. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exception_base.rb +61 -0
  334. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exception_event.rb +46 -0
  335. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exception_message.rb +118 -0
  336. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exception_message_exception.rb +86 -0
  337. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exception_stack_frame.rb +67 -0
  338. data/lib/contrast/agent/telemetry/events/exceptions/telemetry_exceptions.rb +19 -0
  339. data/lib/contrast/agent/telemetry/events/metric_event.rb +28 -0
  340. data/lib/contrast/agent/telemetry/events/startup_metrics_event.rb +123 -0
  341. data/lib/contrast/agent/thread.rb +4 -6
  342. data/lib/contrast/agent/thread_watcher.rb +117 -0
  343. data/lib/contrast/agent/tracepoint_hook.rb +19 -13
  344. data/lib/contrast/agent/version.rb +2 -2
  345. data/lib/contrast/agent/worker_thread.rb +42 -0
  346. data/lib/contrast/agent.rb +83 -50
  347. data/lib/contrast/agent_lib/api/command_injection.rb +46 -0
  348. data/lib/contrast/agent_lib/api/init.rb +101 -0
  349. data/lib/contrast/agent_lib/api/input_tracing.rb +267 -0
  350. data/lib/contrast/agent_lib/api/method_tempering.rb +29 -0
  351. data/lib/contrast/agent_lib/api/panic.rb +87 -0
  352. data/lib/contrast/agent_lib/api/path_semantic_file_security_bypass.rb +40 -0
  353. data/lib/contrast/agent_lib/interface.rb +260 -0
  354. data/lib/contrast/agent_lib/interface_base.rb +118 -0
  355. data/lib/contrast/agent_lib/return_types/eval_result.rb +44 -0
  356. data/lib/contrast/agent_lib/test.rb +29 -0
  357. data/lib/contrast/api/communication/connection_status.rb +59 -0
  358. data/lib/contrast/components/agent.rb +108 -36
  359. data/lib/contrast/components/api.rb +159 -0
  360. data/lib/contrast/components/app_context.rb +124 -134
  361. data/lib/contrast/components/app_context_extend.rb +53 -0
  362. data/lib/contrast/components/assess.rb +187 -24
  363. data/lib/contrast/components/assess_rules.rb +54 -0
  364. data/lib/contrast/components/base.rb +103 -0
  365. data/lib/contrast/components/config/sources.rb +95 -0
  366. data/lib/contrast/components/config.rb +182 -60
  367. data/lib/contrast/components/heap_dump.rb +77 -12
  368. data/lib/contrast/components/inventory.rb +37 -10
  369. data/lib/contrast/components/logger.rb +46 -76
  370. data/lib/contrast/components/polling.rb +36 -0
  371. data/lib/contrast/components/protect.rb +142 -16
  372. data/lib/contrast/components/ruby_component.rb +96 -0
  373. data/lib/contrast/components/sampling.rb +156 -15
  374. data/lib/contrast/components/scope.rb +116 -85
  375. data/lib/contrast/components/security_logger.rb +36 -0
  376. data/lib/contrast/components/settings.rb +197 -90
  377. data/lib/contrast/config/api_proxy_configuration.rb +27 -0
  378. data/lib/contrast/config/base_configuration.rb +20 -94
  379. data/lib/contrast/config/certification_configuration.rb +47 -0
  380. data/lib/contrast/config/config.rb +46 -0
  381. data/lib/contrast/config/diagnostics.rb +114 -0
  382. data/lib/contrast/config/diagnostics_tools.rb +98 -0
  383. data/lib/contrast/config/effective_config.rb +65 -0
  384. data/lib/contrast/config/effective_config_value.rb +32 -0
  385. data/lib/contrast/config/env_variables.rb +18 -0
  386. data/lib/contrast/config/exception_configuration.rb +34 -12
  387. data/lib/contrast/config/protect_rule_configuration.rb +45 -24
  388. data/lib/contrast/config/protect_rules_configuration.rb +97 -22
  389. data/lib/contrast/config/request_audit_configuration.rb +57 -0
  390. data/lib/contrast/config/server_configuration.rb +67 -15
  391. data/lib/contrast/config.rb +6 -22
  392. data/lib/contrast/configuration.rb +231 -108
  393. data/lib/contrast/extension/assess/array.rb +75 -0
  394. data/lib/contrast/extension/assess/erb.rb +61 -0
  395. data/lib/contrast/extension/assess/eval_trigger.rb +47 -0
  396. data/lib/contrast/{core_extensions → extension}/assess/exec_trigger.rb +9 -21
  397. data/lib/contrast/extension/assess/fiber.rb +95 -0
  398. data/lib/contrast/extension/assess/hash.rb +33 -0
  399. data/lib/contrast/extension/assess/kernel.rb +124 -0
  400. data/lib/contrast/extension/assess/marshal.rb +80 -0
  401. data/lib/contrast/extension/assess/regexp.rb +71 -0
  402. data/lib/contrast/extension/assess/string.rb +84 -0
  403. data/lib/contrast/extension/assess.rb +47 -0
  404. data/lib/contrast/{core_extensions → extension}/delegator.rb +3 -1
  405. data/lib/contrast/extension/extension.rb +59 -0
  406. data/lib/contrast/extension/inventory.rb +21 -0
  407. data/lib/contrast/extension/module.rb +16 -0
  408. data/lib/contrast/extension/object.rb +19 -0
  409. data/lib/contrast/extension/protect/psych.rb +7 -0
  410. data/lib/contrast/{core_extensions → extension}/protect.rb +6 -6
  411. data/lib/contrast/extension/thread.rb +50 -0
  412. data/lib/contrast/framework/base_support.rb +78 -0
  413. data/lib/contrast/framework/grape/support.rb +176 -0
  414. data/lib/contrast/framework/manager.rb +158 -0
  415. data/lib/contrast/framework/manager_extend.rb +50 -0
  416. data/lib/contrast/framework/rack/patch/session_cookie.rb +107 -0
  417. data/lib/contrast/framework/rack/patch/support.rb +26 -0
  418. data/lib/contrast/framework/rack/support.rb +23 -0
  419. data/lib/contrast/framework/rails/patch/action_controller_live_buffer.rb +46 -0
  420. data/lib/contrast/framework/rails/patch/assess_configuration.rb +98 -0
  421. data/lib/contrast/framework/rails/patch/rails_application_configuration.rb +31 -0
  422. data/lib/contrast/framework/rails/patch/support.rb +46 -0
  423. data/lib/contrast/framework/rails/railtie.rb +33 -0
  424. data/lib/contrast/framework/rails/support.rb +187 -0
  425. data/lib/contrast/framework/sinatra/support.rb +165 -0
  426. data/lib/contrast/funchook/funchook.rb +45 -0
  427. data/lib/contrast/logger/aliased_logging.rb +101 -0
  428. data/lib/contrast/logger/application.rb +84 -0
  429. data/lib/contrast/logger/cef_log.rb +169 -0
  430. data/lib/contrast/logger/format.rb +61 -0
  431. data/lib/contrast/logger/log.rb +90 -0
  432. data/lib/contrast/logger/request.rb +25 -0
  433. data/lib/contrast/logger/time.rb +57 -0
  434. data/lib/contrast/security_exception.rb +2 -2
  435. data/lib/contrast/tasks/config.rb +144 -0
  436. data/lib/contrast/utils/assess/event_limit_utils.rb +134 -0
  437. data/lib/contrast/utils/assess/object_store.rb +36 -0
  438. data/lib/contrast/utils/assess/propagation_method_utils.rb +155 -0
  439. data/lib/contrast/utils/assess/property/tagged_utils.rb +165 -0
  440. data/lib/contrast/utils/assess/sampling_util.rb +11 -17
  441. data/lib/contrast/utils/assess/source_method_utils.rb +74 -0
  442. data/lib/contrast/utils/assess/split_utils.rb +23 -0
  443. data/lib/contrast/utils/assess/tracking_util.rb +95 -19
  444. data/lib/contrast/utils/assess/trigger_method_utils.rb +132 -0
  445. data/lib/contrast/utils/class_util.rb +125 -38
  446. data/lib/contrast/utils/duck_utils.rb +54 -43
  447. data/lib/contrast/utils/env_configuration_item.rb +4 -3
  448. data/lib/contrast/utils/findings.rb +66 -0
  449. data/lib/contrast/utils/hash_digest.rb +52 -100
  450. data/lib/contrast/utils/hash_digest_extend.rb +129 -0
  451. data/lib/contrast/utils/head_dump_utils_extend.rb +74 -0
  452. data/lib/contrast/utils/heap_dump_util.rb +44 -88
  453. data/lib/contrast/utils/input_classification_base.rb +155 -0
  454. data/lib/contrast/utils/invalid_configuration_util.rb +36 -50
  455. data/lib/contrast/utils/io_util.rb +47 -51
  456. data/lib/contrast/utils/job_servers_running.rb +47 -0
  457. data/lib/contrast/utils/log_utils.rb +254 -0
  458. data/lib/contrast/utils/lru_cache.rb +48 -0
  459. data/lib/contrast/utils/metrics_hash.rb +59 -0
  460. data/lib/contrast/utils/middleware_utils.rb +89 -0
  461. data/lib/contrast/utils/net_http_base.rb +167 -0
  462. data/lib/contrast/utils/object_share.rb +7 -48
  463. data/lib/contrast/utils/os.rb +14 -24
  464. data/lib/contrast/utils/patching/policy/patch_utils.rb +175 -0
  465. data/lib/contrast/utils/patching/policy/patcher_utils.rb +54 -0
  466. data/lib/contrast/utils/reporting/application_activity_batch_utils.rb +81 -0
  467. data/lib/contrast/utils/request_utils.rb +96 -0
  468. data/lib/contrast/utils/resource_loader.rb +2 -2
  469. data/lib/contrast/utils/response_utils.rb +79 -0
  470. data/lib/contrast/utils/routes_sent.rb +60 -0
  471. data/lib/contrast/utils/sha256_builder.rb +9 -21
  472. data/lib/contrast/utils/stack_trace_utils.rb +68 -184
  473. data/lib/contrast/utils/string_utils.rb +82 -52
  474. data/lib/contrast/utils/tag_util.rb +58 -44
  475. data/lib/contrast/utils/telemetry.rb +103 -0
  476. data/lib/contrast/utils/telemetry_client.rb +107 -0
  477. data/lib/contrast/utils/telemetry_hash.rb +65 -0
  478. data/lib/contrast/utils/telemetry_identifier.rb +153 -0
  479. data/lib/contrast/utils/thread_tracker.rb +27 -23
  480. data/lib/contrast/utils/timer.rb +20 -55
  481. data/lib/contrast-agent.rb +2 -2
  482. data/lib/contrast.rb +105 -43
  483. data/resources/assess/policy.json +523 -137
  484. data/resources/deadzone/policy.json +280 -10
  485. data/resources/inventory/policy.json +2 -2
  486. data/resources/protect/policy.json +30 -17
  487. data/ruby-agent.gemspec +114 -45
  488. data/sonar-project.properties +9 -0
  489. metadata +694 -287
  490. data/exe/contrast_service +0 -29
  491. data/ext/cs__assess_active_record_named/cs__active_record_named.c +0 -47
  492. data/ext/cs__assess_active_record_named/cs__active_record_named.h +0 -10
  493. data/ext/cs__assess_active_record_named/extconf.rb +0 -2
  494. data/ext/cs__assess_regexp_track/cs__assess_regexp_track.c +0 -63
  495. data/ext/cs__assess_regexp_track/cs__assess_regexp_track.h +0 -29
  496. data/ext/cs__assess_regexp_track/extconf.rb +0 -2
  497. data/ext/cs__assess_string_interpolation26/cs__assess_string_interpolation26.c +0 -31
  498. data/ext/cs__assess_string_interpolation26/cs__assess_string_interpolation26.h +0 -13
  499. data/ext/cs__assess_string_interpolation26/extconf.rb +0 -2
  500. data/ext/cs__protect_kernel/cs__protect_kernel.c +0 -37
  501. data/ext/cs__protect_kernel/cs__protect_kernel.h +0 -11
  502. data/ext/cs__protect_kernel/extconf.rb +0 -2
  503. data/funchook/Makefile +0 -29
  504. data/funchook/autom4te.cache/output.0 +0 -4976
  505. data/funchook/autom4te.cache/requests +0 -78
  506. data/funchook/autom4te.cache/traces.0 +0 -364
  507. data/funchook/config.log +0 -490
  508. data/funchook/config.status +0 -1016
  509. data/funchook/configure +0 -4976
  510. data/funchook/src/Makefile +0 -70
  511. data/funchook/src/config.h +0 -101
  512. data/funchook/src/config.h.in +0 -100
  513. data/funchook/src/decoder.o +0 -0
  514. data/funchook/src/distorm.o +0 -0
  515. data/funchook/src/funchook.o +0 -0
  516. data/funchook/src/funchook_io.o +0 -0
  517. data/funchook/src/funchook_syscall.o +0 -0
  518. data/funchook/src/funchook_unix.o +0 -0
  519. data/funchook/src/funchook_x86.o +0 -0
  520. data/funchook/src/instructions.o +0 -0
  521. data/funchook/src/insts.o +0 -0
  522. data/funchook/src/libfunchook.so +0 -0
  523. data/funchook/src/mnemonics.o +0 -0
  524. data/funchook/src/operands.o +0 -0
  525. data/funchook/src/os_func.o +0 -0
  526. data/funchook/src/os_func_unix.o +0 -0
  527. data/funchook/src/prefix.o +0 -0
  528. data/funchook/src/printf_base.o +0 -0
  529. data/funchook/src/textdefs.o +0 -0
  530. data/funchook/src/wstring.o +0 -0
  531. data/funchook/test/Makefile +0 -43
  532. data/funchook/test/funchook_test +0 -0
  533. data/funchook/test/libfunchook_test.so +0 -0
  534. data/funchook/test/test_main.o +0 -0
  535. data/funchook/test/x86_64_test.o +0 -0
  536. data/lib/contrast/agent/assess/adjusted_span.rb +0 -25
  537. data/lib/contrast/agent/assess/class_reverter.rb +0 -82
  538. data/lib/contrast/agent/assess/contrast_event.rb +0 -398
  539. data/lib/contrast/agent/assess/frozen_properties.rb +0 -41
  540. data/lib/contrast/agent/assess/insulator.rb +0 -53
  541. data/lib/contrast/agent/assess/policy/rewriter_patch.rb +0 -79
  542. data/lib/contrast/agent/assess/rule/base.rb +0 -72
  543. data/lib/contrast/agent/assess/rule/csrf/csrf_action.rb +0 -28
  544. data/lib/contrast/agent/assess/rule/csrf/csrf_applicator.rb +0 -69
  545. data/lib/contrast/agent/assess/rule/csrf/csrf_watcher.rb +0 -132
  546. data/lib/contrast/agent/assess/rule/csrf.rb +0 -66
  547. data/lib/contrast/agent/assess/rule/redos.rb +0 -68
  548. data/lib/contrast/agent/assess/rule/response_scanning_rule.rb +0 -47
  549. data/lib/contrast/agent/assess/rule/response_watcher.rb +0 -36
  550. data/lib/contrast/agent/assess/rule/watcher.rb +0 -36
  551. data/lib/contrast/agent/assess/rule.rb +0 -18
  552. data/lib/contrast/agent/class_reopener.rb +0 -195
  553. data/lib/contrast/agent/feature_state.rb +0 -379
  554. data/lib/contrast/agent/logger_manager.rb +0 -116
  555. data/lib/contrast/agent/patching/policy/policy_unpatcher.rb +0 -28
  556. data/lib/contrast/agent/protect/rule/csrf/csrf_evaluator.rb +0 -103
  557. data/lib/contrast/agent/protect/rule/csrf/csrf_token_injector.rb +0 -85
  558. data/lib/contrast/agent/protect/rule/csrf.rb +0 -118
  559. data/lib/contrast/agent/railtie.rb +0 -30
  560. data/lib/contrast/agent/reaction_processor.rb +0 -47
  561. data/lib/contrast/agent/require_state.rb +0 -61
  562. data/lib/contrast/agent/rewriter.rb +0 -244
  563. data/lib/contrast/agent/service_heartbeat.rb +0 -37
  564. data/lib/contrast/agent/settings_state.rb +0 -148
  565. data/lib/contrast/agent/socket_client.rb +0 -125
  566. data/lib/contrast/api/connection_status.rb +0 -49
  567. data/lib/contrast/api/dtm_pb.rb +0 -718
  568. data/lib/contrast/api/settings_pb.rb +0 -416
  569. data/lib/contrast/api/socket.rb +0 -43
  570. data/lib/contrast/api/speedracer.rb +0 -206
  571. data/lib/contrast/api/tcp_socket.rb +0 -31
  572. data/lib/contrast/api/unix_socket.rb +0 -25
  573. data/lib/contrast/api.rb +0 -17
  574. data/lib/contrast/common_agent_configuration.rb +0 -86
  575. data/lib/contrast/components/contrast_service.rb +0 -113
  576. data/lib/contrast/components/interface.rb +0 -178
  577. data/lib/contrast/config/agent_configuration.rb +0 -24
  578. data/lib/contrast/config/application_configuration.rb +0 -27
  579. data/lib/contrast/config/assess_configuration.rb +0 -22
  580. data/lib/contrast/config/assess_rules_configuration.rb +0 -18
  581. data/lib/contrast/config/default_value.rb +0 -16
  582. data/lib/contrast/config/heap_dump_configuration.rb +0 -23
  583. data/lib/contrast/config/inventory_configuration.rb +0 -20
  584. data/lib/contrast/config/logger_configuration.rb +0 -20
  585. data/lib/contrast/config/protect_configuration.rb +0 -20
  586. data/lib/contrast/config/root_configuration.rb +0 -26
  587. data/lib/contrast/config/ruby_configuration.rb +0 -39
  588. data/lib/contrast/config/sampling_configuration.rb +0 -22
  589. data/lib/contrast/config/service_configuration.rb +0 -22
  590. data/lib/contrast/core_extensions/assess/array.rb +0 -58
  591. data/lib/contrast/core_extensions/assess/assess_extension.rb +0 -145
  592. data/lib/contrast/core_extensions/assess/basic_object.rb +0 -15
  593. data/lib/contrast/core_extensions/assess/erb.rb +0 -42
  594. data/lib/contrast/core_extensions/assess/fiber.rb +0 -125
  595. data/lib/contrast/core_extensions/assess/hash.rb +0 -22
  596. data/lib/contrast/core_extensions/assess/kernel.rb +0 -95
  597. data/lib/contrast/core_extensions/assess/module.rb +0 -14
  598. data/lib/contrast/core_extensions/assess/regexp.rb +0 -206
  599. data/lib/contrast/core_extensions/assess/string.rb +0 -75
  600. data/lib/contrast/core_extensions/assess/tilt_template_trigger.rb +0 -73
  601. data/lib/contrast/core_extensions/assess.rb +0 -51
  602. data/lib/contrast/core_extensions/eval_trigger.rb +0 -52
  603. data/lib/contrast/core_extensions/inventory/datastores.rb +0 -37
  604. data/lib/contrast/core_extensions/inventory.rb +0 -22
  605. data/lib/contrast/core_extensions/module.rb +0 -42
  606. data/lib/contrast/core_extensions/object.rb +0 -27
  607. data/lib/contrast/core_extensions/protect/applies_command_injection_rule.rb +0 -70
  608. data/lib/contrast/core_extensions/protect/applies_deserialization_rule.rb +0 -58
  609. data/lib/contrast/core_extensions/protect/applies_no_sqli_rule.rb +0 -81
  610. data/lib/contrast/core_extensions/protect/applies_path_traversal_rule.rb +0 -119
  611. data/lib/contrast/core_extensions/protect/applies_sqli_rule.rb +0 -63
  612. data/lib/contrast/core_extensions/protect/applies_xxe_rule.rb +0 -141
  613. data/lib/contrast/core_extensions/protect/kernel.rb +0 -30
  614. data/lib/contrast/core_extensions/protect/psych.rb +0 -7
  615. data/lib/contrast/core_extensions/thread.rb +0 -31
  616. data/lib/contrast/internal_exception.rb +0 -8
  617. data/lib/contrast/rails_extensions/assess/action_controller_inheritance.rb +0 -48
  618. data/lib/contrast/rails_extensions/assess/active_record.rb +0 -32
  619. data/lib/contrast/rails_extensions/assess/active_record_named.rb +0 -61
  620. data/lib/contrast/rails_extensions/assess/configuration.rb +0 -26
  621. data/lib/contrast/rails_extensions/buffer.rb +0 -30
  622. data/lib/contrast/rails_extensions/rack.rb +0 -45
  623. data/lib/contrast/sinatra_extensions/assess/cookie.rb +0 -26
  624. data/lib/contrast/sinatra_extensions/inventory/sinatra_base.rb +0 -59
  625. data/lib/contrast/tasks/service.rb +0 -95
  626. data/lib/contrast/utils/boolean_util.rb +0 -33
  627. data/lib/contrast/utils/cache.rb +0 -69
  628. data/lib/contrast/utils/comment_range.rb +0 -19
  629. data/lib/contrast/utils/data_store_util.rb +0 -23
  630. data/lib/contrast/utils/environment_util.rb +0 -152
  631. data/lib/contrast/utils/freeze_util.rb +0 -36
  632. data/lib/contrast/utils/gemfile_reader.rb +0 -191
  633. data/lib/contrast/utils/inventory_util.rb +0 -126
  634. data/lib/contrast/utils/operating_environment.rb +0 -38
  635. data/lib/contrast/utils/path_util.rb +0 -151
  636. data/lib/contrast/utils/performs_logging.rb +0 -152
  637. data/lib/contrast/utils/preflight_util.rb +0 -13
  638. data/lib/contrast/utils/prevent_serialization.rb +0 -52
  639. data/lib/contrast/utils/rack_assess_session_cookie.rb +0 -104
  640. data/lib/contrast/utils/rails_assess_configuration.rb +0 -95
  641. data/lib/contrast/utils/random_util.rb +0 -22
  642. data/lib/contrast/utils/ruby_ast_rewriter.rb +0 -74
  643. data/lib/contrast/utils/scope_util.rb +0 -99
  644. data/lib/contrast/utils/service_response_util.rb +0 -116
  645. data/lib/contrast/utils/service_sender_util.rb +0 -98
  646. data/lib/contrast/utils/sinatra_helper.rb +0 -49
  647. data/resources/csrf/inject.js +0 -44
  648. data/resources/factory-bot-spec/spec_helper.rb +0 -30
  649. data/resources/rubocops/kernel/catch_cop.rb +0 -37
  650. data/resources/rubocops/kernel/require_cop.rb +0 -37
  651. data/resources/rubocops/kernel/require_relative_cop.rb +0 -33
  652. data/resources/rubocops/module/autoload_cop.rb +0 -37
  653. data/resources/rubocops/module/const_defined_cop.rb +0 -37
  654. data/resources/rubocops/module/const_get_cop.rb +0 -37
  655. data/resources/rubocops/module/const_set_cop.rb +0 -37
  656. data/resources/rubocops/module/constants_cop.rb +0 -37
  657. data/resources/rubocops/module/name_cop.rb +0 -37
  658. data/resources/rubocops/object/class_cop.rb +0 -37
  659. data/resources/rubocops/object/freeze_cop.rb +0 -37
  660. data/resources/rubocops/object/frozen_cop.rb +0 -37
  661. data/resources/rubocops/object/is_a_cop.rb +0 -37
  662. data/resources/rubocops/object/method_cop.rb +0 -37
  663. data/resources/rubocops/object/respond_to_cop.rb +0 -37
  664. data/resources/rubocops/object/singleton_class_cop.rb +0 -37
  665. data/resources/rubocops/regexp/spelling_cop.rb +0 -44
  666. data/resources/rubocops/thread/new_cop.rb +0 -39
  667. data/resources/ruby-spec/ancestors_spec.rb +0 -70
  668. data/resources/ruby-spec/modulo_spec.rb +0 -831
  669. data/resources/ruby-spec/parameters_spec.rb +0 -261
  670. data/resources/ruby-spec/ruby_spec_spec_helper.rb +0 -35
  671. data/service_executables/.gitkeep +0 -0
  672. data/service_executables/VERSION +0 -1
  673. data/service_executables/linux/contrast-service +0 -0
  674. data/service_executables/mac/contrast-service +0 -0
  675. data/shared_libraries/funchook.h +0 -123
  676. data/shared_libraries/libfunchook.so +0 -0
@@ -0,0 +1,23 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/framework/base_support'
5
+ require 'contrast/framework/rack/patch/support'
6
+
7
+ module Contrast
8
+ module Framework
9
+ module Rack
10
+ # Used when Rack is present to define framework specific behavior. For
11
+ # now, the only part of this implemented is the Patch Support.
12
+ module Support
13
+ extend Contrast::Framework::BaseSupport
14
+ extend Contrast::Framework::Rack::Patch::Support
15
+ class << self
16
+ def detection_class
17
+ 'rack -- don\'t let me be detected'
18
+ end
19
+ end
20
+ end
21
+ end
22
+ end
23
+ end
@@ -0,0 +1,46 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/agent/inventory/dependency_usage_analysis'
5
+
6
+ module Contrast
7
+ module Framework
8
+ module Rails
9
+ module Patch
10
+ # This class acts as our patch into the ActionController::Live::Buffer class, allowing us to track the close
11
+ # event on streamed responses.
12
+ module ActionControllerLiveBuffer
13
+ class << self
14
+ def send_messages
15
+ return unless (context = Contrast::Agent::REQUEST_TRACKER.current)
16
+
17
+ [
18
+ context.observed_route
19
+ ].each do |event|
20
+ Contrast::Agent.reporter&.send_event_immediately(event)
21
+ end
22
+
23
+ Contrast::Agent.reporter&.send_event_immediately(context.activity)
24
+ end
25
+
26
+ def instrument
27
+ @_instrument ||= begin
28
+ ::ActionController::Live::Buffer.class_eval do
29
+ # normally pre->in->post filters are applied however, in a streamed response we can run into a case
30
+ # where it's pre -> in -> post -> more infilters in order to submit anything found during the
31
+ # infilters after the response has been written we need to explicitly send them
32
+ alias_method(:cs__close, :close)
33
+ def close
34
+ Contrast::Framework::Rails::Patch::ActionControllerLiveBuffer.send_messages
35
+ cs__close
36
+ end
37
+ end
38
+ true
39
+ end
40
+ end
41
+ end
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,98 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/utils/invalid_configuration_util'
5
+
6
+ module Contrast
7
+ module Framework
8
+ module Rails
9
+ module Patch
10
+ # This module is used to analyze rails session storage configuration for assess vulnerabilities
11
+ module AssessConfiguration
12
+ include Contrast::Components::Logger::InstanceMethods
13
+
14
+ CS__SESSION_TIMEOUT_NAME = 'session-timeout'
15
+ SAFE_SESSION_TIMEOUT = (30 * 60 * 1000)
16
+ CS__SECURE_RULE_NAME = 'secure-flag-missing'
17
+ CS__HTTPONLY_RULE_NAME = 'rails-http-only-disabled'
18
+
19
+ class << self
20
+ include Contrast::Utils::InvalidConfigurationUtil
21
+
22
+ def analyze_session_store *args
23
+ return if ::Contrast::ASSESS.forcibly_disabled?
24
+
25
+ apply_httponly_disabled(*args)
26
+ apply_secure_cookie_disabled(*args)
27
+ apply_session_timeout(*args)
28
+ end
29
+
30
+ private
31
+
32
+ def vulnerable_setting?(setting_key,
33
+ safe_settings_value,
34
+ original_args,
35
+ safe_default: true,
36
+ comparison_type: nil)
37
+ # In most cases, Rails is pretty nice and the default value is safe
38
+ return !safe_default unless original_args && original_args.length > 1
39
+
40
+ # If the user overrode some args, but not ours, fall back on the default
41
+ rails_session_settings = original_args[1]
42
+ return !safe_default unless rails_session_settings&.key?(setting_key)
43
+
44
+ value = rails_session_settings[setting_key]
45
+
46
+ return value.to_i > safe_settings_value.to_i if comparison_type&.to_sym == :greater_than
47
+
48
+ value != safe_settings_value
49
+ end
50
+
51
+ def apply_session_timeout *args
52
+ return if ::Contrast::ASSESS.rule_disabled?(CS__SESSION_TIMEOUT_NAME)
53
+ return unless vulnerable_setting?(:expire_after, SAFE_SESSION_TIMEOUT, args,
54
+ comparison_type: :greater_than, safe_default: false)
55
+
56
+ rails_session_settings = args[1]
57
+ cs__report_finding(CS__SESSION_TIMEOUT_NAME, rails_session_settings, caller_locations(3, 2)[0])
58
+ rescue StandardError => e
59
+ begin
60
+ logger.error('Unable to track call to set session timeout', e)
61
+ rescue StandardError
62
+ nil
63
+ end
64
+ end
65
+
66
+ def apply_secure_cookie_disabled *args
67
+ return if ::Contrast::ASSESS.rule_disabled?(CS__SECURE_RULE_NAME)
68
+ return unless vulnerable_setting?(:secure, true, args)
69
+
70
+ rails_session_settings = args[1]
71
+ cs__report_finding(CS__SECURE_RULE_NAME, rails_session_settings, caller_locations(3, 2)[0])
72
+ rescue StandardError => e
73
+ begin
74
+ logger.error('Unable to track call to disable secure cookies', e)
75
+ rescue StandardError
76
+ nil
77
+ end
78
+ end
79
+
80
+ def apply_httponly_disabled *args
81
+ return if ::Contrast::ASSESS.rule_disabled?(CS__HTTPONLY_RULE_NAME)
82
+ return unless vulnerable_setting?(:httponly, true, args)
83
+
84
+ rails_session_settings = args[1]
85
+ cs__report_finding(CS__HTTPONLY_RULE_NAME, rails_session_settings, caller_locations(3, 2)[0])
86
+ rescue StandardError => e
87
+ begin
88
+ logger.error('Unable to track call to disable httponly in session cookie', e)
89
+ rescue StandardError
90
+ nil
91
+ end
92
+ end
93
+ end
94
+ end
95
+ end
96
+ end
97
+ end
98
+ end
@@ -0,0 +1,31 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/framework/rails/patch/assess_configuration'
5
+ module Contrast
6
+ module Framework
7
+ module Rails
8
+ module Patch
9
+ # Our patch into the Rails::Application::Configuration Class, allowing
10
+ # for the runtime detection of insecure configurations on individual
11
+ # ActionDispatch::Session::AbstractStore instances within the
12
+ # application.
13
+ module RailsApplicationConfiguration
14
+ def self.instrument
15
+ @_instrument ||= begin
16
+ ::Rails::Application::Configuration.class_eval do
17
+ alias_method(:cs__patched_session_store, :session_store)
18
+ def session_store *args, **kwargs
19
+ ret = cs__patched_session_store(*args, **kwargs)
20
+ Contrast::Framework::Rails::Patch::AssessConfiguration.analyze_session_store(*args, **kwargs)
21
+ ret
22
+ end
23
+ end
24
+ true
25
+ end
26
+ end
27
+ end
28
+ end
29
+ end
30
+ end
31
+ end
@@ -0,0 +1,46 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/framework/rails/patch/rails_application_configuration'
5
+ require 'contrast/agent/patching/policy/after_load_patch'
6
+
7
+ module Contrast
8
+ module Framework
9
+ module Rails
10
+ module Patch
11
+ # Extension point allowing for the registration of Patches required to
12
+ # support the Rails framework.
13
+ module Support
14
+ # (See BaseSupport#before_load_patches!)
15
+ def before_load_patches!
16
+ return unless defined?(::Rails)
17
+
18
+ # In Rails, session configuration occurs extremely early & only once.
19
+ # If we defer our patching of the rails session configuration too long
20
+ # (i.e., where we normally patch) we will miss the configuration
21
+ # and will never be able to report session misconfiguration rules.
22
+ Contrast::Framework::Rails::Patch::RailsApplicationConfiguration.instrument
23
+ require('contrast/framework/rails/railtie') if ::Rails::VERSION::MAJOR.to_i >= 3
24
+ end
25
+
26
+ # (See BaseSupport#after_load_patches)
27
+ def after_load_patches
28
+ Set.new([
29
+ Contrast::Agent::Patching::Policy::AfterLoadPatch.new(
30
+ 'ActionController::Live::Buffer',
31
+ 'contrast/framework/rails/patch/action_controller_live_buffer',
32
+ instrumenting_module:
33
+ 'Contrast::Framework::Rails::Patch::ActionControllerLiveBuffer'),
34
+ Contrast::Agent::Patching::Policy::AfterLoadPatch.new(
35
+ 'Rails::Application::Configuration',
36
+ 'contrast/framework/rails/patch/rails_application_configuration',
37
+ method_to_instrument: :session_store,
38
+ instrumenting_module:
39
+ 'Contrast::Framework::Rails::Patch::RailsApplicationConfiguration')
40
+ ])
41
+ end
42
+ end
43
+ end
44
+ end
45
+ end
46
+ end
@@ -0,0 +1,33 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/utils/job_servers_running'
5
+ require 'contrast/components/logger'
6
+
7
+ module Contrast
8
+ module Framework
9
+ module Rails
10
+ # A Railtie to allow for the automatic hooking of the Agent into a Rails application.
11
+ class Railtie < ::Rails::Railtie
12
+ include Contrast::Components::Logger::InstanceMethods
13
+
14
+ initializer 'Contrast Ruby Agent Initializer' do |app|
15
+ log_rails = defined?(Rails) && defined?(Rails.logger)
16
+
17
+ Rails.logger.debug { "In railtie ::#{ app.middleware.inspect }" } if log_rails
18
+
19
+ if ::Contrast::APP_CONTEXT.instrument_middleware_stack?
20
+ ::Contrast::AGENT.insert_middleware(app)
21
+ else
22
+ Rails.logger.debug('Detected a running job server, skipping Contrast middleware insertion.') if log_rails
23
+ logger.debug('Disabling Contrast for process', p_id: Process.pid)
24
+ end
25
+ end
26
+
27
+ rake_tasks do
28
+ load 'contrast/tasks/config.rb'
29
+ end
30
+ end
31
+ end
32
+ end
33
+ end
@@ -0,0 +1,187 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/framework/base_support'
5
+ require 'contrast/framework/rails/patch/support'
6
+ require 'contrast/utils/string_utils'
7
+
8
+ module Contrast
9
+ module Framework
10
+ module Rails
11
+ # Used when Rails is present to define framework specific behavior
12
+ class Support # rubocop:disable Metrics/ClassLength
13
+ extend Contrast::Framework::BaseSupport
14
+ extend Contrast::Framework::Rails::Patch::Support
15
+ include Contrast::Components::Logger::InstanceMethods
16
+ extend Contrast::Components::Logger::InstanceMethods
17
+
18
+ class << self
19
+ RAILS_MODULE_NAME_VERSION = Gem::Version.new('6.0.0')
20
+
21
+ def detection_class
22
+ 'Rails'
23
+ end
24
+
25
+ def version
26
+ ::Rails.version
27
+ end
28
+
29
+ def application_name
30
+ app_class = ::Rails.application.cs__class
31
+ # Rails version 6.0.0 deprecated Rails::Application#parent_name, in Rails 6.1.0 that method will be removed
32
+ # entirely and instead we need to use parent_module_name
33
+ return app_class.parent_module_name if Gem::Version.new(::Rails.version) >= RAILS_MODULE_NAME_VERSION
34
+
35
+ app_class.parent_name
36
+ end
37
+
38
+ def application_root
39
+ ::Rails.root
40
+ end
41
+
42
+ def server_type
43
+ 'rails'
44
+ end
45
+
46
+ # @return [Array<Contrast::Agent::Reporting::DiscoveredRoute>]
47
+ def collect_routes
48
+ find_all_routes(::Rails.application, [])
49
+ end
50
+
51
+ # Find the current route, based on the provided Request wrapper
52
+ #
53
+ # @param request[Contrast::Agent::Request]
54
+ # @return [Contrast::Agent::Reporting::RouteCoverage, nil] a Dtm describing the route
55
+ # matched to the request if a match was found.
56
+ def current_route_coverage request
57
+ return unless ::Rails.cs__respond_to?(:application)
58
+
59
+ # ActionDispatch::Journey::Path::Pattern::MatchData, Hash, ActionDispatch::Journey::Route, Array<String>
60
+ match, _params, route, path = get_full_route(request.rack_request)
61
+ unless route
62
+ logger.warn("Unable to determine the current route of this request: #{ request.rack_request }")
63
+ return
64
+ end
65
+
66
+ original_url = request.rack_request.path_info
67
+ mounted_app = route&.app&.app
68
+ # Route is either the final rails route, or a router that points to a Sinatra controller.
69
+ if mounted_app && Contrast::Framework::Sinatra::Support.sinatra_controller?(mounted_app)
70
+ return mounted_new_sinatra_route(request, match, path, route, original_url)
71
+ end
72
+ if mounted_app && Contrast::Framework::Grape::Support.grape_controller?(mounted_app)
73
+ return mounted_new_grape_route(request, match, path, route, original_url)
74
+ end
75
+
76
+ new_route_coverage = Contrast::Agent::Reporting::RouteCoverage.new
77
+ new_route_coverage&.attach_rails_data(route, original_url)
78
+ new_route_coverage
79
+ rescue StandardError => e
80
+ logger.warn('Unable to determine the current route of this request due to exception: ', e)
81
+ nil
82
+ end
83
+
84
+ # Copy a request for modification.
85
+ #
86
+ # @param env [::ActionDispatch::Request] original env.
87
+ # @return [::ActionDispatch::Request] a copy of original env with rails env merged.
88
+ def retrieve_request env
89
+ rails_env = ::Rails.application.env_config.merge(env)
90
+ ::ActionDispatch::Request.new(rails_env || env)
91
+ end
92
+
93
+ AC_INSTANCE = 'action_controller.instance'
94
+ def streaming? env
95
+ return false unless defined?(::ActionController::Live)
96
+
97
+ env[AC_INSTANCE].cs__class.included_modules.include?(::ActionController::Live)
98
+ end
99
+
100
+ private
101
+
102
+ # Determine if route is a Rails engine route.
103
+ #
104
+ # @param route [Object] app or route that points to a ::Rails::Engine
105
+ # @return [Boolean, nil] whether the router is an engine or not.
106
+ def engine_route? route
107
+ return false unless route&.app&.app
108
+ return false unless route.app.is_a?(::ActionDispatch::Routing::Mapper::Constraints) ||
109
+ route.app.is_a?(::ActionDispatch::Routing::RouteSet::Dispatcher)
110
+
111
+ clazz = route.app.app.is_a?(Class) ? route.app.app : route.app.app.cs__class
112
+ clazz < ::Rails::Engine
113
+ end
114
+
115
+ # Recursively get final route traversing engines as required. Because this can only be called once, we store
116
+ # this match for the duration of our request context.
117
+ #
118
+ # @param request [::Rack::Request] the rack request as will be handed to rails controller.
119
+ # @param top_router [::ActionDispatch::Journey::Router] the current router relative to the previous.
120
+ # @param path [Array<String>] the chunks of path that have been seen.
121
+ # @return [Array<Object>] the final set of rails route classes.
122
+ # ActionDispatch::Journey::Path::Pattern::MatchData, Hash, ActionDispatch::Journey::Route, Array<String>
123
+ def get_full_route request, top_router = ::Rails.application.routes.router, path = []
124
+ return if (route_matches = top_router.send(:find_routes, request)).empty?
125
+
126
+ match, params, route = route_matches.first
127
+
128
+ # If the current routing node points to a sub-app (::Rails::Engine), dive deeper.
129
+ # Have sub-app route the remainder of the url.
130
+ if engine_route?(route)
131
+ new_req = retrieve_request(request.env)
132
+ new_req.path_info = new_req.path_info.gsub(match.to_s, '')
133
+ get_full_route(new_req, route.app.app.routes.router, path << match.to_s)
134
+ else
135
+ [match, params, route, path]
136
+ end
137
+ end
138
+
139
+ # Rails engine routes need to be detected by inspecting Engine class route set
140
+ #
141
+ # @param app [Rails::Application]
142
+ # @param route_list [Array<Contrast::Agent::Reporting::DiscoveredRoute>] the list of discovered routes to
143
+ # which to append and return
144
+ # @return [Array<Contrast::Agent::Reporting::DiscoveredRoute>]
145
+ def find_all_routes app, route_list
146
+ return route_list unless app.cs__respond_to?(:routes) && app.routes.cs__respond_to?(:routes)
147
+
148
+ app.routes.routes.each do |route|
149
+ if route.cs__respond_to?(:app) && route.app.cs__class == ActionDispatch::Routing::RouteSet::Dispatcher
150
+ route_list << Contrast::Agent::Reporting::DiscoveredRoute.from_action_dispatch_journey(route)
151
+ elsif route.app.app.cs__respond_to?(:routes)
152
+ route_list += find_all_routes(route.app.app, [])
153
+ end
154
+ end
155
+ logger.debug("Routes Found: #{ route_list }")
156
+ route_list
157
+ end
158
+
159
+ # @return [Contrast::Agent::Reporting::RouteCoverage, nil]
160
+ def mounted_new_sinatra_route request, match, path, route, original_url
161
+ new_req = unmounted_route(request, match, path)
162
+ Contrast::Framework::Sinatra::Support.current_route_coverage(new_req, route.app.app, original_url)
163
+ end
164
+
165
+ # @return [Contrast::Agent::Reporting::RouteCoverage, nil]
166
+ def mounted_new_grape_route request, match, path, route, original_url
167
+ new_req = unmounted_route(request, match, path)
168
+ Contrast::Framework::Grape::Support.current_route_coverage(new_req, route.app.app, original_url)
169
+ end
170
+
171
+ # Create a request copied from current request, but with the base path removed from path_info, as that's
172
+ # the mount.
173
+ #
174
+ # @param request[Contrast::Agent::Request]
175
+ # @param match []
176
+ # @param path [String] the path of this request
177
+ # @return [::ActionDispatch::Request]
178
+ def unmounted_route request, match, path
179
+ new_req = ::ActionDispatch::Request.new(request.env)
180
+ new_req.path_info = new_req.path_info.gsub((path << match).join, '')
181
+ new_req
182
+ end
183
+ end
184
+ end
185
+ end
186
+ end
187
+ end
@@ -0,0 +1,165 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/framework/base_support'
5
+ require 'contrast/agent/reporting/report'
6
+
7
+ module Contrast
8
+ module Framework
9
+ module Sinatra
10
+ # Used when Sinatra is present to define framework specific behavior
11
+ class Support
12
+ extend Contrast::Framework::BaseSupport
13
+ class << self
14
+ def detection_class
15
+ 'Sinatra'
16
+ end
17
+
18
+ def version
19
+ ::Sinatra::VERSION
20
+ end
21
+
22
+ def application_name
23
+ app_class&.cs__name
24
+ end
25
+
26
+ def application_root
27
+ app_instance&.root
28
+ end
29
+
30
+ def server_type
31
+ 'sinatra'
32
+ end
33
+
34
+ # Given an object, determine if it is a Sinatra controller with routes.
35
+ #
36
+ # @param app [Object] suspected Sinatra app.
37
+ # @return [Boolean]
38
+ def sinatra_controller? app
39
+ # Sinatra is loaded?
40
+ return false unless defined?(::Sinatra) && defined?(::Sinatra::Base)
41
+ # App is a subclass of or actually is ::Sinatra::Base.
42
+ return false unless (app.cs__respond_to?(:<) && app < ::Sinatra::Base) || app == ::Sinatra::Base
43
+
44
+ # App has routes.
45
+ !app.routes.empty?
46
+ end
47
+
48
+ # Find all classes that subclass ::Sinatra::Base. Gather their routes.
49
+ #
50
+ # @return [Array<Contrast::Agent::Reporting::DiscoveredRoute>]
51
+ def collect_routes
52
+ return Contrast::Utils::ObjectShare::EMPTY_ARRAY unless defined?(::Sinatra) && defined?(::Sinatra::Base)
53
+
54
+ routes = []
55
+ sinatra_controllers.each do |controller|
56
+ controller.routes.each_pair do |method, route_triplets|
57
+ # Sinatra stores its routes as a triplet: [Mustermann::Sinatra, [], Proc]
58
+ route_triplets.map(&:first).each do |route_pattern|
59
+ routes << Contrast::Agent::Reporting::DiscoveredRoute.from_sinatra_route(controller, method,
60
+ route_pattern)
61
+ end
62
+ end
63
+ end
64
+ routes
65
+ end
66
+
67
+ # Given the current request - return a RouteCoverage object
68
+
69
+ # @param request [Contrast::Agent::Request] a contrast tracked request.
70
+ # @param controller [::Sinatra::Base] optionally use this controller instead of global ::Sinatra::Base.
71
+ # @return [Contrast::Agent::Reporting::RouteCoverage, nil] a Dtm describing the route
72
+ # matched to the request if a match was found.
73
+ def current_route_coverage request, controller = ::Sinatra::Base, full_route = nil
74
+ return unless sinatra_controller?(controller)
75
+
76
+ method = request.env[::Rack::REQUEST_METHOD] # GET, PUT, POST, etc...
77
+
78
+ # Find route match--checking superclasses if necessary.
79
+ final_controller, route_pattern = _route_recurse(controller, method, _cleaned_route(request))
80
+ return unless final_controller && route_pattern
81
+
82
+ full_route ||= request.env[::Rack::PATH_INFO]
83
+
84
+ new_route_coverage = Contrast::Agent::Reporting::RouteCoverage.new
85
+ new_route_coverage.attach_rack_based_data(final_controller, method, route_pattern, full_route)
86
+ new_route_coverage
87
+ end
88
+
89
+ # Search object space for sinatra controllers--any class that subclasses ::Sinatra::Base.
90
+ #
91
+ # @return [Array<::Sinatra::Base>] sinatra controlelrs
92
+ def sinatra_controllers
93
+ [::Sinatra::Base] + ObjectSpace.each_object(Class).select { |clazz| sinatra_controller?(clazz) }
94
+ end
95
+
96
+ def retrieve_request env
97
+ ::Sinatra::Request.new(env)
98
+ end
99
+
100
+ private
101
+
102
+ # Given a controller and a route to match against, find the route_pattern and class that will serve the
103
+ # route. This is recursive as Sinatra's routing is recursive from subclass to super.
104
+ #
105
+ # @param controller [Sinatra::Base, #routes] a Sinatra application.
106
+ # @param method [::Rack::REQUEST_METHOD] GET, POST, PUT, etc...
107
+ # @param route [String] the relative route passed from Rack.
108
+ # @return [Array[Sinatra::Base, Mustermann::Sinatra], nil] Either the controller that
109
+ # will handle the route along with the route pattern or nil if no match.
110
+ def _route_recurse controller, method, route
111
+ return if controller.nil? || controller.cs__class == NilClass
112
+
113
+ route_patterns = controller.routes.fetch(method) { [] }.
114
+ map(&:first)
115
+ route_pattern = route_patterns&.find do |matcher|
116
+ matcher.params(route) # ::Mustermann::Sinatra match.
117
+ end
118
+
119
+ return controller, route_pattern if route_pattern
120
+
121
+ # Check routes defined in superclass if present.
122
+ return unless controller.superclass&.instance_variable_get(:@routes)
123
+
124
+ _route_recurse(controller.superclass, method, route)
125
+ end
126
+
127
+ # Get route and do some cleanup matching that of Sinatra::Base#process_route.
128
+ #
129
+ # @param request [Contrast::Agent::Request] a contrast tracked request.
130
+ # @return [String] the extracted and cleaned relative route.
131
+ def _cleaned_route request
132
+ settings = ::Sinatra::Base.settings
133
+ route = request.env[::Rack::PATH_INFO]
134
+ return '/' if route.empty? && !settings.empty_path_info?
135
+
136
+ !settings.strict_paths? && route.end_with?('/') ? route[0..-2] : route
137
+ end
138
+
139
+ # Almost an alias to app_instance.
140
+ #
141
+ # @return [::Sinatra::Base] the current controller class as routed by Rack.
142
+ def app_class
143
+ return unless defined?(::Sinatra) && defined?(::Sinatra::Base)
144
+
145
+ app_instance.cs__class
146
+ end
147
+
148
+ # Search the object space for the controller handling this request which will be
149
+ # the class inheriting from ::Sinatra::Base with @app=nil since it is the final servicer
150
+ # in the request/middleware chain.
151
+ #
152
+ # @return [::Sinatra::Base] the current controller as routed by Rack.
153
+ def app_instance
154
+ return unless defined?(::Sinatra) && defined?(::Sinatra::Base)
155
+
156
+ @_app_instance ||= begin
157
+ sinatra_layers = ObjectSpace.each_object(::Sinatra::Base).to_a
158
+ sinatra_layers.find { |layer| layer.app.nil? }
159
+ end
160
+ end
161
+ end
162
+ end
163
+ end
164
+ end
165
+ end
@@ -0,0 +1,45 @@
1
+ # Copyright (c) 2022 Contrast Security, Inc. See https://www.contrastsecurity.com/enduser-terms-0317a for more details.
2
+ # frozen_string_literal: true
3
+
4
+ require 'contrast/components/logger'
5
+
6
+ # This module is used to find funchook library and determine availability
7
+ module Funchook
8
+ extend Contrast::Components::Logger::InstanceMethods
9
+
10
+ attr_accessor :path
11
+
12
+ # Possible platform library files
13
+ ACCEPTABLE_FILES = %w[libfunchook.dylib libfunchook.so].cs__freeze
14
+
15
+ # Top level agent directories that should have the funchook libraries
16
+ SEARCH_DIRS = [File.join('ext'), File.join('shared_libraries'), File.join('funchook', 'src')].cs__freeze
17
+
18
+ AGENT_ROOT = File.join(__dir__, '..', '..', '..')
19
+
20
+ class << self
21
+ def resolve_path!
22
+ actual_path_segments = SEARCH_DIRS.product(ACCEPTABLE_FILES).find do |potential_funchook_path|
23
+ load_path = absolute_path(potential_funchook_path)
24
+ File.exist?(load_path)
25
+ end
26
+
27
+ if actual_path_segments.nil?
28
+ logger.warn('Unable to find funchook')
29
+ else
30
+ @path = absolute_path(actual_path_segments)
31
+ logger.info('Funchook found', path: @path)
32
+ end
33
+ @path
34
+ end
35
+
36
+ def absolute_path path_array
37
+ File.absolute_path(File.join(AGENT_ROOT, *path_array))
38
+ end
39
+
40
+ def available?
41
+ @_available = !!resolve_path! if @_available.nil?
42
+ @_available
43
+ end
44
+ end
45
+ end