2007-01-22 07:43:50 -06:00
|
|
|
require 'sqlite3/constants'
|
|
|
|
require 'sqlite3/errors'
|
|
|
|
|
|
|
|
module SQLite3
|
|
|
|
|
|
|
|
# The ResultSet object encapsulates the enumerability of a query's output.
|
|
|
|
# It is a simple cursor over the data that the query returns. It will
|
|
|
|
# very rarely (if ever) be instantiated directly. Instead, client's should
|
|
|
|
# obtain a ResultSet instance via Statement#execute.
|
|
|
|
class ResultSet
|
|
|
|
include Enumerable
|
|
|
|
|
2008-12-15 14:45:15 -06:00
|
|
|
# The class of which we return an object in case we want an Array as
|
|
|
|
# result. (ArrayFields is installed.)
|
|
|
|
class ArrayWithTypes < Array
|
2007-01-22 07:43:50 -06:00
|
|
|
attr_accessor :types
|
|
|
|
end
|
|
|
|
|
2008-12-15 14:45:15 -06:00
|
|
|
# The class of which we return an object in case we want an Array as
|
|
|
|
# result. (ArrayFields is not installed.)
|
|
|
|
class ArrayWithTypesAndFields < Array
|
|
|
|
attr_accessor :types
|
2007-01-22 07:43:50 -06:00
|
|
|
attr_accessor :fields
|
|
|
|
end
|
|
|
|
|
2008-12-15 14:45:15 -06:00
|
|
|
# The class of which we return an object in case we want a Hash as
|
|
|
|
# result.
|
|
|
|
class HashWithTypes < Hash
|
|
|
|
attr_accessor :types
|
|
|
|
end
|
|
|
|
|
2007-01-22 07:43:50 -06:00
|
|
|
# Create a new ResultSet attached to the given database, using the
|
|
|
|
# given sql text.
|
2010-06-10 22:42:33 -05:00
|
|
|
def initialize db, stmt
|
|
|
|
@db = db
|
2007-01-22 07:43:50 -06:00
|
|
|
@stmt = stmt
|
|
|
|
end
|
|
|
|
|
|
|
|
# Reset the cursor, so that a result set which has reached end-of-file
|
|
|
|
# can be rewound and reiterated.
|
|
|
|
def reset( *bind_params )
|
2010-06-10 22:42:33 -05:00
|
|
|
@stmt.reset!
|
2007-01-22 07:43:50 -06:00
|
|
|
@stmt.bind_params( *bind_params )
|
|
|
|
@eof = false
|
|
|
|
end
|
|
|
|
|
|
|
|
# Query whether the cursor has reached the end of the result set or not.
|
|
|
|
def eof?
|
2010-06-10 22:42:33 -05:00
|
|
|
@stmt.done?
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
# Obtain the next row from the cursor. If there are no more rows to be
|
|
|
|
# had, this will return +nil+. If type translation is active on the
|
|
|
|
# corresponding database, the values in the row will be translated
|
|
|
|
# according to their types.
|
|
|
|
#
|
|
|
|
# The returned value will be an array, unless Database#results_as_hash has
|
|
|
|
# been set to +true+, in which case the returned value will be a hash.
|
|
|
|
#
|
|
|
|
# For arrays, the column names are accessible via the +fields+ property,
|
|
|
|
# and the column types are accessible via the +types+ property.
|
|
|
|
#
|
|
|
|
# For hashes, the column names are the keys of the hash, and the column
|
|
|
|
# types are accessible via the +types+ property.
|
|
|
|
def next
|
2010-06-10 22:42:33 -05:00
|
|
|
row = @stmt.step
|
|
|
|
return nil if @stmt.done?
|
2007-01-22 07:43:50 -06:00
|
|
|
|
2010-06-10 22:42:33 -05:00
|
|
|
if @db.type_translation
|
|
|
|
row = @stmt.types.zip(row).map do |type, value|
|
|
|
|
@db.translator.translate( type, value )
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
2010-06-10 22:42:33 -05:00
|
|
|
end
|
2007-01-22 07:43:50 -06:00
|
|
|
|
2010-06-10 22:42:33 -05:00
|
|
|
if @db.results_as_hash
|
|
|
|
new_row = HashWithTypes[*@stmt.columns.zip(row).flatten]
|
|
|
|
row.each_with_index { |value,idx|
|
|
|
|
new_row[idx] = value
|
|
|
|
}
|
|
|
|
row = new_row
|
|
|
|
else
|
|
|
|
if row.respond_to?(:fields)
|
|
|
|
row = ArrayWithTypes.new(row)
|
2007-01-22 07:43:50 -06:00
|
|
|
else
|
2010-06-10 22:42:33 -05:00
|
|
|
row = ArrayWithTypesAndFields.new(row)
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
2010-06-10 22:42:33 -05:00
|
|
|
row.fields = @stmt.columns
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
|
|
|
|
2010-06-10 22:42:33 -05:00
|
|
|
row.types = @stmt.types
|
|
|
|
row
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
# Required by the Enumerable mixin. Provides an internal iterator over the
|
|
|
|
# rows of the result set.
|
2010-06-10 22:42:33 -05:00
|
|
|
def each( &block )
|
|
|
|
while node = self.next
|
|
|
|
yield node
|
2007-01-22 07:43:50 -06:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
# Closes the statement that spawned this result set.
|
|
|
|
# <em>Use with caution!</em> Closing a result set will automatically
|
|
|
|
# close any other result sets that were spawned from the same statement.
|
|
|
|
def close
|
|
|
|
@stmt.close
|
|
|
|
end
|
|
|
|
|
|
|
|
# Queries whether the underlying statement has been closed or not.
|
|
|
|
def closed?
|
|
|
|
@stmt.closed?
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the types of the columns returned by this result set.
|
|
|
|
def types
|
|
|
|
@stmt.types
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns the names of the columns returned by this result set.
|
|
|
|
def columns
|
|
|
|
@stmt.columns
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|
|
|
|
|
|
|
|
end
|