itsi 0.1.9 → 0.1.12

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 (294) hide show
  1. checksums.yaml +4 -4
  2. data/Cargo.lock +1542 -43
  3. data/Itsi.rb +125 -0
  4. data/Rakefile +8 -4
  5. data/crates/itsi_error/src/lib.rs +9 -0
  6. data/crates/itsi_rb_helpers/Cargo.toml +1 -0
  7. data/crates/itsi_rb_helpers/src/heap_value.rs +18 -0
  8. data/crates/itsi_rb_helpers/src/lib.rs +59 -9
  9. data/crates/itsi_server/Cargo.toml +70 -28
  10. data/crates/itsi_server/src/lib.rs +80 -80
  11. data/crates/itsi_server/src/{body_proxy → ruby_types/itsi_body_proxy}/big_bytes.rs +10 -5
  12. data/{gems/server/ext/itsi_server/src/body_proxy/itsi_body_proxy.rs → crates/itsi_server/src/ruby_types/itsi_body_proxy/mod.rs} +22 -3
  13. data/crates/itsi_server/src/ruby_types/itsi_grpc_request.rs +147 -0
  14. data/crates/itsi_server/src/ruby_types/itsi_grpc_response.rs +19 -0
  15. data/crates/itsi_server/src/ruby_types/itsi_grpc_stream/mod.rs +216 -0
  16. data/crates/itsi_server/src/ruby_types/itsi_http_request.rs +282 -0
  17. data/crates/itsi_server/src/{response/itsi_response.rs → ruby_types/itsi_http_response.rs} +72 -41
  18. data/crates/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +225 -0
  19. data/crates/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +355 -0
  20. data/crates/itsi_server/src/ruby_types/itsi_server.rs +82 -0
  21. data/crates/itsi_server/src/ruby_types/mod.rs +55 -0
  22. data/crates/itsi_server/src/server/bind.rs +29 -17
  23. data/crates/itsi_server/src/server/byte_frame.rs +32 -0
  24. data/crates/itsi_server/src/server/cache_store.rs +74 -0
  25. data/crates/itsi_server/src/server/itsi_service.rs +172 -0
  26. data/crates/itsi_server/src/server/lifecycle_event.rs +3 -0
  27. data/crates/itsi_server/src/server/listener.rs +111 -11
  28. data/crates/itsi_server/src/server/middleware_stack/middleware.rs +153 -0
  29. data/crates/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +47 -0
  30. data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +58 -0
  31. data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +82 -0
  32. data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +321 -0
  33. data/crates/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +139 -0
  34. data/crates/itsi_server/src/server/middleware_stack/middlewares/compression.rs +300 -0
  35. data/crates/itsi_server/src/server/middleware_stack/middlewares/cors.rs +287 -0
  36. data/crates/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +48 -0
  37. data/crates/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +127 -0
  38. data/crates/itsi_server/src/server/middleware_stack/middlewares/etag.rs +191 -0
  39. data/crates/itsi_server/src/server/middleware_stack/middlewares/grpc_service.rs +72 -0
  40. data/crates/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +85 -0
  41. data/crates/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +195 -0
  42. data/crates/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +82 -0
  43. data/crates/itsi_server/src/server/middleware_stack/middlewares/mod.rs +82 -0
  44. data/crates/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +216 -0
  45. data/crates/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +124 -0
  46. data/crates/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +76 -0
  47. data/crates/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +43 -0
  48. data/crates/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +34 -0
  49. data/crates/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +93 -0
  50. data/crates/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +162 -0
  51. data/crates/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +158 -0
  52. data/crates/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +12 -0
  53. data/crates/itsi_server/src/server/middleware_stack/mod.rs +315 -0
  54. data/crates/itsi_server/src/server/mod.rs +8 -1
  55. data/crates/itsi_server/src/server/process_worker.rs +44 -11
  56. data/crates/itsi_server/src/server/rate_limiter.rs +565 -0
  57. data/crates/itsi_server/src/server/request_job.rs +11 -0
  58. data/crates/itsi_server/src/server/serve_strategy/cluster_mode.rs +129 -46
  59. data/crates/itsi_server/src/server/serve_strategy/mod.rs +9 -6
  60. data/crates/itsi_server/src/server/serve_strategy/single_mode.rs +325 -167
  61. data/crates/itsi_server/src/server/signal.rs +20 -4
  62. data/crates/itsi_server/src/server/static_file_server.rs +984 -0
  63. data/crates/itsi_server/src/server/thread_worker.rs +165 -88
  64. data/crates/itsi_server/src/server/tls.rs +1 -1
  65. data/crates/itsi_server/src/server/types.rs +43 -0
  66. data/crates/itsi_server/test.md +14 -0
  67. data/crates/itsi_tracing/Cargo.toml +1 -0
  68. data/crates/itsi_tracing/src/lib.rs +216 -45
  69. data/docs/.gitignore +7 -0
  70. data/docs/.gitpod.yml +15 -0
  71. data/docs/Itsi.rb +17 -0
  72. data/docs/content/_index.md +17 -0
  73. data/docs/content/about.md +6 -0
  74. data/docs/content/docs/_index.md +18 -0
  75. data/docs/content/docs/first-page.md +9 -0
  76. data/docs/content/docs/folder/_index.md +10 -0
  77. data/docs/content/docs/folder/leaf.md +7 -0
  78. data/docs/go.mod +5 -0
  79. data/docs/go.sum +2 -0
  80. data/docs/hugo.yaml +77 -0
  81. data/examples/static_assets_example.rb +83 -0
  82. data/gems/_index.md +18 -0
  83. data/gems/scheduler/CODE_OF_CONDUCT.md +7 -0
  84. data/gems/scheduler/Cargo.lock +75 -14
  85. data/gems/scheduler/README.md +5 -0
  86. data/gems/scheduler/_index.md +7 -0
  87. data/gems/scheduler/itsi-scheduler.gemspec +4 -1
  88. data/gems/scheduler/lib/itsi/scheduler/version.rb +1 -1
  89. data/gems/scheduler/lib/itsi/scheduler.rb +2 -2
  90. data/gems/scheduler/test/test_file_io.rb +0 -1
  91. data/gems/scheduler/test/test_itsi_scheduler.rb +1 -1
  92. data/gems/server/CHANGELOG.md +5 -0
  93. data/gems/server/CODE_OF_CONDUCT.md +7 -0
  94. data/gems/server/Cargo.lock +1543 -43
  95. data/gems/server/README.md +4 -0
  96. data/gems/server/_index.md +6 -0
  97. data/gems/server/exe/itsi +46 -57
  98. data/gems/server/itsi-server.gemspec +3 -2
  99. data/gems/server/lib/itsi/{request.rb → http_request.rb} +29 -5
  100. data/gems/server/lib/itsi/http_response.rb +39 -0
  101. data/gems/server/lib/itsi/server/Itsi.rb +119 -0
  102. data/gems/server/lib/itsi/server/config/dsl.rb +506 -0
  103. data/gems/server/lib/itsi/server/config.rb +131 -0
  104. data/gems/server/lib/itsi/server/default_app/default_app.rb +38 -0
  105. data/gems/server/lib/itsi/server/grpc_interface.rb +213 -0
  106. data/gems/server/lib/itsi/server/rack/handler/itsi.rb +9 -6
  107. data/gems/server/lib/itsi/server/rack_interface.rb +24 -9
  108. data/gems/server/lib/itsi/server/scheduler_interface.rb +1 -1
  109. data/gems/server/lib/itsi/server/scheduler_mode.rb +4 -0
  110. data/gems/server/lib/itsi/server/signal_trap.rb +6 -1
  111. data/gems/server/lib/itsi/server/version.rb +1 -1
  112. data/gems/server/lib/itsi/server.rb +75 -60
  113. data/gems/server/lib/itsi/standard_headers.rb +86 -0
  114. data/gems/server/test/helpers/test_helper.rb +14 -12
  115. data/gems/server/test/test_itsi_server.rb +21 -2
  116. data/lib/itsi/version.rb +1 -1
  117. data/sandbox/itsi_file/Gemfile +11 -0
  118. data/sandbox/itsi_file/Gemfile.lock +69 -0
  119. data/sandbox/itsi_file/Itsi.rb +276 -0
  120. data/sandbox/itsi_file/error.html +2 -0
  121. data/sandbox/itsi_file/organisations_controller.rb +20 -0
  122. data/sandbox/itsi_file/public/assets/image.png +0 -0
  123. data/sandbox/itsi_file/public/assets/index.html +1 -0
  124. data/sandbox/itsi_sandbox_async/Gemfile +1 -1
  125. data/sandbox/itsi_sandbox_hanami/Gemfile.lock +2 -2
  126. data/sandbox/itsi_sandbox_rack/Gemfile.lock +2 -2
  127. data/sandbox/itsi_sandbox_rack/config.ru +2 -15
  128. data/sandbox/itsi_sandbox_rails/.dockerignore +2 -5
  129. data/sandbox/itsi_sandbox_rails/.github/workflows/ci.yml +1 -1
  130. data/sandbox/itsi_sandbox_rails/.gitignore +2 -1
  131. data/sandbox/itsi_sandbox_rails/Dockerfile +6 -9
  132. data/sandbox/itsi_sandbox_rails/Gemfile +16 -22
  133. data/sandbox/itsi_sandbox_rails/Gemfile.lock +100 -225
  134. data/sandbox/itsi_sandbox_rails/app/assets/config/manifest.js +4 -0
  135. data/sandbox/itsi_sandbox_rails/app/assets/stylesheets/application.css +11 -6
  136. data/sandbox/itsi_sandbox_rails/app/channels/application_cable/channel.rb +4 -0
  137. data/sandbox/itsi_sandbox_rails/app/channels/application_cable/connection.rb +4 -0
  138. data/sandbox/itsi_sandbox_rails/app/controllers/live_controller.rb +7 -8
  139. data/sandbox/itsi_sandbox_rails/app/controllers/uploads_controller.rb +0 -3
  140. data/sandbox/itsi_sandbox_rails/app/views/layouts/application.html.erb +2 -7
  141. data/sandbox/itsi_sandbox_rails/bin/docker-entrypoint +3 -4
  142. data/sandbox/itsi_sandbox_rails/bin/setup +8 -5
  143. data/sandbox/itsi_sandbox_rails/config/application.rb +1 -35
  144. data/sandbox/itsi_sandbox_rails/config/cable.yml +3 -10
  145. data/sandbox/itsi_sandbox_rails/config/credentials.yml.enc +1 -1
  146. data/sandbox/itsi_sandbox_rails/config/database.yml +9 -19
  147. data/sandbox/itsi_sandbox_rails/config/environment.rb +1 -1
  148. data/sandbox/itsi_sandbox_rails/config/environments/development.rb +21 -12
  149. data/sandbox/itsi_sandbox_rails/config/environments/production.rb +49 -34
  150. data/sandbox/itsi_sandbox_rails/config/environments/test.rb +19 -5
  151. data/sandbox/itsi_sandbox_rails/config/initializers/assets.rb +5 -0
  152. data/sandbox/itsi_sandbox_rails/config/initializers/filter_parameter_logging.rb +1 -1
  153. data/sandbox/itsi_sandbox_rails/config/initializers/permissions_policy.rb +13 -0
  154. data/sandbox/itsi_sandbox_rails/config/puma.rb +2 -9
  155. data/sandbox/itsi_sandbox_rails/config.ru +0 -1
  156. data/sandbox/itsi_sandbox_rails/db/migrate/20250301041554_create_posts.rb +1 -1
  157. data/sandbox/itsi_sandbox_rails/db/schema.rb +2 -2
  158. data/sandbox/itsi_sandbox_rails/lib/assets/.keep +0 -0
  159. data/sandbox/itsi_sandbox_rails/public/404.html +66 -113
  160. data/sandbox/itsi_sandbox_rails/public/406-unsupported-browser.html +65 -113
  161. data/sandbox/itsi_sandbox_rails/public/422.html +66 -113
  162. data/sandbox/itsi_sandbox_rails/public/500.html +65 -113
  163. data/sandbox/itsi_sandbox_rails/public/icon.png +0 -0
  164. data/sandbox/itsi_sandbox_rails/public/icon.svg +2 -2
  165. data/sandbox/itsi_sandbox_rails/test/channels/application_cable/connection_test.rb +13 -0
  166. data/sandbox/itsi_sandbox_roda/Gemfile.lock +3 -10
  167. data/tasks.txt +72 -12
  168. metadata +94 -139
  169. data/crates/itsi_server/src/body_proxy/itsi_body_proxy.rs +0 -122
  170. data/crates/itsi_server/src/body_proxy/mod.rs +0 -2
  171. data/crates/itsi_server/src/request/itsi_request.rs +0 -305
  172. data/crates/itsi_server/src/request/mod.rs +0 -1
  173. data/crates/itsi_server/src/response/mod.rs +0 -1
  174. data/crates/itsi_server/src/server/itsi_server.rs +0 -294
  175. data/gems/scheduler/ext/itsi_error/Cargo.lock +0 -368
  176. data/gems/scheduler/ext/itsi_error/Cargo.toml +0 -11
  177. data/gems/scheduler/ext/itsi_error/src/from.rs +0 -68
  178. data/gems/scheduler/ext/itsi_error/src/lib.rs +0 -24
  179. data/gems/scheduler/ext/itsi_instrument_entry/Cargo.toml +0 -15
  180. data/gems/scheduler/ext/itsi_instrument_entry/src/lib.rs +0 -31
  181. data/gems/scheduler/ext/itsi_rb_helpers/Cargo.lock +0 -355
  182. data/gems/scheduler/ext/itsi_rb_helpers/Cargo.toml +0 -10
  183. data/gems/scheduler/ext/itsi_rb_helpers/src/heap_value.rs +0 -121
  184. data/gems/scheduler/ext/itsi_rb_helpers/src/lib.rs +0 -178
  185. data/gems/scheduler/ext/itsi_scheduler/Cargo.toml +0 -24
  186. data/gems/scheduler/ext/itsi_scheduler/extconf.rb +0 -6
  187. data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +0 -56
  188. data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +0 -44
  189. data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +0 -44
  190. data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler.rs +0 -308
  191. data/gems/scheduler/ext/itsi_scheduler/src/lib.rs +0 -38
  192. data/gems/scheduler/ext/itsi_server/Cargo.lock +0 -2956
  193. data/gems/scheduler/ext/itsi_server/Cargo.toml +0 -47
  194. data/gems/scheduler/ext/itsi_server/extconf.rb +0 -6
  195. data/gems/scheduler/ext/itsi_server/src/body_proxy/big_bytes.rs +0 -104
  196. data/gems/scheduler/ext/itsi_server/src/body_proxy/itsi_body_proxy.rs +0 -122
  197. data/gems/scheduler/ext/itsi_server/src/body_proxy/mod.rs +0 -2
  198. data/gems/scheduler/ext/itsi_server/src/env.rs +0 -43
  199. data/gems/scheduler/ext/itsi_server/src/lib.rs +0 -112
  200. data/gems/scheduler/ext/itsi_server/src/request/itsi_request.rs +0 -305
  201. data/gems/scheduler/ext/itsi_server/src/request/mod.rs +0 -1
  202. data/gems/scheduler/ext/itsi_server/src/response/itsi_response.rs +0 -357
  203. data/gems/scheduler/ext/itsi_server/src/response/mod.rs +0 -1
  204. data/gems/scheduler/ext/itsi_server/src/server/bind.rs +0 -170
  205. data/gems/scheduler/ext/itsi_server/src/server/bind_protocol.rs +0 -37
  206. data/gems/scheduler/ext/itsi_server/src/server/io_stream.rs +0 -104
  207. data/gems/scheduler/ext/itsi_server/src/server/itsi_server.rs +0 -294
  208. data/gems/scheduler/ext/itsi_server/src/server/lifecycle_event.rs +0 -9
  209. data/gems/scheduler/ext/itsi_server/src/server/listener.rs +0 -318
  210. data/gems/scheduler/ext/itsi_server/src/server/mod.rs +0 -11
  211. data/gems/scheduler/ext/itsi_server/src/server/process_worker.rs +0 -196
  212. data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +0 -254
  213. data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/mod.rs +0 -27
  214. data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/single_mode.rs +0 -263
  215. data/gems/scheduler/ext/itsi_server/src/server/signal.rs +0 -77
  216. data/gems/scheduler/ext/itsi_server/src/server/thread_worker.rs +0 -367
  217. data/gems/scheduler/ext/itsi_server/src/server/tls/locked_dir_cache.rs +0 -132
  218. data/gems/scheduler/ext/itsi_server/src/server/tls.rs +0 -265
  219. data/gems/scheduler/ext/itsi_tracing/Cargo.lock +0 -274
  220. data/gems/scheduler/ext/itsi_tracing/Cargo.toml +0 -16
  221. data/gems/scheduler/ext/itsi_tracing/src/lib.rs +0 -58
  222. data/gems/server/ext/itsi_error/Cargo.lock +0 -368
  223. data/gems/server/ext/itsi_error/Cargo.toml +0 -11
  224. data/gems/server/ext/itsi_error/src/from.rs +0 -68
  225. data/gems/server/ext/itsi_error/src/lib.rs +0 -24
  226. data/gems/server/ext/itsi_instrument_entry/Cargo.toml +0 -15
  227. data/gems/server/ext/itsi_instrument_entry/src/lib.rs +0 -31
  228. data/gems/server/ext/itsi_rb_helpers/Cargo.lock +0 -355
  229. data/gems/server/ext/itsi_rb_helpers/Cargo.toml +0 -10
  230. data/gems/server/ext/itsi_rb_helpers/src/heap_value.rs +0 -121
  231. data/gems/server/ext/itsi_rb_helpers/src/lib.rs +0 -178
  232. data/gems/server/ext/itsi_scheduler/Cargo.toml +0 -24
  233. data/gems/server/ext/itsi_scheduler/extconf.rb +0 -6
  234. data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +0 -56
  235. data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +0 -44
  236. data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +0 -44
  237. data/gems/server/ext/itsi_scheduler/src/itsi_scheduler.rs +0 -308
  238. data/gems/server/ext/itsi_scheduler/src/lib.rs +0 -38
  239. data/gems/server/ext/itsi_server/Cargo.lock +0 -2956
  240. data/gems/server/ext/itsi_server/Cargo.toml +0 -47
  241. data/gems/server/ext/itsi_server/extconf.rb +0 -6
  242. data/gems/server/ext/itsi_server/src/body_proxy/big_bytes.rs +0 -104
  243. data/gems/server/ext/itsi_server/src/body_proxy/mod.rs +0 -2
  244. data/gems/server/ext/itsi_server/src/env.rs +0 -43
  245. data/gems/server/ext/itsi_server/src/lib.rs +0 -112
  246. data/gems/server/ext/itsi_server/src/request/itsi_request.rs +0 -305
  247. data/gems/server/ext/itsi_server/src/request/mod.rs +0 -1
  248. data/gems/server/ext/itsi_server/src/response/itsi_response.rs +0 -357
  249. data/gems/server/ext/itsi_server/src/response/mod.rs +0 -1
  250. data/gems/server/ext/itsi_server/src/server/bind.rs +0 -170
  251. data/gems/server/ext/itsi_server/src/server/bind_protocol.rs +0 -37
  252. data/gems/server/ext/itsi_server/src/server/io_stream.rs +0 -104
  253. data/gems/server/ext/itsi_server/src/server/itsi_server.rs +0 -294
  254. data/gems/server/ext/itsi_server/src/server/lifecycle_event.rs +0 -9
  255. data/gems/server/ext/itsi_server/src/server/listener.rs +0 -318
  256. data/gems/server/ext/itsi_server/src/server/mod.rs +0 -11
  257. data/gems/server/ext/itsi_server/src/server/process_worker.rs +0 -196
  258. data/gems/server/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +0 -254
  259. data/gems/server/ext/itsi_server/src/server/serve_strategy/mod.rs +0 -27
  260. data/gems/server/ext/itsi_server/src/server/serve_strategy/single_mode.rs +0 -263
  261. data/gems/server/ext/itsi_server/src/server/signal.rs +0 -77
  262. data/gems/server/ext/itsi_server/src/server/thread_worker.rs +0 -367
  263. data/gems/server/ext/itsi_server/src/server/tls/locked_dir_cache.rs +0 -132
  264. data/gems/server/ext/itsi_server/src/server/tls.rs +0 -265
  265. data/gems/server/ext/itsi_tracing/Cargo.lock +0 -274
  266. data/gems/server/ext/itsi_tracing/Cargo.toml +0 -16
  267. data/gems/server/ext/itsi_tracing/src/lib.rs +0 -58
  268. data/gems/server/lib/itsi/stream_io.rb +0 -38
  269. data/sandbox/itsi_sandbox_rails/.kamal/hooks/docker-setup.sample +0 -3
  270. data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-app-boot.sample +0 -3
  271. data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-deploy.sample +0 -14
  272. data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-proxy-reboot.sample +0 -3
  273. data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-app-boot.sample +0 -3
  274. data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-build.sample +0 -51
  275. data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-connect.sample +0 -47
  276. data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-deploy.sample +0 -109
  277. data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-proxy-reboot.sample +0 -3
  278. data/sandbox/itsi_sandbox_rails/.kamal/secrets +0 -17
  279. data/sandbox/itsi_sandbox_rails/bin/dev +0 -2
  280. data/sandbox/itsi_sandbox_rails/bin/jobs +0 -6
  281. data/sandbox/itsi_sandbox_rails/bin/kamal +0 -27
  282. data/sandbox/itsi_sandbox_rails/bin/thrust +0 -5
  283. data/sandbox/itsi_sandbox_rails/config/cache.yml +0 -16
  284. data/sandbox/itsi_sandbox_rails/config/deploy.yml +0 -116
  285. data/sandbox/itsi_sandbox_rails/config/queue.yml +0 -18
  286. data/sandbox/itsi_sandbox_rails/config/recurring.yml +0 -10
  287. data/sandbox/itsi_sandbox_rails/db/cable_schema.rb +0 -11
  288. data/sandbox/itsi_sandbox_rails/db/cache_schema.rb +0 -14
  289. data/sandbox/itsi_sandbox_rails/db/queue_schema.rb +0 -129
  290. data/sandbox/itsi_sandbox_rails/public/400.html +0 -114
  291. data/sandbox/itsi_sandbox_rails/test/fixtures/posts.yml +0 -9
  292. data/sandbox/itsi_sandbox_rails/test/models/post_test.rb +0 -7
  293. /data/{sandbox/itsi_sandbox_rails/script/.keep → crates/_index.md} +0 -0
  294. /data/gems/server/lib/itsi/{index.html → server/default_app/index.html} +0 -0
