module SinatraResource class Builder module Helpers # Build a resource, based on +document+, appropriate for +role+. # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [MongoMapper::Document] document # # @param [Hash] resource_config # # @return [Hash Object>] def build_resource(role, document, resource_config) resource = {} resource_config[:properties].each_pair do |property, hash| hide = if params[SHOW_KEY] == "all" false else resource_config[:properties][property][:hide_by_default] end if authorized?(:read, role, resource_config, property) && !hide resource[property.to_s] = value(property, document, hash) end end resource end # Builds a list of resources, based on +documents+, using the # appropriate role for each document. (Delegates to +lookup_role+.) # # @param [Array] documents # # @param [String] api_key # # @param [Hash] resource_config # # @param [Integer] page # # @param [Integer] page_count # # @return [Array Object>>] def build_resources(documents, resource_config, page, page_count, document_count, items_per_page) if page_count > 0 && page > page_count error 400, convert(body_for(:errors, "page (#{page}) must be <= page_count (#{page_count})")) end { 'previous' => page > 1 ? link_to_page(page - 1) : nil, 'next' => page < page_count ? link_to_page(page + 1) : nil, 'page_count' => page_count, 'page_size' => items_per_page, 'document_count' => document_count, 'members' => documents.map do |document| build_resource(lookup_role(document), document, resource_config) end, } end def calculate_page_count(document_count, items_per_page) (document_count.to_f / items_per_page).ceil end # Halt unless the current params are ok for +action+ and +role+. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [Hash] resource_config # # @param [Boolean] leaf # If a simple resource, should be true. # If a nested resource, are we at the 'end' (the leaf)? # # @return [undefined] def check_params(action, role, resource_config, leaf) return unless leaf params_check_action(action) params_check_action_and_role(action, role, resource_config) end # Halt unless the current role has permission to carry out +action+ # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [Hash] resource_config # # @return [undefined] def check_permission(action, role, resource_config) before_authorization(action, role, resource_config) unless authorized?(action, role, resource_config) error 401, convert(body_for(:unauthorized)) end end # Convert +object+ to desired format. # # For example, an application might want to convert +object+ to JSON or # XML. # # Applications must override this method. # # @param [Object] object # # @return [String] def convert raise NotImplementedError end # Display +object+ as appropriate for +action+. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Object] object # # @param [Hash] resource_config # # @return [String] def display(action, object, resource_config, parent_id = nil) case action when :list when :read when :create response.status = 201 response.headers['Location'] = location(object, resource_config, parent_id) when :update when :delete response.status = 204 else raise Error, "Unexpected: #{action.inspect}" end convert(object) end # Execute a callback. # # @param [Symbol] name # Valid values include: # * :before_create, :before_update, :before_destroy # * :after_create, :after_update, :after_destroy # # @param [Hash] resource_config # # @param [MongoMapper::Document, nil] document # # @param [MongoMapper::Document, nil] parent_document # # @return [undefined] def do_callback(name, resource_config, document, parent_document) proc = resource_config[:callbacks][name] return unless proc if document && parent_document proc.call(self, document, parent_document) elsif document proc.call(self, document) elsif parent_document proc.call(self, parent_document) else proc.call(self) end end # Convert a path to a full URI. # # Applications must override this method. # # @param [String] path # # @return [String] def full_uri(path) raise NotImplementedError end # Get the page parameter. # # @param [Hash] params # # @return [Integer] def get_page(params) raw = params.delete('page') return 1 unless raw page = raw.to_i if page < 1 error 400, convert(body_for(:errors, "page must be >= 1")) end page end # Build a hash that contains a URL to +page_number+. # # @param [Integer] page # # @return [Hash] def link_to_page(page_number) q = Rack::Utils.parse_query(request.query_string) q['page'] = page_number query_string = Rack::Utils.build_query(q) { 'href' => "#{request.path}?#{query_string}" } end # Get role, using +model+ and +id+. Delegates to +lookup_role+. # # When +id+ is present, it can help determine 'relative' roles such # as 'ownership' of the current user of a particular document. # # @param [Class] model # # @param [String] id # # @return [Symbol] def role_for(model, id) lookup_role(model.find_by_id(id)) end # Get role for a nested resource situation. Delegates to +lookup_role+. # # @params [MongoMapper::Document] parent # The parent document # # @params [Symbol] child_assoc # Association from the parent to the child # # @params [Class] child_model # # @params [String] child_id # # @return [Symbol] def role_for_nested(parent, child_assoc, child_model, child_id) lookup_role( find_nested_document(parent, child_assoc, child_model, child_id)) end # Return the minimum role required for +action+, and, if specified, # +property+. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Hash] resource_config # # @param [Symbol, nil] property # # @return [Symbol] # a role (such as :anonymous, :basic, or :admin) def minimum_role(action, resource_config, property=nil) if property.nil? p = resource_config[:permission] raise Error, "undefined #{action.inspect} permission" unless p p[action] else hash = resource_config[:properties][property] hash ? hash[to_r_or_w(action)] : :nobody end || :anonymous end protected # Is +role+ authorized for +action+, and, if specified, +property+? # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Hash] resource_config # # @param [Symbol, nil] property # a property of a resource # # @return [Boolean] def authorized?(action, role, resource_config, property=nil) klass = resource_config[:roles] klass.validate_role(role) klass.satisfies?(role, minimum_role(action, resource_config, property)) end # Application-level hook that runs as part of +check_permission+, # before +authorized?(action, role, resource_config)+ is called. # # For example, an application might want to throw custom errors # in certain situations before +authorized?+ runs. # # Applications must override this method. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [Hash] resource_config # # @return [String] def before_authorization(action, role, resource_config) raise NotImplementedError end # Default body message for a +situation+ # # @param [Symbol] situation # # @param [Object] object # # @return [String] def body_for(situation, object=nil) case situation when :errors { "errors" => object } when :internal_server_error { "errors" => "internal_server_error" } when :invalid_document { "errors" => object.errors.errors } when :invalid_params { "errors" => { "invalid_params" => object } } when :invalid_filter { "errors" => { "invalid_filter" => object } } when :no_params { "errors" => "no_params" } when :not_found "" when :unauthorized { "errors" => "unauthorized_api_key" } end end # Return a full URI for +object+. # # @param [Object] object # A resource or a list of resources # # @param [Hash] resource_config # # @param [String] parent_id # # @return [String] def location(object, resource_config, parent_id) o = object c = resource_config path = if c[:parent] raise Error, "expecting parent_id" unless parent_id pc = c[:parent].resource_config "#{pc[:path]}/#{parent_id}/#{c[:path]}/#{o['id']}" else "#{c[:path]}/#{o['id']}" end full_uri(path) end # Do application-specific logging for +event+. # # Applications may override this method. # # @param [Symbol] event # @param [Hash] params # # @return [undefined] def log_event(event, params) end # Lookup the role, using +document+ if specified. # # Applications must override this method. # # @param [MongoMapper::Document, nil] document # # @return [Symbol] def lookup_role(document=nil) raise NotImplementedError end # Are the params suitable for +action+? Raise 400 Bad Request if not. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Boolean] leaf # If a simple resource, should be true. # If a nested resource, are we at the 'end' (the leaf)? # # @return [undefined] def params_check_action(action) case action when :list p = params.reject do |k, v| [FILTER_KEY, SEARCH_KEY].include?(k) end unless p.empty? error 400, convert(body_for(:invalid_params, p)) end when :read p = params.reject { |k, v| k == SHOW_KEY } unless [nil, "all"].include?(params[SHOW_KEY]) error 400, convert(body_for(:invalid_params, { SHOW_KEY => params[SHOW_KEY] })) end unless p.empty? error 400, convert(body_for(:invalid_params, p)) end when :create # No need to complain. If there are problems, # params_check_action_and_role will catch them. when :update if params.empty? error 400, convert(body_for(:no_params)) end when :delete unless params.empty? error 400, convert(body_for(:invalid_params, p)) end else raise Error, "Unexpected: #{action.inspect}" end end # Checks each parameter to make sure it is authorized for +action+ and # +role+. Raises a 400 Bad Request if not authorized. # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @param [Symbol] role # a role (such as :anonymous, :basic, or :admin) # # @param [Hash] resource_config # # @return [undefined] def params_check_action_and_role(action, role, resource_config) invalid = [] params.each_pair do |property, value| next if [FILTER_KEY, SEARCH_KEY, SHOW_KEY].include?(property) if !authorized?(action, role, resource_config, property.intern) invalid << property end end unless invalid.empty? error 400, convert(body_for(:invalid_params, invalid)) end end # Converts +action+ to :r or :w (i.e. read or write). # # @param [Symbol] action # :list, :read, :create, :update, or :delete # # @return [Symbol] # :r or :w def to_r_or_w(action) case action when :list then :r when :read then :r when :create then :w when :update then :w when :delete then :w else raise "Unexpected action : #{action.inspect}" end end # Lookup +attribute+ in +document+ # # @param [Symbol] attribute # an attribute of +document+ # # @param [MongoMapper::Document] document # # @return [undefined] def value(attribute, document, property_hash) proc = property_hash[:read_proc] if proc proc.call(document) else document[attribute == :id ? :_id : attribute] # An alternate way is: # # document.send(attribute == :id ? :_id : attribute) # # This will work; however, it would be confusing to support # properties backed by model methods here if we don't do it # everywhere. And supporting it everywhere would be tricky. # # For example, the filtering code relies on using MongoDB to # search the database. If we supported properties backed by model # methods, filtering / searching would be more complicated and # expensive. end end end end end