module TaskMapper::Provider module Base # This is the base Project class for providers # # Providers should inherit this class and redefine the methods # # Each provider should have their own @system defined. # For example, taskmapper-unfuddle's @system is :unfuddle and taskmapper-lighthouse's # @system is :lighthouse. # # Methods that must be implemented by the provider # # * self.find_by_id # * self.find_by_attributes # # Methods that might need to be implemented by the provider # * tickets # * ticket # * initialize # * update # * destroy # * self.create # # Methods that would probably be okay if the provider left it alone: # # * self.find - although you can define your own to optimize it a bit # * update! # # A provider should define as many attributes as feasibly possible. The list below are # some guidelines as to what attributes are necessary, if your provider's api does not # implement them, point it to an attribute that is close to it. (for example, a name # can point to title. Remember to alias it in your class!) # # * id # * name # * created_at # * updated_at # * description class Project < Hashie::Mash include TaskMapper::Provider::Common include TaskMapper::Provider::Helper extend TaskMapper::Provider::Helper attr_accessor :system, :system_data API = nil #Replace with your api digestor's class. # Find project # You can also retrieve an array of all projects by not specifying any query. # # * find() or find(:all) - Returns an array of all projects # * find(##) - Returns a project based on that id or some other primary (unique) attribute # * find([#, #, #]) - Returns many projects with these ids # * find(:first, :name => 'Project name') - Returns the first project based on the project's attribute(s) # * find(:last, :name => 'Some project') - Returns the last project based on the project's attribute(s) # * find(:all, :name => 'Test Project') - Returns all projects based on the given attribute(s) def self.find(*options) first = options.shift attributes = options.shift if first.nil? or (first == :all and attributes.nil?) self.find_by_attributes elsif first.is_a? Array first.collect { |id| self.find_by_id(id) } elsif first == :first projects = attributes.nil? ? self.find_by_attributes : self.find_by_attributes(attributes) projects.first elsif first == :last projects = attributes.nil? ? self.find_by_attributes : self.find_by_attributes(attributes) projects.last elsif first == :all self.find_by_attributes(attributes) else self.find_by_id(first) end end # The first of whatever project def self.first(*options) self.find(:first, *options) end # The last of whatever project def self.last(*options) self.find(:last, *options) end # Accepts an integer id and returns the single project instance # Must be defined by the provider def self.find_by_id(id) if self::API.is_a? Class self.new self::API.find(id) else raise TaskMapper::Exception.new("#{self.name}::#{this_method} method must be implemented by the provider") end end # Accepts an attributes hash and returns all projects matching those attributes in an array # Should return all projects if the attributes hash is empty # Must be defined by the provider def self.find_by_attributes(attributes = {}) if self::API.is_a? Class self.search(attributes) else raise TaskMapper::Exception.new("#{self.name}::#{this_method} method must be implemented by the provider") end end # This is a helper method to find def self.search(options = {}, limit = 1000) if self::API.is_a? Class projects = self::API.find(:all).collect { |project| self.new project } search_by_attribute(projects, options, limit) else raise TaskMapper::Exception.new("#{self.name}::#{this_method} method must be implemented by the provider") end end # Asks the provider's api for the tickets associated with the project, # returns an array of Ticket objects. def tickets(*options) options.insert 0, id easy_finder(provider_parent(self.class)::Ticket, :all, options, 1) end # Very similar to tickets, and is practically an alias of it # however this returns the ticket class if no parameter is given # unlike tickets which returns an array of all tickets when given no parameters def ticket(*options) options.insert(0, id) if options.length > 0 easy_finder(provider_parent(self.class)::Ticket, :first, options, 1) end # Create a ticket def ticket!(*options) options[0].merge!(:project_id => id) if options.first.is_a?(Hash) provider_parent(self.class)::Ticket.create(*options) end # Define some provider specific initalizations def initialize(*options) # @system_data = {'some' => 'data} super(*options) end end end end