itsi-server 0.1.1 → 0.2.2

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (457) hide show
  1. checksums.yaml +4 -4
  2. data/Cargo.lock +4487 -0
  3. data/Cargo.toml +7 -0
  4. data/README.md +6 -0
  5. data/Rakefile +7 -4
  6. data/exe/itsi +152 -46
  7. data/ext/itsi_acme/Cargo.toml +86 -0
  8. data/ext/itsi_acme/examples/high_level.rs +63 -0
  9. data/ext/itsi_acme/examples/high_level_warp.rs +52 -0
  10. data/ext/itsi_acme/examples/low_level.rs +87 -0
  11. data/ext/itsi_acme/examples/low_level_axum.rs +66 -0
  12. data/ext/itsi_acme/src/acceptor.rs +81 -0
  13. data/ext/itsi_acme/src/acme.rs +354 -0
  14. data/ext/itsi_acme/src/axum.rs +86 -0
  15. data/ext/itsi_acme/src/cache.rs +39 -0
  16. data/ext/itsi_acme/src/caches/boxed.rs +80 -0
  17. data/ext/itsi_acme/src/caches/composite.rs +69 -0
  18. data/ext/itsi_acme/src/caches/dir.rs +106 -0
  19. data/ext/itsi_acme/src/caches/mod.rs +11 -0
  20. data/ext/itsi_acme/src/caches/no.rs +78 -0
  21. data/ext/itsi_acme/src/caches/test.rs +136 -0
  22. data/ext/itsi_acme/src/config.rs +172 -0
  23. data/ext/itsi_acme/src/https_helper.rs +69 -0
  24. data/ext/itsi_acme/src/incoming.rs +142 -0
  25. data/ext/itsi_acme/src/jose.rs +161 -0
  26. data/ext/itsi_acme/src/lib.rs +142 -0
  27. data/ext/itsi_acme/src/resolver.rs +59 -0
  28. data/ext/itsi_acme/src/state.rs +424 -0
  29. data/ext/itsi_error/Cargo.toml +3 -0
  30. data/ext/itsi_error/src/lib.rs +98 -24
  31. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
  32. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
  33. data/ext/itsi_error/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
  34. data/ext/itsi_error/target/debug/build/rb-sys-49f554618693db24/out/bindings-0.9.110-mri-arm64-darwin23-3.4.2.rs +8865 -0
  35. data/ext/itsi_error/target/debug/incremental/itsi_error-1mmt5sux7jb0i/s-h510z7m8v9-0bxu7yd.lock +0 -0
  36. data/ext/itsi_error/target/debug/incremental/itsi_error-2vn3jey74oiw0/s-h5113n0e7e-1v5qzs6.lock +0 -0
  37. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510ykifhe-0tbnep2.lock +0 -0
  38. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510yyocpj-0tz7ug7.lock +0 -0
  39. data/ext/itsi_error/target/debug/incremental/itsi_error-37uv9dicz7awp/s-h510z0xc8g-14ol18k.lock +0 -0
  40. data/ext/itsi_error/target/debug/incremental/itsi_error-3g5qf4y7d54uj/s-h5113n0e7d-1trk8on.lock +0 -0
  41. data/ext/itsi_error/target/debug/incremental/itsi_error-3lpfftm45d3e2/s-h510z7m8r3-1pxp20o.lock +0 -0
  42. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510ykifek-1uxasnk.lock +0 -0
  43. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510yyocki-11u37qm.lock +0 -0
  44. data/ext/itsi_error/target/debug/incremental/itsi_error-3o4qownhl3d7n/s-h510z0xc93-0pmy0zm.lock +0 -0
  45. data/ext/itsi_instrument_entry/Cargo.toml +15 -0
  46. data/ext/itsi_instrument_entry/src/lib.rs +31 -0
  47. data/ext/itsi_rb_helpers/Cargo.toml +3 -0
  48. data/ext/itsi_rb_helpers/src/heap_value.rs +139 -0
  49. data/ext/itsi_rb_helpers/src/lib.rs +141 -10
  50. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/common.rs +355 -0
  51. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/dynamic.rs +276 -0
  52. data/ext/itsi_rb_helpers/target/debug/build/clang-sys-da71b0344e568175/out/macros.rs +49 -0
  53. data/ext/itsi_rb_helpers/target/debug/build/rb-sys-eb9ed4ff3a60f995/out/bindings-0.9.110-mri-arm64-darwin23-3.4.2.rs +8865 -0
  54. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-040pxg6yhb3g3/s-h5113n7a1b-03bwlt4.lock +0 -0
  55. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h51113xnh3-1eik1ip.lock +0 -0
  56. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-131g1u4dzkt1a/s-h5111704jj-0g4rj8x.lock +0 -0
  57. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-1q2d3drtxrzs5/s-h5113n79yl-0bxcqc5.lock +0 -0
  58. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h51113xoox-10de2hp.lock +0 -0
  59. data/ext/itsi_rb_helpers/target/debug/incremental/itsi_rb_helpers-374a9h7ovycj0/s-h5111704w7-0vdq7gq.lock +0 -0
  60. data/ext/itsi_scheduler/Cargo.toml +24 -0
  61. data/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +56 -0
  62. data/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +44 -0
  63. data/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +44 -0
  64. data/ext/itsi_scheduler/src/itsi_scheduler.rs +314 -0
  65. data/ext/itsi_scheduler/src/lib.rs +39 -0
  66. data/ext/itsi_server/Cargo.lock +2956 -0
  67. data/ext/itsi_server/Cargo.toml +75 -14
  68. data/ext/itsi_server/extconf.rb +1 -1
  69. data/ext/itsi_server/src/default_responses/html/401.html +68 -0
  70. data/ext/itsi_server/src/default_responses/html/403.html +68 -0
  71. data/ext/itsi_server/src/default_responses/html/404.html +68 -0
  72. data/ext/itsi_server/src/default_responses/html/413.html +71 -0
  73. data/ext/itsi_server/src/default_responses/html/429.html +68 -0
  74. data/ext/itsi_server/src/default_responses/html/500.html +71 -0
  75. data/ext/itsi_server/src/default_responses/html/502.html +71 -0
  76. data/ext/itsi_server/src/default_responses/html/503.html +68 -0
  77. data/ext/itsi_server/src/default_responses/html/504.html +69 -0
  78. data/ext/itsi_server/src/default_responses/html/index.html +238 -0
  79. data/ext/itsi_server/src/default_responses/json/401.json +6 -0
  80. data/ext/itsi_server/src/default_responses/json/403.json +6 -0
  81. data/ext/itsi_server/src/default_responses/json/404.json +6 -0
  82. data/ext/itsi_server/src/default_responses/json/413.json +6 -0
  83. data/ext/itsi_server/src/default_responses/json/429.json +6 -0
  84. data/ext/itsi_server/src/default_responses/json/500.json +6 -0
  85. data/ext/itsi_server/src/default_responses/json/502.json +6 -0
  86. data/ext/itsi_server/src/default_responses/json/503.json +6 -0
  87. data/ext/itsi_server/src/default_responses/json/504.json +6 -0
  88. data/ext/itsi_server/src/default_responses/mod.rs +11 -0
  89. data/ext/itsi_server/src/env.rs +43 -0
  90. data/ext/itsi_server/src/lib.rs +133 -40
  91. data/ext/itsi_server/src/prelude.rs +2 -0
  92. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/big_bytes.rs +109 -0
  93. data/ext/itsi_server/src/ruby_types/itsi_body_proxy/mod.rs +143 -0
  94. data/ext/itsi_server/src/ruby_types/itsi_grpc_call.rs +344 -0
  95. data/ext/itsi_server/src/ruby_types/itsi_grpc_response_stream/mod.rs +264 -0
  96. data/ext/itsi_server/src/ruby_types/itsi_http_request.rs +362 -0
  97. data/ext/itsi_server/src/ruby_types/itsi_http_response.rs +391 -0
  98. data/ext/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +233 -0
  99. data/ext/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +565 -0
  100. data/ext/itsi_server/src/ruby_types/itsi_server.rs +86 -0
  101. data/ext/itsi_server/src/ruby_types/mod.rs +48 -0
  102. data/ext/itsi_server/src/server/binds/bind.rs +204 -0
  103. data/ext/itsi_server/src/server/binds/bind_protocol.rs +37 -0
  104. data/ext/itsi_server/src/server/binds/listener.rs +444 -0
  105. data/ext/itsi_server/src/server/binds/mod.rs +4 -0
  106. data/ext/itsi_server/src/server/binds/tls/locked_dir_cache.rs +132 -0
  107. data/ext/itsi_server/src/server/binds/tls.rs +278 -0
  108. data/ext/itsi_server/src/server/byte_frame.rs +32 -0
  109. data/ext/itsi_server/src/server/http_message_types.rs +97 -0
  110. data/ext/itsi_server/src/server/io_stream.rs +105 -0
  111. data/ext/itsi_server/src/server/lifecycle_event.rs +12 -0
  112. data/ext/itsi_server/src/server/middleware_stack/middleware.rs +170 -0
  113. data/ext/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +63 -0
  114. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +94 -0
  115. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +94 -0
  116. data/ext/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +343 -0
  117. data/ext/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +151 -0
  118. data/ext/itsi_server/src/server/middleware_stack/middlewares/compression.rs +316 -0
  119. data/ext/itsi_server/src/server/middleware_stack/middlewares/cors.rs +301 -0
  120. data/ext/itsi_server/src/server/middleware_stack/middlewares/csp.rs +193 -0
  121. data/ext/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +64 -0
  122. data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response/default_responses.rs +192 -0
  123. data/ext/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +171 -0
  124. data/ext/itsi_server/src/server/middleware_stack/middlewares/etag.rs +198 -0
  125. data/ext/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +82 -0
  126. data/ext/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +209 -0
  127. data/ext/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +82 -0
  128. data/ext/itsi_server/src/server/middleware_stack/middlewares/max_body.rs +47 -0
  129. data/ext/itsi_server/src/server/middleware_stack/middlewares/mod.rs +116 -0
  130. data/ext/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +411 -0
  131. data/ext/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +142 -0
  132. data/ext/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +55 -0
  133. data/ext/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +54 -0
  134. data/ext/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +51 -0
  135. data/ext/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +126 -0
  136. data/ext/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +187 -0
  137. data/ext/itsi_server/src/server/middleware_stack/middlewares/static_response.rs +55 -0
  138. data/ext/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +173 -0
  139. data/ext/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +31 -0
  140. data/ext/itsi_server/src/server/middleware_stack/mod.rs +381 -0
  141. data/ext/itsi_server/src/server/mod.rs +13 -5
  142. data/ext/itsi_server/src/server/process_worker.rs +247 -0
  143. data/ext/itsi_server/src/server/redirect_type.rs +26 -0
  144. data/ext/itsi_server/src/server/request_job.rs +11 -0
  145. data/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +354 -0
  146. data/ext/itsi_server/src/server/serve_strategy/mod.rs +30 -0
  147. data/ext/itsi_server/src/server/serve_strategy/single_mode.rs +481 -0
  148. data/ext/itsi_server/src/server/signal.rs +77 -0
  149. data/ext/itsi_server/src/server/size_limited_incoming.rs +107 -0
  150. data/ext/itsi_server/src/server/thread_worker.rs +479 -0
  151. data/ext/itsi_server/src/services/cache_store.rs +74 -0
  152. data/ext/itsi_server/src/services/itsi_http_service.rs +257 -0
  153. data/ext/itsi_server/src/services/mime_types.rs +1416 -0
  154. data/ext/itsi_server/src/services/mod.rs +6 -0
  155. data/ext/itsi_server/src/services/password_hasher.rs +83 -0
  156. data/ext/itsi_server/src/services/rate_limiter.rs +580 -0
  157. data/ext/itsi_server/src/services/static_file_server.rs +1340 -0
  158. data/ext/itsi_tracing/Cargo.toml +5 -0
  159. data/ext/itsi_tracing/src/lib.rs +366 -7
  160. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0994n8rpvvt9m/s-h510hfz1f6-1kbycmq.lock +0 -0
  161. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-0bob7bf4yq34i/s-h5113125h5-0lh4rag.lock +0 -0
  162. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2fcodulrxbbxo/s-h510h2infk-0hp5kjw.lock +0 -0
  163. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2iak63r1woi1l/s-h510h2in4q-0kxfzw1.lock +0 -0
  164. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2kk4qj9gn5dg2/s-h5113124kv-0enwon2.lock +0 -0
  165. data/ext/itsi_tracing/target/debug/incremental/itsi_tracing-2mwo0yas7dtw4/s-h510hfz1ha-1udgpei.lock +0 -0
  166. data/lib/itsi/http_request/response_status_shortcodes.rb +76 -0
  167. data/lib/itsi/http_request.rb +218 -0
  168. data/lib/itsi/http_response.rb +42 -0
  169. data/lib/itsi/passfile.rb +108 -0
  170. data/lib/itsi/server/config/config_helpers.rb +105 -0
  171. data/lib/itsi/server/config/dsl.rb +211 -0
  172. data/lib/itsi/server/config/known_paths/KitchensinkDirectories.txt +2346 -0
  173. data/lib/itsi/server/config/known_paths/Randomfiles.txt +24 -0
  174. data/lib/itsi/server/config/known_paths/UnixDotfiles.txt +52 -0
  175. data/lib/itsi/server/config/known_paths/backdoors/ASP_CommonBackdoors.txt +29 -0
  176. data/lib/itsi/server/config/known_paths/backdoors/bot_control_panels.txt +1668 -0
  177. data/lib/itsi/server/config/known_paths/backdoors/shells.txt +1167 -0
  178. data/lib/itsi/server/config/known_paths/cgi/CGI_HTTP_POST.txt +7 -0
  179. data/lib/itsi/server/config/known_paths/cgi/CGI_HTTP_POST_Windows.txt +6 -0
  180. data/lib/itsi/server/config/known_paths/cgi/CGI_Microsoft.txt +79 -0
  181. data/lib/itsi/server/config/known_paths/cgi/CGI_XPlatform.txt +3948 -0
  182. data/lib/itsi/server/config/known_paths/cms/README.md +5 -0
  183. data/lib/itsi/server/config/known_paths/cms/drupal_plugins.txt +6320 -0
  184. data/lib/itsi/server/config/known_paths/cms/drupal_themes.txt +828 -0
  185. data/lib/itsi/server/config/known_paths/cms/joomla_plugins.txt +224 -0
  186. data/lib/itsi/server/config/known_paths/cms/joomla_themes.txt +30 -0
  187. data/lib/itsi/server/config/known_paths/cms/php-nuke.txt +2142 -0
  188. data/lib/itsi/server/config/known_paths/cms/wordpress.txt +1566 -0
  189. data/lib/itsi/server/config/known_paths/cms/wp_common_theme_files.txt +46 -0
  190. data/lib/itsi/server/config/known_paths/cms/wp_plugins.txt +13366 -0
  191. data/lib/itsi/server/config/known_paths/cms/wp_plugins_full.txt +68662 -0
  192. data/lib/itsi/server/config/known_paths/cms/wp_plugins_top225.txt +225 -0
  193. data/lib/itsi/server/config/known_paths/cms/wp_themes.readme +12 -0
  194. data/lib/itsi/server/config/known_paths/cms/wp_themes.txt +7336 -0
  195. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/3CharExtBrute.txt +17576 -0
  196. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/CommonWebExtensions.txt +80 -0
  197. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/Extensions.Backup.txt +14 -0
  198. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/Extensions.Common.txt +865 -0
  199. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/Extensions.Compressed.txt +186 -0
  200. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/Extensions.Mostcommon.txt +30 -0
  201. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/Extensions.Skipfish.txt +93 -0
  202. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/WordlistSkipfish.txt +1918 -0
  203. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/copy_of.txt +8 -0
  204. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-directories-lowercase.txt +56180 -0
  205. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-directories.txt +62290 -0
  206. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-extensions-lowercase.txt +2367 -0
  207. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-extensions.txt +2450 -0
  208. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-files-lowercase.txt +35323 -0
  209. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-files.txt +37037 -0
  210. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-words-lowercase.txt +107982 -0
  211. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-large-words.txt +119600 -0
  212. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-directories-lowercase.txt +26593 -0
  213. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-directories.txt +30009 -0
  214. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-extensions-lowercase.txt +1233 -0
  215. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-extensions.txt +1289 -0
  216. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-files-lowercase.txt +16243 -0
  217. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-files.txt +17128 -0
  218. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-words-lowercase.txt +56293 -0
  219. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-medium-words.txt +63087 -0
  220. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-directories-lowercase.txt +17776 -0
  221. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-directories.txt +20122 -0
  222. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-extensions-lowercase.txt +914 -0
  223. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-extensions.txt +963 -0
  224. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-files-lowercase.txt +10848 -0
  225. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-files.txt +11424 -0
  226. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-words-lowercase.txt +38267 -0
  227. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/raft-small-words.txt +43003 -0
  228. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/spanish.txt +445 -0
  229. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/test_demo.txt +36 -0
  230. data/lib/itsi/server/config/known_paths/filename-dirname-bruteforce/upload_variants.txt +44 -0
  231. data/lib/itsi/server/config/known_paths/login-file-locations/Logins.txt +71 -0
  232. data/lib/itsi/server/config/known_paths/login-file-locations/cfm.txt +294 -0
  233. data/lib/itsi/server/config/known_paths/login-file-locations/html.txt +295 -0
  234. data/lib/itsi/server/config/known_paths/login-file-locations/jsp.txt +294 -0
  235. data/lib/itsi/server/config/known_paths/login-file-locations/php.txt +294 -0
  236. data/lib/itsi/server/config/known_paths/login-file-locations/windows-asp.txt +294 -0
  237. data/lib/itsi/server/config/known_paths/login-file-locations/windows-aspx.txt +294 -0
  238. data/lib/itsi/server/config/known_paths/password-file-locations/Passwords.txt +47 -0
  239. data/lib/itsi/server/config/known_paths/php/PHP.txt +30 -0
  240. data/lib/itsi/server/config/known_paths/php/PHP_CommonBackdoors.txt +5 -0
  241. data/lib/itsi/server/config/known_paths/proxy-conf.txt +31 -0
  242. data/lib/itsi/server/config/known_paths/tftp.txt +79 -0
  243. data/lib/itsi/server/config/known_paths/webservers-appservers/ADFS.txt +86 -0
  244. data/lib/itsi/server/config/known_paths/webservers-appservers/AdobeXML.txt +16 -0
  245. data/lib/itsi/server/config/known_paths/webservers-appservers/Apache.txt +101 -0
  246. data/lib/itsi/server/config/known_paths/webservers-appservers/ApacheTomcat.txt +47 -0
  247. data/lib/itsi/server/config/known_paths/webservers-appservers/Apache_Axis.txt +16 -0
  248. data/lib/itsi/server/config/known_paths/webservers-appservers/ColdFusion.txt +111 -0
  249. data/lib/itsi/server/config/known_paths/webservers-appservers/FatwireCMS.txt +390 -0
  250. data/lib/itsi/server/config/known_paths/webservers-appservers/Frontpage.txt +38 -0
  251. data/lib/itsi/server/config/known_paths/webservers-appservers/HP_System_Mgmt_Homepage.txt +239 -0
  252. data/lib/itsi/server/config/known_paths/webservers-appservers/HTTP_POST_Microsoft.txt +2 -0
  253. data/lib/itsi/server/config/known_paths/webservers-appservers/Hyperion.txt +578 -0
  254. data/lib/itsi/server/config/known_paths/webservers-appservers/IIS.txt +187 -0
  255. data/lib/itsi/server/config/known_paths/webservers-appservers/JBoss.txt +5 -0
  256. data/lib/itsi/server/config/known_paths/webservers-appservers/JRun.txt +13 -0
  257. data/lib/itsi/server/config/known_paths/webservers-appservers/JavaServlets_Common.txt +3 -0
  258. data/lib/itsi/server/config/known_paths/webservers-appservers/Joomla_exploitable.txt +1937 -0
  259. data/lib/itsi/server/config/known_paths/webservers-appservers/LotusNotes.txt +206 -0
  260. data/lib/itsi/server/config/known_paths/webservers-appservers/Netware.txt +18 -0
  261. data/lib/itsi/server/config/known_paths/webservers-appservers/Oracle9i.txt +60 -0
  262. data/lib/itsi/server/config/known_paths/webservers-appservers/OracleAppServer.txt +192 -0
  263. data/lib/itsi/server/config/known_paths/webservers-appservers/README.md +6 -0
  264. data/lib/itsi/server/config/known_paths/webservers-appservers/Ruby_Rails.txt +121 -0
  265. data/lib/itsi/server/config/known_paths/webservers-appservers/SAP.txt +463 -0
  266. data/lib/itsi/server/config/known_paths/webservers-appservers/Sharepoint.txt +1707 -0
  267. data/lib/itsi/server/config/known_paths/webservers-appservers/SiteMinder.txt +19 -0
  268. data/lib/itsi/server/config/known_paths/webservers-appservers/SunAppServerGlassfish.txt +51 -0
  269. data/lib/itsi/server/config/known_paths/webservers-appservers/SuniPlanet.txt +35 -0
  270. data/lib/itsi/server/config/known_paths/webservers-appservers/Vignette.txt +73 -0
  271. data/lib/itsi/server/config/known_paths/webservers-appservers/Weblogic.txt +160 -0
  272. data/lib/itsi/server/config/known_paths/webservers-appservers/Websphere.txt +366 -0
  273. data/lib/itsi/server/config/known_paths/wellknown-rfc5785.txt +30 -0
  274. data/lib/itsi/server/config/known_paths.rb +20 -0
  275. data/lib/itsi/server/config/middleware/_index.md +56 -0
  276. data/lib/itsi/server/config/middleware/allow_list.md +46 -0
  277. data/lib/itsi/server/config/middleware/allow_list.rb +42 -0
  278. data/lib/itsi/server/config/middleware/auth_api_key.md +90 -0
  279. data/lib/itsi/server/config/middleware/auth_api_key.rb +51 -0
  280. data/lib/itsi/server/config/middleware/auth_basic.md +45 -0
  281. data/lib/itsi/server/config/middleware/auth_basic.rb +44 -0
  282. data/lib/itsi/server/config/middleware/auth_jwt.md +82 -0
  283. data/lib/itsi/server/config/middleware/auth_jwt.rb +38 -0
  284. data/lib/itsi/server/config/middleware/cache_control.md +78 -0
  285. data/lib/itsi/server/config/middleware/cache_control.rb +45 -0
  286. data/lib/itsi/server/config/middleware/cidr_to_regex.rb +50 -0
  287. data/lib/itsi/server/config/middleware/compression.md +50 -0
  288. data/lib/itsi/server/config/middleware/compression.rb +37 -0
  289. data/lib/itsi/server/config/middleware/cors.md +93 -0
  290. data/lib/itsi/server/config/middleware/cors.rb +32 -0
  291. data/lib/itsi/server/config/middleware/csp.md +37 -0
  292. data/lib/itsi/server/config/middleware/csp.rb +44 -0
  293. data/lib/itsi/server/config/middleware/deny_list.md +45 -0
  294. data/lib/itsi/server/config/middleware/deny_list.rb +42 -0
  295. data/lib/itsi/server/config/middleware/endpoint/_index.md +159 -0
  296. data/lib/itsi/server/config/middleware/endpoint/controller.md +186 -0
  297. data/lib/itsi/server/config/middleware/endpoint/controller.rb +33 -0
  298. data/lib/itsi/server/config/middleware/endpoint/delete.md +12 -0
  299. data/lib/itsi/server/config/middleware/endpoint/delete.rb +42 -0
  300. data/lib/itsi/server/config/middleware/endpoint/endpoint.rb +99 -0
  301. data/lib/itsi/server/config/middleware/endpoint/get.md +12 -0
  302. data/lib/itsi/server/config/middleware/endpoint/get.rb +42 -0
  303. data/lib/itsi/server/config/middleware/endpoint/http_request.md +44 -0
  304. data/lib/itsi/server/config/middleware/endpoint/http_response.md +39 -0
  305. data/lib/itsi/server/config/middleware/endpoint/patch.md +12 -0
  306. data/lib/itsi/server/config/middleware/endpoint/patch.rb +42 -0
  307. data/lib/itsi/server/config/middleware/endpoint/post.md +12 -0
  308. data/lib/itsi/server/config/middleware/endpoint/post.rb +42 -0
  309. data/lib/itsi/server/config/middleware/endpoint/put.md +12 -0
  310. data/lib/itsi/server/config/middleware/endpoint/put.rb +42 -0
  311. data/lib/itsi/server/config/middleware/endpoint/schemas.md +122 -0
  312. data/lib/itsi/server/config/middleware/error_response.md +61 -0
  313. data/lib/itsi/server/config/middleware/error_response.rb +36 -0
  314. data/lib/itsi/server/config/middleware/etag.md +59 -0
  315. data/lib/itsi/server/config/middleware/etag.rb +27 -0
  316. data/lib/itsi/server/config/middleware/grpc.md +172 -0
  317. data/lib/itsi/server/config/middleware/grpc.rb +54 -0
  318. data/lib/itsi/server/config/middleware/intrusion_protection.md +124 -0
  319. data/lib/itsi/server/config/middleware/intrusion_protection.rb +61 -0
  320. data/lib/itsi/server/config/middleware/location.md +107 -0
  321. data/lib/itsi/server/config/middleware/location.rb +99 -0
  322. data/lib/itsi/server/config/middleware/log_requests.md +65 -0
  323. data/lib/itsi/server/config/middleware/log_requests.rb +31 -0
  324. data/lib/itsi/server/config/middleware/max_body.md +18 -0
  325. data/lib/itsi/server/config/middleware/max_body.rb +21 -0
  326. data/lib/itsi/server/config/middleware/proxy.md +62 -0
  327. data/lib/itsi/server/config/middleware/proxy.rb +41 -0
  328. data/lib/itsi/server/config/middleware/rackup_file.md +54 -0
  329. data/lib/itsi/server/config/middleware/rackup_file.rb +44 -0
  330. data/lib/itsi/server/config/middleware/rate_limit.md +126 -0
  331. data/lib/itsi/server/config/middleware/rate_limit.rb +34 -0
  332. data/lib/itsi/server/config/middleware/rate_limit_store.rb +25 -0
  333. data/lib/itsi/server/config/middleware/redirect.md +55 -0
  334. data/lib/itsi/server/config/middleware/redirect.rb +25 -0
  335. data/lib/itsi/server/config/middleware/request_headers.md +34 -0
  336. data/lib/itsi/server/config/middleware/request_headers.rb +24 -0
  337. data/lib/itsi/server/config/middleware/response_headers.md +33 -0
  338. data/lib/itsi/server/config/middleware/response_headers.rb +25 -0
  339. data/lib/itsi/server/config/middleware/run.md +60 -0
  340. data/lib/itsi/server/config/middleware/run.rb +43 -0
  341. data/lib/itsi/server/config/middleware/static_assets.md +73 -0
  342. data/lib/itsi/server/config/middleware/static_assets.rb +87 -0
  343. data/lib/itsi/server/config/middleware/static_response.md +44 -0
  344. data/lib/itsi/server/config/middleware/static_response.rb +29 -0
  345. data/lib/itsi/server/config/middleware/string_rewrite.md +67 -0
  346. data/lib/itsi/server/config/middleware/token_source.rb +32 -0
  347. data/lib/itsi/server/config/middleware.rb +13 -0
  348. data/lib/itsi/server/config/option.rb +14 -0
  349. data/lib/itsi/server/config/options/_index.md +37 -0
  350. data/lib/itsi/server/config/options/auto_reload_config.md +13 -0
  351. data/lib/itsi/server/config/options/auto_reload_config.rb +41 -0
  352. data/lib/itsi/server/config/options/bind.md +71 -0
  353. data/lib/itsi/server/config/options/bind.rb +26 -0
  354. data/lib/itsi/server/config/options/certificates.md +65 -0
  355. data/lib/itsi/server/config/options/daemonize.md +14 -0
  356. data/lib/itsi/server/config/options/daemonize.rb +19 -0
  357. data/lib/itsi/server/config/options/fiber_scheduler.md +34 -0
  358. data/lib/itsi/server/config/options/fiber_scheduler.rb +21 -0
  359. data/lib/itsi/server/config/options/header_read_timeout.md +17 -0
  360. data/lib/itsi/server/config/options/header_read_timeout.rb +19 -0
  361. data/lib/itsi/server/config/options/hooks/_index.md +11 -0
  362. data/lib/itsi/server/config/options/hooks/after_fork.md +13 -0
  363. data/lib/itsi/server/config/options/hooks/after_fork.rb +28 -0
  364. data/lib/itsi/server/config/options/hooks/after_memory_limit_reached.md +14 -0
  365. data/lib/itsi/server/config/options/hooks/after_memory_limit_reached.rb +28 -0
  366. data/lib/itsi/server/config/options/hooks/after_start.md +12 -0
  367. data/lib/itsi/server/config/options/hooks/after_start.rb +28 -0
  368. data/lib/itsi/server/config/options/hooks/before_fork.md +13 -0
  369. data/lib/itsi/server/config/options/hooks/before_fork.rb +28 -0
  370. data/lib/itsi/server/config/options/hooks/before_restart.md +12 -0
  371. data/lib/itsi/server/config/options/hooks/before_restart.rb +28 -0
  372. data/lib/itsi/server/config/options/hooks/before_shutdown.md +12 -0
  373. data/lib/itsi/server/config/options/hooks/before_shutdown.rb +28 -0
  374. data/lib/itsi/server/config/options/include.md +20 -0
  375. data/lib/itsi/server/config/options/include.rb +36 -0
  376. data/lib/itsi/server/config/options/listen_backlog.md +11 -0
  377. data/lib/itsi/server/config/options/listen_backlog.rb +19 -0
  378. data/lib/itsi/server/config/options/log_format.md +18 -0
  379. data/lib/itsi/server/config/options/log_format.rb +19 -0
  380. data/lib/itsi/server/config/options/log_level.md +34 -0
  381. data/lib/itsi/server/config/options/log_level.rb +20 -0
  382. data/lib/itsi/server/config/options/log_target.md +38 -0
  383. data/lib/itsi/server/config/options/log_target.rb +19 -0
  384. data/lib/itsi/server/config/options/log_target_filters.md +17 -0
  385. data/lib/itsi/server/config/options/log_target_filters.rb +19 -0
  386. data/lib/itsi/server/config/options/multithreaded_reactor.md +27 -0
  387. data/lib/itsi/server/config/options/multithreaded_reactor.rb +24 -0
  388. data/lib/itsi/server/config/options/nodelay.md +16 -0
  389. data/lib/itsi/server/config/options/nodelay.rb +19 -0
  390. data/lib/itsi/server/config/options/oob_gc_responses_threshold.md +19 -0
  391. data/lib/itsi/server/config/options/oob_gc_responses_threshold.rb +18 -0
  392. data/lib/itsi/server/config/options/pin_worker_cores.md +17 -0
  393. data/lib/itsi/server/config/options/pin_worker_cores.rb +19 -0
  394. data/lib/itsi/server/config/options/preload.md +21 -0
  395. data/lib/itsi/server/config/options/preload.rb +18 -0
  396. data/lib/itsi/server/config/options/recv_buffer_size.md +15 -0
  397. data/lib/itsi/server/config/options/recv_buffer_size.rb +19 -0
  398. data/lib/itsi/server/config/options/redirect_http_to_https.md +21 -0
  399. data/lib/itsi/server/config/options/redirect_http_to_https.rb +30 -0
  400. data/lib/itsi/server/config/options/request_timeout.md +23 -0
  401. data/lib/itsi/server/config/options/request_timeout.rb +19 -0
  402. data/lib/itsi/server/config/options/reuse_address.md +16 -0
  403. data/lib/itsi/server/config/options/reuse_address.rb +19 -0
  404. data/lib/itsi/server/config/options/reuse_port.md +16 -0
  405. data/lib/itsi/server/config/options/reuse_port.rb +19 -0
  406. data/lib/itsi/server/config/options/scheduler_threads.md +34 -0
  407. data/lib/itsi/server/config/options/scheduler_threads.rb +17 -0
  408. data/lib/itsi/server/config/options/shutdown_timeout.md +17 -0
  409. data/lib/itsi/server/config/options/shutdown_timeout.rb +19 -0
  410. data/lib/itsi/server/config/options/stream_body.md +32 -0
  411. data/lib/itsi/server/config/options/stream_body.rb +18 -0
  412. data/lib/itsi/server/config/options/threads.md +44 -0
  413. data/lib/itsi/server/config/options/threads.rb +17 -0
  414. data/lib/itsi/server/config/options/watch.md +16 -0
  415. data/lib/itsi/server/config/options/watch.rb +28 -0
  416. data/lib/itsi/server/config/options/worker_memory_limit.md +22 -0
  417. data/lib/itsi/server/config/options/worker_memory_limit.rb +18 -0
  418. data/lib/itsi/server/config/options/workers.md +42 -0
  419. data/lib/itsi/server/config/options/workers.rb +17 -0
  420. data/lib/itsi/server/config/typed_struct.rb +242 -0
  421. data/lib/itsi/server/config.rb +289 -0
  422. data/lib/itsi/server/default_app/default_app.rb +34 -0
  423. data/lib/itsi/server/default_app/index.html +115 -0
  424. data/lib/itsi/server/default_config/Itsi.rb +107 -0
  425. data/lib/itsi/server/grpc/grpc_call.rb +246 -0
  426. data/lib/itsi/server/grpc/grpc_interface.rb +107 -0
  427. data/lib/itsi/server/grpc/reflection/v1/reflection_pb.rb +26 -0
  428. data/lib/itsi/server/grpc/reflection/v1/reflection_services_pb.rb +122 -0
  429. data/lib/itsi/server/rack/handler/itsi.rb +27 -0
  430. data/lib/itsi/server/rack_interface.rb +94 -0
  431. data/lib/itsi/server/route_tester.rb +157 -0
  432. data/lib/itsi/server/scheduler_interface.rb +21 -0
  433. data/lib/itsi/server/scheduler_mode.rb +10 -0
  434. data/lib/itsi/server/signal_trap.rb +33 -0
  435. data/lib/itsi/server/typed_handlers/param_parser.rb +196 -0
  436. data/lib/itsi/server/typed_handlers/source_parser.rb +56 -0
  437. data/lib/itsi/server/typed_handlers.rb +25 -0
  438. data/lib/itsi/server/version.rb +1 -1
  439. data/lib/itsi/server.rb +265 -9
  440. data/lib/itsi/standard_headers.rb +86 -0
  441. data/lib/ruby_lsp/itsi/addon.rb +129 -0
  442. data/lib/shell_completions/completions.rb +26 -0
  443. metadata +454 -28
  444. data/CHANGELOG.md +0 -5
  445. data/CODE_OF_CONDUCT.md +0 -132
  446. data/LICENSE.txt +0 -21
  447. data/ext/itsi_server/src/request/itsi_request.rs +0 -143
  448. data/ext/itsi_server/src/request/mod.rs +0 -1
  449. data/ext/itsi_server/src/server/bind.rs +0 -138
  450. data/ext/itsi_server/src/server/itsi_ca/itsi_ca.crt +0 -32
  451. data/ext/itsi_server/src/server/itsi_ca/itsi_ca.key +0 -52
  452. data/ext/itsi_server/src/server/itsi_server.rs +0 -182
  453. data/ext/itsi_server/src/server/listener.rs +0 -218
  454. data/ext/itsi_server/src/server/tls.rs +0 -138
  455. data/ext/itsi_server/src/server/transfer_protocol.rs +0 -23
  456. data/ext/itsi_server/src/stream_writer/mod.rs +0 -21
  457. data/lib/itsi/request.rb +0 -39
