mirror of
https://gitea.invidious.io/iv-org/shard-crystal-db.git
synced 2024-08-15 00:53:32 +00:00
#5 remove read_object. allow read / read? to receive any type
This commit is contained in:
parent
0daa1c18d7
commit
9f3f0a9836
2 changed files with 74 additions and 31 deletions
|
@ -1,10 +1,7 @@
|
||||||
require "./spec_helper"
|
require "./spec_helper"
|
||||||
|
|
||||||
class GenericResultSet(T) < DB::ResultSet
|
module GenericResultSet
|
||||||
def initialize(statement, @row : Array(T))
|
|
||||||
super(statement)
|
|
||||||
@index = 0
|
@index = 0
|
||||||
end
|
|
||||||
|
|
||||||
def move_next
|
def move_next
|
||||||
@index = 0
|
@index = 0
|
||||||
|
@ -26,11 +23,11 @@ class GenericResultSet(T) < DB::ResultSet
|
||||||
{% for t in DB::TYPES %}
|
{% for t in DB::TYPES %}
|
||||||
# Reads the next column as a nillable {{t}}.
|
# Reads the next column as a nillable {{t}}.
|
||||||
def read?(t : {{t}}.class) : {{t}}?
|
def read?(t : {{t}}.class) : {{t}}?
|
||||||
read_object as {{t}}?
|
read_and_move_next_column as {{t}}?
|
||||||
end
|
end
|
||||||
{% end %}
|
{% end %}
|
||||||
|
|
||||||
def read_object
|
def read_and_move_next_column
|
||||||
@index += 1
|
@index += 1
|
||||||
@row[@index - 1]
|
@row[@index - 1]
|
||||||
end
|
end
|
||||||
|
@ -69,13 +66,25 @@ class FooDriver < DB::Driver
|
||||||
|
|
||||||
class FooStatement < DB::Statement
|
class FooStatement < DB::Statement
|
||||||
protected def perform_query(args : Enumerable) : DB::ResultSet
|
protected def perform_query(args : Enumerable) : DB::ResultSet
|
||||||
GenericResultSet(Any).new(self, FooDriver.fake_row)
|
FooResultSet.new(self, FooDriver.fake_row)
|
||||||
end
|
end
|
||||||
|
|
||||||
protected def perform_exec(args : Enumerable) : DB::ExecResult
|
protected def perform_exec(args : Enumerable) : DB::ExecResult
|
||||||
DB::ExecResult.new 0, 0i64
|
DB::ExecResult.new 0, 0i64
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class FooResultSet < DB::ResultSet
|
||||||
|
include GenericResultSet
|
||||||
|
|
||||||
|
def initialize(statement, @row : Array(FooDriver::Any))
|
||||||
|
super(statement)
|
||||||
|
end
|
||||||
|
|
||||||
|
def read?(t : FooValue.class) : FooValue?
|
||||||
|
read_and_move_next_column.as(FooValue?)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
DB.register_driver "foo", FooDriver
|
DB.register_driver "foo", FooDriver
|
||||||
|
@ -111,13 +120,25 @@ class BarDriver < DB::Driver
|
||||||
|
|
||||||
class BarStatement < DB::Statement
|
class BarStatement < DB::Statement
|
||||||
protected def perform_query(args : Enumerable) : DB::ResultSet
|
protected def perform_query(args : Enumerable) : DB::ResultSet
|
||||||
GenericResultSet(Any).new(self, BarDriver.fake_row)
|
BarResultSet.new(self, BarDriver.fake_row)
|
||||||
end
|
end
|
||||||
|
|
||||||
protected def perform_exec(args : Enumerable) : DB::ExecResult
|
protected def perform_exec(args : Enumerable) : DB::ExecResult
|
||||||
DB::ExecResult.new 0, 0i64
|
DB::ExecResult.new 0, 0i64
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
class BarResultSet < DB::ResultSet
|
||||||
|
include GenericResultSet
|
||||||
|
|
||||||
|
def initialize(statement, @row : Array(BarDriver::Any))
|
||||||
|
super(statement)
|
||||||
|
end
|
||||||
|
|
||||||
|
def read?(t : BarValue.class) : BarValue?
|
||||||
|
read_and_move_next_column.as(BarValue?)
|
||||||
|
end
|
||||||
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
DB.register_driver "bar", BarDriver
|
DB.register_driver "bar", BarDriver
|
||||||
|
@ -138,7 +159,7 @@ describe DB do
|
||||||
rs.move_next
|
rs.move_next
|
||||||
rs.read?(Int32).should eq(1)
|
rs.read?(Int32).should eq(1)
|
||||||
rs.read?(String).should eq("string")
|
rs.read?(String).should eq("string")
|
||||||
(rs.read_object.as(FooValue)).value.should eq(3)
|
rs.read(FooValue).value.should eq(3)
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
@ -150,7 +171,7 @@ describe DB do
|
||||||
db.query "query" do |rs|
|
db.query "query" do |rs|
|
||||||
w.check
|
w.check
|
||||||
rs.move_next
|
rs.move_next
|
||||||
(rs.read_object.as(BarValue)).value.should eq(4)
|
rs.read(BarValue).value.should eq(4)
|
||||||
rs.read?(String).should eq("lorem")
|
rs.read?(String).should eq("lorem")
|
||||||
rs.read?(Float64).should eq(1.0)
|
rs.read?(Float64).should eq(1.0)
|
||||||
end
|
end
|
||||||
|
@ -158,6 +179,34 @@ describe DB do
|
||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
|
it "Foo and Bar drivers should not implement each other read" do
|
||||||
|
with_witness do |w|
|
||||||
|
DB.open("foo://host") do |db|
|
||||||
|
FooDriver.fake_row = [1] of FooDriver::Any
|
||||||
|
db.query "query" do |rs|
|
||||||
|
rs.move_next
|
||||||
|
expect_raises Exception, "read?(t : BarValue) is not implemented in FooDriver::FooResultSet" do
|
||||||
|
w.check
|
||||||
|
rs.read(BarValue)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
with_witness do |w|
|
||||||
|
DB.open("bar://host") do |db|
|
||||||
|
BarDriver.fake_row = [1] of BarDriver::Any
|
||||||
|
db.query "query" do |rs|
|
||||||
|
rs.move_next
|
||||||
|
expect_raises Exception, "read?(t : FooValue) is not implemented in BarDriver::BarResultSet" do
|
||||||
|
w.check
|
||||||
|
rs.read(FooValue)
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
it "allow custom types to be used as arguments for query" do
|
it "allow custom types to be used as arguments for query" do
|
||||||
DB.open("foo://host") do |db|
|
DB.open("foo://host") do |db|
|
||||||
FooDriver.fake_row = [1, "string"] of FooDriver::Any
|
FooDriver.fake_row = [1, "string"] of FooDriver::Any
|
||||||
|
|
|
@ -16,14 +16,10 @@ module DB
|
||||||
# ### Note to implementors
|
# ### Note to implementors
|
||||||
#
|
#
|
||||||
# 1. Override `#move_next` to move to the next row.
|
# 1. Override `#move_next` to move to the next row.
|
||||||
# 2. Override `#read?(t)` for all `t` in `DB::TYPES`.
|
# 2. Override `#read?(t)` for all `t` in `DB::TYPES` and any other types the driver should handle.
|
||||||
# 3. (Optional) Override `#read(t)` for all `t` in `DB::TYPES`.
|
# 3. (Optional) Override `#read(t)` for all `t` in `DB::TYPES` and any other.
|
||||||
# 4. Override `#column_count`, `#column_name`.
|
# 4. Override `#column_count`, `#column_name`.
|
||||||
# 5. Override `#column_type`. It must return a type in `DB::TYPES`.
|
# 5. Override `#column_type`. It must return a type in `DB::TYPES`.
|
||||||
# 6. Override `#read_object` to return other data types not included in `DB::TYPES`. This
|
|
||||||
# will create a union type, so user will be forced to cast result type. Usually `#read`
|
|
||||||
# should be used to avoid unnecesary intermediate union type values. Calling `#read_object`
|
|
||||||
# should also move to the next column.
|
|
||||||
abstract class ResultSet
|
abstract class ResultSet
|
||||||
include Disposable
|
include Disposable
|
||||||
|
|
||||||
|
@ -63,28 +59,26 @@ module DB
|
||||||
# The result is one of `DB::TYPES`.
|
# The result is one of `DB::TYPES`.
|
||||||
abstract def column_type(index : Int32)
|
abstract def column_type(index : Int32)
|
||||||
|
|
||||||
# list datatypes that must be supported form the driver
|
def read(t)
|
||||||
# users will call read(String) or read?(String) for nillables
|
read?(t).not_nil!
|
||||||
|
|
||||||
{% for t in DB::TYPES %}
|
|
||||||
# Reads the next column as a nillable {{t}}.
|
|
||||||
abstract def read?(t : {{t}}.class) : {{t}}?
|
|
||||||
|
|
||||||
# Reads the next column as a {{t}}.
|
|
||||||
def read(t : {{t}}.class) : {{t}}
|
|
||||||
read?({{t}}).not_nil!
|
|
||||||
end
|
end
|
||||||
{% end %}
|
|
||||||
|
|
||||||
# Reads the next column as a Nil.
|
# Reads the next column as a Nil.
|
||||||
def read(t : Nil.class) : Nil
|
def read(t : Nil.class) : Nil
|
||||||
read?(Nil)
|
read?(Nil)
|
||||||
end
|
end
|
||||||
|
|
||||||
def read_object
|
def read?(t)
|
||||||
raise "Not implemented"
|
raise "read?(t : #{t}) is not implemented in #{self.class}"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
# list datatypes that must be supported form the driver
|
||||||
|
# users will call read(String) or read?(String) for nillables
|
||||||
|
{% for t in DB::TYPES %}
|
||||||
|
# Reads the next column as a nillable {{t}}.
|
||||||
|
abstract def read?(t : {{t}}.class) : {{t}}?
|
||||||
|
{% end %}
|
||||||
|
|
||||||
# def read_blob
|
# def read_blob
|
||||||
# yield ... io ....
|
# yield ... io ....
|
||||||
# end
|
# end
|
||||||
|
|
Loading…
Reference in a new issue