I ran a test and I can still find "database is locked" even if I use
busy_handler(threadsafe=2, 2 connections).
When thread 1 executing a writing transaction, thread 2 runs the code below
at the same time:
sqlite3_exec("BEGIN")
//SELECT
sqlite3_prepare_v2("SELECT * FROM t1;");
sqlite3_step;
sqlite3_reset;
//INSERT
sqlite3_exec("INSERT INTO t1 VALUES(1, 1, \"aweagsr\")");//database is
locked
sqlite3_exec("COMMIT");

Writing in thread 1 will no block SELECTs in thread 2 as I use WAL. But the
INSERT within the transaction of thread 2 still returns SQLITE_BUSY.
I think I have used sqlite3_busy_timeout() in right way and I find that
sqliteDefaultBusyCallback() did not be called.

Is it expected?

It's almost like you are not reading the posts by Keith - which leads me to believe you might have messages channeled to spam for some reason, it's worth checking the spam folder.

To answer the questions: YES, it is expected.

To elaborate:
A - When starting a writing transaction from ANY thread, then ANY other write attempt (threaded or otherwise) will find the DB/Table(s) locked, that's by design and an absolute necessity in any RDBMS.

B - Should this happen, sqlite /CAN/ use a default busy handler that does nothing more than waits a bit, then see if the in-progress lock is lifted before continuing. Rinse and repeat until busy timeout time is exhausted.  It is important to note that this busy handler ONLY functions if there is an allowed time-out set on the connection. (See pragma timeout, or it can be set via the api too).

C - You can provide a busy-handler of your own making that will fire whenever an SQLITE_BUSY signal is received, regardless of timeouts. You can specify via this handler if the operation should wait, cancel, etc.


Usually setting the internal busy timeout is enough, and with large DBs running huge transactions, a very long timeout is common (sometimes in the order of minutes) - you will need to study your DB use case and decide what a good time is to give up on waiting.  If you /DO/ expect a lot of timeout waiting on an interactive user-based system, probably best to indicate to the user and not look like you died - perhaps even prompt them to find if they would like to wait for xxx to finish or cancel - in which case a combination of setting a busy handler and also a progress handler will be best.

More information on all the above is easily found in the sqlite site's search - if you do decide which to do and have any difficulty with a specific method, or fail to find information - please post again.
Good luck!
Ryan

_______________________________________________
sqlite-users mailing list
sqlite-users@mailinglists.sqlite.org
http://mailinglists.sqlite.org/cgi-bin/mailman/listinfo/sqlite-users

Reply via email to