@@ -0,0 +1,198 @@
1
+ use crate::{
2
+ server::http_message_types::{HttpRequest, HttpResponse},
3
+ services::itsi_http_service::HttpRequestContext,
4
+ };
5
+
6
+ use super::{FromValue, MiddlewareLayer};
7
+ use async_trait::async_trait;
8
+ use base64::{engine::general_purpose, Engine as _};
9
+ use bytes::{Bytes, BytesMut};
10
+ use either::Either;
11
+ use futures::TryStreamExt;
12
+ use http::{header, HeaderValue, Response, StatusCode};
13
+ use http_body_util::{combinators::BoxBody, BodyExt, Empty, Full};
14
+ use hyper::body::Body;
15
+ use magnus::error::Result;
16
+ use serde::Deserialize;
17
+ use sha2::{Digest, Sha256};
18
+ use tracing::debug;
19
+
20
+ #[derive(Debug, Clone, Copy, Deserialize, Default)]
21
+ pub enum ETagType {
22
+ #[serde(rename = "strong")]
23
+ #[default]
24
+ Strong,
25
+ #[serde(rename = "weak")]
26
+ Weak,
27
+ }
28
+
29
+ #[derive(Debug, Clone, Copy, Deserialize, Default)]
30
+ pub enum HashAlgorithm {
31
+ #[serde(rename = "sha256")]
32
+ #[default]
33
+ Sha256,
34
+ #[serde(rename = "md5")]
35
+ Md5,
36
+ }
37
+
38
+ #[derive(Debug, Clone, Deserialize)]
39
+ pub struct ETag {
40
+ #[serde(default)]
41
+ pub r#type: ETagType,
42
+ #[serde(default)]
43
+ pub algorithm: HashAlgorithm,
44
+ #[serde(default)]
45
+ pub min_body_size: usize,
46
+ #[serde(default = "default_true")]
47
+ pub handle_if_none_match: bool,
48
+ }
49
+
50
+ fn default_true() -> bool {
51
+ true
52
+ }
53
+
54
+ #[async_trait]
55
+ impl MiddlewareLayer for ETag {
56
+ async fn before(
57
+ &self,
58
+ req: HttpRequest,
59
+ context: &mut HttpRequestContext,
60
+ ) -> Result<Either<HttpRequest, HttpResponse>> {
61
+ // Store if-none-match header in context if present for later use in after hook
62
+ if self.handle_if_none_match {
63
+ if let Some(if_none_match) = req.headers().get(header::IF_NONE_MATCH) {
64
+ debug!(target: "middleware::etag", "Received If-None-Match header: {:?}", if_none_match);
65
+ if let Ok(etag_value) = if_none_match.to_str() {
66
+ context.set_if_none_match(Some(etag_value.to_string()));
67
+ }
68
+ }
69
+ }
70
+ Ok(Either::Left(req))
71
+ }
72
+
73
+ async fn after(&self, resp: HttpResponse, context: &mut HttpRequestContext) -> HttpResponse {
74
+ // Skip for error responses or responses that shouldn't have ETags
75
+ match resp.status() {
76
+ StatusCode::OK
77
+ | StatusCode::CREATED
78
+ | StatusCode::ACCEPTED
79
+ | StatusCode::NON_AUTHORITATIVE_INFORMATION
80
+ | StatusCode::NO_CONTENT
81
+ | StatusCode::PARTIAL_CONTENT => {}
82
+ _ => {
83
+ debug!(target: "middleware::etag", "Skipping ETag middleware for ineligible response");
84
+ return resp;
85
+ }
86
+ }
87
+
88
+ if resp.headers().contains_key(header::ETAG) {
89
+ debug!(target: "middleware::etag", "Forwarding response with existing ETag");
90
+ return resp;
91
+ }
92
+
93
+ if let Some(cache_control) = resp.headers().get(header::CACHE_CONTROL) {
94
+ if let Ok(cache_control_str) = cache_control.to_str() {
95
+ if cache_control_str.contains("no-store") {
96
+ debug!(target: "middleware::etag", "Skipping ETag for no-store response");
97
+ return resp;
98
+ }
99
+ }
100
+ }
101
+
102
+ let body_size = resp.size_hint().exact();
103
+
104
+ if body_size.is_none() {
105
+ debug!(target: "middleware::etag", "Skipping ETag for streaming response");
106
+ return resp;
107
+ }
108
+
109
+ if body_size.unwrap_or(0) < self.min_body_size as u64 {
110
+ debug!(target: "middleware::etag", "Skipping ETag for small response");
111
+ return resp;
112
+ }
113
+
114
+ let (mut parts, mut body) = resp.into_parts();
115
+ let etag_value = if let Some(existing_etag) = parts.headers.get(header::ETAG) {
116
+ existing_etag.to_str().unwrap_or("").to_string()
117
+ } else {
118
+ // Get the full bytes from the body
119
+ let full_bytes: Bytes = match body
120
+ .into_data_stream()
121
+ .try_fold(BytesMut::new(), |mut acc, chunk| async move {
122
+ acc.extend_from_slice(&chunk);
123
+ Ok(acc)
124
+ })
125
+ .await
126
+ {
127
+ Ok(bytes_mut) => bytes_mut.freeze(),
128
+ Err(_) => return Response::from_parts(parts, BoxBody::new(Empty::new())),
129
+ };
130
+
131
+ let computed_etag = match self.algorithm {
132
+ HashAlgorithm::Sha256 => {
133
+ let mut hasher = Sha256::new();
134
+ hasher.update(&full_bytes);
135
+ let result = hasher.finalize();
136
+ general_purpose::STANDARD.encode(result)
137
+ }
138
+ HashAlgorithm::Md5 => {
139
+ let digest = md5::compute(&full_bytes);
140
+ format!("{:x}", digest)
141
+ }
142
+ };
143
+
144
+ let formatted_etag = match self.r#type {
145
+ ETagType::Strong => format!("\"{}\"", computed_etag),
146
+ ETagType::Weak => format!("W/\"{}\"", computed_etag),
147
+ };
148
+
149
+ debug!(target: "middleware::etag", "Computed ETag for response {}", formatted_etag);
150
+ if let Ok(value) = HeaderValue::from_str(&formatted_etag) {
151
+ parts.headers.insert(header::ETAG, value);
152
+ }
153
+
154
+ body = Full::new(full_bytes).boxed();
155
+ formatted_etag
156
+ };
157
+
158
+ if self.handle_if_none_match {
159
+ if let Some(if_none_match) = context.get_if_none_match() {
160
+ if if_none_match == etag_value || if_none_match == "*" {
161
+ // Return 304 Not Modified without the body
162
+ let mut not_modified = Response::new(BoxBody::new(Empty::new()));
163
+ *not_modified.status_mut() = StatusCode::NOT_MODIFIED;
164
+ // Copy headers we want to preserve
165
+ for (name, value) in parts.headers.iter() {
166
+ if matches!(
167
+ name,
168
+ &header::CACHE_CONTROL
169
+ | &header::CONTENT_LOCATION
170
+ | &header::DATE
171
+ | &header::ETAG
172
+ | &header::EXPIRES
173
+ | &header::VARY
174
+ ) {
175
+ not_modified.headers_mut().insert(name, value.clone());
176
+ }
177
+ }
178
+ return not_modified;
179
+ }
180
+ }
181
+ }
182
+
183
+ Response::from_parts(parts, body)
184
+ }
185
+ }
186
+
187
+ impl Default for ETag {
188
+ fn default() -> Self {
189
+ Self {
190
+ r#type: ETagType::Strong,
191
+ algorithm: HashAlgorithm::Sha256,
192
+ min_body_size: 0,
193
+ handle_if_none_match: true,
194
+ }
195
+ }
196
+ }
197
+
198
+ impl FromValue for ETag {}
@@ -0,0 +1,82 @@
1
+ use http::{header::GetAll, HeaderValue};
2
+
3
+ /// Given a list of header values (which may be comma-separated and may have quality parameters)
4
+ /// and a list of supported items (each supported item is a full value or a prefix ending with '*'),
5
+ /// return Some(supported_item) for the first supported item that matches any header value, or None.
6
+ pub fn find_first_supported<'a, I>(header_values: &[HeaderValue], supported: I) -> Option<&'a str>
7
+ where
8
+ I: IntoIterator<Item = &'a str> + Clone,
9
+ {
10
+ // best candidate: (quality, supported_index, candidate)
11
+ let mut best: Option<(f32, usize, &'a str)> = None;
12
+
13
+ for value in header_values.iter() {
14
+ if let Ok(s) = value.to_str() {
15
+ for token in s.split(',') {
16
+ let token = token.trim();
17
+ if token.is_empty() {
18
+ continue;
19
+ }
20
+ let mut parts = token.split(';');
21
+ let enc = parts.next()?.trim();
22
+ if enc.is_empty() {
23
+ continue;
24
+ }
25
+ let quality = parts
26
+ .find_map(|p| {
27
+ let p = p.trim();
28
+ if let Some(q_str) = p.strip_prefix("q=") {
29
+ q_str.parse::<f32>().ok()
30
+ } else {
31
+ None
32
+ }
33
+ })
34
+ .unwrap_or(1.0);
35
+
36
+ // For each supported encoding, iterate over a clone of the iterable.
37
+ for (i, supp) in supported.clone().into_iter().enumerate() {
38
+ let is_match = if supp == "*" {
39
+ true
40
+ } else if let Some(prefix) = supp.strip_suffix('*') {
41
+ enc.starts_with(prefix)
42
+ } else {
43
+ enc.eq_ignore_ascii_case(supp)
44
+ };
45
+
46
+ if is_match {
47
+ best = match best {
48
+ Some((best_q, best_idx, _))
49
+ if quality > best_q || (quality == best_q && i < best_idx) =>
50
+ {
51
+ Some((quality, i, supp))
52
+ }
53
+ None => Some((quality, i, supp)),
54
+ _ => best,
55
+ };
56
+ }
57
+ }
58
+ }
59
+ }
60
+ }
61
+
62
+ best.map(|(_, _, candidate)| candidate)
63
+ }
64
+
65
+ pub fn header_contains(header_values: &GetAll<HeaderValue>, needle: &str) -> bool {
66
+ if needle == "*" {
67
+ return true;
68
+ }
69
+ let mut headers = header_values
70
+ .iter()
71
+ .flat_map(|value| value.to_str().unwrap_or("").split(','))
72
+ .map(|s| s.trim().split(';').next().unwrap_or(""))
73
+ .filter(|s| !s.is_empty());
74
+
75
+ let needle_lower = needle;
76
+ if needle.ends_with('*') {
77
+ let prefix = &needle_lower[..needle_lower.len() - 1];
78
+ headers.any(|h| h.starts_with(prefix))
79
+ } else {
80
+ headers.any(|h| h == needle_lower)
81
+ }
82
+ }
@@ -0,0 +1,209 @@
1
+ use crate::server::http_message_types::{HttpRequest, HttpResponse, RequestExt};
2
+ use crate::services::itsi_http_service::HttpRequestContext;
3
+ use crate::services::rate_limiter::{
4
+ get_ban_manager, get_rate_limiter, BanManager, RateLimiter, RateLimiterConfig,
5
+ };
6
+
7
+ use super::token_source::TokenSource;
8
+ use super::{ErrorResponse, FromValue, MiddlewareLayer};
9
+
10
+ use async_trait::async_trait;
11
+ use either::Either;
12
+ use itsi_tracing::*;
13
+ use magnus::error::Result;
14
+ use regex::RegexSet;
15
+ use serde::Deserialize;
16
+ use std::time::Duration;
17
+ use std::{
18
+ collections::HashMap,
19
+ sync::{Arc, OnceLock},
20
+ };
21
+
22
+ #[derive(Debug, Clone, Deserialize)]
23
+ pub struct IntrusionProtection {
24
+ #[serde(skip_deserializing)]
25
+ pub banned_url_pattern_matcher: OnceLock<RegexSet>,
26
+ #[serde(default)]
27
+ pub banned_url_patterns: Vec<String>,
28
+ #[serde(skip_deserializing)]
29
+ pub banned_header_pattern_matchers: OnceLock<HashMap<String, RegexSet>>,
30
+ #[serde(default)]
31
+ pub banned_header_patterns: HashMap<String, Vec<String>>,
32
+ pub banned_time_seconds: f64,
33
+ #[serde(skip_deserializing)]
34
+ pub rate_limiter: OnceLock<Arc<dyn RateLimiter>>,
35
+ #[serde(skip_deserializing)]
36
+ pub ban_manager: OnceLock<BanManager>,
37
+ pub store_config: RateLimiterConfig,
38
+ pub trusted_proxies: HashMap<String, TokenSource>,
39
+ #[serde(default = "forbidden_error_response")]
40
+ pub error_response: ErrorResponse,
41
+ }
42
+
43
+ fn forbidden_error_response() -> ErrorResponse {
44
+ ErrorResponse::forbidden()
45
+ }
46
+
47
+ #[async_trait]
48
+ impl MiddlewareLayer for IntrusionProtection {
49
+ async fn initialize(&self) -> Result<()> {
50
+ // Initialize regex matchers for URL patterns
51
+ if !self.banned_url_patterns.is_empty() {
52
+ match RegexSet::new(&self.banned_url_patterns) {
53
+ Ok(regex_set) => {
54
+ debug!(target: "middleware::intrusion_protection", "Compiled URL regex patterns: {} items.", regex_set.len());
55
+ let _ = self.banned_url_pattern_matcher.set(regex_set);
56
+ }
57
+ Err(e) => {
58
+ error!("Failed to compile URL regex patterns: {:?}", e);
59
+ }
60
+ }
61
+ }
62
+
63
+ // Initialize regex matchers for header patterns
64
+ if !self.banned_header_patterns.is_empty() {
65
+ let mut header_matchers = HashMap::new();
66
+ for (header_name, patterns) in &self.banned_header_patterns {
67
+ if !patterns.is_empty() {
68
+ match RegexSet::new(patterns) {
69
+ Ok(regex_set) => {
70
+ debug!(target: "middleware::intrusion_protection", "Compiled header regex patterns for {}: {} items.", header_name, regex_set.len());
71
+ header_matchers.insert(header_name.clone(), regex_set);
72
+ }
73
+ Err(e) => {
74
+ error!(
75
+ "Failed to compile header regex patterns for {}: {:?}",
76
+ header_name, e
77
+ );
78
+ }
79
+ }
80
+ }
81
+ }
82
+ let _ = self.banned_header_pattern_matchers.set(header_matchers);
83
+ }
84
+
85
+ // Initialize rate limiter (used for tracking bans)
86
+ // This will automatically fall back to in-memory if Redis fails
87
+ if let Ok(limiter) = get_rate_limiter(&self.store_config).await {
88
+ debug!(target: "middleware::intrusion_protection", "Initialized rate limiter.");
89
+ let _ = self.rate_limiter.set(limiter);
90
+ }
91
+
92
+ // Initialize ban manager
93
+ // This will automatically fall back to in-memory if Redis fails
94
+ if let Ok(manager) = get_ban_manager(&self.store_config).await {
95
+ debug!(target: "middleware::intrusion_protection", "Initialized ban manager.");
96
+ let _ = self.ban_manager.set(manager);
97
+ }
98
+
99
+ Ok(())
100
+ }
101
+
102
+ async fn before(
103
+ &self,
104
+ req: HttpRequest,
105
+ context: &mut HttpRequestContext,
106
+ ) -> Result<Either<HttpRequest, HttpResponse>> {
107
+ // Get client IP address from context's service
108
+ let client_ip = if self.trusted_proxies.contains_key(&context.addr) {
109
+ let source = self.trusted_proxies.get(&context.addr).unwrap();
110
+ source.extract_token(&req).unwrap_or(&context.addr)
111
+ } else {
112
+ &context.addr
113
+ };
114
+
115
+ // Check if the IP is already banned
116
+ if let Some(ban_manager) = self.ban_manager.get() {
117
+ match ban_manager.is_banned(client_ip).await {
118
+ Ok(Some(_)) => {
119
+ debug!(target: "middleware::intrusion_protection", "IP {} is banned.", client_ip);
120
+ return Ok(Either::Right(
121
+ self.error_response
122
+ .to_http_response(req.accept().into())
123
+ .await,
124
+ ));
125
+ }
126
+ Err(e) => {
127
+ error!("Error checking IP ban status: {:?}", e);
128
+ // Continue processing - fail open
129
+ }
130
+ _ => {}
131
+ }
132
+ } else {
133
+ warn!("No ban manager available for intrusion protection");
134
+ }
135
+
136
+ // Check for banned URL patterns
137
+ if let Some(url_matcher) = self.banned_url_pattern_matcher.get() {
138
+ let path = req.uri().path_and_query().map(|p| p.as_str()).unwrap_or("");
139
+
140
+ if url_matcher.is_match(path) {
141
+ debug!(target: "middleware::intrusion_protection", "Banned URL pattern detected: {}", path);
142
+ if let Some(ban_manager) = self.ban_manager.get() {
143
+ match ban_manager
144
+ .ban_ip(
145
+ client_ip,
146
+ &format!("Banned URL pattern detected: {}", path),
147
+ Duration::from_secs_f64(self.banned_time_seconds),
148
+ )
149
+ .await
150
+ {
151
+ Ok(_) => {}
152
+ Err(e) => error!("Failed to ban IP {}: {:?}", client_ip, e),
153
+ }
154
+ }
155
+
156
+ // Always return the error response even if banning failed
157
+ return Ok(Either::Right(
158
+ self.error_response
159
+ .to_http_response(req.accept().into())
160
+ .await,
161
+ ));
162
+ }
163
+ }
164
+
165
+ // Check for banned header patterns
166
+ if let Some(header_matchers) = self.banned_header_pattern_matchers.get() {
167
+ for (header_name, pattern_set) in header_matchers {
168
+ if let Some(header_value) = req.header(header_name) {
169
+ if pattern_set.is_match(header_value) {
170
+ debug!(target: "middleware::intrusion_protection", "Banned header pattern detected: {} in {}", header_value, header_name);
171
+
172
+ // Ban the IP address if possible
173
+ if let Some(ban_manager) = self.ban_manager.get() {
174
+ match ban_manager
175
+ .ban_ip(
176
+ client_ip,
177
+ &format!(
178
+ "Banned header pattern detected: {} in {}",
179
+ header_value, header_name
180
+ ),
181
+ Duration::from_secs_f64(self.banned_time_seconds),
182
+ )
183
+ .await
184
+ {
185
+ Ok(_) => info!(
186
+ "Successfully banned IP {} for {} seconds",
187
+ client_ip, self.banned_time_seconds
188
+ ),
189
+ Err(e) => error!("Failed to ban IP {}: {:?}", client_ip, e),
190
+ }
191
+ }
192
+
193
+ // Always return the error response even if banning failed
194
+ return Ok(Either::Right(
195
+ self.error_response
196
+ .to_http_response(req.accept().into())
197
+ .await,
198
+ ));
199
+ }
200
+ }
201
+ }
202
+ }
203
+
204
+ // No intrusion detected
205
+ Ok(Either::Left(req))
206
+ }
207
+ }
208
+
209
+ impl FromValue for IntrusionProtection {}
@@ -0,0 +1,82 @@
1
+ use async_trait::async_trait;
2
+ use either::Either;
3
+ use itsi_tracing::*;
4
+ use magnus::error::Result;
5
+ use serde::Deserialize;
6
+
7
+ use crate::server::http_message_types::{HttpRequest, HttpResponse};
8
+ use crate::services::itsi_http_service::HttpRequestContext;
9
+
10
+ use super::string_rewrite::StringRewrite;
11
+ use super::{FromValue, MiddlewareLayer};
12
+
13
+ /// Logging middleware for HTTP requests and responses
14
+ ///
15
+ /// Supports customizable log formats with placeholders
16
+ #[derive(Debug, Clone, Deserialize)]
17
+ pub struct LogRequests {
18
+ pub before: Option<LogConfig>,
19
+ pub after: Option<LogConfig>,
20
+ }
21
+
22
+ #[derive(Debug, Clone, Deserialize)]
23
+ pub struct LogConfig {
24
+ level: LogMiddlewareLevel,
25
+ format: StringRewrite,
26
+ }
27
+
28
+ #[derive(Debug, Clone, Deserialize)]
29
+ pub enum LogMiddlewareLevel {
30
+ #[serde(rename(deserialize = "INFO"))]
31
+ Info,
32
+ #[serde(rename(deserialize = "TRACE"))]
33
+ Trace,
34
+ #[serde(rename(deserialize = "DEBUG"))]
35
+ Debug,
36
+ #[serde(rename(deserialize = "WARN"))]
37
+ Warn,
38
+ #[serde(rename(deserialize = "ERROR"))]
39
+ Error,
40
+ }
41
+
42
+ impl LogMiddlewareLevel {
43
+ pub fn log(&self, message: String) {
44
+ match self {
45
+ LogMiddlewareLevel::Trace => trace!(target: "middleware::log_requests", message),
46
+ LogMiddlewareLevel::Debug => debug!(target: "middleware::log_requests", message),
47
+ LogMiddlewareLevel::Info => info!(target: "middleware::log_requests", message),
48
+ LogMiddlewareLevel::Warn => warn!(target: "middleware::log_requests", message),
49
+ LogMiddlewareLevel::Error => error!(target: "middleware::log_requests", message),
50
+ }
51
+ }
52
+ }
53
+
54
+ #[async_trait]
55
+ impl MiddlewareLayer for LogRequests {
56
+ async fn initialize(&self) -> Result<()> {
57
+ Ok(())
58
+ }
59
+
60
+ async fn before(
61
+ &self,
62
+ req: HttpRequest,
63
+ context: &mut HttpRequestContext,
64
+ ) -> Result<Either<HttpRequest, HttpResponse>> {
65
+ context.track_start_time();
66
+ if let Some(LogConfig { level, format }) = self.before.as_ref() {
67
+ level.log(format.rewrite_request(&req, context));
68
+ }
69
+
70
+ Ok(Either::Left(req))
71
+ }
72
+
73
+ async fn after(&self, resp: HttpResponse, context: &mut HttpRequestContext) -> HttpResponse {
74
+ if let Some(LogConfig { level, format }) = self.after.as_ref() {
75
+ level.log(format.rewrite_response(&resp, context));
76
+ }
77
+
78
+ resp
79
+ }
80
+ }
81
+
82
+ impl FromValue for LogRequests {}
@@ -0,0 +1,47 @@
1
+ use crate::{
2
+ server::http_message_types::{HttpRequest, HttpResponse, RequestExt},
3
+ services::itsi_http_service::HttpRequestContext,
4
+ };
5
+
6
+ use super::{ErrorResponse, FromValue, MiddlewareLayer};
7
+ use async_trait::async_trait;
8
+ use either::Either;
9
+ use http::StatusCode;
10
+ use magnus::error::Result;
11
+ use serde::Deserialize;
12
+ use std::sync::atomic::Ordering;
13
+
14
+ #[derive(Debug, Clone, Deserialize)]
15
+ pub struct MaxBody {
16
+ pub limit_bytes: usize,
17
+ #[serde(default = "payload_too_large_error_response")]
18
+ pub error_response: ErrorResponse,
19
+ }
20
+
21
+ fn payload_too_large_error_response() -> ErrorResponse {
22
+ ErrorResponse::payload_too_large()
23
+ }
24
+
25
+ #[async_trait]
26
+ impl MiddlewareLayer for MaxBody {
27
+ async fn before(
28
+ &self,
29
+ req: HttpRequest,
30
+ context: &mut HttpRequestContext,
31
+ ) -> Result<Either<HttpRequest, HttpResponse>> {
32
+ req.body().limit.store(self.limit_bytes, Ordering::Relaxed);
33
+ context.set_response_format(req.accept().into());
34
+ Ok(Either::Left(req))
35
+ }
36
+
37
+ async fn after(&self, resp: HttpResponse, context: &mut HttpRequestContext) -> HttpResponse {
38
+ if resp.status() == StatusCode::PAYLOAD_TOO_LARGE {
39
+ self.error_response
40
+ .to_http_response(context.response_format().clone())
41
+ .await
42
+ } else {
43
+ resp
44
+ }
45
+ }
46
+ }
47
+ impl FromValue for MaxBody {}