package org.thoughtcrime.securesms.jobs; import android.content.Context; import android.support.annotation.NonNull; import android.support.annotation.Nullable; import android.telephony.SmsMessage; import org.thoughtcrime.securesms.jobmanager.SafeData; import org.thoughtcrime.securesms.logging.Log; import org.thoughtcrime.securesms.database.DatabaseFactory; import org.thoughtcrime.securesms.database.MessagingDatabase.InsertResult; import org.thoughtcrime.securesms.database.SmsDatabase; import org.thoughtcrime.securesms.jobmanager.JobParameters; import org.thoughtcrime.securesms.notifications.MessageNotifier; import org.thoughtcrime.securesms.recipients.Recipient; import org.thoughtcrime.securesms.sms.IncomingTextMessage; import org.thoughtcrime.securesms.util.Base64; import org.thoughtcrime.securesms.util.TextSecurePreferences; import org.whispersystems.libsignal.util.guava.Optional; import java.io.IOException; import java.util.LinkedList; import java.util.List; import androidx.work.Data; public class SmsReceiveJob extends ContextJob { private static final long serialVersionUID = 1L; private static final String TAG = SmsReceiveJob.class.getSimpleName(); private static final String KEY_PDUS = "pdus"; private static final String KEY_SUBSCRIPTION_ID = "subscription_id"; private @Nullable Object[] pdus; private int subscriptionId; public SmsReceiveJob() { super(null, null); } public SmsReceiveJob(@NonNull Context context, @Nullable Object[] pdus, int subscriptionId) { super(context, JobParameters.newBuilder() .withSqlCipherRequirement() .create()); this.pdus = pdus; this.subscriptionId = subscriptionId; } @Override protected void initialize(@NonNull SafeData data) { String[] encoded = data.getStringArray(KEY_PDUS); pdus = new Object[encoded.length]; try { for (int i = 0; i < encoded.length; i++) { pdus[i] = Base64.decode(encoded[i]); } } catch (IOException e) { throw new AssertionError(e); } subscriptionId = data.getInt(KEY_SUBSCRIPTION_ID); } @Override protected @NonNull Data serialize(@NonNull Data.Builder dataBuilder) { String[] encoded = new String[pdus.length]; for (int i = 0; i < pdus.length; i++) { encoded[i] = Base64.encodeBytes((byte[]) pdus[i]); } return dataBuilder.putStringArray(KEY_PDUS, encoded) .putInt(KEY_SUBSCRIPTION_ID, subscriptionId) .build(); } @Override public void onRun() throws MigrationPendingException { Log.i(TAG, "onRun()"); Optional message = assembleMessageFragments(pdus, subscriptionId); if (message.isPresent() && !isBlocked(message.get())) { Optional insertResult = storeMessage(message.get()); if (insertResult.isPresent()) { MessageNotifier.updateNotification(context, insertResult.get().getThreadId()); } } else if (message.isPresent()) { Log.w(TAG, "*** Received blocked SMS, ignoring..."); } else { Log.w(TAG, "*** Failed to assemble message fragments!"); } } @Override public void onCanceled() { } @Override public boolean onShouldRetry(Exception exception) { return exception instanceof MigrationPendingException; } private boolean isBlocked(IncomingTextMessage message) { if (message.getSender() != null) { Recipient recipient = Recipient.from(context, message.getSender(), false); return recipient.isBlocked(); } return false; } private Optional storeMessage(IncomingTextMessage message) throws MigrationPendingException { SmsDatabase database = DatabaseFactory.getSmsDatabase(context); database.ensureMigration(); if (TextSecurePreferences.getNeedsSqlCipherMigration(context)) { throw new MigrationPendingException(); } if (message.isSecureMessage()) { IncomingTextMessage placeholder = new IncomingTextMessage(message, ""); Optional insertResult = database.insertMessageInbox(placeholder); database.markAsLegacyVersion(insertResult.get().getMessageId()); return insertResult; } else { return database.insertMessageInbox(message); } } private Optional assembleMessageFragments(@Nullable Object[] pdus, int subscriptionId) { if (pdus == null) { return Optional.absent(); } List messages = new LinkedList<>(); for (Object pdu : pdus) { messages.add(new IncomingTextMessage(context, SmsMessage.createFromPdu((byte[])pdu), subscriptionId)); } if (messages.isEmpty()) { return Optional.absent(); } return Optional.of(new IncomingTextMessage(messages)); } private class MigrationPendingException extends Exception { } }