Skip to content

Commit 0dcfc89

Browse files
committedFeb 3, 2014
database/sql: add WebSQL wrapping
1 parent 93aec20 commit 0dcfc89

File tree

2 files changed

+324
-0
lines changed

2 files changed

+324
-0
lines changed
 

‎opal/browser/database/sql.rb

+194
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,194 @@
1+
require 'promise'
2+
require 'ostruct'
3+
4+
module Browser; module Database
5+
6+
class SQL
7+
# Check if the browser supports WebSQL.
8+
def self.supported?
9+
Browser.supports? 'WebSQL'
10+
end
11+
12+
class Error < StandardError
13+
def self.new(error)
14+
return super if self != Error
15+
16+
[Unknown, Database, Version, TooLarge, Quota, Syntax, Constraint, Timeout] \
17+
[`error.code`].new(`error.message`)
18+
end
19+
20+
Unknown = Class.new(self)
21+
Database = Class.new(self)
22+
Version = Class.new(self)
23+
TooLarge = Class.new(self)
24+
Quota = Class.new(self)
25+
Syntax = Class.new(self)
26+
Constraint = Class.new(self)
27+
Timeout = Class.new(self)
28+
end
29+
30+
include Native
31+
32+
# @return [String] the name of the database
33+
attr_reader :name
34+
35+
# @return [String] the description for the database
36+
attr_reader :description
37+
38+
# @return [Integer] the size constraint in bytes
39+
attr_reader :size
40+
41+
# Open a database with the given name and options.
42+
#
43+
# @param name [String] the name for the database
44+
# @param options [Hash] options to open the database
45+
#
46+
# @option options [String] :description the description for the database
47+
# @option options [String] :version ('') the expected version of the database
48+
# @option options [Integer] :size (5 * 1024 * 1024) the size constraint in bytes
49+
def initialize(name, options = {})
50+
@name = name
51+
@description = options[:description] || name
52+
@version = options[:version] || ''
53+
@size = options[:size] || 5 * 1024 * 1024
54+
55+
super(`window.openDatabase(#{name}, #{@version}, #{@description}, #{@size})`)
56+
end
57+
58+
# @overload version()
59+
#
60+
# Get the version of the database.
61+
#
62+
# @return [String]
63+
#
64+
# @overload version(from, to, &block)
65+
#
66+
# Migrate the database to a new version.
67+
#
68+
# @param from [String] the version you're migrating from
69+
# @param to [String] the version you're migrating to
70+
#
71+
# @yieldparam transaction [Transaction] the transaction to work with
72+
def version(from = nil, to = nil, &block)
73+
return `#@native.version` unless block
74+
75+
`#@native.changeVersion(#{from}, #{to},
76+
#{-> t { block.call(Transaction.new(self, t)) }})`
77+
end
78+
79+
# Start a transaction on the database.
80+
#
81+
# @yieldparam transaction [Transaction] the transaction to work on
82+
def transaction(&block)
83+
raise ArgumentError, 'no block given' unless block
84+
85+
`#@native.transaction(#{-> t { block.call(Transaction.new(self, t)) }})`
86+
end
87+
88+
# Allows you to make changes to the database or read data from it.
89+
class Transaction
90+
include Native
91+
92+
# @return [Database] the database the transaction has been created from
93+
attr_reader :database
94+
95+
# @private
96+
def initialize(database, transaction)
97+
@database = database
98+
99+
super(transaction)
100+
end
101+
102+
# Query the database.
103+
#
104+
# @param query [String] the SQL query to send
105+
# @param parameters [Array] optional bind parameters for the query
106+
#
107+
# @return [Promise]
108+
def query(query, *parameters)
109+
promise = Promise.new
110+
111+
`#@native.executeSql(#{query}, #{parameters},
112+
#{-> _, r { promise.resolve(Result.new(self, r)) }},
113+
#{-> _, e { promise.reject(Error.new(e)) }})`
114+
115+
promise
116+
end
117+
end
118+
119+
# Represents a row.
120+
class Row < OpenStruct
121+
# @private
122+
def initialize(row)
123+
super(Hash.new(row))
124+
end
125+
126+
def inspect
127+
"#<SQL::Row: #{Hash.new(@native).inspect}>"
128+
end
129+
end
130+
131+
class Result
132+
include Native
133+
134+
# @return [Transaction] the transaction the result came from
135+
attr_reader :transaction
136+
137+
# @return [SQL] the database the result came from
138+
attr_reader :database
139+
140+
# @private
141+
def initialize(transaction, result)
142+
@transaction = transaction
143+
@database = transaction.database
144+
145+
super(result)
146+
end
147+
148+
include Enumerable
149+
150+
# Get a row from the result.
151+
#
152+
# @param index [Integer] the index for the row
153+
#
154+
# @return [Row]
155+
def [](index)
156+
if index < 0
157+
index += length
158+
end
159+
160+
unless index < 0 || index >= length
161+
Row.new(`#@native.rows.item(index)`)
162+
end
163+
end
164+
165+
# Enumerate over the rows.
166+
#
167+
# @yieldparam row [Row]
168+
#
169+
# @return [self]
170+
def each(&block)
171+
return enum_for :each unless block
172+
173+
%x{
174+
for (var i = 0, length = #@native.rows.length; i < length; i++) {
175+
#{block.call(self[`i`])};
176+
}
177+
}
178+
179+
self
180+
end
181+
182+
# @!attribute [r] length
183+
# @return [Integer] number of rows in the result
184+
def length
185+
`#@native.rows.length`
186+
end
187+
188+
# @!attribute [r] affected
189+
# @return [Integer] number of affected rows
190+
alias_native :affected, :rowsAffected
191+
end
192+
end
193+
194+
end; end

