Skip to content

Commit

Permalink
update to include libsql if-defs to make continuous sync trivial
Browse files Browse the repository at this point in the history
  • Loading branch information
tantaman committed Nov 10, 2023
1 parent 165401f commit 680d1ce
Show file tree
Hide file tree
Showing 8 changed files with 116 additions and 29 deletions.
5 changes: 3 additions & 2 deletions libsql-sqlite3/ext/crr/Makefile
Original file line number Diff line number Diff line change
Expand Up @@ -4,8 +4,9 @@ else
CC:=$(CI_GCC)
endif

LOADABLE_CFLAGS=-std=c99 -fPIC -shared -Wall
STATIC_CFLAGS=-std=c99 -fPIC -c -Wall
# SHARED_CFLAGS=-DLIBSQL=1
LOADABLE_CFLAGS=-std=c99 -fPIC -shared -Wall $(SHARED_CFLAGS)
STATIC_CFLAGS=-std=c99 -fPIC -c -Wall $(SHARED_CFLAGS)

ifeq ($(shell uname -s),Darwin)
CONFIG_DARWIN=y
Expand Down
28 changes: 22 additions & 6 deletions libsql-sqlite3/ext/crr/rs/core/src/changes_vtab_write.rs
Original file line number Diff line number Diff line change
Expand Up @@ -28,12 +28,14 @@ use crate::util::slab_rowid;
*/
fn did_cid_win(
db: *mut sqlite3,
ext_data: *mut crsql_ExtData,
insert_tbl: &str,
tbl_info: &TableInfo,
unpacked_pks: &Vec<ColumnValue>,
key: sqlite::int64,
col_name: &str,
insert_val: *mut sqlite::value,
insert_site_id: &[u8],
col_name: &str,
col_version: sqlite::int64,
errmsg: *mut *mut c_char,
) -> Result<bool, ResultCode> {
Expand Down Expand Up @@ -76,9 +78,19 @@ fn did_cid_win(
}

// versions are equal
// need to pull the current value and compare
// we could compare on site_id if we can guarantee site_id is always provided.
// would be slightly more performant..
// need to compare site ids
let ret = unsafe {
let my_site_id = core::slice::from_raw_parts((*ext_data).siteId, 16);
insert_site_id.cmp(my_site_id) as c_int
};

// site id lost.
if ret <= 0 {
return Ok(false);
}

// site id won
// last thing, compare values to ensure we're not changing state on equal values
let col_val_stmt_ref = tbl_info.get_col_value_stmt(db, col_name)?;
let col_val_stmt = col_val_stmt_ref.as_ref().ok_or(ResultCode::ERROR)?;

Expand All @@ -94,7 +106,9 @@ fn did_cid_win(
let local_value = col_val_stmt.column_value(0)?;
let ret = crsql_compare_sqlite_values(insert_val, local_value);
reset_cached_stmt(col_val_stmt.stmt)?;
return Ok(ret > 0);
// insert site id won and values differ. We should take the update.
// if values are the same (ret == 0) then we return false and do not take the update
return Ok(ret != 0);
}
_ => {
// ResultCode::DONE would happen if clock values exist but actual values are missing.
Expand Down Expand Up @@ -585,12 +599,14 @@ unsafe fn merge_insert(
|| !row_exists_locally
|| did_cid_win(
db,
(*tab).pExtData,
insert_tbl,
&tbl_info,
&unpacked_pks,
key,
insert_col,
insert_val,
insert_site_id,
insert_col,
insert_col_vrsn,
errmsg,
)?;
Expand Down
16 changes: 0 additions & 16 deletions libsql-sqlite3/ext/crr/rs/integration-check/Cargo.toml

This file was deleted.

Original file line number Diff line number Diff line change
Expand Up @@ -806,6 +806,7 @@ pub trait Context {
fn result_error_code(&self, code: ResultCode);
fn result_value(&self, value: *mut value);
fn result_double(&self, value: f64);
fn result_int(&self, value: i32);
fn result_int64(&self, value: int64);
fn result_null(&self);
fn db_handle(&self) -> *mut sqlite3;
Expand Down Expand Up @@ -906,6 +907,11 @@ impl Context for *mut context {
result_int64(*self, value);
}

#[inline]
fn result_int(&self, value: i32) {
result_int(*self, value);
}

#[inline]
fn db_handle(&self) -> *mut sqlite3 {
context_db_handle(*self)
Expand Down
7 changes: 6 additions & 1 deletion libsql-sqlite3/ext/crr/src/changes-vtab.c
Original file line number Diff line number Diff line change
Expand Up @@ -176,4 +176,9 @@ sqlite3_module crsql_changesModule = {
/* xSavepoint */ 0,
/* xRelease */ 0,
/* xRollbackTo */ 0,
/* xShadowName */ 0};
/* xShadowName */ 0
#ifdef LIBSQL
,
/* xPreparedSql */ 0
#endif
};
20 changes: 19 additions & 1 deletion libsql-sqlite3/ext/crr/src/crsqlite.c
Original file line number Diff line number Diff line change
Expand Up @@ -287,17 +287,32 @@ static void rollbackHook(void *pUserData) {
pExtData->readDbVersionThisTx = 0;
}

#ifdef LIBSQL
static void closeHook(void *pUserData, sqlite3 *db) {
crsql_ExtData *pExtData = (crsql_ExtData *)pUserData;
crsql_finalize(pExtData);
}
#endif

int sqlite3_crsqlrustbundle_init(sqlite3 *db, char **pzErrMsg,
const sqlite3_api_routines *pApi);

#ifdef _WIN32
__declspec(dllexport)
#endif
int sqlite3_crsqlite_init(sqlite3 *db, char **pzErrMsg,
const sqlite3_api_routines *pApi) {
const sqlite3_api_routines *pApi
#ifdef LIBSQL
,
const libsql_api_routines *pLibsqlApi
#endif
) {
int rc = SQLITE_OK;

SQLITE_EXTENSION_INIT2(pApi);
#ifdef LIBSQL
LIBSQL_EXTENSION_INIT2(pLibsqlApi);
#endif

// TODO: should be moved lower once we finish migrating to rust.
// RN it is safe here since the rust bundle init is largely just reigstering
Expand Down Expand Up @@ -432,6 +447,9 @@ __declspec(dllexport)
}

if (rc == SQLITE_OK) {
#ifdef LIBSQL
libsql_close_hook(db, closeHook, pExtData);
#endif
// TODO: get the prior callback so we can call it rather than replace
// it?
sqlite3_commit_hook(db, commitHook, pExtData);
Expand Down
29 changes: 29 additions & 0 deletions libsql-sqlite3/ext/crr/src/crsqlite.test.c
Original file line number Diff line number Diff line change
Expand Up @@ -440,6 +440,20 @@ static sqlite3_int64 getDbVersion(sqlite3 *db) {
return db2v;
}

static void *getSiteId(sqlite3 *db) {
sqlite3_stmt *pStmt = 0;
int rc = sqlite3_prepare_v2(db, "SELECT crsql_site_id()", -1, &pStmt, 0);
if (rc != SQLITE_OK) {
return 0;
}

sqlite3_step(pStmt);
void *site = sqlite3_column_blob(pStmt, 0);
sqlite3_finalize(pStmt);

return site;
}

static void testLamportCondition() {
printf("LamportCondition\n");
// syncing from A -> B, while no changes happen on B, moves up
Expand Down Expand Up @@ -511,6 +525,17 @@ static void noopsDoNotMoveClocks() {
rc += sqlite3_open(":memory:", &db1);
rc += sqlite3_open(":memory:", &db2);

void *db1SiteId = getSiteId(db1);
void *db2SiteId = getSiteId(db2);
int cmp = memcmp(db1SiteId, db2SiteId, 16);
if (cmp > 0) {
sqlite3 *temp = db1;
db1 = db2;
db2 = temp;
}

// swap dbs based on site id compare to make it a noop.

rc += sqlite3_exec(
db1, "CREATE TABLE \"hoot\" (\"a\", \"b\" primary key not null, \"c\")",
0, 0, 0);
Expand Down Expand Up @@ -545,6 +570,10 @@ static void noopsDoNotMoveClocks() {
sqlite3_int64 db1vPost = getDbVersion(db1);
sqlite3_int64 db2vPost = getDbVersion(db2);

// TODO: we still need to compare values so as not to bump the db_version
// forward on a no-difference
// this fails sometimes because site id winning.
printf("db1 pre: %lld db2 post: %lld", db1vPre, db2vPost);
assert(db1vPre == db2vPost);
assert(db1vPre == db1vPost);

Expand Down
34 changes: 31 additions & 3 deletions libsql-sqlite3/ext/crr/src/rows-impacted.test.c
Original file line number Diff line number Diff line change
Expand Up @@ -314,7 +314,7 @@ static void testCreateThatDoesNotChangeAnything() {
printf("\t\e[0;32mSuccess\e[0m\n");
}

static void testValueWin() {
static void testValueWouldWinButSiteIdLoses() {
printf("ValueWin\n");
int rc = SQLITE_OK;
char *err = 0;
Expand All @@ -326,10 +326,37 @@ static void testValueWin() {
rc = sqlite3_exec(db, "BEGIN", 0, 0, 0);
rc += sqlite3_exec(db,
"INSERT INTO crsql_changes VALUES ('foo', X'010901', 'b', "
"3, 1, 1, NULL, 1, 1)",
"3, 1, 1, X'00000000000000000000000000000000', 1, 1)",
0, 0, &err);
sqlite3_prepare_v2(db, "SELECT crsql_rows_impacted()", -1, &pStmt, 0);
sqlite3_step(pStmt);
// value is greater but site id lower, a loss and now rows changed.
assert(sqlite3_column_int(pStmt, 0) == 0);
sqlite3_finalize(pStmt);
rc += sqlite3_exec(db, "COMMIT", 0, 0, 0);
assert(rc == SQLITE_OK);

crsql_close(db);
printf("\t\e[0;32mSuccess\e[0m\n");
}

static void testSiteIdWin() {
printf("SiteIdWin\n");
int rc = SQLITE_OK;
char *err = 0;
sqlite3 *db = createDb();
sqlite3_stmt *pStmt = 0;

rc = sqlite3_exec(db, "INSERT INTO foo VALUES (1, 2)", 0, 0, 0);

rc = sqlite3_exec(db, "BEGIN", 0, 0, 0);
rc += sqlite3_exec(db,
"INSERT INTO crsql_changes VALUES ('foo', X'010901', 'b', "
"3, 1, 1, X'FFFFFFFFFFFFFFFFFFFFFFFFFFFFFFFF', 1, 1)",
0, 0, &err);
sqlite3_prepare_v2(db, "SELECT crsql_rows_impacted()", -1, &pStmt, 0);
sqlite3_step(pStmt);
// site id is larger, a win
assert(sqlite3_column_int(pStmt, 0) == 1);
sqlite3_finalize(pStmt);
rc += sqlite3_exec(db, "COMMIT", 0, 0, 0);
Expand Down Expand Up @@ -374,7 +401,8 @@ void rowsImpactedTestSuite() {
testUpdateThatDoesNotChangeAnything();
testDeleteThatDoesNotChangeAnything();
testCreateThatDoesNotChangeAnything();
testValueWin();
testValueWouldWinButSiteIdLoses();
testSiteIdWin();
testClockWin();
testDelete();
}

0 comments on commit 680d1ce

Please sign in to comment.