Skip to content

Commit

Permalink
Initial Ractor support
Browse files Browse the repository at this point in the history
  • Loading branch information
sandlerr committed Dec 29, 2022
1 parent cc3fb2e commit e116ba5
Show file tree
Hide file tree
Showing 7 changed files with 108 additions and 1 deletion.
3 changes: 3 additions & 0 deletions ext/sqlite3/extconf.rb
Original file line number Diff line number Diff line change
Expand Up @@ -108,6 +108,9 @@ def configure_extension
# Functions defined in 2.1 but not 2.0
have_func('rb_integer_pack')

# Functions defined in 3.0 but not 2.7
have_func('rb_ext_ractor_safe')

# These functions may not be defined
have_func('sqlite3_initialize')
have_func('sqlite3_backup_init')
Expand Down
5 changes: 5 additions & 0 deletions ext/sqlite3/sqlite3.c
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,11 @@ static VALUE threadsafe_p(VALUE UNUSED(klass))

void init_sqlite3_constants()
{
#ifdef HAVE_RB_EXT_RACTOR_SAFE
if (sqlite3_threadsafe()) {
rb_ext_ractor_safe(true);
}
#endif
VALUE mSqlite3Constants;
VALUE mSqlite3Open;

Expand Down
3 changes: 3 additions & 0 deletions lib/sqlite3.rb
Original file line number Diff line number Diff line change
Expand Up @@ -12,4 +12,7 @@
module SQLite3
# Was sqlite3 compiled with thread safety on?
def self.threadsafe?; threadsafe > 0; end

# Is the gem's C extension marked as Ractor-safe?
def self.ractor_safe?; threadsafe? && !defined?(Ractor).nil?; end
end
9 changes: 8 additions & 1 deletion lib/sqlite3/database.rb
Original file line number Diff line number Diff line change
Expand Up @@ -724,7 +724,14 @@ def translate_from_db types, row

private

NULL_TRANSLATOR = lambda { |_, row| row }
# NULL_TRANSLATOR used to be a lambda, but a lambda can't be frozen (properly)
# and so can't work with ractors.
class NullTranslatorImplementation
def self.call(_, row)
row
end
end
NULL_TRANSLATOR = NullTranslatorImplementation

def make_type_translator should_translate
if should_translate
Expand Down
1 change: 1 addition & 0 deletions sqlite3.gemspec
Original file line number Diff line number Diff line change
Expand Up @@ -84,6 +84,7 @@ Gem::Specification.new do |s|
"test/test_integration_aggregate.rb",
"test/test_integration_open_close.rb",
"test/test_integration_pending.rb",
"test/test_integration_ractor.rb",
"test/test_integration_resultset.rb",
"test/test_integration_statement.rb",
"test/test_result_set.rb",
Expand Down
1 change: 1 addition & 0 deletions test/helper.rb
Original file line number Diff line number Diff line change
Expand Up @@ -9,6 +9,7 @@
puts "info: sqlite3 version: #{SQLite3::SQLITE_VERSION}/#{SQLite3::SQLITE_LOADED_VERSION}"
puts "info: sqlcipher?: #{SQLite3.sqlcipher?}"
puts "info: threadsafe?: #{SQLite3.threadsafe?}"
puts "info: ractor_safe?: #{SQLite3.ractor_safe?}"

unless RUBY_VERSION >= "1.9"
require 'iconv'
Expand Down
87 changes: 87 additions & 0 deletions test/test_integration_ractor.rb
Original file line number Diff line number Diff line change
@@ -0,0 +1,87 @@
# frozen_string_literal: true

require 'helper'
require 'fileutils'

class TC_Integration_Ractor < SQLite3::TestCase
STRESS_DB_NAME = "stress.db"

def setup
teardown
end

def teardown
FileUtils.rm_rf(Dir.glob "#{STRESS_DB_NAME}*")
end

def test_ractor_safe
skip unless RUBY_VERSION >= '3.0' && SQLite3.threadsafe?
assert SQLite3.ractor_safe?
end

def test_ractor_share_database
skip('Requires Ruby with Ractors') unless SQLite3.ractor_safe?

db_receiver = Ractor.new do
db = Ractor.receive
Ractor.yield db.object_id
begin
db.execute("create table test_table ( b integer primary key)")
raise "Should have raised an exception in db.execute()"
rescue => e
Ractor.yield e
end
end
db_creator = Ractor.new(db_receiver) do |db_receiver|
db = SQLite3::Database.open(":memory:")
Ractor.yield db.object_id
db_receiver.send(db)
sleep 0.1
db.execute("create table test_table ( a integer primary key)")
end
first_oid = db_creator.take
second_oid = db_receiver.take
assert_not_equal first_oid, second_oid
ex = db_receiver.take
# For now, let's assert that you can't pass database connections around
# between different Ractors. Letting a live DB connection exist in two
# threads that are running concurrently might expose us to footguns and
# lead to data corruption, so we should avoid this possibility and wait
# until connections can be given away using `yield` or `send`.
assert_equal "prepare called on a closed database", ex.message
end

def test_ractor_stress
skip('Requires Ruby with Ractors') unless SQLite3.ractor_safe?

# Testing with a file instead of :memory: since it can be more realistic
# compared with real production use, and so discover problems that in-
# memory testing won't find. Trivial example: STRESS_DB_NAME needs to be
# frozen to pass into the Ractor, but :memory: might avoid that problem by
# using a literal string.
db = SQLite3::Database.open(STRESS_DB_NAME)
db.execute("PRAGMA journal_mode=WAL") # A little slow without this
db.execute("create table stress_test (a integer primary_key, b text)")
random = Random.new.freeze
ractors = (0..9).map do |ractor_number|
Ractor.new(random, ractor_number) do |random, ractor_number|
db_in_ractor = SQLite3::Database.open(STRESS_DB_NAME)
db_in_ractor.busy_handler do
sleep random.rand / 100 # Lots of busy errors happen with multiple concurrent writers
true
end
100.times do |i|
db_in_ractor.execute("insert into stress_test(a, b) values (#{ractor_number * 100 + i}, '#{random.rand}')")
end
end
end
ractors.each {|r| r.take}
final_check = Ractor.new do
db_in_ractor = SQLite3::Database.open(STRESS_DB_NAME)
res = db_in_ractor.execute("select count(*) from stress_test")
Ractor.yield res
end
res = final_check.take
assert_equal 1000, res[0][0]
end
end

0 comments on commit e116ba5

Please sign in to comment.