‎spec/database/sql_spec.rb

+130
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,130 @@
1+
require 'spec_helper'
2+
require 'browser/database/sql'
3+
4+
describe Browser::Database::SQL do
5+
SQL = Browser::Database::SQL
6+
7+
describe '#new' do
8+
it 'sets the attributes properly' do
9+
sql = SQL.new('test', description: 'trains', size: 1024)
10+
11+
expect(sql.name).to eq('test')
12+
expect(sql.description).to eq('trains')
13+
expect(sql.size).to eq(1024)
14+
end
15+
16+
it 'sets the version properly' do
17+
sql = SQL.new('test2', version: '1.0')
18+
expect(sql.version).to eq('1.0')
19+
20+
sql = SQL.new('test')
21+
expect(sql.version).to eq('')
22+
end
23+
end
24+
25+
describe '#transaction' do
26+
async 'calls the block with the transaction' do
27+
sql = SQL.new('test')
28+
29+
sql.transaction {|t|
30+
async {
31+
expect(t).to be_a(SQL::Transaction)
32+
}
33+
}
34+
end
35+
36+
async 'the transaction database is the right one' do
37+
sql = SQL.new('test')
38+
39+
sql.transaction {|t|
40+
async {
41+
expect(t.database).to eq(sql)
42+
}
43+
}
44+
end
45+
end
46+
47+
describe SQL::Transaction do
48+
describe '#query' do
49+
async 'returns a promise' do
50+
sql = SQL.new('test')
51+
52+
sql.transaction {|t|
53+
async {
54+
expect(t.query('hue')).to be_a(Promise)
55+
}
56+
}
57+
end
58+
59+
async 'resolves on success' do
60+
sql = SQL.new('test')
61+
62+
sql.transaction {|t|
63+
t.query('CREATE TABLE IF NOT EXISTS test(ID INTEGER PRIMARY KEY ASC, a TEXT)').then {|r|
64+
async {
65+
expect(r).to be_a(SQL::Result)
66+
}
67+
}
68+
}
69+
end
70+
71+
async 'rejects on failure' do
72+
sql = SQL.new('test')
73+
74+
sql.transaction {|t|
75+
t.query('huehue').rescue {|e|
76+
async {
77+
expect(e).to be_a(SQL::Error::Syntax)
78+
}
79+
}
80+
}
81+
end
82+
end
83+
end
84+
85+
describe SQL::Result do
86+
describe '#length' do
87+
async 'has the proper length' do
88+
sql = SQL.new('test')
89+
90+
sql.transaction {|t|
91+
t.query('SELECT 1').then {|r|
92+
async {
93+
expect(r.length).to eq(1)
94+
}
95+
}
96+
}
97+
end
98+
end
99+
100+
describe '#[]' do
101+
async 'returns a row' do
102+
sql = SQL.new('test')
103+
104+
sql.transaction {|t|
105+
t.query('SELECT 1, 2, 3').then {|r|
106+
async {
107+
expect(r[0]).to be_a(SQL::Row)
108+
expect(r[-1]).to be_a(SQL::Row)
109+
110+
expect(r[0]).to eq(r[-1])
111+
}
112+
}
113+
}
114+
end
115+
116+
async 'returns nil on missing row' do
117+
sql = SQL.new('test')
118+
119+
sql.transaction {|t|
120+
t.query('SELECT 1, 2, 3').then {|r|
121+
async {
122+
expect(r[5]).to be_nil
123+
expect(r[-5]).to be_nil
124+
}
125+
}
126+
}
127+
end
128+
end
129+
end
130+
end if Browser::Database::SQL.supported?

0 commit comments

Comments
 (0)
Please sign in to comment.