Sometimes we are facing this issue, as it gets triggered during the data-saving process into the database, but the data is not being saved. We encounter the following error:
Note: We are unable to catch the error all the time, but errors are being recorded in production, and data is not being saved.
Error: “must be called during a transaction”
(CouchbaseLite Android v3.1.0-425@33 (EE/release, Commit/3f009a25f8@934af6e32f2a Core/3.1.0 (356) at 2023-04-14T04:06:14.503123Z) on Java; Android 12; SM-T500)
Code:
return withContext(backgroundDispatcher) {
val queryResult = kotlin.runCatching {
val map = converter.toMap(model).toMutableMap().apply {
this.remove("_id")
}
val document = MutableDocument(id, map)
cbHelper.getDataBase().defaultCollection?.save(
document
) { newDoc, curDoc ->
if (curDoc == null) {
// No existing document, treat it as creation
true
} else {
val resolvedDoc = MutableDocument(newDoc.id)
// Remove the _id key from newDoc before merging
val newDataMap: MutableMap<String, Any> = newDoc.toMap().toMutableMap()
newDataMap.remove("_id")
// Merge the data from newDoc and curDoc
val mergedDataMap: MutableMap<String, Any> = mutableMapOf()
mergedDataMap.putAll(curDoc.toMap())
mergedDataMap.putAll(newDataMap)
resolvedDoc.setData(mergedDataMap)
true
}
}
document.id
}
queryResult.getOrNull()
?.let { docId -> getDocument(docId) }
?: kotlin.run {
val error =queryResult.exceptionOrNull()!!
Log.d("Error Save/Update",error.toString())
QueryResult.error(error)
}
}
Thanks for the reply but we are not able to get this in development as i told, we are able to see logs in production, please find the logs below,
QueryResultError -StackTrace : CouchbaseLiteException{CouchbaseLite,17,‘must be called during a transaction
(CouchbaseLite Android v3.1.0-425@33 (EE/release, Commit/3f009a25f8@934af6e32f2a Core/3.1.0 (356) at 2023-04-14T04:06:14.503123Z) on Java; Android 13; SM-T575)’}
at com.couchbase.lite.Collection.saveInTransaction(Collection.java:904)
at com.couchbase.lite.Collection.saveLocked(Collection.java:670)
at com.couchbase.lite.Collection.saveWithConflictHandler(Collection.java:613)
at com.couchbase.lite.Collection.save(Collection.java:257)
at xxxx.xxxxxxxx.xxxxx.data.internal.impl.DataSourceImplSession$saveUserSession$$inlined$saveDocument$1.invokeSuspend(DeModelQuery.kt:96)
at xxxx.xxxxxxxx.xxxxx.data.internal.impl.DataSourceImplSession$saveUserSession$$inlined$saveDocument$1.invoke(Unknown Source:8)
at xxxx.xxxxxxxx.xxxxx.data.internal.impl.DataSourceImplSession$saveUserSession$$inlined$saveDocument$1.invoke(Unknown Source:4)
at kotlinx.coroutines.intrinsics.UndispatchedKt.startUndispatchedOrReturn(Undispatched.kt:89)
at kotlinx.coroutines.BuildersKt__Builders_commonKt.withContext(Builders.common.kt:166)
at kotlinx.coroutines.BuildersKt.withContext(Unknown Source:1)
at xxxx.xxxxxxxx.xxxxx.data.internal.impl.DataSourceImplSession.saveUserSession(DataSourceImplSession.kt:123)
at xxxx.xxxxxxxx.xxxxx.data.dataSources.repos.RepoSession.saveUserSession(RepoSession.kt:17)
at xxxx.xxxxxxxx.xxxxx.features.session.ForegroundBackgroundListener.createNewAppSession$addAppSessionToUserSession(ForegroundBackgroundListener.kt:123)
at xxxx.xxxxxxxx.xxxxx.features.session.ForegroundBackgroundListener.access$createNewAppSession$addAppSessionToUserSession(ForegroundBackgroundListener.kt:21)
at xxxx.xxxxxxxx.xxxxx.features.session.ForegroundBackgroundListener$createNewAppSession$1.invokeSuspend(ForegroundBackgroundListener.kt:145)
at kotlin.coroutines.jvm.internal.BaseContinuationImpl.resumeWith(ContinuationImpl.kt:33)
at kotlinx.coroutines.DispatchedTask.run(DispatchedTask.kt:106)
at kotlinx.coroutines.internal.LimitedDispatcher.run(LimitedDispatcher.kt:42)
at kotlinx.coroutines.scheduling.TaskImpl.run(Tasks.kt:95)
at kotlinx.coroutines.scheduling.CoroutineScheduler.runSafely(CoroutineScheduler.kt:570)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.executeTask(CoroutineScheduler.kt:749)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.runWorker(CoroutineScheduler.kt:677)
at kotlinx.coroutines.scheduling.CoroutineScheduler$Worker.run(CoroutineScheduler.kt:664)
Caused by: LiteCoreException{1, 17, “must be called during a transaction”}
at com.couchbase.lite.internal.core.C4Document.createFromSlice(Native Method)
at com.couchbase.lite.internal.core.C4Document.create(C4Document.java:52)
at com.couchbase.lite.internal.core.C4Collection.createDocument(C4Collection.java:182)
at com.couchbase.lite.Collection.createC4Document(Collection.java:585)
at com.couchbase.lite.Collection.saveInTransaction(Collection.java:898)
… 22 more
I’m not entirely sure what is going on, here: I haven’t had time, yet, to examine it in detail. Looking at the code, I would say that it is impossible.
I’m not exactly sure of what you mean when you say “if the database becomes locked”.
Here’s what I can say: Database use is, essentially, single threaded. Nearly every call to a method that uses the database will seize a lock that prevents any other access until the first call completes. It is possible to do some concurrent processing by opening two different instances of the same database… but even then, file access is single threaded: a long running query, for instance, will block all other access until it completes.
Can you tell me which of the methods in saveUserSession are suspend methods?
As I say, if you can share logs that would be tremendously helpful.
@blake.meike saveDocument(userSession) is the one is suspend function, As I told earlier we didn’t faced this in development environment, If I get it will post it immediately.