module QBFC class QuickbooksClosedError < RuntimeError #:nodoc: end class SetValueMissing < RuntimeError#:nodoc: end class QBXMLVersionError < RuntimeError#:nodoc: end class UnknownRequestError < RuntimeError#:nodoc: end class BaseClassNewError < RuntimeError#:nodoc: end class NotSavableError < RuntimeError#:nodoc: end # Encapsulates a QBFC session. # # QBFC::Session.open(:app_name => 'Test Application') do |qb| # qb.customers.find(:all).each do |customer| # puts customer.full_name # end # end # # qb = QBFC::Session.new(:app_name => 'Test Application') # qb.customers.find(:all).each do |customer| # puts customer.full_name # end # qb.close # # A QBFC::Session abstracts the ole_methods so that more conventional Ruby method names are used, # e.g. full_name instead of FullName.GetValue(). # # This also allows a shortcut for setting up Quickbooks objects: # # - session.customers.find(:all) instead of QBFC::Customer.find(session, :all) # - session.customer('CustomerFullName') instead of QBFC::Customer.find(session, 'CustomerFullName') # - session.customer instead of QBFC::Customer.find(session, :first) class Session class << self # Open a QBFC session. Takes options as a hash, and an optional block. Options are: # # - +app_name+: Name that the application sends to Quickbooks (used for allowing/denying access) # (defaults to 'Ruby QBFC Application' # - +app_id+: Per the Quickbooks SDK (QBFC Language Reference): # 'Normally not assigned. Use an empty string for appID.' An empty string is passed by default. # - +conn_type+: QBFC_CONST::CtUnknown, CtLocalQBD, CtRemoteQBD, CtLocalQBDLaunchUI, or CtRemoteQBOE. # Default is QBFC_CONST::CtLocalQBD (1) # - +filename+: The full path to the Quickbooks file; leave blank to connect to the currently # open company file. Default is an empty string (Quickbooks should be running). # - +open_mode+: The desired access mode. It can be one of three values: # - QBFC_CONST::OmSingleUser (specifies single-user mode) # - QBFC_CONST::OmMultiUser (specifies multi-user mode) # - QBFC_CONST::OmDontCare (accept whatever mode is currently in effect, or single-user mode if no other mode is in effect) # Default is QBFC_CONST::OmDontCare # # If given a block, it yields the Session object and closes the Session and Connection # when the block closes. # # Otherwise, it returns the new Session object. def open(*options, &block) qb = new(*options) if block_given? begin yield qb ensure qb.close end else qb end end end # See Session.open for initialization options. def initialize(options = {}) ole_object = WIN32OLE.new("QBFC6.QBSessionManager") ole_object.OpenConnection2(options[:app_id].to_s, (options[:app_name] || "Ruby QBFC Application"), (options[:conn_type] || QBFC_CONST::CtLocalQBD)) begin ole_object.BeginSession(options[:filename].to_s, (options[:open_mode] || QBFC_CONST::OmDontCare)) rescue WIN32OLERuntimeError ole_object.CloseConnection ole_object = nil raise(QBFC::QuickbooksClosedError, "BeginSession failed: Quickbooks must be open or a valid filename specified.\n\n#{$!}") end @ole_object = QBFC::OLEWrapper.new(ole_object) end # Close the session with Quickbooks. If this is ommitted, Quickbooks will not close. # Using a block with Session.open ensures the session is closed. def close @ole_object.EndSession @ole_object.CloseConnection @ole_object = nil end # Generate a report with the given +name+ and +args+. # (See QBFC::Report.get for details). def report(name, *args) Report.get(self, name, *args) end # The classes method allows using session.classes.find instead of # session.q_b_classes.find, for finds on QBClass. def classes QBFC::QBCollection.new(self, :QBClass) end # Responsible for the conversion of ole_method name to more convential Ruby method names. # This specifies the methods for setting up an Entity, such as a Customer, directly, which is # not included in OLEWrapper (setting up entities that are children of another entity is). # Send other missing methods on to OLE Wrapper def method_missing(symbol, *params) #:nodoc: if (('a'..'z') === symbol.to_s[0].chr && symbol.to_s[-1].chr != '=') camelized_method = symbol.to_s.camelize.to_sym if camelized_method.to_s =~ /Terms\Z/ single_camelized_method = camelized_method else single_camelized_method = symbol.to_s.singularize.camelize.to_sym end if QBFC.const_defined?(camelized_method) && camelized_method.to_s !~ /Terms\Z/ if params[0] return QBFC::const_get(camelized_method).find_by_unique_id(self, params[0]) else return QBFC::const_get(camelized_method).find(self, :first) end elsif QBFC.const_defined?(single_camelized_method) return QBFC::QBCollection.new(self, single_camelized_method) end end # Don't want to pass an OLEWrapper to a WIN32OLE method. params = params.collect{ |p| p.respond_to?(:ole_object) ? p.ole_object : p } @ole_object.qbfc_method_missing(self, symbol, *params) end end end