=begin rdoc The Videojuicer::Resource module is a mixin that provides RESTful model features to classes within the Videojuicer SDK. By including Videojuicer::Resource in a model class, it will gain a series of class and instance methods that *approximate* the interface provided by a DataMapper or ActiveRecord-style object. =end module Videojuicer module Resource include Videojuicer::Exceptions include Videojuicer::Configurable include Videojuicer::OAuth::ProxyFactory include Videojuicer::Resource::Inferrable include Videojuicer::Resource::PropertyRegistry include Videojuicer::Resource::Types include Videojuicer::Resource::Relationships::BelongsTo def self.included(base) base.extend(ClassMethods) Inferrable.included(base) PropertyRegistry.included(base) Relationships::BelongsTo.included(base) end # Determines if this instance is a new record. For the purposes of the SDK, # this really means "does the item already have an ID?" - because if i reconstitute # a known item without first retrieving it from the API, then saving that # object should push changes to the correct resource rather than creating a new # object. def new_record? (id.to_i > 0)? false : true end # Saves the record, creating it if is new and updating it if it is not. # Returns TRUE on success and FALSE on fail. def save proxy = proxy_for(config) param_key = self.class.parameter_name response = if new_record? proxy.post(resource_path, param_key=>returnable_attributes) else proxy.put(resource_path, param_key=>returnable_attributes) end # Parse and handle response return validate_committed_response(response) end # Attempts to validate this object with the remote service. Returns TRUE on success, # and returns FALSE on failure. Failure will also populate the #errors object on # this instance. def valid? proxy = proxy_for(config) param_key = self.class.parameter_name response = if new_record? proxy.post(resource_path(:validate, :nested=>false), param_key=>returnable_attributes) else proxy.put(resource_path(:validate, :nested=>false), param_key=>returnable_attributes) end # Parse and handle response return validate_response(response) end # The hash of errors on this object - keyed by attribute name, # with the error description as the value. def errors=(arg); @errors = Videojuicer::Resource::Errors.new(arg); end def errors; @errors ||= Videojuicer::Resource::Errors.new({}); end def errors_on(key); errors.on(key); end # Takes a *change-making* response from the API and cleans the dirty attributes if it was successful. def validate_committed_response(response) result = validate_response(response) clean_dirty_attributes! if result return result end # Takes a response from the API and performs the appropriate actions. def validate_response(response) body = response.body attribs = (body.is_a?(Hash))? body : JSON.parse(body) rescue raise(JSON::ParserError, "Could not parse #{body}: \n\n #{body}") attribs.each do |prop, value| next if (prop == "id") and (value.to_i < 1) self.send("#{prop}=", value) rescue next end if e = attribs["errors"] self.errors = e return false else self.id = attribs["id"].to_i self.errors = {} return true end end # Updates the attributes and saves the record in one go. def update_attributes(attrs) self.attributes = attrs return save end # Makes a call to the API for the current attributes on this object. # Overwrites the current instance attribute values. def reload raise NoResource, "Cannot load remote attributes for new records" if new_record? response = proxy_for(config).get(resource_path(nil, :nested=>false)) return validate_committed_response(response) end # Attempts to delete this record. Will raise an exception if the record is new # or if it has already been deleted. def destroy proxy_for(config).delete(resource_path) end module ClassMethods # Finds all objects matching the criteria. Also allows filterable methods. # Use :limit to throttle the amount of records returned. # Use :offset to return all objects after a certain index. Combine with :limit for pagination. # You may specify comparators on attributes by giving them in the following forms: # > "id"=>"9" #=> Returns only records with ID 9 # > "id.gt" => "9" #=> Returns only records with ID greater than 9 # > "id.lt" => "9" #=> Returns only records with ID less than 9 # > "id.gte" => "9" #=> Returns only records with ID greater than or equal to 9 # > "id.lte" => "9" #=> Returns only records with ID less than or equal to than 9 # Range operations are also supported e.g. # > :id => '1,2,3' def all(options={}) # Get a proxy options = (options.empty?)? {} : {parameter_name=>options} # FIXME this is a hacky workaround for singleton scope. response = instance_proxy.get(base_path(:nested=>false), options) op = JSON.parse(response.body) items = (op["items"] rescue op) # If "items" is on the returned object then this is a collection hash. # Instantiate objects items = items.collect do |attrs| o = new; o.attributes = attrs o.clean_dirty_attributes! o end return (op.is_a?(Hash))? Videojuicer::Resource::Collection.new(items, op["count"], op["offset"], op["limit"]) : items end def first(options={}) all(options.merge(:limit=>1)).first end def create(attrs={}) o = new(attrs) o.save return o end # Fetches an object given an ID. Straight forward. def get(id) o = new(:id=>id) begin o.reload o rescue Videojuicer::Exceptions::NoResource nil end end def destroy(id) o = new(:id=>id) o.destroy end def instance_proxy o = new o.proxy_for(o.config) end end end end