package org.thoughtcrime.securesms.jobs; import android.content.Context; import android.util.Log; import org.thoughtcrime.securesms.ApplicationContext; import org.thoughtcrime.securesms.crypto.MasterSecret; import org.thoughtcrime.securesms.crypto.storage.TextSecureAxolotlStore; import org.thoughtcrime.securesms.database.DatabaseFactory; import org.thoughtcrime.securesms.database.EncryptingSmsDatabase; import org.thoughtcrime.securesms.database.NoSuchMessageException; import org.thoughtcrime.securesms.database.SmsDatabase; import org.thoughtcrime.securesms.database.model.SmsMessageRecord; import org.thoughtcrime.securesms.dependencies.InjectableType; import org.thoughtcrime.securesms.notifications.MessageNotifier; import org.thoughtcrime.securesms.recipients.Recipient; import org.thoughtcrime.securesms.recipients.RecipientFactory; import org.thoughtcrime.securesms.recipients.RecipientFormattingException; import org.thoughtcrime.securesms.recipients.Recipients; import org.thoughtcrime.securesms.transport.InsecureFallbackApprovalException; import org.thoughtcrime.securesms.transport.RetryLaterException; import org.thoughtcrime.securesms.transport.SecureFallbackApprovalException; import org.whispersystems.libaxolotl.state.AxolotlStore; import org.whispersystems.textsecure.api.TextSecureMessageSender; import org.whispersystems.textsecure.api.crypto.UntrustedIdentityException; import org.whispersystems.textsecure.api.messages.TextSecureMessage; import org.whispersystems.textsecure.api.push.TextSecureAddress; import org.whispersystems.textsecure.api.push.exceptions.UnregisteredUserException; import org.whispersystems.textsecure.api.util.InvalidNumberException; import java.io.IOException; import javax.inject.Inject; import static org.thoughtcrime.securesms.dependencies.TextSecureCommunicationModule.TextSecureMessageSenderFactory; public class PushTextSendJob extends PushSendJob implements InjectableType { private static final String TAG = PushTextSendJob.class.getSimpleName(); @Inject transient TextSecureMessageSenderFactory messageSenderFactory; private final long messageId; public PushTextSendJob(Context context, long messageId, String destination) { super(context, constructParameters(context, destination, false)); this.messageId = messageId; } @Override public void onAdded() { SmsDatabase smsDatabase = DatabaseFactory.getSmsDatabase(context); smsDatabase.markAsSending(messageId); smsDatabase.markAsPush(messageId); } @Override public void onSend(MasterSecret masterSecret) throws NoSuchMessageException, RetryLaterException, RecipientFormattingException { EncryptingSmsDatabase database = DatabaseFactory.getEncryptingSmsDatabase(context); SmsMessageRecord record = database.getMessage(masterSecret, messageId); String destination = record.getIndividualRecipient().getNumber(); try { Log.w(TAG, "Sending message: " + messageId); if (deliver(masterSecret, record, destination)) { database.markAsPush(messageId); database.markAsSecure(messageId); database.markAsSent(messageId); } } catch (InsecureFallbackApprovalException e) { Log.w(TAG, e); database.markAsPendingInsecureSmsFallback(record.getId()); MessageNotifier.notifyMessageDeliveryFailed(context, record.getRecipients(), record.getThreadId()); } catch (SecureFallbackApprovalException e) { Log.w(TAG, e); database.markAsPendingSecureSmsFallback(record.getId()); MessageNotifier.notifyMessageDeliveryFailed(context, record.getRecipients(), record.getThreadId()); } catch (UntrustedIdentityException e) { Log.w(TAG, e); Recipients recipients = RecipientFactory.getRecipientsFromString(context, e.getE164Number(), false); long recipientId = recipients.getPrimaryRecipient().getRecipientId(); database.addMismatchedIdentity(record.getId(), recipientId, e.getIdentityKey()); database.markAsSentFailed(record.getId()); database.markAsPush(record.getId()); } } @Override public boolean onShouldRetryThrowable(Exception exception) { if (exception instanceof RetryLaterException) return true; return false; } @Override public void onCanceled() { DatabaseFactory.getSmsDatabase(context).markAsSentFailed(messageId); long threadId = DatabaseFactory.getSmsDatabase(context).getThreadIdForMessage(messageId); Recipients recipients = DatabaseFactory.getThreadDatabase(context).getRecipientsForThreadId(threadId); MessageNotifier.notifyMessageDeliveryFailed(context, recipients, threadId); } private boolean deliver(MasterSecret masterSecret, SmsMessageRecord message, String destination) throws UntrustedIdentityException, SecureFallbackApprovalException, InsecureFallbackApprovalException, RetryLaterException { boolean isSmsFallbackSupported = isSmsFallbackSupported(context, destination, false); try { TextSecureAddress address = getPushAddress(message.getIndividualRecipient()); TextSecureMessageSender messageSender = messageSenderFactory.create(masterSecret); if (message.isEndSession()) { messageSender.sendMessage(address, new TextSecureMessage(message.getDateSent(), null, null, null, true, true)); } else { messageSender.sendMessage(address, new TextSecureMessage(message.getDateSent(), message.getBody().getBody())); } return true; } catch (InvalidNumberException | UnregisteredUserException e) { Log.w(TAG, e); if (isSmsFallbackSupported) fallbackOrAskApproval(masterSecret, message, destination); else DatabaseFactory.getSmsDatabase(context).markAsSentFailed(messageId); } catch (IOException e) { Log.w(TAG, e); if (isSmsFallbackSupported) fallbackOrAskApproval(masterSecret, message, destination); else throw new RetryLaterException(e); } return false; } private void fallbackOrAskApproval(MasterSecret masterSecret, SmsMessageRecord smsMessage, String destination) throws SecureFallbackApprovalException, InsecureFallbackApprovalException { Recipient recipient = smsMessage.getIndividualRecipient(); boolean isSmsFallbackApprovalRequired = isSmsFallbackApprovalRequired(destination, false); AxolotlStore axolotlStore = new TextSecureAxolotlStore(context, masterSecret); if (!isSmsFallbackApprovalRequired) { Log.w(TAG, "Falling back to SMS"); DatabaseFactory.getSmsDatabase(context).markAsForcedSms(smsMessage.getId()); ApplicationContext.getInstance(context).getJobManager().add(new SmsSendJob(context, messageId, destination)); } else if (!axolotlStore.containsSession(recipient.getRecipientId(), TextSecureAddress.DEFAULT_DEVICE_ID)) { Log.w(TAG, "Marking message as pending insecure fallback."); throw new InsecureFallbackApprovalException("Pending user approval for fallback to insecure SMS"); } else { Log.w(TAG, "Marking message as pending secure fallback."); throw new SecureFallbackApprovalException("Pending user approval for fallback to secure SMS"); } } }