itsi 0.1.11 → 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.
- checksums.yaml +4 -4
- data/Cargo.lock +1535 -45
- data/{sandbox/itsi_itsi_file/Itsi.rb → Itsi.rb} +19 -13
- data/Rakefile +8 -7
- data/crates/itsi_error/src/lib.rs +9 -0
- data/crates/itsi_rb_helpers/Cargo.toml +1 -0
- data/crates/itsi_rb_helpers/src/heap_value.rs +18 -0
- data/crates/itsi_rb_helpers/src/lib.rs +34 -7
- data/crates/itsi_server/Cargo.toml +69 -30
- data/crates/itsi_server/src/lib.rs +79 -147
- data/crates/itsi_server/src/{body_proxy → ruby_types/itsi_body_proxy}/big_bytes.rs +10 -5
- data/crates/itsi_server/src/{body_proxy/itsi_body_proxy.rs → ruby_types/itsi_body_proxy/mod.rs} +22 -3
- data/crates/itsi_server/src/ruby_types/itsi_grpc_request.rs +147 -0
- data/crates/itsi_server/src/ruby_types/itsi_grpc_response.rs +19 -0
- data/crates/itsi_server/src/ruby_types/itsi_grpc_stream/mod.rs +216 -0
- data/{gems/server/ext/itsi_server/src/request/itsi_request.rs → crates/itsi_server/src/ruby_types/itsi_http_request.rs} +101 -117
- data/crates/itsi_server/src/{response/itsi_response.rs → ruby_types/itsi_http_response.rs} +72 -41
- data/crates/itsi_server/src/ruby_types/itsi_server/file_watcher.rs +225 -0
- data/crates/itsi_server/src/ruby_types/itsi_server/itsi_server_config.rs +355 -0
- data/crates/itsi_server/src/ruby_types/itsi_server.rs +82 -0
- data/crates/itsi_server/src/ruby_types/mod.rs +55 -0
- data/crates/itsi_server/src/server/bind.rs +13 -5
- data/crates/itsi_server/src/server/byte_frame.rs +32 -0
- data/crates/itsi_server/src/server/cache_store.rs +74 -0
- data/crates/itsi_server/src/server/itsi_service.rs +172 -0
- data/crates/itsi_server/src/server/lifecycle_event.rs +3 -0
- data/crates/itsi_server/src/server/listener.rs +102 -2
- data/crates/itsi_server/src/server/middleware_stack/middleware.rs +153 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/allow_list.rs +47 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_api_key.rs +58 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_basic.rs +82 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/auth_jwt.rs +321 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/cache_control.rs +139 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/compression.rs +300 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/cors.rs +287 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/deny_list.rs +48 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/error_response.rs +127 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/etag.rs +191 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/grpc_service.rs +72 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/header_interpretation.rs +85 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/intrusion_protection.rs +195 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/log_requests.rs +82 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/mod.rs +82 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/proxy.rs +216 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/rate_limit.rs +124 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/redirect.rs +76 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/request_headers.rs +43 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/response_headers.rs +34 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/ruby_app.rs +93 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/static_assets.rs +162 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/string_rewrite.rs +158 -0
- data/crates/itsi_server/src/server/middleware_stack/middlewares/token_source.rs +12 -0
- data/crates/itsi_server/src/server/middleware_stack/mod.rs +315 -0
- data/crates/itsi_server/src/server/mod.rs +8 -1
- data/crates/itsi_server/src/server/process_worker.rs +38 -12
- data/crates/itsi_server/src/server/rate_limiter.rs +565 -0
- data/crates/itsi_server/src/server/request_job.rs +11 -0
- data/crates/itsi_server/src/server/serve_strategy/cluster_mode.rs +119 -42
- data/crates/itsi_server/src/server/serve_strategy/mod.rs +9 -6
- data/crates/itsi_server/src/server/serve_strategy/single_mode.rs +256 -111
- data/crates/itsi_server/src/server/signal.rs +19 -0
- data/crates/itsi_server/src/server/static_file_server.rs +984 -0
- data/crates/itsi_server/src/server/thread_worker.rs +139 -94
- data/crates/itsi_server/src/server/types.rs +43 -0
- data/crates/itsi_server/test.md +14 -0
- data/crates/itsi_tracing/Cargo.toml +1 -0
- data/crates/itsi_tracing/src/lib.rs +216 -45
- data/docs/.gitignore +7 -0
- data/docs/.gitpod.yml +15 -0
- data/docs/Itsi.rb +17 -0
- data/docs/content/_index.md +17 -0
- data/docs/content/about.md +6 -0
- data/docs/content/docs/_index.md +18 -0
- data/docs/content/docs/first-page.md +9 -0
- data/docs/content/docs/folder/_index.md +10 -0
- data/docs/content/docs/folder/leaf.md +7 -0
- data/docs/go.mod +5 -0
- data/docs/go.sum +2 -0
- data/docs/hugo.yaml +77 -0
- data/examples/static_assets_example.rb +83 -0
- data/gems/_index.md +18 -0
- data/gems/scheduler/CODE_OF_CONDUCT.md +7 -0
- data/gems/scheduler/Cargo.lock +75 -14
- data/gems/scheduler/README.md +5 -0
- data/gems/scheduler/_index.md +7 -0
- data/gems/scheduler/itsi-scheduler.gemspec +4 -1
- data/gems/scheduler/lib/itsi/scheduler/version.rb +1 -1
- data/gems/scheduler/lib/itsi/scheduler.rb +2 -2
- data/gems/scheduler/test/test_file_io.rb +0 -1
- data/gems/scheduler/test/test_itsi_scheduler.rb +1 -1
- data/gems/server/CHANGELOG.md +5 -0
- data/gems/server/CODE_OF_CONDUCT.md +7 -0
- data/gems/server/Cargo.lock +1536 -45
- data/gems/server/README.md +4 -0
- data/gems/server/_index.md +6 -0
- data/gems/server/exe/itsi +33 -74
- data/gems/server/itsi-server.gemspec +3 -2
- data/gems/server/lib/itsi/{request.rb → http_request.rb} +29 -5
- data/gems/server/lib/itsi/http_response.rb +39 -0
- data/gems/server/lib/itsi/server/Itsi.rb +11 -19
- data/gems/server/lib/itsi/server/config/dsl.rb +506 -0
- data/gems/server/lib/itsi/server/config.rb +103 -8
- data/gems/server/lib/itsi/server/default_app/default_app.rb +38 -0
- data/gems/server/lib/itsi/server/grpc_interface.rb +213 -0
- data/gems/server/lib/itsi/server/rack/handler/itsi.rb +8 -17
- data/gems/server/lib/itsi/server/rack_interface.rb +23 -4
- data/gems/server/lib/itsi/server/scheduler_interface.rb +1 -1
- data/gems/server/lib/itsi/server/scheduler_mode.rb +4 -0
- data/gems/server/lib/itsi/server/signal_trap.rb +7 -1
- data/gems/server/lib/itsi/server/version.rb +1 -1
- data/gems/server/lib/itsi/server.rb +74 -63
- data/gems/server/lib/itsi/standard_headers.rb +86 -0
- data/gems/server/test/helpers/test_helper.rb +12 -12
- data/gems/server/test/test_itsi_server.rb +2 -2
- data/lib/itsi/version.rb +1 -1
- data/sandbox/itsi_file/Gemfile +11 -0
- data/sandbox/itsi_file/Gemfile.lock +69 -0
- data/sandbox/itsi_file/Itsi.rb +276 -0
- data/sandbox/itsi_file/error.html +2 -0
- data/sandbox/itsi_file/organisations_controller.rb +20 -0
- data/sandbox/itsi_file/public/assets/image.png +0 -0
- data/sandbox/itsi_file/public/assets/index.html +1 -0
- data/sandbox/itsi_sandbox_hanami/Gemfile.lock +2 -2
- data/sandbox/itsi_sandbox_rack/Gemfile.lock +2 -2
- data/sandbox/itsi_sandbox_rack/config.ru +2 -15
- data/sandbox/itsi_sandbox_rails/.dockerignore +2 -5
- data/sandbox/itsi_sandbox_rails/.github/workflows/ci.yml +1 -1
- data/sandbox/itsi_sandbox_rails/.gitignore +2 -1
- data/sandbox/itsi_sandbox_rails/Dockerfile +6 -9
- data/sandbox/itsi_sandbox_rails/Gemfile +16 -22
- data/sandbox/itsi_sandbox_rails/Gemfile.lock +100 -225
- data/sandbox/itsi_sandbox_rails/app/assets/config/manifest.js +4 -0
- data/sandbox/itsi_sandbox_rails/app/assets/stylesheets/application.css +11 -6
- data/sandbox/itsi_sandbox_rails/app/channels/application_cable/channel.rb +4 -0
- data/sandbox/itsi_sandbox_rails/app/channels/application_cable/connection.rb +4 -0
- data/sandbox/itsi_sandbox_rails/app/controllers/live_controller.rb +7 -8
- data/sandbox/itsi_sandbox_rails/app/controllers/uploads_controller.rb +0 -3
- data/sandbox/itsi_sandbox_rails/app/views/layouts/application.html.erb +2 -7
- data/sandbox/itsi_sandbox_rails/bin/docker-entrypoint +3 -4
- data/sandbox/itsi_sandbox_rails/bin/setup +8 -5
- data/sandbox/itsi_sandbox_rails/config/application.rb +1 -35
- data/sandbox/itsi_sandbox_rails/config/cable.yml +3 -10
- data/sandbox/itsi_sandbox_rails/config/credentials.yml.enc +1 -1
- data/sandbox/itsi_sandbox_rails/config/database.yml +9 -19
- data/sandbox/itsi_sandbox_rails/config/environment.rb +1 -1
- data/sandbox/itsi_sandbox_rails/config/environments/development.rb +21 -12
- data/sandbox/itsi_sandbox_rails/config/environments/production.rb +49 -34
- data/sandbox/itsi_sandbox_rails/config/environments/test.rb +19 -5
- data/sandbox/itsi_sandbox_rails/config/initializers/assets.rb +5 -0
- data/sandbox/itsi_sandbox_rails/config/initializers/filter_parameter_logging.rb +1 -1
- data/sandbox/itsi_sandbox_rails/config/initializers/permissions_policy.rb +13 -0
- data/sandbox/itsi_sandbox_rails/config/puma.rb +2 -9
- data/sandbox/itsi_sandbox_rails/config.ru +0 -1
- data/sandbox/itsi_sandbox_rails/db/migrate/20250301041554_create_posts.rb +1 -1
- data/sandbox/itsi_sandbox_rails/db/schema.rb +2 -2
- data/sandbox/itsi_sandbox_rails/lib/assets/.keep +0 -0
- data/sandbox/itsi_sandbox_rails/public/404.html +66 -113
- data/sandbox/itsi_sandbox_rails/public/406-unsupported-browser.html +65 -113
- data/sandbox/itsi_sandbox_rails/public/422.html +66 -113
- data/sandbox/itsi_sandbox_rails/public/500.html +65 -113
- data/sandbox/itsi_sandbox_rails/public/icon.png +0 -0
- data/sandbox/itsi_sandbox_rails/public/icon.svg +2 -2
- data/sandbox/itsi_sandbox_rails/test/channels/application_cable/connection_test.rb +13 -0
- data/sandbox/itsi_sandbox_roda/Gemfile.lock +3 -10
- data/tasks.txt +72 -35
- metadata +89 -139
- data/crates/itsi_server/src/body_proxy/mod.rs +0 -2
- data/crates/itsi_server/src/request/itsi_request.rs +0 -298
- data/crates/itsi_server/src/request/mod.rs +0 -1
- data/crates/itsi_server/src/response/mod.rs +0 -1
- data/crates/itsi_server/src/server/itsi_server.rs +0 -288
- data/gems/scheduler/ext/itsi_error/Cargo.lock +0 -368
- data/gems/scheduler/ext/itsi_error/Cargo.toml +0 -11
- data/gems/scheduler/ext/itsi_error/src/from.rs +0 -68
- data/gems/scheduler/ext/itsi_error/src/lib.rs +0 -24
- data/gems/scheduler/ext/itsi_instrument_entry/Cargo.toml +0 -15
- data/gems/scheduler/ext/itsi_instrument_entry/src/lib.rs +0 -31
- data/gems/scheduler/ext/itsi_rb_helpers/Cargo.lock +0 -355
- data/gems/scheduler/ext/itsi_rb_helpers/Cargo.toml +0 -10
- data/gems/scheduler/ext/itsi_rb_helpers/src/heap_value.rs +0 -121
- data/gems/scheduler/ext/itsi_rb_helpers/src/lib.rs +0 -201
- data/gems/scheduler/ext/itsi_scheduler/Cargo.toml +0 -24
- data/gems/scheduler/ext/itsi_scheduler/extconf.rb +0 -6
- data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +0 -56
- data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +0 -44
- data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +0 -44
- data/gems/scheduler/ext/itsi_scheduler/src/itsi_scheduler.rs +0 -308
- data/gems/scheduler/ext/itsi_scheduler/src/lib.rs +0 -38
- data/gems/scheduler/ext/itsi_server/Cargo.lock +0 -2956
- data/gems/scheduler/ext/itsi_server/Cargo.toml +0 -50
- data/gems/scheduler/ext/itsi_server/extconf.rb +0 -6
- data/gems/scheduler/ext/itsi_server/src/body_proxy/big_bytes.rs +0 -104
- data/gems/scheduler/ext/itsi_server/src/body_proxy/itsi_body_proxy.rs +0 -122
- data/gems/scheduler/ext/itsi_server/src/body_proxy/mod.rs +0 -2
- data/gems/scheduler/ext/itsi_server/src/env.rs +0 -43
- data/gems/scheduler/ext/itsi_server/src/lib.rs +0 -180
- data/gems/scheduler/ext/itsi_server/src/request/itsi_request.rs +0 -298
- data/gems/scheduler/ext/itsi_server/src/request/mod.rs +0 -1
- data/gems/scheduler/ext/itsi_server/src/response/itsi_response.rs +0 -357
- data/gems/scheduler/ext/itsi_server/src/response/mod.rs +0 -1
- data/gems/scheduler/ext/itsi_server/src/server/bind.rs +0 -174
- data/gems/scheduler/ext/itsi_server/src/server/bind_protocol.rs +0 -37
- data/gems/scheduler/ext/itsi_server/src/server/io_stream.rs +0 -104
- data/gems/scheduler/ext/itsi_server/src/server/itsi_server.rs +0 -288
- data/gems/scheduler/ext/itsi_server/src/server/lifecycle_event.rs +0 -9
- data/gems/scheduler/ext/itsi_server/src/server/listener.rs +0 -318
- data/gems/scheduler/ext/itsi_server/src/server/mod.rs +0 -11
- data/gems/scheduler/ext/itsi_server/src/server/process_worker.rs +0 -203
- data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +0 -260
- data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/mod.rs +0 -27
- data/gems/scheduler/ext/itsi_server/src/server/serve_strategy/single_mode.rs +0 -276
- data/gems/scheduler/ext/itsi_server/src/server/signal.rs +0 -74
- data/gems/scheduler/ext/itsi_server/src/server/thread_worker.rs +0 -399
- data/gems/scheduler/ext/itsi_server/src/server/tls/locked_dir_cache.rs +0 -132
- data/gems/scheduler/ext/itsi_server/src/server/tls.rs +0 -265
- data/gems/scheduler/ext/itsi_tracing/Cargo.lock +0 -274
- data/gems/scheduler/ext/itsi_tracing/Cargo.toml +0 -16
- data/gems/scheduler/ext/itsi_tracing/src/lib.rs +0 -58
- data/gems/server/ext/itsi_error/Cargo.lock +0 -368
- data/gems/server/ext/itsi_error/Cargo.toml +0 -11
- data/gems/server/ext/itsi_error/src/from.rs +0 -68
- data/gems/server/ext/itsi_error/src/lib.rs +0 -24
- data/gems/server/ext/itsi_instrument_entry/Cargo.toml +0 -15
- data/gems/server/ext/itsi_instrument_entry/src/lib.rs +0 -31
- data/gems/server/ext/itsi_rb_helpers/Cargo.lock +0 -355
- data/gems/server/ext/itsi_rb_helpers/Cargo.toml +0 -10
- data/gems/server/ext/itsi_rb_helpers/src/heap_value.rs +0 -121
- data/gems/server/ext/itsi_rb_helpers/src/lib.rs +0 -201
- data/gems/server/ext/itsi_scheduler/Cargo.toml +0 -24
- data/gems/server/ext/itsi_scheduler/extconf.rb +0 -6
- data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/io_helpers.rs +0 -56
- data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/io_waiter.rs +0 -44
- data/gems/server/ext/itsi_scheduler/src/itsi_scheduler/timer.rs +0 -44
- data/gems/server/ext/itsi_scheduler/src/itsi_scheduler.rs +0 -308
- data/gems/server/ext/itsi_scheduler/src/lib.rs +0 -38
- data/gems/server/ext/itsi_server/Cargo.lock +0 -2956
- data/gems/server/ext/itsi_server/Cargo.toml +0 -50
- data/gems/server/ext/itsi_server/extconf.rb +0 -6
- data/gems/server/ext/itsi_server/src/body_proxy/big_bytes.rs +0 -104
- data/gems/server/ext/itsi_server/src/body_proxy/itsi_body_proxy.rs +0 -122
- data/gems/server/ext/itsi_server/src/body_proxy/mod.rs +0 -2
- data/gems/server/ext/itsi_server/src/env.rs +0 -43
- data/gems/server/ext/itsi_server/src/lib.rs +0 -180
- data/gems/server/ext/itsi_server/src/request/mod.rs +0 -1
- data/gems/server/ext/itsi_server/src/response/itsi_response.rs +0 -357
- data/gems/server/ext/itsi_server/src/response/mod.rs +0 -1
- data/gems/server/ext/itsi_server/src/server/bind.rs +0 -174
- data/gems/server/ext/itsi_server/src/server/bind_protocol.rs +0 -37
- data/gems/server/ext/itsi_server/src/server/io_stream.rs +0 -104
- data/gems/server/ext/itsi_server/src/server/itsi_server.rs +0 -288
- data/gems/server/ext/itsi_server/src/server/lifecycle_event.rs +0 -9
- data/gems/server/ext/itsi_server/src/server/listener.rs +0 -318
- data/gems/server/ext/itsi_server/src/server/mod.rs +0 -11
- data/gems/server/ext/itsi_server/src/server/process_worker.rs +0 -203
- data/gems/server/ext/itsi_server/src/server/serve_strategy/cluster_mode.rs +0 -260
- data/gems/server/ext/itsi_server/src/server/serve_strategy/mod.rs +0 -27
- data/gems/server/ext/itsi_server/src/server/serve_strategy/single_mode.rs +0 -276
- data/gems/server/ext/itsi_server/src/server/signal.rs +0 -74
- data/gems/server/ext/itsi_server/src/server/thread_worker.rs +0 -399
- data/gems/server/ext/itsi_server/src/server/tls/locked_dir_cache.rs +0 -132
- data/gems/server/ext/itsi_server/src/server/tls.rs +0 -265
- data/gems/server/ext/itsi_tracing/Cargo.lock +0 -274
- data/gems/server/ext/itsi_tracing/Cargo.toml +0 -16
- data/gems/server/ext/itsi_tracing/src/lib.rs +0 -58
- data/gems/server/lib/itsi/server/options_dsl.rb +0 -401
- data/gems/server/lib/itsi/stream_io.rb +0 -38
- data/location_dsl.rb +0 -381
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/docker-setup.sample +0 -3
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-app-boot.sample +0 -3
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-deploy.sample +0 -14
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/post-proxy-reboot.sample +0 -3
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-app-boot.sample +0 -3
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-build.sample +0 -51
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-connect.sample +0 -47
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-deploy.sample +0 -109
- data/sandbox/itsi_sandbox_rails/.kamal/hooks/pre-proxy-reboot.sample +0 -3
- data/sandbox/itsi_sandbox_rails/.kamal/secrets +0 -17
- data/sandbox/itsi_sandbox_rails/bin/dev +0 -2
- data/sandbox/itsi_sandbox_rails/bin/jobs +0 -6
- data/sandbox/itsi_sandbox_rails/bin/kamal +0 -27
- data/sandbox/itsi_sandbox_rails/bin/thrust +0 -5
- data/sandbox/itsi_sandbox_rails/config/cache.yml +0 -16
- data/sandbox/itsi_sandbox_rails/config/deploy.yml +0 -116
- data/sandbox/itsi_sandbox_rails/config/queue.yml +0 -18
- data/sandbox/itsi_sandbox_rails/config/recurring.yml +0 -10
- data/sandbox/itsi_sandbox_rails/db/cable_schema.rb +0 -11
- data/sandbox/itsi_sandbox_rails/db/cache_schema.rb +0 -14
- data/sandbox/itsi_sandbox_rails/db/queue_schema.rb +0 -129
- data/sandbox/itsi_sandbox_rails/public/400.html +0 -114
- data/sandbox/itsi_sandbox_rails/test/fixtures/posts.yml +0 -9
- data/sandbox/itsi_sandbox_rails/test/models/post_test.rb +0 -7
- /data/{sandbox/itsi_sandbox_rails/script/.keep → crates/_index.md} +0 -0
- /data/gems/server/lib/itsi/{index.html → server/default_app/index.html} +0 -0
data/location_dsl.rb
DELETED
@@ -1,381 +0,0 @@
|
|
1
|
-
class RouterDSL
|
2
|
-
attr_reader :parent, :children, :filters, :endpoint_defs, :controller_class
|
3
|
-
|
4
|
-
def initialize(parent = nil, route_specs = [], &block)
|
5
|
-
@parent = parent
|
6
|
-
@children = []
|
7
|
-
@filters = {}
|
8
|
-
@endpoint_defs = [] # Each is [subpath, *endpoint_args]
|
9
|
-
@controller_class = nil
|
10
|
-
|
11
|
-
# We'll store our array of route specs (strings or a single Regexp).
|
12
|
-
@route_specs = Array(route_specs).flatten
|
13
|
-
|
14
|
-
validate_path_specs!(@route_specs)
|
15
|
-
|
16
|
-
instance_eval(&block) if block_given?
|
17
|
-
end
|
18
|
-
|
19
|
-
################################
|
20
|
-
# DSL: location
|
21
|
-
################################
|
22
|
-
|
23
|
-
# location can accept multiple strings (and at most one Regexp).
|
24
|
-
# E.g.: location '/foo', '/bar/:id(\d+)' do ... end
|
25
|
-
def location(*route_specs, &block)
|
26
|
-
route_specs = route_specs.flatten
|
27
|
-
child = RouterDSL.new(self, route_specs, &block)
|
28
|
-
@children << child
|
29
|
-
end
|
30
|
-
|
31
|
-
# define endpoints
|
32
|
-
def endpoint(subpath, *args)
|
33
|
-
@endpoint_defs << [subpath, *args]
|
34
|
-
end
|
35
|
-
|
36
|
-
def controller(klass)
|
37
|
-
@controller_class = klass
|
38
|
-
end
|
39
|
-
|
40
|
-
# define some filters
|
41
|
-
def basic_auth(**args)
|
42
|
-
@filters[:basic_auth] = args
|
43
|
-
end
|
44
|
-
|
45
|
-
# define some filters
|
46
|
-
def redirect(**args)
|
47
|
-
@filters[:redirect] = args
|
48
|
-
end
|
49
|
-
|
50
|
-
def jwt_auth(**args)
|
51
|
-
@filters[:jwt_auth] = args
|
52
|
-
end
|
53
|
-
|
54
|
-
def api_key_auth(**args)
|
55
|
-
@filters[:api_key_auth] = args
|
56
|
-
end
|
57
|
-
|
58
|
-
def compress(**args)
|
59
|
-
@filters[:compress] = args
|
60
|
-
end
|
61
|
-
|
62
|
-
def rate_limit(name, **args)
|
63
|
-
@filters[:rate_limit] = { name: name }.merge(args)
|
64
|
-
end
|
65
|
-
|
66
|
-
def cors(**args)
|
67
|
-
@filters[:cors] = args
|
68
|
-
end
|
69
|
-
|
70
|
-
def file_server(**args)
|
71
|
-
@filters[:file_server] = args
|
72
|
-
end
|
73
|
-
|
74
|
-
################################
|
75
|
-
# Flattening logic
|
76
|
-
################################
|
77
|
-
|
78
|
-
def flatten_routes
|
79
|
-
# We produce an array of route-hashes:
|
80
|
-
# { route: Regexp, filters: [ {type:, params:}, ... ] }
|
81
|
-
#
|
82
|
-
# The order: children first (most specific), then endpoints, then the location route.
|
83
|
-
|
84
|
-
# 1) Flatten children (most specific first)
|
85
|
-
child_routes = @children.flat_map(&:flatten_routes)
|
86
|
-
|
87
|
-
# 2) Build the expansions for the "parent portion" (this location) → we'll call them "my_base_expansions"
|
88
|
-
# That is effectively the parent's expansions combined with mine (OR pattern).
|
89
|
-
my_base_expansions = combined_paths_from_parent
|
90
|
-
|
91
|
-
# 3) Endpoint routes: for each endpoint subpath, produce a single OR pattern that covers
|
92
|
-
# (base expansions) x (endpoint expansion).
|
93
|
-
endpoint_routes = @endpoint_defs.map do |(endpoint_subpath, *endpoint_args)|
|
94
|
-
# Expand subpath
|
95
|
-
ep_expansions = expand_single_subpath(endpoint_subpath)
|
96
|
-
|
97
|
-
# Combine base expansions with endpoint expansions in a cartesian product, then OR them
|
98
|
-
final_regex_str = or_pattern_for(cartesian_combine(my_base_expansions, ep_expansions))
|
99
|
-
|
100
|
-
{
|
101
|
-
route: Regexp.new("^#{final_regex_str}$"),
|
102
|
-
filters: effective_filters_with_endpoint(endpoint_args)
|
103
|
-
}
|
104
|
-
end
|
105
|
-
|
106
|
-
# 4) A route for this location block itself (without subpaths).
|
107
|
-
# The OR pattern for my_base_expansions (like '^/(?:foo|bar)$' ).
|
108
|
-
# If I have route specs, we produce that route. If you want a route even with no route_specs, adapt.
|
109
|
-
location_route = unless @route_specs.empty?
|
110
|
-
pattern_str = or_pattern_for(my_base_expansions) # the expansions themselves
|
111
|
-
{
|
112
|
-
route: Regexp.new("^#{pattern_str}$"),
|
113
|
-
filters: effective_filters
|
114
|
-
}
|
115
|
-
end
|
116
|
-
|
117
|
-
# Final array: child routes first, then endpoints, then my location route
|
118
|
-
result = []
|
119
|
-
result.concat(child_routes)
|
120
|
-
result.concat(endpoint_routes)
|
121
|
-
result << location_route if location_route
|
122
|
-
result
|
123
|
-
end
|
124
|
-
|
125
|
-
################################
|
126
|
-
# Helpers
|
127
|
-
################################
|
128
|
-
|
129
|
-
def validate_path_specs!(specs)
|
130
|
-
# 1) If there's more than one raw Ruby Regexp, raise an error
|
131
|
-
# 2) If there's 1 raw Ruby Regexp + anything else, also raise an error
|
132
|
-
# 3) We can allow *multiple strings*, but if any is a Regexp => can't nest children
|
133
|
-
# We'll actually raise an error if the user tries to create children anyway.
|
134
|
-
regexes = specs.select { |s| s.is_a?(Regexp) }
|
135
|
-
return unless regexes.size > 1
|
136
|
-
|
137
|
-
raise ArgumentError, 'Cannot have multiple raw Regex route specs in a single location.'
|
138
|
-
end
|
139
|
-
|
140
|
-
# Called by flatten_routes to get expansions from the parent's expansions combined with mine
|
141
|
-
def combined_paths_from_parent
|
142
|
-
if parent
|
143
|
-
# get parent's expansions
|
144
|
-
pex = parent.combined_paths_from_parent_for_children
|
145
|
-
# combine with my route specs
|
146
|
-
cartesian_combine(pex, expansions_for(@route_specs))
|
147
|
-
else
|
148
|
-
# top-level: no parent expansions
|
149
|
-
expansions_for(@route_specs)
|
150
|
-
end
|
151
|
-
end
|
152
|
-
|
153
|
-
# If the parent is a raw Regexp route, that parent wouldn't allow children anyway.
|
154
|
-
# But let's define a method the child uses:
|
155
|
-
def combined_paths_from_parent_for_children
|
156
|
-
# The parent's "my_base_expansions" is the expansions from the parent's route specs,
|
157
|
-
# ignoring endpoints. Because a parent's endpoints produce *its own* routes, not expansions for children.
|
158
|
-
parent ? parent.combined_paths_from_parent_for_children : [] # recursion upward
|
159
|
-
end
|
160
|
-
|
161
|
-
# We override that in a simpler way so it includes the parent's expansions.
|
162
|
-
# But to keep it straightforward, we'll store our expansions so children can see them.
|
163
|
-
|
164
|
-
def combined_paths_from_parent_for_children
|
165
|
-
# For me, the expansions are expansions_for(@route_specs) combined with parent's expansions
|
166
|
-
# if there is one. (Essentially the same logic as combined_paths_from_parent,
|
167
|
-
# but we do it for the child's perspective.)
|
168
|
-
if parent
|
169
|
-
pex = parent.combined_paths_from_parent_for_children
|
170
|
-
cartesian_combine(pex, expansions_for(@route_specs))
|
171
|
-
else
|
172
|
-
expansions_for(@route_specs)
|
173
|
-
end
|
174
|
-
end
|
175
|
-
|
176
|
-
# Given a single subpath from an endpoint call (a string or a "*"), produce expansions
|
177
|
-
def expand_single_subpath(subpath)
|
178
|
-
expansions_for([subpath]) # just treat it as a mini specs array
|
179
|
-
end
|
180
|
-
|
181
|
-
# Turn an array of specs (strings or at most one Regexp) into an array of expansions (strings).
|
182
|
-
# If there's exactly one raw Regexp, we store a single special marker that indicates "raw regexp".
|
183
|
-
def expansions_for(specs)
|
184
|
-
return [] if specs.empty?
|
185
|
-
|
186
|
-
# If we have a single raw Ruby Regexp, we do not expand it further;
|
187
|
-
# but that also means no nesting is allowed. We just store that "as-is".
|
188
|
-
if specs.any? { |s| s.is_a? Regexp }
|
189
|
-
# Return something that indicates we have a raw Regexp route:
|
190
|
-
# We'll store it as an array with one element `[:raw_regex, that_regexp]`
|
191
|
-
# so cartesian_combine logic can handle it specially.
|
192
|
-
# Let's say we raise if there's more than one
|
193
|
-
raise 'Cannot combine a raw Regexp with other strings in the same location.' if specs.size > 1
|
194
|
-
|
195
|
-
[[:raw_regex, specs.first]]
|
196
|
-
else
|
197
|
-
# We have multiple strings. Convert each to a "fully expanded" sub-regex piece,
|
198
|
-
# but do NOT add ^...$ here. We'll do that later.
|
199
|
-
# We'll simply store them as strings in "regex-ready" form, i.e. leading slash is included if needed.
|
200
|
-
# Actually we only produce the "inner piece" so we can do `^(?: piece1 | piece2 )$`.
|
201
|
-
specs.map do |string_spec|
|
202
|
-
# remove leading slash
|
203
|
-
string_spec = string_spec.sub(%r{^/}, '')
|
204
|
-
# if empty => it means "/", so let's keep it blank
|
205
|
-
string_spec
|
206
|
-
end
|
207
|
-
end
|
208
|
-
end
|
209
|
-
|
210
|
-
# Combine two arrays of expansions in a cartesian way.
|
211
|
-
# If either array has a raw_regexp, we raise if the other is non-empty or also raw.
|
212
|
-
# If both are pure string expansions, we produce new expansions for each combination.
|
213
|
-
def cartesian_combine(parent_exps, child_exps)
|
214
|
-
return child_exps if parent_exps.empty?
|
215
|
-
return parent_exps if child_exps.empty?
|
216
|
-
|
217
|
-
# If parent_exps has raw_regexp
|
218
|
-
if parent_exps.size == 1 && parent_exps.first.is_a?(Array) && parent_exps.first.first == :raw_regex
|
219
|
-
# That means parent's route is a raw Regexp => no children allowed
|
220
|
-
# (the problem statement: "if any route is a raw Ruby Regexp, no nesting allowed")
|
221
|
-
raise 'Cannot nest under a raw Regexp route.'
|
222
|
-
end
|
223
|
-
|
224
|
-
if child_exps.size == 1 && child_exps.first.is_a?(Array) && child_exps.first.first == :raw_regex
|
225
|
-
# child is also a raw regex => not allowed in combination
|
226
|
-
raise 'Cannot nest a raw Regexp route under a parent string route.'
|
227
|
-
end
|
228
|
-
|
229
|
-
# Both are purely strings => cartesian
|
230
|
-
results = []
|
231
|
-
parent_exps.each do |p|
|
232
|
-
child_exps.each do |c|
|
233
|
-
# combine with a slash if needed (unless p or c are empty)
|
234
|
-
joined = [p, c].reject(&:empty?).join('/')
|
235
|
-
results << joined
|
236
|
-
end
|
237
|
-
end
|
238
|
-
results
|
239
|
-
end
|
240
|
-
|
241
|
-
# Turn an array of expansions (which are strings like "users/(?<id>[^/]+)" or blank "")
|
242
|
-
# into a big OR group. e.g. (?: /users/xxx | /whatever )
|
243
|
-
# We'll inject a leading slash if not empty, then create the group.
|
244
|
-
def or_pattern_for(expansions)
|
245
|
-
return '' if expansions.empty?
|
246
|
-
|
247
|
-
# If expansions has an array that starts with [:raw_regex, /someRegex/], that means no strings
|
248
|
-
# but the user is making a direct Regexp. But we already handle that by disallowing nesting.
|
249
|
-
# So we shouldn't see it here unless it's from the top location with no parent.
|
250
|
-
# In that case let's do a direct pass-through:
|
251
|
-
if expansions.size == 1 && expansions.first.is_a?(Array) && expansions.first.first == :raw_regex
|
252
|
-
raw = expansions.first.last
|
253
|
-
return raw.source # Use the raw Regexp's source
|
254
|
-
end
|
255
|
-
|
256
|
-
# For each expansion, we do a slash plus expansion if expansion isn't blank
|
257
|
-
# If expansion is blank => it's basically "/"
|
258
|
-
pattern_pieces = expansions.map do |exp|
|
259
|
-
if exp.empty?
|
260
|
-
'' # => means top-level "/"
|
261
|
-
else
|
262
|
-
# We'll parse param placeholders into named captures.
|
263
|
-
# So let's do the same param expansions as in the earlier snippet:
|
264
|
-
segment_to_regex_with_slash(exp)
|
265
|
-
end
|
266
|
-
end
|
267
|
-
|
268
|
-
# Join them with '|'
|
269
|
-
joined = pattern_pieces.join('|')
|
270
|
-
|
271
|
-
"(?:#{joined})"
|
272
|
-
end
|
273
|
-
|
274
|
-
# Take a string like "users/:id(\d+)" and produce "users/(?<id>\d+)" with the correct param expansions,
|
275
|
-
# then put a leading slash. If blank => just slash. We'll do a quick parse (like the earlier snippet).
|
276
|
-
def segment_to_regex_with_slash(path_str)
|
277
|
-
return '' if path_str == ''
|
278
|
-
|
279
|
-
segments = path_str.split('/')
|
280
|
-
|
281
|
-
converted = segments.map do |seg|
|
282
|
-
# wildcard?
|
283
|
-
next '.*' if seg == '*'
|
284
|
-
|
285
|
-
# :param(...)?
|
286
|
-
if seg =~ /^:([A-Za-z_]\w*)(?:\(([^)]*)\))?$/
|
287
|
-
param_name = Regexp.last_match(1)
|
288
|
-
custom = Regexp.last_match(2)
|
289
|
-
if custom && !custom.empty?
|
290
|
-
"(?<#{param_name}>#{custom})"
|
291
|
-
else
|
292
|
-
"(?<#{param_name}>[^/]+)"
|
293
|
-
end
|
294
|
-
else
|
295
|
-
Regexp.escape(seg)
|
296
|
-
end
|
297
|
-
end
|
298
|
-
|
299
|
-
converted.join('/')
|
300
|
-
end
|
301
|
-
|
302
|
-
################################
|
303
|
-
# Filter merging
|
304
|
-
################################
|
305
|
-
|
306
|
-
def effective_filters
|
307
|
-
# gather from root -> self, overriding duplicates
|
308
|
-
merged = merge_ancestor_filters
|
309
|
-
# turn into array
|
310
|
-
merged.map { |k, v| { type: k, params: v } }
|
311
|
-
end
|
312
|
-
|
313
|
-
def effective_filters_with_endpoint(endpoint_args)
|
314
|
-
arr = effective_filters
|
315
|
-
# endpoint filter last
|
316
|
-
ep_filter_params = endpoint_args.dup
|
317
|
-
ep_filter_params << @controller_class if @controller_class
|
318
|
-
arr << { type: :endpoint, params: ep_filter_params }
|
319
|
-
arr
|
320
|
-
end
|
321
|
-
|
322
|
-
def merge_ancestor_filters
|
323
|
-
chain = []
|
324
|
-
node = self
|
325
|
-
while node
|
326
|
-
chain << node
|
327
|
-
node = node.parent
|
328
|
-
end
|
329
|
-
chain.reverse!
|
330
|
-
|
331
|
-
merged = {}
|
332
|
-
chain.each do |n|
|
333
|
-
n.filters.each do |k, v|
|
334
|
-
merged[k] = v
|
335
|
-
end
|
336
|
-
end
|
337
|
-
merged
|
338
|
-
end
|
339
|
-
end
|
340
|
-
|
341
|
-
################################
|
342
|
-
# Example usage
|
343
|
-
################################
|
344
|
-
|
345
|
-
if $0 == __FILE__
|
346
|
-
dsl = RouterDSL.new do
|
347
|
-
location 'http://*' do
|
348
|
-
redirect to: 'https://{host}/{path}', status: 301
|
349
|
-
end
|
350
|
-
|
351
|
-
location 'http://www.google.com/', 'https://www.foo.bar.com' do
|
352
|
-
# Authentication
|
353
|
-
basic_auth credential_pairs: [[ENV['BASIC_USER_EMAIL'], ENV['BASIC_USER_PASSWORD']]]
|
354
|
-
jwt_auth secret: ENV['JWT_SECRET'], algorithm: 'HS256', verification: 'RS256', public_key: 'public.pem'
|
355
|
-
api_key_auth allowed_keys: ENV['API_KEYS'] ? ENV['API_KEYS'].split(',') : []
|
356
|
-
|
357
|
-
# Compression
|
358
|
-
compress types: ['text/html', 'text/css', 'application/javascript'],
|
359
|
-
threshold: 1024,
|
360
|
-
algorithms: %w[gzip deflate]
|
361
|
-
|
362
|
-
# Rate Limiting
|
363
|
-
rate_limit :api_rate, limit: '1000 requests/hour', burst: 50
|
364
|
-
|
365
|
-
# CORS
|
366
|
-
cors allow_origins: ['https://example.com'], methods: %i[GET POST], apply_to: [:api]
|
367
|
-
|
368
|
-
# Static assets
|
369
|
-
|
370
|
-
location '/users', '/members' do
|
371
|
-
file_server dir: 'public', index: 'index.html', default: 'index.html'
|
372
|
-
controller 'User'
|
373
|
-
endpoint ':id(\d+)', :get_user
|
374
|
-
end
|
375
|
-
end
|
376
|
-
end
|
377
|
-
|
378
|
-
flattened = dsl.flatten_routes
|
379
|
-
require 'debug'
|
380
|
-
bb
|
381
|
-
end
|
@@ -1,14 +0,0 @@
|
|
1
|
-
#!/bin/sh
|
2
|
-
|
3
|
-
# A sample post-deploy hook
|
4
|
-
#
|
5
|
-
# These environment variables are available:
|
6
|
-
# KAMAL_RECORDED_AT
|
7
|
-
# KAMAL_PERFORMER
|
8
|
-
# KAMAL_VERSION
|
9
|
-
# KAMAL_HOSTS
|
10
|
-
# KAMAL_ROLE (if set)
|
11
|
-
# KAMAL_DESTINATION (if set)
|
12
|
-
# KAMAL_RUNTIME
|
13
|
-
|
14
|
-
echo "$KAMAL_PERFORMER deployed $KAMAL_VERSION to $KAMAL_DESTINATION in $KAMAL_RUNTIME seconds"
|
@@ -1,51 +0,0 @@
|
|
1
|
-
#!/bin/sh
|
2
|
-
|
3
|
-
# A sample pre-build hook
|
4
|
-
#
|
5
|
-
# Checks:
|
6
|
-
# 1. We have a clean checkout
|
7
|
-
# 2. A remote is configured
|
8
|
-
# 3. The branch has been pushed to the remote
|
9
|
-
# 4. The version we are deploying matches the remote
|
10
|
-
#
|
11
|
-
# These environment variables are available:
|
12
|
-
# KAMAL_RECORDED_AT
|
13
|
-
# KAMAL_PERFORMER
|
14
|
-
# KAMAL_VERSION
|
15
|
-
# KAMAL_HOSTS
|
16
|
-
# KAMAL_ROLE (if set)
|
17
|
-
# KAMAL_DESTINATION (if set)
|
18
|
-
|
19
|
-
if [ -n "$(git status --porcelain)" ]; then
|
20
|
-
echo "Git checkout is not clean, aborting..." >&2
|
21
|
-
git status --porcelain >&2
|
22
|
-
exit 1
|
23
|
-
fi
|
24
|
-
|
25
|
-
first_remote=$(git remote)
|
26
|
-
|
27
|
-
if [ -z "$first_remote" ]; then
|
28
|
-
echo "No git remote set, aborting..." >&2
|
29
|
-
exit 1
|
30
|
-
fi
|
31
|
-
|
32
|
-
current_branch=$(git branch --show-current)
|
33
|
-
|
34
|
-
if [ -z "$current_branch" ]; then
|
35
|
-
echo "Not on a git branch, aborting..." >&2
|
36
|
-
exit 1
|
37
|
-
fi
|
38
|
-
|
39
|
-
remote_head=$(git ls-remote $first_remote --tags $current_branch | cut -f1)
|
40
|
-
|
41
|
-
if [ -z "$remote_head" ]; then
|
42
|
-
echo "Branch not pushed to remote, aborting..." >&2
|
43
|
-
exit 1
|
44
|
-
fi
|
45
|
-
|
46
|
-
if [ "$KAMAL_VERSION" != "$remote_head" ]; then
|
47
|
-
echo "Version ($KAMAL_VERSION) does not match remote HEAD ($remote_head), aborting..." >&2
|
48
|
-
exit 1
|
49
|
-
fi
|
50
|
-
|
51
|
-
exit 0
|
@@ -1,47 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
|
3
|
-
# A sample pre-connect check
|
4
|
-
#
|
5
|
-
# Warms DNS before connecting to hosts in parallel
|
6
|
-
#
|
7
|
-
# These environment variables are available:
|
8
|
-
# KAMAL_RECORDED_AT
|
9
|
-
# KAMAL_PERFORMER
|
10
|
-
# KAMAL_VERSION
|
11
|
-
# KAMAL_HOSTS
|
12
|
-
# KAMAL_ROLE (if set)
|
13
|
-
# KAMAL_DESTINATION (if set)
|
14
|
-
# KAMAL_RUNTIME
|
15
|
-
|
16
|
-
hosts = ENV["KAMAL_HOSTS"].split(",")
|
17
|
-
results = nil
|
18
|
-
max = 3
|
19
|
-
|
20
|
-
elapsed = Benchmark.realtime do
|
21
|
-
results = hosts.map do |host|
|
22
|
-
Thread.new do
|
23
|
-
tries = 1
|
24
|
-
|
25
|
-
begin
|
26
|
-
Socket.getaddrinfo(host, 0, Socket::AF_UNSPEC, Socket::SOCK_STREAM, nil, Socket::AI_CANONNAME)
|
27
|
-
rescue SocketError
|
28
|
-
if tries < max
|
29
|
-
puts "Retrying DNS warmup: #{host}"
|
30
|
-
tries += 1
|
31
|
-
sleep rand
|
32
|
-
retry
|
33
|
-
else
|
34
|
-
puts "DNS warmup failed: #{host}"
|
35
|
-
host
|
36
|
-
end
|
37
|
-
end
|
38
|
-
|
39
|
-
tries
|
40
|
-
end
|
41
|
-
end.map(&:value)
|
42
|
-
end
|
43
|
-
|
44
|
-
retries = results.sum - hosts.size
|
45
|
-
nopes = results.count { |r| r == max }
|
46
|
-
|
47
|
-
puts "Prewarmed %d DNS lookups in %.2f sec: %d retries, %d failures" % [ hosts.size, elapsed, retries, nopes ]
|
@@ -1,109 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
|
3
|
-
# A sample pre-deploy hook
|
4
|
-
#
|
5
|
-
# Checks the Github status of the build, waiting for a pending build to complete for up to 720 seconds.
|
6
|
-
#
|
7
|
-
# Fails unless the combined status is "success"
|
8
|
-
#
|
9
|
-
# These environment variables are available:
|
10
|
-
# KAMAL_RECORDED_AT
|
11
|
-
# KAMAL_PERFORMER
|
12
|
-
# KAMAL_VERSION
|
13
|
-
# KAMAL_HOSTS
|
14
|
-
# KAMAL_COMMAND
|
15
|
-
# KAMAL_SUBCOMMAND
|
16
|
-
# KAMAL_ROLE (if set)
|
17
|
-
# KAMAL_DESTINATION (if set)
|
18
|
-
|
19
|
-
# Only check the build status for production deployments
|
20
|
-
if ENV["KAMAL_COMMAND"] == "rollback" || ENV["KAMAL_DESTINATION"] != "production"
|
21
|
-
exit 0
|
22
|
-
end
|
23
|
-
|
24
|
-
require "bundler/inline"
|
25
|
-
|
26
|
-
# true = install gems so this is fast on repeat invocations
|
27
|
-
gemfile(true, quiet: true) do
|
28
|
-
source "https://rubygems.org"
|
29
|
-
|
30
|
-
gem "octokit"
|
31
|
-
gem "faraday-retry"
|
32
|
-
end
|
33
|
-
|
34
|
-
MAX_ATTEMPTS = 72
|
35
|
-
ATTEMPTS_GAP = 10
|
36
|
-
|
37
|
-
def exit_with_error(message)
|
38
|
-
$stderr.puts message
|
39
|
-
exit 1
|
40
|
-
end
|
41
|
-
|
42
|
-
class GithubStatusChecks
|
43
|
-
attr_reader :remote_url, :git_sha, :github_client, :combined_status
|
44
|
-
|
45
|
-
def initialize
|
46
|
-
@remote_url = `git config --get remote.origin.url`.strip.delete_prefix("https://github.com/")
|
47
|
-
@git_sha = `git rev-parse HEAD`.strip
|
48
|
-
@github_client = Octokit::Client.new(access_token: ENV["GITHUB_TOKEN"])
|
49
|
-
refresh!
|
50
|
-
end
|
51
|
-
|
52
|
-
def refresh!
|
53
|
-
@combined_status = github_client.combined_status(remote_url, git_sha)
|
54
|
-
end
|
55
|
-
|
56
|
-
def state
|
57
|
-
combined_status[:state]
|
58
|
-
end
|
59
|
-
|
60
|
-
def first_status_url
|
61
|
-
first_status = combined_status[:statuses].find { |status| status[:state] == state }
|
62
|
-
first_status && first_status[:target_url]
|
63
|
-
end
|
64
|
-
|
65
|
-
def complete_count
|
66
|
-
combined_status[:statuses].count { |status| status[:state] != "pending"}
|
67
|
-
end
|
68
|
-
|
69
|
-
def total_count
|
70
|
-
combined_status[:statuses].count
|
71
|
-
end
|
72
|
-
|
73
|
-
def current_status
|
74
|
-
if total_count > 0
|
75
|
-
"Completed #{complete_count}/#{total_count} checks, see #{first_status_url} ..."
|
76
|
-
else
|
77
|
-
"Build not started..."
|
78
|
-
end
|
79
|
-
end
|
80
|
-
end
|
81
|
-
|
82
|
-
|
83
|
-
$stdout.sync = true
|
84
|
-
|
85
|
-
puts "Checking build status..."
|
86
|
-
attempts = 0
|
87
|
-
checks = GithubStatusChecks.new
|
88
|
-
|
89
|
-
begin
|
90
|
-
loop do
|
91
|
-
case checks.state
|
92
|
-
when "success"
|
93
|
-
puts "Checks passed, see #{checks.first_status_url}"
|
94
|
-
exit 0
|
95
|
-
when "failure"
|
96
|
-
exit_with_error "Checks failed, see #{checks.first_status_url}"
|
97
|
-
when "pending"
|
98
|
-
attempts += 1
|
99
|
-
end
|
100
|
-
|
101
|
-
exit_with_error "Checks are still pending, gave up after #{MAX_ATTEMPTS * ATTEMPTS_GAP} seconds" if attempts == MAX_ATTEMPTS
|
102
|
-
|
103
|
-
puts checks.current_status
|
104
|
-
sleep(ATTEMPTS_GAP)
|
105
|
-
checks.refresh!
|
106
|
-
end
|
107
|
-
rescue Octokit::NotFound
|
108
|
-
exit_with_error "Build status could not be found"
|
109
|
-
end
|
@@ -1,17 +0,0 @@
|
|
1
|
-
# Secrets defined here are available for reference under registry/password, env/secret, builder/secrets,
|
2
|
-
# and accessories/*/env/secret in config/deploy.yml. All secrets should be pulled from either
|
3
|
-
# password manager, ENV, or a file. DO NOT ENTER RAW CREDENTIALS HERE! This file needs to be safe for git.
|
4
|
-
|
5
|
-
# Example of extracting secrets from 1password (or another compatible pw manager)
|
6
|
-
# SECRETS=$(kamal secrets fetch --adapter 1password --account your-account --from Vault/Item KAMAL_REGISTRY_PASSWORD RAILS_MASTER_KEY)
|
7
|
-
# KAMAL_REGISTRY_PASSWORD=$(kamal secrets extract KAMAL_REGISTRY_PASSWORD ${SECRETS})
|
8
|
-
# RAILS_MASTER_KEY=$(kamal secrets extract RAILS_MASTER_KEY ${SECRETS})
|
9
|
-
|
10
|
-
# Use a GITHUB_TOKEN if private repositories are needed for the image
|
11
|
-
# GITHUB_TOKEN=$(gh config get -h github.com oauth_token)
|
12
|
-
|
13
|
-
# Grab the registry password from ENV
|
14
|
-
KAMAL_REGISTRY_PASSWORD=$KAMAL_REGISTRY_PASSWORD
|
15
|
-
|
16
|
-
# Improve security by using a password manager. Never check config/master.key into git!
|
17
|
-
RAILS_MASTER_KEY=$(cat config/master.key)
|
@@ -1,27 +0,0 @@
|
|
1
|
-
#!/usr/bin/env ruby
|
2
|
-
# frozen_string_literal: true
|
3
|
-
|
4
|
-
#
|
5
|
-
# This file was generated by Bundler.
|
6
|
-
#
|
7
|
-
# The application 'kamal' is installed as part of a gem, and
|
8
|
-
# this file is here to facilitate running it.
|
9
|
-
#
|
10
|
-
|
11
|
-
ENV["BUNDLE_GEMFILE"] ||= File.expand_path("../Gemfile", __dir__)
|
12
|
-
|
13
|
-
bundle_binstub = File.expand_path("bundle", __dir__)
|
14
|
-
|
15
|
-
if File.file?(bundle_binstub)
|
16
|
-
if File.read(bundle_binstub, 300).include?("This file was generated by Bundler")
|
17
|
-
load(bundle_binstub)
|
18
|
-
else
|
19
|
-
abort("Your `bin/bundle` was not generated by Bundler, so this binstub cannot run.
|
20
|
-
Replace `bin/bundle` by running `bundle binstubs bundler --force`, then run this command again.")
|
21
|
-
end
|
22
|
-
end
|
23
|
-
|
24
|
-
require "rubygems"
|
25
|
-
require "bundler/setup"
|
26
|
-
|
27
|
-
load Gem.bin_path("kamal", "kamal")
|