@@ -13,7 +13,7 @@ use itsi_tracing::error;
13
13
  use magnus::error::Result as MagnusResult;
14
14
  use parking_lot::RwLock;
15
15
  use std::{
16
- convert::Infallible,
16
+ collections::HashMap,
17
17
  io,
18
18
  os::{fd::FromRawFd, unix::net::UnixStream},
19
19
  str::FromStr,
@@ -31,31 +31,32 @@ use tokio_stream::wrappers::ReceiverStream;
31
31
  use tokio_util::io::ReaderStream;
32
32
  use tracing::warn;
33
33
 
34
- use crate::server::serve_strategy::single_mode::RunningPhase;
34
+ use crate::server::{
35
+ byte_frame::ByteFrame, serve_strategy::single_mode::RunningPhase, types::HttpResponse,
36
+ };
35
37
 
36
- #[magnus::wrap(class = "Itsi::Response", free_immediately, size)]
38
+ #[magnus::wrap(class = "Itsi::HttpResponse", free_immediately, size)]
37
39
  #[derive(Debug, Clone)]
38
- pub struct ItsiResponse {
40
+ pub struct ItsiHttpResponse {
39
41
  pub data: Arc<ResponseData>,
40
- pub accept: String,
41
42
  }
42
43
 
43
44
  #[derive(Debug)]
44
45
  pub struct ResponseData {
45
- pub response: RwLock<Option<Response<BoxBody<Bytes, Infallible>>>>,
46
- pub response_writer: RwLock<Option<mpsc::Sender<Option<Bytes>>>>,
46
+ pub response: RwLock<Option<HttpResponse>>,
47
+ pub response_writer: RwLock<Option<mpsc::Sender<ByteFrame>>>,
47
48
  pub response_buffer: RwLock<BytesMut>,
48
49
  pub hijacked_socket: RwLock<Option<UnixStream>>,
49
50
  pub parts: Parts,
50
51
  }
51
52
 
52
- impl ItsiResponse {
53
+ impl ItsiHttpResponse {
53
54
  pub async fn build(
54
55
  &self,
55
- first_frame: Option<Bytes>,
56
- receiver: mpsc::Receiver<Option<Bytes>>,
56
+ first_frame: ByteFrame,
57
+ receiver: mpsc::Receiver<ByteFrame>,
57
58
  shutdown_rx: watch::Receiver<RunningPhase>,
58
- ) -> Response<BoxBody<Bytes, Infallible>> {
59
+ ) -> HttpResponse {
59
60
  if self.is_hijacked() {
60
61
  return match self.process_hijacked_response().await {
61
62
  Ok(result) => result,
@@ -65,31 +66,30 @@ impl ItsiResponse {
65
66
  }
66
67
  };
67
68
  }
68
-
69
69
  let mut response = self.data.response.write().take().unwrap();
70
- *response.body_mut() = if first_frame.is_none() {
70
+ *response.body_mut() = if matches!(first_frame, ByteFrame::Empty) {
71
71
  BoxBody::new(Empty::new())
72
- } else if receiver.is_closed() && receiver.is_empty() {
73
- BoxBody::new(Full::new(first_frame.unwrap()))
72
+ } else if matches!(first_frame, ByteFrame::End(_)) {
73
+ BoxBody::new(Full::new(first_frame.into()))
74
74
  } else {
75
- let initial_frame = tokio_stream::once(Ok(Frame::data(first_frame.unwrap())));
75
+ let initial_frame = tokio_stream::once(Ok(Frame::data(Bytes::from(first_frame))));
76
76
  let frame_stream = unfold(
77
77
  (ReceiverStream::new(receiver), shutdown_rx),
78
78
  |(mut receiver, mut shutdown_rx)| async move {
79
79
  if let RunningPhase::ShutdownPending = *shutdown_rx.borrow() {
80
- warn!("Disconnecting streaming client.");
81
80
  return None;
82
81
  }
83
82
  loop {
84
83
  tokio::select! {
85
84
  maybe_bytes = receiver.next() => {
86
- if let Some(bytes) = maybe_bytes {
87
- // We assume `bytes` is Some(Bytes) here.
88
- return Some((Ok(Frame::data(bytes.unwrap())), (receiver, shutdown_rx)));
89
- } else {
90
- // Receiver closed, end the stream.
91
- return None;
85
+ match maybe_bytes {
86
+ Some(ByteFrame::Data(bytes)) | Some(ByteFrame::End(bytes)) => {
87
+ return Some((Ok(Frame::data(bytes)), (receiver, shutdown_rx)));
92
88
  }
89
+ _ => {
90
+ return None;
91
+ }
92
+ }
93
93
  },
94
94
  _ = shutdown_rx.changed() => {
95
95
  match *shutdown_rx.borrow() {
@@ -185,7 +185,7 @@ impl ItsiResponse {
185
185
  Ok((headers, status, requires_upgrade, reader))
186
186
  }
187
187
 
188
- pub async fn process_hijacked_response(&self) -> Result<Response<BoxBody<Bytes, Infallible>>> {
188
+ pub async fn process_hijacked_response(&self) -> Result<HttpResponse> {
189
189
  let (headers, status, requires_upgrade, reader) = self.read_hijacked_headers().await?;
190
190
  let mut response = if requires_upgrade {
191
191
  let parts = self.data.parts.clone();
@@ -262,27 +262,31 @@ impl ItsiResponse {
262
262
  }
263
263
  }
264
264
 
265
- pub fn send_frame(&self, frame: Bytes) -> MagnusResult<usize> {
266
- self.send_frame_into(frame, &self.data.response_writer)
265
+ pub fn send_frame(&self, frame: Bytes) -> MagnusResult<()> {
266
+ self.send_frame_into(ByteFrame::Data(frame), &self.data.response_writer)
267
267
  }
268
268
 
269
- pub fn send_and_close(&self, frame: Bytes) -> MagnusResult<usize> {
270
- let result = self.send_frame_into(frame, &self.data.response_writer);
269
+ pub fn recv_frame(&self) {
270
+ // not implemented
271
+ }
272
+
273
+ pub fn send_and_close(&self, frame: Bytes) -> MagnusResult<()> {
274
+ let result = self.send_frame_into(ByteFrame::End(frame), &self.data.response_writer);
271
275
  self.data.response_writer.write().take();
272
276
  result
273
277
  }
274
278
 
275
279
  pub fn send_frame_into(
276
280
  &self,
277
- frame: Bytes,
278
- writer: &RwLock<Option<mpsc::Sender<Option<Bytes>>>>,
279
- ) -> MagnusResult<usize> {
281
+ frame: ByteFrame,
282
+ writer: &RwLock<Option<mpsc::Sender<ByteFrame>>>,
283
+ ) -> MagnusResult<()> {
280
284
  if let Some(writer) = writer.write().as_ref() {
281
- writer
282
- .blocking_send(Some(frame))
283
- .map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)?;
285
+ return Ok(writer
286
+ .blocking_send(frame)
287
+ .map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)?);
284
288
  }
285
- Ok(0)
289
+ Ok(())
286
290
  }
287
291
 
288
292
  pub fn is_hijacked(&self) -> bool {
@@ -293,20 +297,28 @@ impl ItsiResponse {
293
297
  self.data.response_writer.write().take();
294
298
  Ok(true)
295
299
  }
300
+ fn accept_str(&self) -> &str {
301
+ self.data
302
+ .parts
303
+ .headers
304
+ .get("Content-Type")
305
+ .and_then(|hv| hv.to_str().ok()) // handle invalid utf-8
306
+ .unwrap_or("application/x-www-form-urlencoded")
307
+ }
296
308
 
297
309
  pub fn is_html(&self) -> bool {
298
- self.accept.starts_with("text/html")
310
+ self.accept_str().starts_with("text/html")
299
311
  }
300
312
 
301
313
  pub fn is_json(&self) -> bool {
302
- self.accept.starts_with("application/json")
314
+ self.accept_str().starts_with("application/json")
303
315
  }
304
316
 
305
317
  pub fn close_read(&self) -> MagnusResult<bool> {
306
318
  Ok(true)
307
319
  }
308
320
 
309
- pub fn new(parts: Parts, response_writer: mpsc::Sender<Option<Bytes>>, accept: String) -> Self {
321
+ pub fn new(parts: Parts, response_writer: mpsc::Sender<ByteFrame>) -> Self {
310
322
  Self {
311
323
  data: Arc::new(ResponseData {
312
324
  response: RwLock::new(Some(Response::new(BoxBody::new(Empty::new())))),
@@ -315,7 +327,6 @@ impl ItsiResponse {
315
327
  hijacked_socket: RwLock::new(None),
316
328
  parts,
317
329
  }),
318
- accept,
319
330
  }
320
331
  }
321
332
 
@@ -330,6 +341,26 @@ impl ItsiResponse {
330
341
  Ok(())
331
342
  }
332
343
 
344
+ pub fn add_headers(&self, headers: HashMap<Bytes, Vec<Bytes>>) -> MagnusResult<()> {
345
+ if let Some(ref mut resp) = *self.data.response.write() {
346
+ let headers_mut = resp.headers_mut();
347
+ for (name, values) in headers {
348
+ let header_name = HeaderName::from_bytes(&name).map_err(|e| {
349
+ itsi_error::ItsiError::InvalidInput(format!(
350
+ "Invalid header name {:?}: {:?}",
351
+ name, e
352
+ ))
353
+ })?;
354
+ for value in values {
355
+ let header_value = unsafe { HeaderValue::from_maybe_shared_unchecked(value) };
356
+ headers_mut.insert(&header_name, header_value);
357
+ }
358
+ }
359
+ }
360
+
361
+ Ok(())
362
+ }
363
+
333
364
  pub fn set_status(&self, status: u16) -> MagnusResult<()> {
334
365
  if let Some(ref mut resp) = *self.data.response.write() {
335
366
  *resp.status_mut() = StatusCode::from_u16(status).map_err(|e| {
@@ -348,8 +379,8 @@ impl ItsiResponse {
348
379
  *self.data.hijacked_socket.write() = Some(stream);
349
380
  if let Some(writer) = self.data.response_writer.write().as_ref() {
350
381
  writer
351
- .blocking_send(None)
352
- .map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)?;
382
+ .blocking_send(ByteFrame::Empty)
383
+ .map_err(|_| itsi_error::ItsiError::ClientConnectionClosed)?
353
384
  }
354
385
  self.close();
355
386
  Ok(())
@@ -0,0 +1,225 @@
1
+ use derive_more::Debug;
2
+ use globset::{Glob, GlobSet, GlobSetBuilder};
3
+ use magnus::error::Result;
4
+ use nix::unistd::{close, fork, pipe, read};
5
+ use notify::{event::ModifyKind, EventKind, RecommendedWatcher};
6
+ use notify::{Event, RecursiveMode, Watcher};
7
+ use std::path::Path;
8
+ use std::sync::mpsc::Sender;
9
+ use std::time::{Duration, Instant};
10
+ use std::{collections::HashSet, fs};
11
+ use std::{
12
+ os::fd::{AsRawFd, IntoRawFd, OwnedFd},
13
+ path::PathBuf,
14
+ process::Command,
15
+ sync::mpsc,
16
+ thread::{self},
17
+ };
18
+
19
+ /// Represents a set of patterns and commands.
20
+ #[derive(Debug, Clone)]
21
+ struct PatternGroup {
22
+ base_dir: PathBuf,
23
+ glob_set: GlobSet,
24
+ pattern: String,
25
+ commands: Vec<Vec<String>>,
26
+ last_triggered: Option<Instant>,
27
+ }
28
+
29
+ /// Extracts the base directory from a wildcard pattern by taking the portion up to the first
30
+ /// component that contains a wildcard character.
31
+ fn extract_and_canonicalize_base_dir(pattern: &str) -> PathBuf {
32
+ if !(pattern.contains("*") || pattern.contains("?") || pattern.contains('[')) {
33
+ if let Ok(metadata) = fs::metadata(pattern) {
34
+ if metadata.is_dir() {
35
+ return fs::canonicalize(pattern).unwrap();
36
+ }
37
+ if metadata.is_file() {
38
+ return fs::canonicalize(pattern)
39
+ .unwrap()
40
+ .parent()
41
+ .unwrap()
42
+ .to_path_buf();
43
+ }
44
+ }
45
+ }
46
+
47
+ let path = Path::new(pattern);
48
+ let mut base = PathBuf::new();
49
+ for comp in path.components() {
50
+ let comp_str = comp.as_os_str().to_string_lossy();
51
+ if comp_str.contains('*') || comp_str.contains('?') || comp_str.contains('[') {
52
+ break;
53
+ } else {
54
+ base.push(comp);
55
+ }
56
+ }
57
+ // If no base was built, default to "."
58
+ let base = if base.as_os_str().is_empty() || !base.exists() {
59
+ PathBuf::from(".")
60
+ } else {
61
+ base
62
+ };
63
+
64
+ // Canonicalize to get the absolute path.
65
+ fs::canonicalize(&base).unwrap_or(base)
66
+ }
67
+
68
+ /// Minimum time between triggering the same pattern group (debounce time)
69
+ const DEBOUNCE_DURATION: Duration = Duration::from_millis(500);
70
+
71
+ pub fn watch_groups(pattern_groups: Vec<(String, Vec<Vec<String>>)>) -> Result<Option<OwnedFd>> {
72
+ let (r_fd, w_fd): (OwnedFd, OwnedFd) = pipe().map_err(|e| {
73
+ magnus::Error::new(
74
+ magnus::exception::exception(),
75
+ format!("Failed to create watcher pipe: {}", e),
76
+ )
77
+ })?;
78
+
79
+ let fork_result = unsafe {
80
+ fork().map_err(|e| {
81
+ magnus::Error::new(
82
+ magnus::exception::exception(),
83
+ format!("Failed to fork file watcher: {}", e),
84
+ )
85
+ })
86
+ }?;
87
+
88
+ if fork_result.is_child() {
89
+ let _ = close(w_fd.into_raw_fd());
90
+ thread::spawn(move || {
91
+ let mut buf = [0u8; 1];
92
+ loop {
93
+ match read(r_fd.as_raw_fd(), &mut buf) {
94
+ Ok(0) => {
95
+ std::process::exit(0);
96
+ }
97
+ Ok(_) => {}
98
+ Err(_) => {
99
+ std::process::exit(0);
100
+ }
101
+ }
102
+ }
103
+ });
104
+
105
+ let mut groups = Vec::new();
106
+ for (pattern, commands) in pattern_groups.into_iter() {
107
+ let base_dir = extract_and_canonicalize_base_dir(&pattern);
108
+ let glob = Glob::new(&pattern).map_err(|e| {
109
+ magnus::Error::new(
110
+ magnus::exception::exception(),
111
+ format!("Failed to create watch glob: {}", e),
112
+ )
113
+ })?;
114
+ let glob_set = GlobSetBuilder::new().add(glob).build().map_err(|e| {
115
+ magnus::Error::new(
116
+ magnus::exception::exception(),
117
+ format!("Failed to create watch glob set: {}", e),
118
+ )
119
+ })?;
120
+ groups.push(PatternGroup {
121
+ base_dir,
122
+ glob_set,
123
+ pattern,
124
+ commands,
125
+ last_triggered: None,
126
+ });
127
+ }
128
+
129
+ // Create a channel and a watcher.
130
+ let (tx, rx) = mpsc::channel::<notify::Result<Event>>();
131
+ let sender = tx.clone();
132
+ fn event_fn(sender: Sender<notify::Result<Event>>) -> impl Fn(notify::Result<Event>) {
133
+ move |res| match res {
134
+ Ok(event) => {
135
+ sender.send(Ok(event)).unwrap();
136
+ }
137
+ Err(e) => println!("watch error: {:?}", e),
138
+ }
139
+ }
140
+
141
+ let mut watched_dirs = HashSet::new();
142
+ let mut watcher: RecommendedWatcher =
143
+ notify::recommended_watcher(event_fn(sender)).expect("Failed to create watcher");
144
+ for group in &groups {
145
+ if watched_dirs.insert(group.base_dir.clone()) {
146
+ watcher
147
+ .watch(&group.base_dir, RecursiveMode::Recursive)
148
+ .expect("Failed to add watch");
149
+ }
150
+ }
151
+
152
+ // Main event loop.
153
+ for res in rx {
154
+ match res {
155
+ Ok(event) => {
156
+ if !matches!(event.kind, EventKind::Modify(ModifyKind::Metadata(_))) {
157
+ continue;
158
+ }
159
+ let now = Instant::now();
160
+ for group in &mut groups {
161
+ for path in event.paths.iter() {
162
+ if let Ok(rel_path) = path.strip_prefix(&group.base_dir) {
163
+ if group.glob_set.is_match(rel_path)
164
+ || rel_path.to_str().is_some_and(|s| s == group.pattern)
165
+ {
166
+ // Check if we should debounce this event
167
+ if let Some(last_triggered) = group.last_triggered {
168
+ if now.duration_since(last_triggered) < DEBOUNCE_DURATION {
169
+ // Skip this event as we've recently triggered for this pattern
170
+ continue;
171
+ }
172
+ }
173
+
174
+ // Update the last triggered time
175
+ group.last_triggered = Some(now);
176
+
177
+ // Execute the commands for this group.
178
+ for command in &group.commands {
179
+ if command.is_empty() {
180
+ continue;
181
+ }
182
+ let mut cmd = Command::new(&command[0]);
183
+ if command.len() > 1 {
184
+ cmd.args(&command[1..]);
185
+ }
186
+ match cmd.spawn() {
187
+ Ok(mut child) => {
188
+ if let Err(e) = child.wait() {
189
+ eprintln!(
190
+ "Command {:?} failed: {:?}",
191
+ command, e
192
+ );
193
+ }
194
+ }
195
+ Err(e) => {
196
+ eprintln!(
197
+ "Failed to execute command {:?}: {:?}",
198
+ command, e
199
+ );
200
+ }
201
+ }
202
+ }
203
+ break;
204
+ }
205
+ }
206
+ }
207
+ }
208
+ }
209
+ Err(e) => println!("Watch error: {:?}", e),
210
+ }
211
+ }
212
+
213
+ // Clean up the watches.
214
+ for group in &groups {
215
+ watcher
216
+ .unwatch(&group.base_dir)
217
+ .expect("Failed to remove watch");
218
+ }
219
+ drop(watcher);
220
+ std::process::exit(0);
221
+ } else {
222
+ let _ = close(r_fd.into_raw_fd());
223
+ Ok(Some(w_fd))
224
+ }
225
+ }