Module: Tina4::Router
- Defined in:
- lib/tina4/router.rb
Defined Under Namespace
Classes: GroupContext
Class Method Summary collapse
- .add(method, path, handler, auth_handler: nil, swagger_meta: {}, middleware: [], template: nil) ⇒ Object
- .any(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
- .clear! ⇒ Object (also: clear)
- .delete(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
- .find_route(method, path) ⇒ Object
-
.find_ws_route(path) ⇒ Object
Find a matching WebSocket route for a given path.
-
.get(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Convenience registration methods.
- .get_routes ⇒ Object
-
.get_web_socket_routes ⇒ Object
Parity alias — returns all registered WebSocket routes.
- .group(prefix, auth_handler: nil, middleware: [], &block) ⇒ Object
-
.head(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Register an explicit HEAD route.
- .list_routes ⇒ Object
-
.load_routes(directory) ⇒ Object
Load route files from a directory (file-based route discovery).
-
.match(method, path) ⇒ Object
Find a route matching method + path.
-
.method_index ⇒ Object
Routes indexed by HTTP method for O(1) method lookup.
-
.methods_allowed_for_path(path) ⇒ Object
Return the list of HTTP methods registered for “path“, in the order GET / POST / PUT / PATCH / DELETE / HEAD / OPTIONS.
-
.options(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Register an explicit OPTIONS route.
- .patch(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
- .post(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
- .put(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
-
.record_broken_route_import(file, error) ⇒ Object
Write a .broken sentinel so /health and the dev dashboard surface auto-discover failures instead of swallowing them into a log line.
-
.rescan_routes! ⇒ Object
Re-run the most recent load_routes — called by /__dev/api/reload so files dropped into src/routes/ after server boot get picked up without a restart.
-
.reset_route_discovery! ⇒ Object
Test-only helper — reset the loaded-files state so tests can scan the same directory multiple times with different file contents.
- .routes ⇒ Object
-
.secure_websocket(path, &block) ⇒ Object
Register a SECURED WebSocket route (auth required on the upgrade).
-
.trailing_slash_redirect? ⇒ Boolean
When TINA4_TRAILING_SLASH_REDIRECT is truthy, the rack app uses this to detect whether the original (un-stripped) path differed from the canonical form so it can issue a 301 redirect.
-
.use(klass) ⇒ Object
Register a class-based middleware globally.
-
.websocket(path, secure: false, &block) ⇒ Object
Register a WebSocket route.
-
.ws_routes ⇒ Object
Registered WebSocket routes.
Class Method Details
.add(method, path, handler, auth_handler: nil, swagger_meta: {}, middleware: [], template: nil) ⇒ Object
354 355 356 357 358 359 360 361 362 363 364 365 366 367 368 369 370 371 372 373 374 375 376 377 378 379 380 381 382 383 384 385 386 |
# File 'lib/tina4/router.rb', line 354 def add(method, path, handler, auth_handler: nil, swagger_meta: {}, middleware: [], template: nil) route = Route.new(method, path, handler, auth_handler: auth_handler, swagger_meta: , middleware: middleware, template: template) # Replace semantics: re-registering the same (method, path) overwrites # the existing entry in place rather than appending a second one. # This is what makes dev hot-reload work — when a changed route file is # re-loaded, its Router.get("/x") call runs again with a fresh handler, # and #find_route returns the FIRST match, so a stale leftover would # otherwise shadow the new handler forever. Overwriting keeps the # registry free of duplicates and ensures the latest handler wins. # Distinct (method, path) pairs are untouched — only an exact dup # collapses onto the prior slot, preserving its position/order. bucket = method_index[route.method] existing_index = routes.index { |r| r.method == route.method && r.path == route.path } if existing_index routes[existing_index] = route bucket_index = bucket.index { |r| r.path == route.path } if bucket_index bucket[bucket_index] = route else bucket << route end Tina4::Log.debug("Route replaced: #{route.method} #{route.path}") else routes << route bucket << route Tina4::Log.debug("Route registered: #{route.method} #{route.path}") end route end |
.any(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
408 409 410 |
# File 'lib/tina4/router.rb', line 408 def any(path, middleware: [], swagger_meta: {}, template: nil, &block) add("ANY", path, block, middleware: middleware, swagger_meta: , template: template) end |
.clear! ⇒ Object Also known as: clear
527 528 529 530 531 |
# File 'lib/tina4/router.rb', line 527 def clear! @routes = [] @method_index = Hash.new { |h, k| h[k] = [] } @ws_routes = [] end |
.delete(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
404 405 406 |
# File 'lib/tina4/router.rb', line 404 def delete(path, middleware: [], swagger_meta: {}, template: nil, &block) add("DELETE", path, block, middleware: middleware, swagger_meta: , template: template) end |
.find_route(method, path) ⇒ Object
431 432 433 434 435 436 437 438 439 440 441 442 443 444 445 446 447 448 449 450 451 452 453 454 455 456 457 |
# File 'lib/tina4/router.rb', line 431 def find_route(method, path) normalized_method = method.upcase # Normalize path once (not per-route) normalized_path = path.gsub("\\", "/") normalized_path = "/#{normalized_path}" unless normalized_path.start_with?("/") normalized_path = normalized_path.chomp("/") unless normalized_path == "/" # Check ANY routes first, then method-specific routes candidates = (method_index["ANY"] || []) + (method_index[normalized_method] || []) candidates.each do |route| params = route.match_path(normalized_path) return [route, params] if params end # RFC 9110 §9.3.2: HEAD is identical to GET except for the absence # of a response body. If no explicit HEAD route matched, fall back # to the GET route — the dispatcher strips the body on the way out # so the handler doesn't need to know HEAD even happened. if normalized_method == "HEAD" (method_index["GET"] || []).each do |route| params = route.match_path(normalized_path) return [route, params] if params end end nil end |
.find_ws_route(path) ⇒ Object
Find a matching WebSocket route for a given path. Returns [ws_route, params] or nil.
337 338 339 340 341 342 343 344 345 346 347 |
# File 'lib/tina4/router.rb', line 337 def find_ws_route(path) normalized = path.gsub("\\", "/") normalized = "/#{normalized}" unless normalized.start_with?("/") normalized = normalized.chomp("/") unless normalized == "/" ws_routes.each do |ws_route| params = ws_route.match?(normalized) return [ws_route, params] if params end nil end |
.get(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Convenience registration methods
388 389 390 |
# File 'lib/tina4/router.rb', line 388 def get(path, middleware: [], swagger_meta: {}, template: nil, &block) add("GET", path, block, middleware: middleware, swagger_meta: , template: template) end |
.get_routes ⇒ Object
293 294 295 |
# File 'lib/tina4/router.rb', line 293 def get_routes routes end |
.get_web_socket_routes ⇒ Object
Parity alias — returns all registered WebSocket routes.
307 308 309 |
# File 'lib/tina4/router.rb', line 307 def get_web_socket_routes ws_routes end |
.group(prefix, auth_handler: nil, middleware: [], &block) ⇒ Object
534 535 536 |
# File 'lib/tina4/router.rb', line 534 def group(prefix, auth_handler: nil, middleware: [], &block) GroupContext.new(prefix, auth_handler, middleware).instance_eval(&block) end |
.head(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Register an explicit HEAD route. By default the framework auto-handles HEAD by falling back to the GET route and stripping the body (RFC 9110 §9.3.2). Use this only when you need a HEAD handler that does something different from GET — e.g. cheaper existence-check logic, custom validator headers without the cost of building the body. The framework still strips the response body for you on the way out.
418 419 420 |
# File 'lib/tina4/router.rb', line 418 def head(path, middleware: [], swagger_meta: {}, template: nil, &block) add("HEAD", path, block, middleware: middleware, swagger_meta: , template: template) end |
.list_routes ⇒ Object
297 298 299 |
# File 'lib/tina4/router.rb', line 297 def list_routes routes end |
.load_routes(directory) ⇒ Object
Load route files from a directory (file-based route discovery).
mtime-tracked & re-runnable so re-discovery on /__dev/api/reload is cheap and picks up edits without a server restart:
* NEW file (not seen before) → load it, record its mtime.
* CHANGED file (mtime newer than seen) → load it again. Ruby's `load`
RE-EXECUTES the file, so its Router.get(...) calls run afresh and
#add replaces the (method, path) in place — the new handler wins
instead of being shadowed by the stale one.
* UNCHANGED file (present, same mtime) → skip (keeps reload cheap).
Scope guard: the glob is rooted at the user’s routes/‘src` `directory`, so only application route files are ever (re)loaded — framework files are never touched. Records the directory so #rescan_routes! can re-run without re-passing it.
554 555 556 557 558 559 560 561 562 563 564 565 566 567 568 569 570 571 572 573 574 575 576 577 578 579 580 581 582 583 584 585 586 587 |
# File 'lib/tina4/router.rb', line 554 def load_routes(directory) return unless Dir.exist?(directory) @loaded_route_files ||= {} @last_routes_dir = directory files = Dir.glob(File.join(directory, "**/*.rb")).sort total = files.length files.each do |file| current_mtime = File.mtime(file).to_i # Skip only when we've seen this file AND it hasn't changed since. next if @loaded_route_files.key?(file) && current_mtime <= @loaded_route_files[file] begin load file @loaded_route_files[file] = current_mtime Tina4::Log.debug("Route loaded: #{file}") rescue ScriptError, StandardError => e # ScriptError catches SyntaxError, which is NOT a StandardError — # a bare `rescue => e` would let a syntax-broken route file crash # the whole discovery pass. Tina4::Log.error("Failed to load route #{file}: #{e.}") record_broken_route_import(file, e) end end # Zero-routes warning — src/routes/ has .rb files but the router # is still empty. Almost certainly the user forgot Tina4::Router.get. if total > 0 && routes.empty? Tina4::Log.warning( "Auto-discover found #{total} .rb file(s) in #{directory} but no routes registered. " \ "Each route file must call Tina4::Router.get / .post / etc." ) end end |
.match(method, path) ⇒ Object
Find a route matching method + path. Returns [route, params] or nil. match(method, path) — consistent with Python, PHP, and Node.
507 508 509 |
# File 'lib/tina4/router.rb', line 507 def match(method, path) find_route(method, path) end |
.method_index ⇒ Object
Routes indexed by HTTP method for O(1) method lookup
350 351 352 |
# File 'lib/tina4/router.rb', line 350 def method_index @method_index ||= Hash.new { |h, k| h[k] = [] } end |
.methods_allowed_for_path(path) ⇒ Object
Return the list of HTTP methods registered for “path“, in the order GET / POST / PUT / PATCH / DELETE / HEAD / OPTIONS. Used by the dispatcher to build the “Allow:“ header on 405 / OPTIONS responses (RFC 9110 §10.2.1, §9.3.7).
If GET is registered for the path, HEAD is appended implicitly (HEAD auto-fallback). OPTIONS is appended whenever the path has any registered method (the framework auto-handles OPTIONS).
467 468 469 470 471 472 473 474 475 476 477 478 479 480 481 482 483 484 485 486 487 488 489 490 491 492 493 494 495 |
# File 'lib/tina4/router.rb', line 467 def methods_allowed_for_path(path) normalized_path = path.gsub("\\", "/") normalized_path = "/#{normalized_path}" unless normalized_path.start_with?("/") normalized_path = normalized_path.chomp("/") unless normalized_path == "/" method_order = %w[GET POST PUT PATCH DELETE HEAD OPTIONS] seen = [] any_matched = false method_index.each do |m, routes_for_method| next if routes_for_method.empty? matched = routes_for_method.any? { |r| r.match_path(normalized_path) } next unless matched if m == "ANY" any_matched = true elsif method_order.include?(m) seen << m unless seen.include?(m) end end seen = method_order.dup if any_matched if !seen.empty? seen << "HEAD" if seen.include?("GET") && !seen.include?("HEAD") seen << "OPTIONS" unless seen.include?("OPTIONS") end method_order.select { |m| seen.include?(m) } end |
.options(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
Register an explicit OPTIONS route. By default the framework auto- handles OPTIONS by building an Allow header from every method registered for the path and returning 204 (RFC 9110 §9.3.7). Use this to take over that behaviour — e.g. to return a richer OPTIONS payload describing the resource.
427 428 429 |
# File 'lib/tina4/router.rb', line 427 def (path, middleware: [], swagger_meta: {}, template: nil, &block) add("OPTIONS", path, block, middleware: middleware, swagger_meta: , template: template) end |
.patch(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
400 401 402 |
# File 'lib/tina4/router.rb', line 400 def patch(path, middleware: [], swagger_meta: {}, template: nil, &block) add("PATCH", path, block, middleware: middleware, swagger_meta: , template: template) end |
.post(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
392 393 394 |
# File 'lib/tina4/router.rb', line 392 def post(path, middleware: [], swagger_meta: {}, template: nil, &block) add("POST", path, block, middleware: middleware, swagger_meta: , template: template) end |
.put(path, middleware: [], swagger_meta: {}, template: nil, &block) ⇒ Object
396 397 398 |
# File 'lib/tina4/router.rb', line 396 def put(path, middleware: [], swagger_meta: {}, template: nil, &block) add("PUT", path, block, middleware: middleware, swagger_meta: , template: template) end |
.record_broken_route_import(file, error) ⇒ Object
Write a .broken sentinel so /health and the dev dashboard surface auto-discover failures instead of swallowing them into a log line.
610 611 612 613 614 615 616 617 618 619 620 621 622 623 |
# File 'lib/tina4/router.rb', line 610 def record_broken_route_import(file, error) broken_dir = File.join(Dir.pwd, "data", ".broken") FileUtils.mkdir_p(broken_dir) unless Dir.exist?(broken_dir) slug = file.gsub(%r{[/\\]}, "_") payload = JSON.generate( type: "auto_discover_failure", file: file, error: "#{error.class}: #{error.}" ) File.write(File.join(broken_dir, "discover_#{slug}.broken"), payload) rescue StandardError # If the .broken write itself fails, the original error is already # in the log — nothing more to do. end |
.rescan_routes! ⇒ Object
Re-run the most recent load_routes — called by /__dev/api/reload so files dropped into src/routes/ after server boot get picked up without a restart. No-op if load_routes has never been called.
592 593 594 595 596 597 598 599 |
# File 'lib/tina4/router.rb', line 592 def rescan_routes! return [] if @last_routes_dir.nil? || @last_routes_dir.empty? before = routes.length load_routes(@last_routes_dir) added = routes.length - before Tina4::Log.info("Re-discovered #{added} new route(s) on reload") if added.positive? added end |
.reset_route_discovery! ⇒ Object
Test-only helper — reset the loaded-files state so tests can scan the same directory multiple times with different file contents.
603 604 605 606 |
# File 'lib/tina4/router.rb', line 603 def reset_route_discovery! @loaded_route_files = {} @last_routes_dir = nil end |
.routes ⇒ Object
289 290 291 |
# File 'lib/tina4/router.rb', line 289 def routes @routes ||= [] end |
.secure_websocket(path, &block) ⇒ Object
Register a SECURED WebSocket route (auth required on the upgrade). The declarative sibling of Tina4::Router.websocket(…).secure — mirrors the secure_get/secure_post pair for HTTP routes.
331 332 333 |
# File 'lib/tina4/router.rb', line 331 def secure_websocket(path, &block) websocket(path, secure: true, &block) end |
.trailing_slash_redirect? ⇒ Boolean
When TINA4_TRAILING_SLASH_REDIRECT is truthy, the rack app uses this to detect whether the original (un-stripped) path differed from the canonical form so it can issue a 301 redirect. Default false — silent match keeps backward compatibility.
501 502 503 |
# File 'lib/tina4/router.rb', line 501 def trailing_slash_redirect? %w[true 1 yes on].include?(ENV.fetch("TINA4_TRAILING_SLASH_REDIRECT", "").to_s.strip.downcase) end |
.use(klass) ⇒ Object
Register a class-based middleware globally. The class should define static before_* and/or after_* methods. Example:
class AuthMiddleware
def self.before_auth(request, response)
unless request.headers["authorization"]
return [request, response.json({ error: "Unauthorized" }, 401)]
end
[request, response]
end
end
Tina4::Router.use(AuthMiddleware)
523 524 525 |
# File 'lib/tina4/router.rb', line 523 def use(klass) Tina4::Middleware.use(klass) end |
.websocket(path, secure: false, &block) ⇒ Object
Register a WebSocket route. The handler block receives (connection, event, data) where:
connection — WebSocketConnection with #send, #broadcast, #close, #params
event — :open, :message, or :close
data — String payload for :message, nil for :open/:close
PUBLIC by default (mirrors GET). Pass secure: true (the declarative way) OR chain .secure on the returned route (the imperative way) to require a valid JWT on the upgrade — both set the same auth_required flag, exactly like the HTTP routes support both a decorator/docblock and .secure.
321 322 323 324 325 326 |
# File 'lib/tina4/router.rb', line 321 def websocket(path, secure: false, &block) ws_route = WebSocketRoute.new(path, block, auth_required: secure) ws_routes << ws_route Tina4::Log.debug("WebSocket route registered: #{path}#{secure ? ' (secured)' : ''}") ws_route end |
.ws_routes ⇒ Object
Registered WebSocket routes
302 303 304 |
# File 'lib/tina4/router.rb', line 302 def ws_routes @ws_routes ||= [] end |