rudimentary XEP-0490 implementation

This commit is contained in:
Arne 2024-06-01 15:04:13 +02:00
parent 4c88e1b789
commit 9efccacd4b
8 changed files with 248 additions and 31 deletions

View file

@ -126,6 +126,7 @@ import androidx.annotation.Nullable;
import androidx.appcompat.app.AlertDialog;
import com.google.common.collect.ComparisonChain;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.Lists;
import org.json.JSONArray;
@ -654,6 +655,17 @@ public class Conversation extends AbstractEntity implements Blockable, Comparabl
return null;
}
public Message findReceivedWithRemoteId(final String id) {
synchronized (this.messages) {
for (final Message message : this.messages) {
if (message.getStatus() == Message.STATUS_RECEIVED && id.equals(message.getRemoteMsgId())) {
return message;
}
}
}
return null;
}
public Message findMessageWithServerMsgId(String id) {
synchronized (this.messages) {
for (Message message : this.messages) {
@ -945,20 +957,20 @@ public class Conversation extends AbstractEntity implements Blockable, Comparabl
return (this.messages.size() == 0) || this.messages.get(this.messages.size() - 1).isRead();
}
public List<Message> markRead(String upToUuid) {
final List<Message> unread = new ArrayList<>();
public List<Message> markRead(final String upToUuid) {
final ImmutableList.Builder<Message> unread = new ImmutableList.Builder<>();
synchronized (this.messages) {
for (Message message : this.messages) {
for (final Message message : this.messages) {
if (!message.isRead()) {
message.markRead();
unread.add(message);
}
if (message.getUuid().equals(upToUuid)) {
return unread;
return unread.build();
}
}
}
return unread;
return unread.build();
}
public Message getLatestMessage() {

View file

@ -44,7 +44,8 @@ public abstract class AbstractGenerator {
Namespace.NICK + "+notify",
"urn:xmpp:ping",
"jabber:iq:version",
"http://jabber.org/protocol/chatstates"
"http://jabber.org/protocol/chatstates",
Namespace.MDS_DISPLAYED + "+notify"
};
private final String[] MESSAGE_CONFIRMATION_FEATURES = {
"urn:xmpp:chat-markers:0",

View file

@ -153,6 +153,10 @@ public class IqGenerator extends AbstractGenerator {
return retrieve(Namespace.BOOKMARKS2, null);
}
public IqPacket retrieveMds() {
return retrieve(Namespace.MDS_DISPLAYED, null);
}
public IqPacket publishNick(String nick) {
final Element item = new Element("item");
item.setAttribute("id", "current");
@ -295,6 +299,24 @@ public class IqGenerator extends AbstractGenerator {
return conference;
}
public Element mdsDisplayed(final String stanzaId, final Conversation conversation) {
final Jid by;
if (conversation.getMode() == Conversation.MODE_MULTI) {
by = conversation.getJid().asBareJid();
} else {
by = conversation.getAccount().getJid().asBareJid();
}
return mdsDisplayed(stanzaId, by);
}
private Element mdsDisplayed(final String stanzaId, final Jid by) {
final Element displayed = new Element("displayed", Namespace.MDS_DISPLAYED);
final Element stanzaIdElement = displayed.addChild("stanza-id", Namespace.STANZA_IDS);
stanzaIdElement.setAttribute("id", stanzaId);
stanzaIdElement.setAttribute("by", by);
return displayed;
}
public IqPacket publishBundles(final SignedPreKeyRecord signedPreKeyRecord, final IdentityKey identityKey,
final Set<PreKeyRecord> preKeyRecords, final int deviceId, Bundle publishOptions) {
final Element item = new Element("item");

View file

@ -381,6 +381,9 @@ public class MessageParser extends AbstractParser implements OnMessagePacketRece
mXmppConnectionService.updateConversationUi();
}
}
} else if (Namespace.MDS_DISPLAYED.equals(node) && account.getJid().asBareJid().equals(from)) {
final Element item = items.findChild("item");
mXmppConnectionService.processMdsItem(account, item);
} else {
Log.d(Config.LOGTAG, account.getJid().asBareJid() + " received pubsub notification for node=" + node);
}
@ -1355,12 +1358,18 @@ public class MessageParser extends AbstractParser implements OnMessagePacketRece
}
}
}
Element displayed = packet.findChild("displayed", "urn:xmpp:chat-markers:0");
final Element displayed = packet.findChild("displayed", "urn:xmpp:chat-markers:0");
if (displayed != null) {
final String id = displayed.getAttribute("id");
final Jid sender = InvalidJid.getNullForInvalid(displayed.getAttributeAsJid("sender"));
if (packet.fromAccount(account) && !selfAddressed) {
dismissNotification(account, counterpart, query, id);
final Conversation c =
mXmppConnectionService.find(account, counterpart.asBareJid());
final Message message =
(c == null || id == null) ? null : c.findReceivedWithRemoteId(id);
if (message != null && (query == null || query.isCatchup())) {
mXmppConnectionService.markReadUpTo(c, message);
}
if (query == null) {
activateGracePeriod(account);
}
@ -1382,7 +1391,7 @@ public class MessageParser extends AbstractParser implements OnMessagePacketRece
final boolean trueJidMatchesAccount = account.getJid().asBareJid().equals(trueJid == null ? null : trueJid.asBareJid());
if (trueJidMatchesAccount || conversation.getMucOptions().isSelf(counterpart)) {
if (!message.isRead() && (query == null || query.isCatchup())) { //checking if message is unread fixes race conditions with reflections
mXmppConnectionService.markRead(conversation);
mXmppConnectionService.markReadUpTo(conversation, message);
}
} else if (!counterpart.isBareJid() && trueJid != null) {
final ReadByMarker readByMarker = ReadByMarker.from(counterpart, trueJid);

View file

@ -100,11 +100,15 @@ import eu.siacs.conversations.xmpp.Jid;
import androidx.annotation.BoolRes;
import androidx.annotation.IntegerRes;
import androidx.annotation.Nullable;
import androidx.core.app.RemoteInput;
import androidx.core.content.ContextCompat;
import androidx.annotation.NonNull;
import com.google.common.base.Objects;
import com.google.common.base.Strings;
import com.google.common.collect.Collections2;
import com.google.common.collect.Iterables;
import com.otaliastudios.transcoder.strategy.DefaultAudioStrategy;
import de.monocles.chat.WebxdcUpdate;
@ -213,6 +217,7 @@ import eu.siacs.conversations.utils.WakeLockHelper;
import eu.siacs.conversations.utils.XmppUri;
import eu.siacs.conversations.xml.Element;
import eu.siacs.conversations.xml.LocalizedContent;
import eu.siacs.conversations.xmpp.InvalidJid;
import eu.siacs.conversations.xmpp.OnBindListener;
import eu.siacs.conversations.xmpp.OnContactStatusChanged;
import eu.siacs.conversations.xmpp.OnGatewayPromptResult;
@ -455,6 +460,12 @@ public class XmppConnectionService extends Service {
} else if (!account.getXmppConnection().getFeatures().bookmarksConversion()) {
fetchBookmarks(account);
}
if (connection.getFeatures().mds()) {
fetchMessageDisplayedSynchronization(account);
} else {
Log.d(Config.LOGTAG,account.getJid()+": server has no support for mds");
}
final boolean flexible = account.getXmppConnection().getFeatures().flexibleOfflineMessageRetrieval();
final boolean catchup = getMessageArchiveService().inCatchup(account);
if (flexible && catchup && account.getXmppConnection().isMamPreferenceAlways()) {
@ -2540,18 +2551,89 @@ public class XmppConnectionService extends Service {
public void fetchBookmarks2(final Account account) {
final IqPacket retrieve = mIqGenerator.retrieveBookmarks();
sendIqPacket(account, retrieve, new OnIqPacketReceived() {
@Override
public void onIqPacketReceived(final Account account, final IqPacket response) {
if (response.getType() == IqPacket.TYPE.RESULT) {
final Element pubsub = response.findChild("pubsub", Namespace.PUBSUB);
final Map<Jid, Bookmark> bookmarks = Bookmark.parseFromPubsub(pubsub, account);
processBookmarksInitial(account, bookmarks, true);
}
sendIqPacket(account, retrieve, (a, response) -> {
if (response.getType() == IqPacket.TYPE.RESULT) {
final Element pubsub = response.findChild("pubsub", Namespace.PUBSUB);
final Map<Jid, Bookmark> bookmarks = Bookmark.parseFromPubsub(pubsub, a);
processBookmarksInitial(a, bookmarks, true);
}
});
}
private void fetchMessageDisplayedSynchronization(final Account account) {
Log.d(Config.LOGTAG, account.getJid() + ": retrieve mds");
final var retrieve = mIqGenerator.retrieveMds();
sendIqPacket(
account,
retrieve,
(a, response) -> {
if (response.getType() != IqPacket.TYPE.RESULT) {
return;
}
final var pubSub = response.findChild("pubsub", Namespace.PUBSUB);
final Element items = pubSub == null ? null : pubSub.findChild("items");
if (items == null
|| !Namespace.MDS_DISPLAYED.equals(items.getAttribute("node"))) {
return;
}
for (final Element child : items.getChildren()) {
if ("item".equals(child.getName())) {
processMdsItem(account, child);
}
}
});
}
public void processMdsItem(final Account account, final Element item) {
final Jid jid =
item == null ? null : InvalidJid.getNullForInvalid(item.getAttributeAsJid("id"));
if (jid == null) {
return;
}
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": processing mds item for " + jid);
final Element displayed = item.findChild("displayed", Namespace.MDS_DISPLAYED);
final Element stanzaId =
displayed == null ? null : displayed.findChild("stanza-id", Namespace.STANZA_IDS);
final String id = stanzaId == null ? null : stanzaId.getAttribute("id");
final Conversation conversation = find(account, jid);
if (id != null && conversation != null) {
markReadUpToStanzaId(conversation, id);
}
}
public void markReadUpToStanzaId(final Conversation conversation, final String stanzaId) {
final Message message = conversation.findMessageWithServerMsgId(stanzaId);
if (message == null) { // do we want to check if isRead?
return;
}
markReadUpTo(conversation, message);
}
public void markReadUpTo(final Conversation conversation, final Message message) {
final boolean isDismissNotification = isDismissNotification(message);
final var uuid = message.getUuid();
Log.d(
Config.LOGTAG,
conversation.getAccount().getJid().asBareJid()
+ ": mark "
+ conversation.getJid().asBareJid()
+ " as read up to "
+ uuid);
markRead(conversation, uuid, isDismissNotification);
}
private static boolean isDismissNotification(final Message message) {
Message next = message.next();
while (next != null) {
if (message.getStatus() == Message.STATUS_RECEIVED) {
return false;
}
next = next.next();
}
return true;
}
public void processBookmarksInitial(Account account, Map<Jid, Bookmark> bookmarks, final boolean pep) {
final Set<Jid> previousBookmarks = account.getBookmarkedJids();
final boolean synchronizeWithBookmarks = synchronizeWithBookmarks();
@ -2718,7 +2800,7 @@ public class XmppConnectionService extends Service {
}
});
} else {
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": error publishing bookmarks (retry=" + retry + ") " + response);
Log.d(Config.LOGTAG, account.getJid().asBareJid() + ": error publishing "+node+" (retry=" + retry + ") " + response);
}
});
}
@ -5595,24 +5677,101 @@ public class XmppConnectionService extends Service {
mDatabaseWriterExecutor.execute(runnable);
}
public void sendReadMarker(final Conversation conversation, String upToUuid) {
final boolean isPrivateAndNonAnonymousMuc = conversation.getMode() == Conversation.MODE_MULTI && conversation.isPrivateAndNonAnonymous();
public void sendReadMarker(final Conversation conversation, final String upToUuid) {
final boolean isPrivateAndNonAnonymousMuc =
conversation.getMode() == Conversation.MODE_MULTI
&& conversation.isPrivateAndNonAnonymous();
final List<Message> readMessages = this.markRead(conversation, upToUuid, true);
if (readMessages.size() > 0) {
updateConversationUi();
if (readMessages.isEmpty()) {
return;
}
final Message markable = Conversation.getLatestMarkableMessage(readMessages, isPrivateAndNonAnonymousMuc);
if (confirmMessages()
&& markable != null
&& (markable.trusted() || isPrivateAndNonAnonymousMuc)
&& markable.getRemoteMsgId() != null) {
Log.d(Config.LOGTAG, conversation.getAccount().getJid().asBareJid() + ": sending read marker to " + markable.getCounterpart().toString());
final Account account = conversation.getAccount();
final MessagePacket packet = mMessageGenerator.confirm(markable);
final var account = conversation.getAccount();
final var connection = account.getXmppConnection();
updateConversationUi();
final var last =
Iterables.getLast(
Collections2.filter(
readMessages,
m ->
!m.isPrivateMessage()
&& m.getStatus() == Message.STATUS_RECEIVED),
null);
if (last == null) {
return;
}
final boolean sendDisplayedMarker =
confirmMessages()
&& (last.trusted() || isPrivateAndNonAnonymousMuc)
&& last.getRemoteMsgId() != null
&& (last.markable || isPrivateAndNonAnonymousMuc);
final boolean serverAssist =
connection != null && connection.getFeatures().mdsServerAssist();
final String stanzaId = last.getServerMsgId();
if (sendDisplayedMarker && serverAssist) {
final var mdsDisplayed = mIqGenerator.mdsDisplayed(stanzaId, conversation);
final MessagePacket packet = mMessageGenerator.confirm(last);
packet.addChild(mdsDisplayed);
if (!last.isPrivateMessage()) {
packet.setTo(packet.getTo().asBareJid());
}
Log.d(Config.LOGTAG,account.getJid().asBareJid()+": server assisted "+packet);
this.sendMessagePacket(account, packet);
} else {
publishMds(last);
// read markers will be sent after MDS to flush the CSI stanza queue
if (sendDisplayedMarker) {
Log.d(
Config.LOGTAG,
conversation.getAccount().getJid().asBareJid()
+ ": sending displayed marker to "
+ last.getCounterpart().toString());
final MessagePacket packet = mMessageGenerator.confirm(last);
this.sendMessagePacket(account, packet);
}
}
}
private void publishMds(@Nullable final Message message) {
final String stanzaId = message == null ? null : message.getServerMsgId();
if (Strings.isNullOrEmpty(stanzaId)) {
return;
}
final Conversation conversation;
final var conversational = message.getConversation();
if (conversational instanceof Conversation c) {
conversation = c;
} else {
return;
}
final var account = conversation.getAccount();
final var connection = account.getXmppConnection();
if (connection == null || !connection.getFeatures().mds()) {
return;
}
final Jid itemId;
if (message.isPrivateMessage()) {
itemId = message.getCounterpart();
} else {
itemId = conversation.getJid().asBareJid();
}
Log.d(Config.LOGTAG,"publishing mds for "+itemId+"/"+stanzaId);
publishMds(account, itemId, stanzaId, conversation);
}
private void publishMds(
final Account account, final Jid itemId, final String stanzaId, final Conversation conversation) {
final var item = mIqGenerator.mdsDisplayed(stanzaId, conversation);
pushNodeAndEnforcePublishOptions(
account,
Namespace.MDS_DISPLAYED,
item,
itemId.toEscapedString(),
PublishOptions.persistentWhitelistAccessMaxItems());
}
public MemorizingTrustManager getMemorizingTrustManager() {
return this.mMemorizingTrustManager;
}

View file

@ -24,6 +24,7 @@ public final class Namespace {
public static final String TLS = "urn:ietf:params:xml:ns:xmpp-tls";
public static final String PUBSUB = "http://jabber.org/protocol/pubsub";
public static final String PUBSUB_PUBLISH_OPTIONS = PUBSUB + "#publish-options";
public static final String PUBSUB_CONFIG_NODE_MAX = PUBSUB + "#config-node-max";
public static final String PUBSUB_ERROR = PUBSUB + "#errors";
public static final String PUBSUB_OWNER = PUBSUB + "#owner";
public static final String NICK = "http://jabber.org/protocol/nick";
@ -72,4 +73,6 @@ public final class Namespace {
public static final String SDP_OFFER_ANSWER = "urn:ietf:rfc:3264";
public static final String REPORTING = "urn:xmpp:reporting:1";
public static final String REPORTING_REASON_SPAM = "urn:xmpp:reporting:spam";
public static final String MDS_DISPLAYED = "urn:xmpp:mds:displayed:0";
public static final String MDS_SERVER_ASSIST = "urn:xmpp:mds:server-assist:0";
}

View file

@ -3087,6 +3087,10 @@ public class XmppConnection implements Runnable {
return hasDiscoFeature(account.getJid().asBareJid(), Namespace.PUBSUB_PUBLISH_OPTIONS);
}
public boolean pepConfigNodeMax() {
return hasDiscoFeature(account.getJid().asBareJid(), Namespace.PUBSUB_CONFIG_NODE_MAX);
}
public boolean pepOmemoWhitelisted() {
return hasDiscoFeature(
account.getJid().asBareJid(), AxolotlService.PEP_OMEMO_WHITELISTED);
@ -3186,5 +3190,13 @@ public class XmppConnection implements Runnable {
public boolean externalServiceDiscovery() {
return hasDiscoFeature(account.getDomain(), Namespace.EXTERNAL_SERVICE_DISCOVERY);
}
public boolean mds() {
return pepPublishOptions() && pepConfigNodeMax();
}
public boolean mdsServerAssist() {
return hasDiscoFeature(account.getJid().asBareJid(), Namespace.MDS_DISPLAYED);
}
}
}

View file

@ -31,7 +31,6 @@ public class PublishOptions {
options.putString("pubsub#access_model", "whitelist");
options.putString("pubsub#send_last_published_item", "never");
options.putString("pubsub#max_items", "max");
options.putString("pubsub#notify_delete", "true");
options.putString("pubsub#notify_retract", "true"); //one could also set notify=true on the retract
return options;