package org.thoughtcrime.securesms.jobs; import android.support.annotation.NonNull; import com.annimon.stream.Stream; import org.thoughtcrime.securesms.crypto.UnidentifiedAccessUtil; import org.thoughtcrime.securesms.database.DatabaseFactory; import org.thoughtcrime.securesms.dependencies.InjectableType; import org.thoughtcrime.securesms.jobmanager.Data; import org.thoughtcrime.securesms.jobmanager.Job; import org.thoughtcrime.securesms.logging.Log; import org.thoughtcrime.securesms.loki.MultiDeviceUtilities; import org.thoughtcrime.securesms.recipients.Recipient; import org.thoughtcrime.securesms.util.GroupUtil; import org.thoughtcrime.securesms.util.TextSecurePreferences; import org.whispersystems.libsignal.util.guava.Optional; import org.whispersystems.signalservice.api.SignalServiceMessageSender; import org.whispersystems.signalservice.api.crypto.UnidentifiedAccessPair; import org.whispersystems.signalservice.api.messages.SignalServiceTypingMessage; import org.whispersystems.signalservice.api.messages.SignalServiceTypingMessage.Action; import org.whispersystems.signalservice.api.push.SignalServiceAddress; import org.whispersystems.signalservice.loki.utilities.PromiseUtil; import java.util.Collections; import java.util.List; import java.util.concurrent.TimeUnit; import javax.inject.Inject; public class TypingSendJob extends BaseJob implements InjectableType { public static final String KEY = "TypingSendJob"; private static final String TAG = TypingSendJob.class.getSimpleName(); private static final String KEY_THREAD_ID = "thread_id"; private static final String KEY_TYPING = "typing"; private long threadId; private boolean typing; @Inject SignalServiceMessageSender messageSender; public TypingSendJob(long threadId, boolean typing) { this(new Job.Parameters.Builder() .setQueue("TYPING_" + threadId) .setMaxAttempts(1) .setLifespan(TimeUnit.SECONDS.toMillis(5)) .build(), threadId, typing); } private TypingSendJob(@NonNull Job.Parameters parameters, long threadId, boolean typing) { super(parameters); this.threadId = threadId; this.typing = typing; } @Override public @NonNull Data serialize() { return new Data.Builder().putLong(KEY_THREAD_ID, threadId) .putBoolean(KEY_TYPING, typing) .build(); } @Override public @NonNull String getFactoryKey() { return KEY; } @Override public void onRun() throws Exception { if (!TextSecurePreferences.isTypingIndicatorsEnabled(context)) { return; } Log.d(TAG, "Sending typing " + (typing ? "started" : "stopped") + " for thread " + threadId); Recipient recipient = DatabaseFactory.getThreadDatabase(context).getRecipientForThreadId(threadId); if (recipient == null) { throw new IllegalStateException("Tried to send a typing indicator to a non-existent thread."); } List recipients = Collections.singletonList(recipient); Optional groupId = Optional.absent(); if (recipient.isGroupRecipient()) { recipients = DatabaseFactory.getGroupDatabase(context).getGroupMembers(recipient.getAddress().toGroupString(), false); groupId = Optional.of(GroupUtil.getDecodedId(recipient.getAddress().toGroupString())); } List addresses = Stream.of(recipients).map(r -> new SignalServiceAddress(r.getAddress().serialize())).toList(); List> unidentifiedAccess = Stream.of(recipients).map(r -> UnidentifiedAccessUtil.getAccessFor(context, r)).toList(); SignalServiceTypingMessage typingMessage = new SignalServiceTypingMessage(typing ? Action.STARTED : Action.STOPPED, System.currentTimeMillis(), groupId); // Loki - Don't send typing indicators in group chats or to ourselves if (recipient.isGroupRecipient()) { return; } boolean isOurDevice = PromiseUtil.get(MultiDeviceUtilities.isOneOfOurDevices(context, recipient.getAddress()), false); if (!isOurDevice) { messageSender.sendTyping(0, addresses, unidentifiedAccess, typingMessage); } } @Override public void onCanceled() { } @Override protected boolean onShouldRetry(@NonNull Exception exception) { return false; } public static final class Factory implements Job.Factory { @Override public @NonNull TypingSendJob create(@NonNull Parameters parameters, @NonNull Data data) { return new TypingSendJob(parameters, data.getLong(KEY_THREAD_ID), data.getBoolean(KEY_TYPING)); } } }