require 'active_support' module Saddle # This base endpoint is what all implementation endpoints should inherit # from. It automatically provides tree construction and traversal # functionality. It also abstracts away url construction and requests to # the underlying requester instance. class BaseEndpoint attr_reader :requester, :relative_path, :parent # Each endpoint needs to have a requester in order to ... make ... uh ... requests. def initialize(requester, relative_path_override=nil, parent=nil) @requester = requester @parent = parent @relative_path = relative_path_override || _relative_path() end # Provide GET functionality for the implementer class def get(action, params={}, options={}) _request(:get, action, params, options) end # Provide POST functionality for the implementer class def post(action, params={}, options={}) _request(:post, action, params, options) end # Provide PUT functionality for the implementer class def put(action, params={}, options={}) _request(:put, action, params, options) end # Provide DELETE functionality for the implementer class def delete(action, params={}, options={}) _request(:delete, action, params, options) end # This will create a resource endpoint, based upon the parameters # of this current node endpoint def create_resource_endpoint(endpoint_class, resource_id) endpoint_class.new(@requester, resource_id, self) end # Create an endpoint instance and foist it upon this node # Not private, but not part of the public interface for an endpoint def _build_and_attach_node(endpoint_class, method_name=nil) # Create the new endpoint endpoint_instance = endpoint_class.new(@requester, method_name, self) # Attach the endpoint as an instance variable and method method_name ||= ActiveSupport::Inflector.underscore( ActiveSupport::Inflector.demodulize( endpoint_class.name ) ) self.instance_variable_set("@#{method_name}", endpoint_instance) self.class.class_eval { define_method(method_name) { endpoint_instance } } endpoint_instance end protected def _request(method, action, params={}, options={}) # Augment in interesting options options[:saddle] = { :client_name => _client_name(), :call_chain => _path_array(), :action => action, } @requester.send(method, _path(action), params, options) end # Get the url path for this endpoint/action combo def _path(action=nil) paths = _path_array() paths << action unless action.nil? '/' + paths.join('/') end def _path_array _endpoint_chain().map(&:relative_path).reject{|p| p.nil?} end # Get the parent chain that led to this endpoint def _endpoint_chain chain = [] node = self while node.is_a?(BaseEndpoint) chain << node node = node.parent end chain.reverse() end # Traverse back until we find the original client def _client node = self while node.is_a?(BaseEndpoint) node = node.parent end node end # Underscore name of the client def _client_name ActiveSupport::Inflector.underscore(self._client.name.split('::')[-2]) end # If the parent is not an endpoint, it is a root node def _is_root? !@parent.is_a?(BaseEndpoint) end # Build the default relative path for this endpoint node # If this is a root node, use nil. # Otherwise use the underscored version of the class name # # Override this if needed for specific endpoints def _relative_path if _is_root? nil else ActiveSupport::Inflector.underscore( ActiveSupport::Inflector.demodulize( self.class.name ) ) end end end # This endpoint will be automatically constructed into the node # traversal tree. class TraversalEndpoint < BaseEndpoint; end # This endpoint is used for constructing resource-style endpoints. This # means it will NOT be automatically added into the traversal tree. class ResourceEndpoint < BaseEndpoint; end end