session-android/app/src/main/java/org/thoughtcrime/securesms/database/SessionJobDatabase.kt

143 lines
6.7 KiB
Kotlin
Raw Normal View History

2021-07-09 05:25:57 +02:00
package org.thoughtcrime.securesms.database
2021-01-28 05:24:27 +01:00
import android.content.ContentValues
import android.content.Context
import net.sqlcipher.Cursor
import org.session.libsession.messaging.jobs.AttachmentUploadJob
import org.session.libsession.messaging.jobs.GroupAvatarDownloadJob
import org.session.libsession.messaging.jobs.Job
import org.session.libsession.messaging.jobs.MessageReceiveJob
import org.session.libsession.messaging.jobs.MessageSendJob
import org.session.libsession.messaging.jobs.SessionJobInstantiator
import org.session.libsession.messaging.jobs.SessionJobManagerFactories
2021-05-13 02:31:06 +02:00
import org.session.libsession.messaging.utilities.Data
2021-05-18 01:12:33 +02:00
import org.session.libsignal.utilities.Log
2021-01-28 05:24:27 +01:00
import org.thoughtcrime.securesms.database.helpers.SQLCipherOpenHelper
import org.thoughtcrime.securesms.jobmanager.impl.JsonDataSerializer
class SessionJobDatabase(context: Context, helper: SQLCipherOpenHelper) : Database(context, helper) {
companion object {
2021-05-21 01:08:16 +02:00
const val sessionJobTable = "session_job_database"
2021-03-17 06:09:01 +01:00
const val jobID = "job_id"
const val jobType = "job_type"
const val failureCount = "failure_count"
const val serializedData = "serialized_data"
2021-05-12 08:17:25 +02:00
@JvmStatic val createSessionJobTableCommand
= "CREATE TABLE $sessionJobTable ($jobID INTEGER PRIMARY KEY, $jobType STRING, $failureCount INTEGER DEFAULT 0, $serializedData TEXT);"
2021-01-28 05:24:27 +01:00
}
fun persistJob(job: Job) {
val database = databaseHelper.writableDatabase
2021-03-03 05:14:45 +01:00
val contentValues = ContentValues(4)
contentValues.put(jobID, job.id!!)
2021-01-28 05:24:27 +01:00
contentValues.put(jobType, job.getFactoryKey())
contentValues.put(failureCount, job.failureCount)
contentValues.put(serializedData, SessionJobHelper.dataSerializer.serialize(job.serialize()))
database.insertOrUpdate(sessionJobTable, contentValues, "$jobID = ?", arrayOf( job.id!! ))
2021-01-28 05:24:27 +01:00
}
2021-05-12 08:17:25 +02:00
fun markJobAsSucceeded(jobID: String) {
databaseHelper.writableDatabase.delete(sessionJobTable, "${Companion.jobID} = ?", arrayOf( jobID ))
2021-01-28 05:24:27 +01:00
}
fun markJobAsFailedPermanently(jobID: String) {
databaseHelper.writableDatabase.delete(sessionJobTable, "${Companion.jobID} = ?", arrayOf( jobID ))
2021-01-28 05:24:27 +01:00
}
fun getAllPendingJobs(type: String): Map<String, Job?> {
2021-01-28 05:24:27 +01:00
val database = databaseHelper.readableDatabase
2021-05-12 08:17:25 +02:00
return database.getAll(sessionJobTable, "$jobType = ?", arrayOf( type )) { cursor ->
val jobID = cursor.getString(jobID)
try {
2021-05-12 08:17:25 +02:00
jobID to jobFromCursor(cursor)
} catch (e: Exception) {
2021-05-12 08:17:25 +02:00
Log.e("Loki", "Error deserializing job of type: $type.", e)
jobID to null
}
}.toMap()
2021-01-28 05:24:27 +01:00
}
fun getAttachmentUploadJob(attachmentID: Long): AttachmentUploadJob? {
val database = databaseHelper.readableDatabase
2021-05-12 08:17:25 +02:00
val result = mutableListOf<AttachmentUploadJob>()
database.getAll(sessionJobTable, "$jobType = ?", arrayOf( AttachmentUploadJob.KEY )) { cursor ->
val job = jobFromCursor(cursor) as AttachmentUploadJob?
if (job != null) { result.add(job) }
2021-01-28 05:24:27 +01:00
}
2021-03-02 07:22:56 +01:00
return result.firstOrNull { job -> job.attachmentID == attachmentID }
2021-01-28 05:24:27 +01:00
}
fun getMessageSendJob(messageSendJobID: String): MessageSendJob? {
val database = databaseHelper.readableDatabase
2021-05-12 08:17:25 +02:00
return database.get(sessionJobTable, "$jobID = ? AND $jobType = ?", arrayOf( messageSendJobID, MessageSendJob.KEY )) { cursor ->
jobFromCursor(cursor) as MessageSendJob?
2021-01-28 05:24:27 +01:00
}
}
fun getMessageReceiveJob(messageReceiveJobID: String): MessageReceiveJob? {
val database = databaseHelper.readableDatabase
return database.get(sessionJobTable, "$jobID = ? AND $jobType = ?", arrayOf( messageReceiveJobID, MessageReceiveJob.KEY )) { cursor ->
jobFromCursor(cursor) as MessageReceiveJob?
}
}
fun getGroupAvatarDownloadJob(server: String, room: String): GroupAvatarDownloadJob? {
val database = databaseHelper.readableDatabase
return database.getAll(sessionJobTable, "$jobType = ?", arrayOf(GroupAvatarDownloadJob.KEY)) {
jobFromCursor(it) as GroupAvatarDownloadJob?
}.filterNotNull().find { it.server == server && it.room == room }
}
2021-05-14 05:09:40 +02:00
fun cancelPendingMessageSendJobs(threadID: Long) {
val database = databaseHelper.writableDatabase
val attachmentUploadJobKeys = mutableListOf<String>()
database.getAll(sessionJobTable, "$jobType = ?", arrayOf( AttachmentUploadJob.KEY )) { cursor ->
val job = jobFromCursor(cursor) as AttachmentUploadJob?
if (job != null && job.threadID == threadID.toString()) { attachmentUploadJobKeys.add(job.id!!) }
}
val messageSendJobKeys = mutableListOf<String>()
database.getAll(sessionJobTable, "$jobType = ?", arrayOf( MessageSendJob.KEY )) { cursor ->
val job = jobFromCursor(cursor) as MessageSendJob?
if (job != null && job.message.threadID == threadID) { messageSendJobKeys.add(job.id!!) }
}
if (attachmentUploadJobKeys.isNotEmpty()) {
val attachmentUploadJobKeysAsString = attachmentUploadJobKeys.joinToString(", ")
database.delete(sessionJobTable, "${Companion.jobType} = ? AND ${Companion.jobID} IN (?)",
arrayOf( AttachmentUploadJob.KEY, attachmentUploadJobKeysAsString ))
}
if (messageSendJobKeys.isNotEmpty()) {
val messageSendJobKeysAsString = messageSendJobKeys.joinToString(", ")
database.delete(sessionJobTable, "${Companion.jobType} = ? AND ${Companion.jobID} IN (?)",
arrayOf( MessageSendJob.KEY, messageSendJobKeysAsString ))
}
}
2021-01-28 05:24:27 +01:00
fun isJobCanceled(job: Job): Boolean {
val database = databaseHelper.readableDatabase
var cursor: android.database.Cursor? = null
try {
cursor = database.rawQuery("SELECT * FROM $sessionJobTable WHERE $jobID = ?", arrayOf( job.id!! ))
return cursor == null || !cursor.moveToFirst()
2021-01-28 05:24:27 +01:00
} catch (e: Exception) {
// Do nothing
} finally {
cursor?.close()
}
return false
}
2021-05-12 08:17:25 +02:00
private fun jobFromCursor(cursor: Cursor): Job? {
2021-01-28 05:24:27 +01:00
val type = cursor.getString(jobType)
val data = SessionJobHelper.dataSerializer.deserialize(cursor.getString(serializedData))
2021-05-12 08:17:25 +02:00
val job = SessionJobHelper.sessionJobInstantiator.instantiate(type, data) ?: return null
2021-01-28 05:24:27 +01:00
job.id = cursor.getString(jobID)
job.failureCount = cursor.getInt(failureCount)
return job
}
}
2021-03-17 02:06:44 +01:00
object SessionJobHelper {
val dataSerializer: Data.Serializer = JsonDataSerializer()
val sessionJobInstantiator: SessionJobInstantiator = SessionJobInstantiator(SessionJobManagerFactories.getSessionJobFactories())
}