feat: use models during export, try to update conversation date during import

This commit is contained in:
darthpaul
2021-09-18 21:05:06 +01:00
parent e10a410788
commit d78776e288
11 changed files with 216 additions and 149 deletions

View File

@@ -260,7 +260,7 @@ fun Context.getConversationIds(): List<Long> {
val projection = arrayOf(Threads._ID) val projection = arrayOf(Threads._ID)
val selection = "${Threads.MESSAGE_COUNT} > ?" val selection = "${Threads.MESSAGE_COUNT} > ?"
val selectionArgs = arrayOf("0") val selectionArgs = arrayOf("0")
val sortOrder = "${Threads.DATE} DESC" val sortOrder = "${Threads.DATE} ASC"
val conversationIds = mutableListOf<Long>() val conversationIds = mutableListOf<Long>()
queryCursor(uri, projection, selection, selectionArgs, sortOrder, true) { cursor -> queryCursor(uri, projection, selection, selectionArgs, sortOrder, true) { cursor ->
val id = cursor.getLongValue(Threads._ID) val id = cursor.getLongValue(Threads._ID)

View File

@@ -22,7 +22,6 @@ const val EXPORT_MIME_TYPE = "application/json"
const val EXPORT_FILE_EXT = ".json" const val EXPORT_FILE_EXT = ".json"
const val IMPORT_SMS = "import_sms" const val IMPORT_SMS = "import_sms"
const val IMPORT_MMS = "import_mms" const val IMPORT_MMS = "import_mms"
const val MMS_CONTENT = "mms_content"
private const val PATH = "com.simplemobiletools.smsmessenger.action." private const val PATH = "com.simplemobiletools.smsmessenger.action."
const val MARK_AS_READ = PATH + "mark_as_read" const val MARK_AS_READ = PATH + "mark_as_read"

View File

@@ -1,6 +1,7 @@
package com.simplemobiletools.smsmessenger.helpers package com.simplemobiletools.smsmessenger.helpers
import android.content.Context import android.content.Context
import com.google.gson.Gson
import com.google.gson.stream.JsonWriter import com.google.gson.stream.JsonWriter
import com.simplemobiletools.commons.helpers.ensureBackgroundThread import com.simplemobiletools.commons.helpers.ensureBackgroundThread
import com.simplemobiletools.smsmessenger.extensions.config import com.simplemobiletools.smsmessenger.extensions.config
@@ -14,6 +15,7 @@ class MessagesExporter(private val context: Context) {
private val config = context.config private val config = context.config
private val messageReader = MessagesReader(context) private val messageReader = MessagesReader(context)
private val gson = Gson()
fun exportMessages( fun exportMessages(
outputStream: OutputStream?, outputStream: OutputStream?,
@@ -27,33 +29,33 @@ class MessagesExporter(private val context: Context) {
/* /*
* We should have json in this format * We should have json in this format
* { * [
* "threadId" : { * {
* "threadId": ""
* "sms": [{ smses }], * "sms": [{ smses }],
* "mms": [{ mmses }] * "mms": [{ mmses }]
* } * },
* } * {
* "sms": [{ smses }],
* "mms": [{ mmses }]
* }
* ]
* *
* */ * */
val writer = JsonWriter(outputStream.bufferedWriter()) val writer = JsonWriter(outputStream.bufferedWriter())
writer.use { writer.use {
try { try {
var written = 0 var written = 0
writer.beginObject() writer.beginArray()
val conversationIds = context.getConversationIds() val conversationIds = context.getConversationIds()
val totalMessages = messageReader.getMessagesCount()
for (threadId in conversationIds) { for (threadId in conversationIds) {
writer.name(threadId.toString())
writer.beginObject() writer.beginObject()
writer.name("threadId")
writer.value(threadId)
if (config.exportSms) { if (config.exportSms) {
writer.name("sms") writer.name("sms")
writer.beginArray() writer.beginArray()
//write all sms //write all sms
messageReader.forEachSms(threadId) { messageReader.forEachSms(threadId) {
JsonObjectWriter(writer).write(it) writer.jsonValue(gson.toJson(it))
written++ written++
} }
writer.endArray() writer.endArray()
@@ -64,7 +66,7 @@ class MessagesExporter(private val context: Context) {
writer.beginArray() writer.beginArray()
//write all mms //write all mms
messageReader.forEachMms(threadId) { messageReader.forEachMms(threadId) {
JsonObjectWriter(writer).write(it) writer.jsonValue(gson.toJson(it))
written++ written++
} }
@@ -73,7 +75,7 @@ class MessagesExporter(private val context: Context) {
writer.endObject() writer.endObject()
} }
writer.endObject() writer.endArray()
callback.invoke(ExportResult.EXPORT_OK) callback.invoke(ExportResult.EXPORT_OK)
} catch (e: Exception) { } catch (e: Exception) {
callback.invoke(ExportResult.EXPORT_FAIL) callback.invoke(ExportResult.EXPORT_FAIL)

View File

@@ -3,6 +3,7 @@ package com.simplemobiletools.smsmessenger.helpers
import android.content.Context import android.content.Context
import android.net.Uri import android.net.Uri
import android.provider.Telephony import android.provider.Telephony
import android.provider.Telephony.*
import android.util.Log import android.util.Log
import com.google.gson.Gson import com.google.gson.Gson
import com.google.gson.reflect.TypeToken import com.google.gson.reflect.TypeToken
@@ -32,9 +33,6 @@ class MessagesImporter(private val context: Context) {
return@ensureBackgroundThread return@ensureBackgroundThread
} }
//read data from path
// parse json
// write data to sql db
val inputStream = if (path.contains("/")) { val inputStream = if (path.contains("/")) {
File(path).inputStream() File(path).inputStream()
} else { } else {
@@ -44,33 +42,32 @@ class MessagesImporter(private val context: Context) {
inputStream.bufferedReader().use { inputStream.bufferedReader().use {
try { try {
val json = it.readText() val json = it.readText()
Log.d(TAG, "importMessages: json== ${json.length}") Log.d(TAG, "importMessages: json== $json")
val type = object : TypeToken<Map<String, ExportedMessage>>() {}.type val type = object : TypeToken<List<ExportedMessage>>() {}.type
val data = gson.fromJson<Map<String, ExportedMessage>>(json, type) val messages = gson.fromJson<List<ExportedMessage>>(json, type)
Log.d(TAG, "importMessages: ${data.size}") Log.d(TAG, "importMessages: ${messages.size}")
for (message in data.values) { for (message in messages) {
// add sms // add sms
if (config.importSms) { if (config.importSms) {
message.sms.forEach(messageWriter::writeSmsMessage) message.sms.forEach(messageWriter::writeSmsMessage)
} }
// add mms // add mms
if (config.importMms) { if (config.importMms) {
message.mms.forEach(messageWriter::writeMmsMessage) message.mms.forEach(messageWriter::writeMmsMessage)
} }
// messageWriter.updateAllSmsThreads()
val conversations = context.getConversations() context.queryCursor(Threads.CONTENT_URI) { cursor ->
val conversationIds = context.getConversationIds() val json = cursor.rowsToJson()
Log.w(TAG, "conversations = $conversations") Log.w(TAG, "converations = $json")
Log.w(TAG, "conversationIds = $conversationIds") }
context.queryCursor(Telephony.Sms.CONTENT_URI) { cursor ->
context.queryCursor(Sms.CONTENT_URI) { cursor ->
val json = cursor.rowsToJson() val json = cursor.rowsToJson()
Log.w(TAG, "smses = $json") Log.w(TAG, "smses = $json")
} }
context.queryCursor(Telephony.Mms.CONTENT_URI) { cursor -> context.queryCursor(Mms.CONTENT_URI) { cursor ->
val json = cursor.rowsToJson() val json = cursor.rowsToJson()
Log.w(TAG, "mmses = $json") Log.w(TAG, "mmses = $json")
} }
@@ -80,10 +77,8 @@ class MessagesImporter(private val context: Context) {
Log.w(TAG, "parts = $json") Log.w(TAG, "parts = $json")
} }
refreshMessages() refreshMessages()
callback.invoke(ImportResult.IMPORT_OK)
} }
} catch (e: Exception) { } catch (e: Exception) {
Log.e(TAG, "importMessages: ", e) Log.e(TAG, "importMessages: ", e)

View File

@@ -3,16 +3,16 @@ package com.simplemobiletools.smsmessenger.helpers
import android.annotation.SuppressLint import android.annotation.SuppressLint
import android.content.Context import android.content.Context
import android.net.Uri import android.net.Uri
import android.provider.Telephony import android.provider.Telephony.*
import android.util.Base64 import android.util.Base64
import android.util.Log import android.util.Log
import com.google.gson.JsonArray
import com.google.gson.JsonObject
import com.simplemobiletools.commons.extensions.* import com.simplemobiletools.commons.extensions.*
import com.simplemobiletools.commons.helpers.isQPlus import com.simplemobiletools.commons.helpers.isQPlus
import com.simplemobiletools.smsmessenger.extensions.optLong
import com.simplemobiletools.smsmessenger.extensions.optString
import com.simplemobiletools.smsmessenger.extensions.rowsToJson import com.simplemobiletools.smsmessenger.extensions.rowsToJson
import com.simplemobiletools.smsmessenger.models.MmsAddress
import com.simplemobiletools.smsmessenger.models.MmsBackup
import com.simplemobiletools.smsmessenger.models.MmsPart
import com.simplemobiletools.smsmessenger.models.SmsBackup
import java.io.IOException import java.io.IOException
import java.io.InputStream import java.io.InputStream
@@ -21,22 +21,63 @@ class MessagesReader(private val context: Context) {
private const val TAG = "MessagesReader" private const val TAG = "MessagesReader"
} }
fun forEachSms(threadId: Long, block: (JsonObject) -> Unit) { fun forEachSms(threadId: Long, block: (SmsBackup) -> Unit) {
val selection = "${Telephony.Sms.THREAD_ID} = ?" val projection = arrayOf(
Sms.SUBSCRIPTION_ID,
Sms.ADDRESS,
Sms.BODY,
Sms.DATE,
Sms.DATE_SENT,
Sms.LOCKED,
Sms.PROTOCOL,
Sms.READ,
Sms.STATUS,
Sms.TYPE,
Sms.SERVICE_CENTER
)
val selection = "${Sms.THREAD_ID} = ?"
val selectionArgs = arrayOf(threadId.toString()) val selectionArgs = arrayOf(threadId.toString())
context.queryCursor(Telephony.Sms.CONTENT_URI, null, selection, selectionArgs) { cursor -> context.queryCursor(Sms.CONTENT_URI, projection, selection, selectionArgs) { cursor ->
val json = cursor.rowsToJson() val subscriptionId = cursor.getLongValue(Sms.SUBSCRIPTION_ID)
block(json) val address = cursor.getStringValue(Sms.ADDRESS)
val body = cursor.getStringValueOrNull(Sms.BODY)
val date = cursor.getLongValue(Sms.DATE)
val dateSent = cursor.getLongValue(Sms.DATE_SENT)
val locked = cursor.getIntValue(Sms.DATE_SENT)
val protocol = cursor.getStringValueOrNull(Sms.PROTOCOL)
val read = cursor.getIntValue(Sms.READ)
val status = cursor.getIntValue(Sms.STATUS)
val type = cursor.getIntValue(Sms.TYPE)
val serviceCenter = cursor.getStringValueOrNull(Sms.SERVICE_CENTER)
block(SmsBackup(subscriptionId, address, body, date, dateSent, locked, protocol, read, status, type, serviceCenter))
} }
} }
// all mms from simple sms are non-text messages // all mms from simple sms are non-text messages
fun forEachMms(threadId: Long, includeTextOnlyAttachment: Boolean = false, block: (JsonObject) -> Unit) { fun forEachMms(threadId: Long, includeTextOnlyAttachment: Boolean = false, block: (MmsBackup) -> Unit) {
val projection = arrayOf(
Mms._ID,
Mms.CREATOR,
Mms.CONTENT_TYPE,
Mms.DELIVERY_REPORT,
Mms.DATE,
Mms.DATE_SENT,
Mms.LOCKED,
Mms.MESSAGE_TYPE,
Mms.MESSAGE_BOX,
Mms.READ,
Mms.READ_REPORT,
Mms.SEEN,
Mms.TEXT_ONLY,
Mms.STATUS,
Mms.SUBJECT_CHARSET,
Mms.SUBSCRIPTION_ID,
Mms.TRANSACTION_ID
)
val selection = if (includeTextOnlyAttachment) { val selection = if (includeTextOnlyAttachment) {
"${Telephony.Mms.THREAD_ID} = ? AND ${Telephony.Mms.TEXT_ONLY} = ?" "${Mms.THREAD_ID} = ? AND ${Mms.TEXT_ONLY} = ?"
} else { } else {
"${Telephony.Mms.THREAD_ID} = ?" "${Mms.THREAD_ID} = ?"
} }
val selectionArgs = if (includeTextOnlyAttachment) { val selectionArgs = if (includeTextOnlyAttachment) {
@@ -44,61 +85,95 @@ class MessagesReader(private val context: Context) {
} else { } else {
arrayOf(threadId.toString()) arrayOf(threadId.toString())
} }
context.queryCursor(Telephony.Mms.CONTENT_URI, null, selection, selectionArgs) { cursor -> context.queryCursor(Mms.CONTENT_URI, projection, selection, selectionArgs) { cursor ->
val json = cursor.rowsToJson() val mmsId = cursor.getLongValue(Mms._ID)
json.add("parts", getParts(json.getAsJsonPrimitive(Telephony.Mms._ID).asLong)) val creator = cursor.getStringValueOrNull(Mms.CREATOR)
json.add("addresses", getMMSAddresses(json.getAsJsonPrimitive(Telephony.Mms._ID).asLong)) val contentType = cursor.getStringValueOrNull(Mms.CONTENT_TYPE)
block(json) val deliveryReport = cursor.getIntValue(Mms.DELIVERY_REPORT)
val date = cursor.getLongValue(Mms.DATE)
val dateSent = cursor.getLongValue(Mms.DATE_SENT)
val locked = cursor.getIntValue(Mms.LOCKED)
val messageType = cursor.getIntValue(Mms.MESSAGE_TYPE)
val messageBox = cursor.getIntValue(Mms.MESSAGE_BOX)
val read = cursor.getIntValue(Mms.READ)
val readReport = cursor.getIntValue(Mms.READ_REPORT)
val seen = cursor.getIntValue(Mms.SEEN)
val textOnly = cursor.getIntValue(Mms.TEXT_ONLY)
val status = cursor.getStringValueOrNull(Mms.STATUS)
val subject = cursor.getStringValueOrNull(Mms.SUBJECT)
val subjectCharSet = cursor.getStringValueOrNull(Mms.SUBJECT_CHARSET)
val subscriptionId = cursor.getLongValue(Mms.SUBSCRIPTION_ID)
val transactionId = cursor.getStringValueOrNull(Mms.TRANSACTION_ID)
val parts = getParts(mmsId)
val addresses = getMMSAddresses(mmsId)
block(MmsBackup(creator, contentType, deliveryReport, date, dateSent, locked, messageType, messageBox, read, readReport, seen, textOnly, status, subject, subjectCharSet, subscriptionId, transactionId, addresses, parts))
} }
} }
@SuppressLint("NewApi") @SuppressLint("NewApi")
private fun getParts(mmsId: Long): JsonArray { private fun getParts(mmsId: Long): List<MmsPart> {
val jsonArray = JsonArray() val parts = mutableListOf<MmsPart>()
val uri = if (isQPlus()) { val uri = if (isQPlus()) {
Telephony.Mms.Part.CONTENT_URI Mms.Part.CONTENT_URI
} else { } else {
Uri.parse("content://mms/part") Uri.parse("content://mms/part")
} }
val projection = arrayOf(
val selection = "${Telephony.Mms.Part.MSG_ID}= ?" Mms.Part._ID,
Mms.Part.CONTENT_DISPOSITION,
Mms.Part.CHARSET,
Mms.Part.CONTENT_ID,
Mms.Part.CONTENT_LOCATION,
Mms.Part.CONTENT_TYPE,
Mms.Part.CT_START,
Mms.Part.CT_TYPE,
Mms.Part.FILENAME,
Mms.Part.NAME,
Mms.Part.SEQ,
Mms.Part.TEXT
)
val selection = "${Mms.Part.MSG_ID}= ?"
val selectionArgs = arrayOf(mmsId.toString()) val selectionArgs = arrayOf(mmsId.toString())
context.queryCursor(uri, null, selection, selectionArgs) { cursor -> context.queryCursor(uri, projection, selection, selectionArgs) { cursor ->
val part = cursor.rowsToJson() val partId = cursor.getLongValue(Mms.Part._ID)
val contentDisposition = cursor.getStringValueOrNull(Mms.Part.CONTENT_DISPOSITION)
val hasTextValue = (part.has(Telephony.Mms.Part.TEXT) && !part.get(Telephony.Mms.Part.TEXT).optString.isNullOrEmpty()) val charset = cursor.getStringValueOrNull(Mms.Part.CHARSET)
val contentId = cursor.getStringValueOrNull(Mms.Part.CONTENT_ID)
when { val contentLocation = cursor.getStringValueOrNull(Mms.Part.CONTENT_LOCATION)
hasTextValue -> { val contentType = cursor.getStringValue(Mms.Part.CONTENT_TYPE)
part.addProperty(MMS_CONTENT, "") val ctStart = cursor.getStringValueOrNull(Mms.Part.CT_START)
} val ctType = cursor.getStringValueOrNull(Mms.Part.CT_TYPE)
val filename = cursor.getStringValueOrNull(Mms.Part.FILENAME)
part.get(Telephony.Mms.Part.CONTENT_TYPE).optString?.startsWith("text/") == true -> { val name = cursor.getStringValueOrNull(Mms.Part.NAME)
part.addProperty(MMS_CONTENT, usePart(part.get(Telephony.Mms.Part._ID).asLong) { stream -> val sequenceOrder = cursor.getIntValue(Mms.Part.SEQ)
val text = cursor.getStringValueOrNull(Mms.Part.TEXT)
val data = when {
contentType.startsWith("text/") -> {
usePart(partId) { stream ->
stream.readBytes().toString(Charsets.UTF_8) stream.readBytes().toString(Charsets.UTF_8)
}) }
} }
else -> { else -> {
part.addProperty(MMS_CONTENT, usePart(part.get(Telephony.Mms.Part._ID).asLong) { stream -> usePart(partId) { stream ->
val arr = stream.readBytes() val arr = stream.readBytes()
Log.d(TAG, "getParts: $arr") Log.d(TAG, "getParts: $arr")
Log.d(TAG, "getParts: size = ${arr.size}") Log.d(TAG, "getParts: size = ${arr.size}")
Log.d(TAG, "getParts: US_ASCII-> ${arr.toString(Charsets.US_ASCII)}") Log.d(TAG, "getParts: US_ASCII-> ${arr.toString(Charsets.US_ASCII)}")
Log.d(TAG, "getParts: UTF_8-> ${arr.toString(Charsets.UTF_8)}") Log.d(TAG, "getParts: UTF_8-> ${arr.toString(Charsets.UTF_8)}")
Base64.encodeToString(arr, Base64.DEFAULT) Base64.encodeToString(arr, Base64.DEFAULT)
}) }
} }
} }
jsonArray.add(part) parts.add(MmsPart(contentDisposition, charset, contentId, contentLocation, contentType, ctStart, ctType, filename, name, sequenceOrder, text, data))
} }
return parts
return jsonArray
} }
@SuppressLint("NewApi") @SuppressLint("NewApi")
private fun usePart(partId: Long, block: (InputStream) -> String): String { private fun usePart(partId: Long, block: (InputStream) -> String): String {
val partUri = if (isQPlus()) { val partUri = if (isQPlus()) {
Telephony.Mms.Part.CONTENT_URI.buildUpon().appendPath(partId.toString()).build() Mms.Part.CONTENT_URI.buildUpon().appendPath(partId.toString()).build()
} else { } else {
Uri.parse("content://mms/part/$partId") Uri.parse("content://mms/part/$partId")
} }
@@ -120,26 +195,28 @@ class MessagesReader(private val context: Context) {
} }
@SuppressLint("NewApi") @SuppressLint("NewApi")
private fun getMMSAddresses(messageId: Long): JsonArray { private fun getMMSAddresses(messageId: Long): List<MmsAddress> {
val jsonArray = JsonArray() val addresses = mutableListOf<MmsAddress>()
val addressUri = if (isQPlus()) { val uri = if (isQPlus()) {
Telephony.Mms.Addr.getAddrUriForMessage(messageId.toString()) Mms.Addr.getAddrUriForMessage(messageId.toString())
} else { } else {
Uri.parse("content://mms/$messageId/addr") Uri.parse("content://mms/$messageId/addr")
} }
val projection = arrayOf(Mms.Addr.ADDRESS, Mms.Addr.TYPE, Mms.Addr.CHARSET)
val projection = arrayOf(Telephony.Mms.Addr.ADDRESS, Telephony.Mms.Addr.TYPE) val selection = "${Mms.Addr.MSG_ID}= ?"
val selection = "${Telephony.Mms.Addr.MSG_ID}= ?"
val selectionArgs = arrayOf(messageId.toString()) val selectionArgs = arrayOf(messageId.toString())
context.queryCursor(addressUri, null, selection, selectionArgs) { cursor -> context.queryCursor(uri, projection, selection, selectionArgs) { cursor ->
val part = cursor.rowsToJson() val address = cursor.getStringValue(Mms.Addr.ADDRESS)
jsonArray.add(part) val type = cursor.getIntValue(Mms.Addr.TYPE)
// when (cursor.getIntValue(Telephony.Mms.Addr.TYPE)) { val charset = cursor.getIntValue(Mms.Addr.CHARSET)
// PduHeaders.FROM, PduHeaders.TO, PduHeaders.CC, PduHeaders.BCC -> jsonArray.add(cursor.getStringValue(Telephony.Mms.Addr.ADDRESS)) addresses.add(MmsAddress(address, type, charset))
// }
} }
return jsonArray return addresses
}
fun getMessagesCount(): Long {
return 0
} }
} }

View File

@@ -5,20 +5,19 @@ import android.content.ContentValues
import android.content.Context import android.content.Context
import android.net.Uri import android.net.Uri
import android.provider.Telephony import android.provider.Telephony
import android.provider.Telephony.Mms import android.provider.Telephony.*
import android.util.Base64 import android.util.Base64
import android.util.Log import android.util.Log
import androidx.core.content.contentValuesOf
import com.google.android.mms.pdu_alt.PduHeaders import com.google.android.mms.pdu_alt.PduHeaders
import com.klinker.android.send_message.Utils import com.klinker.android.send_message.Utils
import com.simplemobiletools.commons.extensions.getLongValue import com.simplemobiletools.commons.extensions.getLongValue
import com.simplemobiletools.commons.extensions.queryCursor import com.simplemobiletools.commons.extensions.queryCursor
import com.simplemobiletools.commons.helpers.isQPlus import com.simplemobiletools.commons.helpers.isQPlus
import com.simplemobiletools.smsmessenger.extensions.getThreadId
import com.simplemobiletools.smsmessenger.models.MmsAddress import com.simplemobiletools.smsmessenger.models.MmsAddress
import com.simplemobiletools.smsmessenger.models.MmsBackup import com.simplemobiletools.smsmessenger.models.MmsBackup
import com.simplemobiletools.smsmessenger.models.MmsPart import com.simplemobiletools.smsmessenger.models.MmsPart
import com.simplemobiletools.smsmessenger.models.SmsBackup import com.simplemobiletools.smsmessenger.models.SmsBackup
import java.nio.charset.Charset
class MessagesWriter(private val context: Context) { class MessagesWriter(private val context: Context) {
companion object { companion object {
@@ -31,66 +30,79 @@ class MessagesWriter(private val context: Context) {
fun writeSmsMessage(smsBackup: SmsBackup) { fun writeSmsMessage(smsBackup: SmsBackup) {
Log.w(TAG, "writeSmsMessage: smsBackup=$smsBackup") Log.w(TAG, "writeSmsMessage: smsBackup=$smsBackup")
val contentValues = smsBackup.toContentValues() val contentValues = smsBackup.toContentValues()
replaceSmsThreadId(contentValues) val threadId = Utils.getOrCreateThreadId(context, smsBackup.address)
contentValues.put(Sms.THREAD_ID, threadId)
Log.d(TAG, "writeSmsMessage: contentValues=$contentValues") Log.d(TAG, "writeSmsMessage: contentValues=$contentValues")
Log.d(TAG, "writeSmsMessage: type=${smsBackup.type}") Log.d(TAG, "writeSmsMessage: type=${smsBackup.type}")
if ((smsBackup.type == Telephony.Sms.MESSAGE_TYPE_INBOX || smsBackup.type == Telephony.Sms.MESSAGE_TYPE_SENT) && !smsExist(smsBackup)) { if (!smsExist(smsBackup)) {
Log.d(TAG, "writeSmsMessage: Inserting SMS...") Log.d(TAG, "writeSmsMessage: Inserting SMS...")
val uri = Telephony.Sms.CONTENT_URI val uri = Sms.CONTENT_URI
Log.d(TAG, "writeSmsMessage: uri=$uri") Log.d(TAG, "writeSmsMessage: uri=$uri")
contentResolver.insert(Telephony.Sms.CONTENT_URI, contentValues) contentResolver.insert(Sms.CONTENT_URI, contentValues)
} else {
Log.w(TAG, "SMS already exists")
} }
// update conversation date
updateThreadDate(threadId, smsBackup.date)
} }
private fun replaceSmsThreadId(contentValues: ContentValues) { private fun updateThreadDate(
val address = contentValues.get(Telephony.Sms.ADDRESS) threadId: Long,
val threadId = Utils.getOrCreateThreadId(context, address.toString()) date: Long,
contentValues.put(Telephony.Sms.THREAD_ID, threadId) ) {
Log.d(TAG, "updateThreadDate: threadId=$threadId -- date=$date")
val selection = "${Threads._ID} = ?"
val selectionArgs = arrayOf(threadId.toString())
val threadValues = contentValuesOf(Threads.DATE to date)
Log.d(TAG, "threadValues=$threadValues")
val result = contentResolver.update(Threads.CONTENT_URI.buildUpon().appendPath(threadId.toString()).build(), threadValues, null, null)
Log.d(TAG, "updateThreadDate: id=$result")
} }
private fun smsExist(smsBackup: SmsBackup): Boolean { private fun smsExist(smsBackup: SmsBackup): Boolean {
val uri = Telephony.Sms.CONTENT_URI val uri = Sms.CONTENT_URI
val projection = arrayOf(Telephony.Sms._ID) val projection = arrayOf(Sms._ID)
val selection = "${Telephony.Sms.DATE} = ? AND ${Telephony.Sms.ADDRESS} = ? AND ${Telephony.Sms.TYPE} = ?" val selection = "${Sms.DATE} = ? AND ${Sms.ADDRESS} = ? AND ${Sms.TYPE} = ?"
val selectionArgs = arrayOf(smsBackup.date.toString(), smsBackup.address, smsBackup.type.toString()) val selectionArgs = arrayOf(smsBackup.date.toString(), smsBackup.address, smsBackup.type.toString())
var exists = false var exists = false
context.queryCursor(uri, projection, selection, selectionArgs) { context.queryCursor(uri, projection, selection, selectionArgs) {
exists = it.count > 0 exists = it.count > 0
Log.i(TAG, "smsExist After: $exists") Log.i(TAG, "smsExist After: $exists")
} }
Log.i(TAG, "smsExist: $exists") Log.i(TAG, "smsExist: $exists")
return exists return exists
} }
fun writeMmsMessage(mmsBackup: MmsBackup) { fun writeMmsMessage(mmsBackup: MmsBackup) {
// 1. write mms msg, get the msg_id, check if mms exists before writing // 1. write mms msg, get the msg_id, check if mms exists before writing
// 2. write parts - parts depend on the msg id, check if part exist before writing, store _data in Downloads directory // 2. write parts - parts depend on the msg id, check if part exist before writing, write data if it is a non-text part
// 3. write the addresses, address depends on msg id too, check if address exist before writing // 3. write the addresses, address depends on msg id too, check if address exist before writing
Log.w(TAG, "writeMmsMessage: backup=$mmsBackup") Log.w(TAG, "writeMmsMessage: backup=$mmsBackup")
val contentValues = mmsBackup.toContentValues() val contentValues = mmsBackup.toContentValues()
val threadId = getMmsThreadId(mmsBackup) val threadId = getMmsThreadId(mmsBackup)
if (threadId != INVALID_ID) { if (threadId != INVALID_ID) {
contentValues.put(Telephony.Mms.THREAD_ID, threadId) contentValues.put(Mms.THREAD_ID, threadId)
Log.w(TAG, "writeMmsMessage: backup=$mmsBackup") Log.w(TAG, "writeMmsMessage: backup=$mmsBackup")
//write mms //write mms
if ((mmsBackup.messageBox == Telephony.Mms.MESSAGE_BOX_INBOX || mmsBackup.messageBox == Telephony.Mms.MESSAGE_BOX_SENT) && !mmsExist(mmsBackup)) { if (!mmsExist(mmsBackup)) {
contentResolver.insert(Telephony.Mms.CONTENT_URI, contentValues) contentResolver.insert(Mms.CONTENT_URI, contentValues)
updateThreadDate(threadId, mmsBackup.date)
} else {
Log.w(TAG, "mms already exists")
} }
val messageId = getMmsId(mmsBackup) val messageId = getMmsId(mmsBackup)
if (messageId != INVALID_ID) { if (messageId != INVALID_ID) {
Log.d(TAG, "writing mms addresses") Log.d(TAG, "writing mms addresses")
//write addresses //write addresses
mmsBackup.addresses.forEach { writeMmsAddress(it, messageId) } mmsBackup.addresses.forEach { writeMmsAddress(it, messageId) }
mmsBackup.mmsParts.forEach { writeMmsPart(it, messageId) } mmsBackup.parts.forEach { writeMmsPart(it, messageId) }
} else { } else {
Log.d(TAG, "failed to write mms message, invalid mms id") Log.d(TAG, "failed to write mms message, invalid mms id")
} }
} else { } else {
Log.d(TAG, "failed to write mms message, invalid thread id") Log.d(TAG, "failed to write mms message, invalid thread id")
} }
} }
private fun getMmsThreadId(mmsBackup: MmsBackup): Long { private fun getMmsThreadId(mmsBackup: MmsBackup): Long {
@@ -111,14 +123,14 @@ class MessagesWriter(private val context: Context) {
private fun getMmsId(mmsBackup: MmsBackup): Long { private fun getMmsId(mmsBackup: MmsBackup): Long {
val threadId = getMmsThreadId(mmsBackup) val threadId = getMmsThreadId(mmsBackup)
val uri = Telephony.Mms.CONTENT_URI val uri = Mms.CONTENT_URI
val projection = arrayOf(Telephony.Mms._ID) val projection = arrayOf(Mms._ID)
val selection = "${Telephony.Mms.DATE} = ? AND ${Telephony.Mms.DATE_SENT} = ? AND ${Telephony.Mms.THREAD_ID} = ? AND ${Telephony.Mms.MESSAGE_BOX} = ?" val selection = "${Mms.DATE} = ? AND ${Mms.DATE_SENT} = ? AND ${Mms.THREAD_ID} = ? AND ${Mms.MESSAGE_BOX} = ?"
val selectionArgs = arrayOf(mmsBackup.date.toString(), mmsBackup.dateSent.toString(), threadId.toString(), mmsBackup.messageBox.toString()) val selectionArgs = arrayOf(mmsBackup.date.toString(), mmsBackup.dateSent.toString(), threadId.toString(), mmsBackup.messageBox.toString())
var id = INVALID_ID var id = INVALID_ID
context.queryCursor(uri, projection, selection, selectionArgs) { context.queryCursor(uri, projection, selection, selectionArgs) {
id = it.getLongValue(Telephony.Mms._ID) id = it.getLongValue(Mms._ID)
Log.i(TAG, "getMmsId After: $id") Log.i(TAG, "getMmsId After: $id")
} }
Log.i(TAG, "getMmsId: $id") Log.i(TAG, "getMmsId: $id")
@@ -133,12 +145,12 @@ class MessagesWriter(private val context: Context) {
@SuppressLint("NewApi") @SuppressLint("NewApi")
private fun mmsAddressExist(mmsAddress: MmsAddress, messageId: Long): Boolean { private fun mmsAddressExist(mmsAddress: MmsAddress, messageId: Long): Boolean {
val addressUri = if (isQPlus()) { val addressUri = if (isQPlus()) {
Telephony.Mms.Addr.getAddrUriForMessage(messageId.toString()) Mms.Addr.getAddrUriForMessage(messageId.toString())
} else { } else {
Uri.parse("content://mms/$messageId/addr") Uri.parse("content://mms/$messageId/addr")
} }
val projection = arrayOf(Telephony.Mms.Addr._ID) val projection = arrayOf(Mms.Addr._ID)
val selection = "${Telephony.Mms.Addr.TYPE} = ? AND ${Telephony.Mms.Addr.ADDRESS} = ? AND ${Telephony.Mms.Addr.MSG_ID} = ?" val selection = "${Mms.Addr.TYPE} = ? AND ${Mms.Addr.ADDRESS} = ? AND ${Mms.Addr.MSG_ID} = ?"
val selectionArgs = arrayOf(mmsAddress.type.toString(), mmsAddress.address.toString(), messageId.toString()) val selectionArgs = arrayOf(mmsAddress.type.toString(), mmsAddress.address.toString(), messageId.toString())
var exists = false var exists = false
@@ -155,13 +167,13 @@ class MessagesWriter(private val context: Context) {
private fun writeMmsAddress(mmsAddress: MmsAddress, messageId: Long) { private fun writeMmsAddress(mmsAddress: MmsAddress, messageId: Long) {
if (!mmsAddressExist(mmsAddress, messageId)) { if (!mmsAddressExist(mmsAddress, messageId)) {
val addressUri = if (isQPlus()) { val addressUri = if (isQPlus()) {
Telephony.Mms.Addr.getAddrUriForMessage(messageId.toString()) Mms.Addr.getAddrUriForMessage(messageId.toString())
} else { } else {
Uri.parse("content://mms/$messageId/addr") Uri.parse("content://mms/$messageId/addr")
} }
val contentValues = mmsAddress.toContentValues() val contentValues = mmsAddress.toContentValues()
contentValues.put(Telephony.Mms.Addr.MSG_ID, messageId) contentValues.put(Mms.Addr.MSG_ID, messageId)
contentResolver.insert(addressUri, contentValues) contentResolver.insert(addressUri, contentValues)
} else { } else {
Log.w(TAG, "writeMmsAddress: Skip already exists") Log.w(TAG, "writeMmsAddress: Skip already exists")
@@ -182,14 +194,13 @@ class MessagesWriter(private val context: Context) {
if (partUri != null) { if (partUri != null) {
if (mmsPart.isNonText()) { if (mmsPart.isNonText()) {
contentResolver.openOutputStream(partUri).use { contentResolver.openOutputStream(partUri).use {
val arr = Base64.decode(mmsPart.mmsContent, Base64.DEFAULT) val arr = Base64.decode(mmsPart.data, Base64.DEFAULT)
it!!.write(arr) it!!.write(arr)
Log.d(TAG, "Wrote part data $mmsPart") Log.d(TAG, "Wrote part data $mmsPart")
} }
} else { } else {
Log.w(TAG, "skip writing text content") Log.w(TAG, "skip writing text content")
} }
} else { } else {
Log.e(TAG, "invalid uri while writing part") Log.e(TAG, "invalid uri while writing part")
} }
@@ -202,9 +213,9 @@ class MessagesWriter(private val context: Context) {
@SuppressLint("NewApi") @SuppressLint("NewApi")
private fun mmsPartExist(mmsPart: MmsPart, messageId: Long): Boolean { private fun mmsPartExist(mmsPart: MmsPart, messageId: Long): Boolean {
val uri = Uri.parse("content://mms/${messageId}/part") val uri = Uri.parse("content://mms/${messageId}/part")
val projection = arrayOf(Telephony.Mms.Part._ID) val projection = arrayOf(Mms.Part._ID)
val selection = val selection =
"${Telephony.Mms.Part.CONTENT_LOCATION} = ? AND ${Telephony.Mms.Part.CT_TYPE} = ? AND ${Mms.Part.MSG_ID} = ? AND ${Telephony.Mms.Part.CONTENT_ID} = ?" "${Mms.Part.CONTENT_LOCATION} = ? AND ${Mms.Part.CT_TYPE} = ? AND ${Mms.Part.MSG_ID} = ? AND ${Mms.Part.CONTENT_ID} = ?"
val selectionArgs = arrayOf(mmsPart.contentLocation.toString(), mmsPart.contentType.toString(), messageId.toString(), mmsPart.contentId.toString()) val selectionArgs = arrayOf(mmsPart.contentLocation.toString(), mmsPart.contentType.toString(), messageId.toString(), mmsPart.contentId.toString())
var exists = false var exists = false
context.queryCursor(uri, projection, selection, selectionArgs) { context.queryCursor(uri, projection, selection, selectionArgs) {
@@ -220,6 +231,6 @@ class MessagesWriter(private val context: Context) {
// thread dates + states might be wrong, we need to force a full update // thread dates + states might be wrong, we need to force a full update
// unfortunately there's no direct way to do that in the SDK, but passing a // unfortunately there's no direct way to do that in the SDK, but passing a
// negative conversation id to delete should to the trick // negative conversation id to delete should to the trick
contentResolver.delete(Telephony.Sms.Conversations.CONTENT_URI.buildUpon().appendPath("-1").build(), null, null) contentResolver.delete(Sms.Conversations.CONTENT_URI.buildUpon().appendPath("-1").build(), null, null)
} }
} }

View File

@@ -3,8 +3,6 @@ package com.simplemobiletools.smsmessenger.models
import com.google.gson.annotations.SerializedName import com.google.gson.annotations.SerializedName
data class ExportedMessage( data class ExportedMessage(
@SerializedName("threadId")
val threadId: Long,
@SerializedName("sms") @SerializedName("sms")
val sms: List<SmsBackup>, val sms: List<SmsBackup>,
@SerializedName("mms") @SerializedName("mms")

View File

@@ -8,8 +8,6 @@ import com.google.gson.annotations.SerializedName
data class MmsAddress( data class MmsAddress(
@SerializedName("address") @SerializedName("address")
val address: String, val address: String,
@SerializedName("msg_id")
val msgId: Int,
@SerializedName("type") @SerializedName("type")
val type: Int, val type: Int,
@SerializedName("charset") @SerializedName("charset")

View File

@@ -18,8 +18,6 @@ data class MmsBackup(
val dateSent: Long, val dateSent: Long,
@SerializedName("locked") @SerializedName("locked")
val locked: Int, val locked: Int,
@SerializedName("m_id")
val messageId: String?,
@SerializedName("m_type") @SerializedName("m_type")
val messageType: Int, val messageType: Int,
@SerializedName("msg_box") @SerializedName("msg_box")
@@ -40,14 +38,12 @@ data class MmsBackup(
val subjectCharSet: String?, val subjectCharSet: String?,
@SerializedName("sub_id") @SerializedName("sub_id")
val subscriptionId: Long, val subscriptionId: Long,
@SerializedName("thread_id")
val threadId: Long,
@SerializedName("tr_id") @SerializedName("tr_id")
val transactionId: String?, val transactionId: String?,
@SerializedName("addresses") @SerializedName("addresses")
val addresses: List<MmsAddress>, val addresses: List<MmsAddress>,
@SerializedName("parts") @SerializedName("parts")
val mmsParts: List<MmsPart>, val parts: List<MmsPart>,
) { ) {
fun toContentValues(): ContentValues { fun toContentValues(): ContentValues {

View File

@@ -20,22 +20,16 @@ data class MmsPart(
val ctStart: String?, val ctStart: String?,
@SerializedName("ctt_t") @SerializedName("ctt_t")
val ctType: String?, val ctType: String?,
@SerializedName("_data")
val `data`: String?,
@SerializedName("fn") @SerializedName("fn")
val filename: String?, val filename: String?,
@SerializedName("_id")
val id: Long,
@SerializedName("mid")
val messageId: Long,
@SerializedName("name") @SerializedName("name")
val name: String, val name: String?,
@SerializedName("seq") @SerializedName("seq")
val sequenceOrder: Int, val sequenceOrder: Int,
@SerializedName("text") @SerializedName("text")
val text: String?, val text: String?,
@SerializedName("mms_content") @SerializedName("data")
val mmsContent: String?, val data: String?,
) { ) {
fun toContentValues(): ContentValues { fun toContentValues(): ContentValues {

View File

@@ -7,14 +7,12 @@ import androidx.core.content.contentValuesOf
import com.google.gson.annotations.SerializedName import com.google.gson.annotations.SerializedName
data class SmsBackup( data class SmsBackup(
@SerializedName("thread_id")
val threadId: Long,
@SerializedName("sub_id") @SerializedName("sub_id")
val subscriptionId: Long, val subscriptionId: Long,
@SerializedName("address") @SerializedName("address")
val address: String, val address: String,
@SerializedName("body") @SerializedName("body")
val body: String, val body: String?,
@SerializedName("date") @SerializedName("date")
val date: Long, val date: Long,
@SerializedName("date_sent") @SerializedName("date_sent")
@@ -35,7 +33,6 @@ data class SmsBackup(
fun toContentValues(): ContentValues { fun toContentValues(): ContentValues {
return contentValuesOf( return contentValuesOf(
Telephony.Sms.THREAD_ID to threadId,
Telephony.Sms.SUBSCRIPTION_ID to subscriptionId, Telephony.Sms.SUBSCRIPTION_ID to subscriptionId,
Telephony.Sms.ADDRESS to address, Telephony.Sms.ADDRESS to address,
Telephony.Sms.BODY to body, Telephony.Sms.BODY to body,