session-android/app/src/main/java/org/thoughtcrime/securesms/service/ExpiringMessageManager.java

237 lines
9.1 KiB
Java
Raw Normal View History

package org.thoughtcrime.securesms.service;
import android.content.Context;
2021-01-13 07:11:30 +01:00
import org.jetbrains.annotations.NotNull;
import org.jetbrains.annotations.Nullable;
2021-03-25 04:55:23 +01:00
import org.session.libsession.messaging.messages.control.ExpirationTimerUpdate;
2021-01-13 07:11:30 +01:00
import org.session.libsession.messaging.threads.Address;
import org.session.libsession.messaging.threads.recipients.Recipient;
2021-03-25 04:55:23 +01:00
import org.session.libsession.utilities.GroupUtil;
2021-01-13 07:11:30 +01:00
import org.session.libsession.utilities.SSKEnvironment;
import org.session.libsignal.libsignal.util.guava.Optional;
import org.session.libsignal.service.api.messages.SignalServiceGroup;
import org.session.libsignal.service.internal.push.SignalServiceProtos;
import org.session.libsignal.service.internal.push.SignalServiceProtos.GroupContext;
2021-02-03 02:22:40 +01:00
import org.session.libsignal.utilities.logging.Log;
import org.thoughtcrime.securesms.database.DatabaseFactory;
import org.thoughtcrime.securesms.database.MmsDatabase;
import org.thoughtcrime.securesms.database.SmsDatabase;
import org.thoughtcrime.securesms.database.model.MessageRecord;
2021-03-12 05:23:29 +01:00
import org.session.libsession.messaging.messages.signal.IncomingMediaMessage;
2021-01-13 07:11:30 +01:00
import org.thoughtcrime.securesms.mms.MmsException;
2021-03-25 04:55:23 +01:00
import java.io.IOException;
import java.util.Comparator;
import java.util.TreeSet;
import java.util.concurrent.Executor;
import java.util.concurrent.Executors;
2021-01-13 07:11:30 +01:00
public class ExpiringMessageManager implements SSKEnvironment.MessageExpirationManagerProtocol {
private static final String TAG = ExpiringMessageManager.class.getSimpleName();
private final TreeSet<ExpiringMessageReference> expiringMessageReferences = new TreeSet<>(new ExpiringMessageComparator());
private final Executor executor = Executors.newSingleThreadExecutor();
private final SmsDatabase smsDatabase;
private final MmsDatabase mmsDatabase;
private final Context context;
public ExpiringMessageManager(Context context) {
this.context = context.getApplicationContext();
this.smsDatabase = DatabaseFactory.getSmsDatabase(context);
this.mmsDatabase = DatabaseFactory.getMmsDatabase(context);
executor.execute(new LoadTask());
executor.execute(new ProcessTask());
}
public void scheduleDeletion(long id, boolean mms, long expiresInMillis) {
scheduleDeletion(id, mms, System.currentTimeMillis(), expiresInMillis);
}
public void scheduleDeletion(long id, boolean mms, long startedAtTimestamp, long expiresInMillis) {
long expiresAtMillis = startedAtTimestamp + expiresInMillis;
synchronized (expiringMessageReferences) {
expiringMessageReferences.add(new ExpiringMessageReference(id, mms, expiresAtMillis));
expiringMessageReferences.notifyAll();
}
}
public void checkSchedule() {
synchronized (expiringMessageReferences) {
expiringMessageReferences.notifyAll();
}
}
2021-01-13 07:11:30 +01:00
@Override
2021-03-25 04:55:23 +01:00
public void setExpirationTimer(@NotNull ExpirationTimerUpdate message) {
MmsDatabase database = DatabaseFactory.getMmsDatabase(context);
String senderPublicKey = message.getSender();
int duration = message.getDuration();
String groupPK = message.getGroupPublicKey();
Long sentTimestamp = message.getSentTimestamp();
Optional<SignalServiceGroup> groupInfo = Optional.absent();
Address address;
2021-01-13 07:11:30 +01:00
try {
2021-03-25 04:55:23 +01:00
if (groupPK != null) {
String groupID = GroupUtil.doubleEncodeGroupID(groupPK);
groupInfo = Optional.of(new SignalServiceGroup(GroupUtil.getDecodedGroupIDAsData(groupID), SignalServiceGroup.GroupType.SIGNAL));
address = Address.fromSerialized(groupID);
} else {
address = Address.fromSerialized(senderPublicKey);
}
2021-01-13 07:11:30 +01:00
Recipient recipient = Recipient.from(context, address, false);
if (recipient.isBlocked()) return;
2021-03-25 04:55:23 +01:00
IncomingMediaMessage mediaMessage = new IncomingMediaMessage(address, sentTimestamp, -1,
2021-01-13 07:11:30 +01:00
duration * 1000L, true,
false,
Optional.absent(),
groupInfo,
Optional.absent(),
Optional.absent(),
Optional.absent(),
Optional.absent());
2021-03-25 04:55:23 +01:00
//insert the timer update message
2021-01-13 07:11:30 +01:00
database.insertSecureDecryptedMessageInbox(mediaMessage, -1);
2021-03-25 04:55:23 +01:00
//set the timer to the conversation
2021-01-13 07:11:30 +01:00
DatabaseFactory.getRecipientDatabase(context).setExpireMessages(recipient, duration);
2021-03-25 04:55:23 +01:00
if (message.getId() != null) {
DatabaseFactory.getSmsDatabase(context).deleteMessage(message.getId());
2021-01-13 07:11:30 +01:00
}
} catch (MmsException e) {
Log.e("Loki", "Failed to insert expiration update message.");
2021-03-25 04:55:23 +01:00
} catch (IOException ioe) {
Log.e("Loki", "Failed to insert expiration update message.");
2021-01-13 07:11:30 +01:00
}
}
@Override
2021-03-25 04:55:23 +01:00
public void disableExpirationTimer(@NotNull ExpirationTimerUpdate message) {
setExpirationTimer(message);
2021-01-13 07:11:30 +01:00
}
2021-01-20 06:29:52 +01:00
@Override
2021-03-02 02:24:09 +01:00
public void startAnyExpiration(long timestamp, @NotNull String author) {
MessageRecord messageRecord = DatabaseFactory.getMmsSmsDatabase(context).getMessageFor(timestamp, author);
2021-01-20 06:29:52 +01:00
if (messageRecord != null) {
boolean mms = messageRecord.isMms();
Recipient recipient = messageRecord.getRecipient();
2021-03-02 02:24:09 +01:00
if (recipient.getExpireMessages() <= 0) return;
2021-01-20 06:29:52 +01:00
if (mms) {
2021-03-02 02:24:09 +01:00
mmsDatabase.markExpireStarted(messageRecord.getId());
2021-01-20 06:29:52 +01:00
} else {
2021-03-02 02:24:09 +01:00
smsDatabase.markExpireStarted(messageRecord.getId());
2021-01-20 06:29:52 +01:00
}
2021-03-04 07:14:12 +01:00
scheduleDeletion(messageRecord.getId(), mms, recipient.getExpireMessages() * 1000);
2021-01-20 06:29:52 +01:00
}
}
private class LoadTask implements Runnable {
public void run() {
SmsDatabase.Reader smsReader = smsDatabase.readerFor(smsDatabase.getExpirationStartedMessages());
MmsDatabase.Reader mmsReader = mmsDatabase.getExpireStartedMessages();
MessageRecord messageRecord;
while ((messageRecord = smsReader.getNext()) != null) {
expiringMessageReferences.add(new ExpiringMessageReference(messageRecord.getId(),
messageRecord.isMms(),
messageRecord.getExpireStarted() + messageRecord.getExpiresIn()));
}
while ((messageRecord = mmsReader.getNext()) != null) {
expiringMessageReferences.add(new ExpiringMessageReference(messageRecord.getId(),
messageRecord.isMms(),
messageRecord.getExpireStarted() + messageRecord.getExpiresIn()));
}
smsReader.close();
mmsReader.close();
}
}
@SuppressWarnings("InfiniteLoopStatement")
private class ProcessTask implements Runnable {
public void run() {
while (true) {
ExpiringMessageReference expiredMessage = null;
synchronized (expiringMessageReferences) {
try {
while (expiringMessageReferences.isEmpty()) expiringMessageReferences.wait();
ExpiringMessageReference nextReference = expiringMessageReferences.first();
long waitTime = nextReference.expiresAtMillis - System.currentTimeMillis();
if (waitTime > 0) {
ExpirationListener.setAlarm(context, waitTime);
expiringMessageReferences.wait(waitTime);
} else {
expiredMessage = nextReference;
expiringMessageReferences.remove(nextReference);
}
} catch (InterruptedException e) {
Log.w(TAG, e);
}
}
if (expiredMessage != null) {
if (expiredMessage.mms) mmsDatabase.delete(expiredMessage.id);
else smsDatabase.deleteMessage(expiredMessage.id);
}
}
}
}
private static class ExpiringMessageReference {
private final long id;
private final boolean mms;
private final long expiresAtMillis;
private ExpiringMessageReference(long id, boolean mms, long expiresAtMillis) {
this.id = id;
this.mms = mms;
this.expiresAtMillis = expiresAtMillis;
}
@Override
public boolean equals(Object other) {
if (other == null) return false;
if (!(other instanceof ExpiringMessageReference)) return false;
ExpiringMessageReference that = (ExpiringMessageReference)other;
return this.id == that.id && this.mms == that.mms && this.expiresAtMillis == that.expiresAtMillis;
}
@Override
public int hashCode() {
return (int)this.id ^ (mms ? 1 : 0) ^ (int)expiresAtMillis;
}
}
private static class ExpiringMessageComparator implements Comparator<ExpiringMessageReference> {
@Override
public int compare(ExpiringMessageReference lhs, ExpiringMessageReference rhs) {
if (lhs.expiresAtMillis < rhs.expiresAtMillis) return -1;
else if (lhs.expiresAtMillis > rhs.expiresAtMillis) return 1;
else if (lhs.id < rhs.id) return -1;
else if (lhs.id > rhs.id) return 1;
else if (!lhs.mms && rhs.mms) return -1;
else if (lhs.mms && !rhs.mms) return 1;
else return 0;
}
}
}