# frozen_string_literal: true # class Roda module RodaPlugins # This plugin makes it easier to to respond to specific request data types. User agents can request # specific data types by either supplying an appropriate +Accept+ request header # or by appending it as file extension to the path. # # Example: # # plugin :type_routing # # route do |r| # r.get 'a' do # r.html{ "

This is the HTML response

" } # r.json{ '{"json": "ok"}' } # r.xml{ "This is the XML response" } # "Unsupported data type" # end # end # # This application will handle the following paths: # /a.html :: HTML response # /a.json :: JSON response # /a.xml :: XML response # /a :: HTML, JSON, or XML response, depending on the Accept header # # The response +Content-Type+ header will be set to a suitable value when # the +r.html+, +r.json+, or +r.xml+ block is matched. # # Note that if no match is found, code will continue to execute, which can # result in unexpected behaviour. This should only happen if you do not # handle all supported/configured types. If you want to simplify handling, # you can just place the html handling after the other types, without using # a separate block: # # route do |r| # r.get 'a' do # r.json{ '{"json": "ok"}' } # r.xml{ "This is the XML response" } # # "

This is the HTML response

" # end # end # # This works correctly because Roda's default Content-Type is text/html. Note that # if you use this approach, the type_routing plugin's :html content type will not be # used for html responses, since you aren't using an +r.html+ block. Instead, the # Content-Type header will be set to Roda's default (which you can override via # the default_headers plugin). # # If the type routing is based on the +Accept+ request header and not the file extension, # then an appropriate +Vary+ header will be set or appended to, so that HTTP caches do # not serve the same result for requests with different +Accept+ headers. # # To match custom extensions, use the :types option: # # plugin :type_routing, types: { # yaml: 'application/x-yaml', # js: 'application/javascript; charset=utf-8' # } # # route do |r| # r.get 'a' do # r.yaml{ YAML.dump "YAML data" } # r.js{ "JavaScript code" } # # or: # r.on_type(:js){ "JavaScript code" } # "Unsupported data type" # end # end # # = Plugin options # # The following plugin options are supported: # # :default_type :: The default data type to assume if the client did not # provide one. Defaults to +:html+. # :exclude :: Exclude one or more types from the default set (default set # is :html, :xml, :json). # :types :: Mapping from a data type to its MIME-Type. Used both to match # incoming requests and to provide +Content-Type+ values. If the # value is +nil+, no +Content-Type+ will be set. The type may # contain media type parameters, which will be sent to the client # but ignored for request matching. # :use_extension :: Whether to take the path extension into account. # Default is +true+. # :use_header :: Whether to take the +Accept+ header into account. # Default is +true+. module TypeRouting CONFIGURATION = { :mimes => { 'text/json' => :json, 'application/json' => :json, 'text/xml' => :xml, 'application/xml' => :xml, 'text/html' => :html, }.freeze, :types => { :json => 'application/json'.freeze, :xml => 'application/xml'.freeze, :html => 'text/html'.freeze, }.freeze, :use_extension => true, :use_header => true, :default_type => :html }.freeze def self.configure(app, opts = {}) config = (app.opts[:type_routing] || CONFIGURATION).dup [:use_extension, :use_header, :default_type].each do |key| config[key] = opts[key] if opts.has_key?(key) end types = config[:types] = config[:types].dup mimes = config[:mimes] = config[:mimes].dup Array(opts[:exclude]).each do |type| types.delete(type) mimes.reject!{|_, v| v == type} end if mapping = opts[:types] types.merge!(mapping) mapping.each do |k, v| if v mimes[v.split(';', 2).first] = k end end end types.freeze mimes.freeze type_keys = config[:types].keys config[:extension_regexp] = /(.*?)\.(#{Regexp.union(type_keys.map(&:to_s))})\z/ type_keys.each do |type| app::RodaRequest.send(:define_method, type) do |&block| on_type(type, &block) end app::RodaRequest.send(:alias_method, type, type) end app.opts[:type_routing] = config.freeze end module RequestMethods # Yields if the given +type+ matches the requested data type and halts # the request afterwards, returning the result of the block. def on_type(type, &block) return unless type == requested_type response['Content-Type'] ||= @scope.opts[:type_routing][:types][type] always(&block) end # Returns the data type the client requests. def requested_type return @requested_type if defined?(@requested_type) opts = @scope.opts[:type_routing] @requested_type = accept_response_type if opts[:use_header] @requested_type ||= opts[:default_type] end # Append the type routing extension back to the path if it was # removed before routing. def real_remaining_path if defined?(@type_routing_extension) "#{super}.#{@type_routing_extension}" else super end end private # Removes a trailing file extension from the path, and sets # the requested type if so. def _remaining_path(env) opts = scope.opts[:type_routing] path = super if opts[:use_extension] if m = opts[:extension_regexp].match(path) @type_routing_extension = @requested_type = m[2].to_sym path = m[1] end end path end # The response type indicated by the Accept request header. def accept_response_type mimes = @scope.opts[:type_routing][:mimes] @env['HTTP_ACCEPT'].to_s.split(/\s*,\s*/).map do |part| mime, _= part.split(/\s*;\s*/, 2) if sym = mimes[mime] response['Vary'] = (vary = response['Vary']) ? "#{vary}, Accept" : 'Accept' return sym end end nil end end end register_plugin(:type_routing, TypeRouting) end end