mirror of
https://github.com/AsamK/signal-cli
synced 2025-08-29 10:30:38 +00:00
Remove legacy EncapsulatedExceptions from manager
This commit is contained in:
parent
d94a7511dd
commit
a634b46eb2
5 changed files with 108 additions and 141 deletions
|
@ -8,16 +8,18 @@ import org.asamk.signal.manager.Manager;
|
||||||
import org.asamk.signal.manager.NotAGroupMemberException;
|
import org.asamk.signal.manager.NotAGroupMemberException;
|
||||||
import org.asamk.signal.util.GroupIdFormatException;
|
import org.asamk.signal.util.GroupIdFormatException;
|
||||||
import org.asamk.signal.util.Util;
|
import org.asamk.signal.util.Util;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.EncapsulatedExceptions;
|
import org.whispersystems.libsignal.util.Pair;
|
||||||
|
import org.whispersystems.signalservice.api.messages.SendMessageResult;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleAssertionError;
|
import static org.asamk.signal.util.ErrorUtils.handleAssertionError;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleEncapsulatedExceptions;
|
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleGroupIdFormatException;
|
import static org.asamk.signal.util.ErrorUtils.handleGroupIdFormatException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleGroupNotFoundException;
|
import static org.asamk.signal.util.ErrorUtils.handleGroupNotFoundException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleIOException;
|
import static org.asamk.signal.util.ErrorUtils.handleIOException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleNotAGroupMemberException;
|
import static org.asamk.signal.util.ErrorUtils.handleNotAGroupMemberException;
|
||||||
|
import static org.asamk.signal.util.ErrorUtils.handleTimestampAndSendMessageResults;
|
||||||
|
|
||||||
public class QuitGroupCommand implements LocalCommand {
|
public class QuitGroupCommand implements LocalCommand {
|
||||||
|
|
||||||
|
@ -36,14 +38,11 @@ public class QuitGroupCommand implements LocalCommand {
|
||||||
}
|
}
|
||||||
|
|
||||||
try {
|
try {
|
||||||
m.sendQuitGroupMessage(Util.decodeGroupId(ns.getString("group")));
|
final Pair<Long, List<SendMessageResult>> results = m.sendQuitGroupMessage(Util.decodeGroupId(ns.getString("group")));
|
||||||
return 0;
|
return handleTimestampAndSendMessageResults(results.first(), results.second());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
handleIOException(e);
|
handleIOException(e);
|
||||||
return 3;
|
return 3;
|
||||||
} catch (EncapsulatedExceptions e) {
|
|
||||||
handleEncapsulatedExceptions(e);
|
|
||||||
return 3;
|
|
||||||
} catch (AssertionError e) {
|
} catch (AssertionError e) {
|
||||||
handleAssertionError(e);
|
handleAssertionError(e);
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -9,18 +9,20 @@ import org.asamk.signal.manager.Manager;
|
||||||
import org.asamk.signal.manager.NotAGroupMemberException;
|
import org.asamk.signal.manager.NotAGroupMemberException;
|
||||||
import org.asamk.signal.util.GroupIdFormatException;
|
import org.asamk.signal.util.GroupIdFormatException;
|
||||||
import org.asamk.signal.util.Util;
|
import org.asamk.signal.util.Util;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.EncapsulatedExceptions;
|
import org.whispersystems.libsignal.util.Pair;
|
||||||
|
import org.whispersystems.signalservice.api.messages.SendMessageResult;
|
||||||
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleAssertionError;
|
import static org.asamk.signal.util.ErrorUtils.handleAssertionError;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleEncapsulatedExceptions;
|
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleGroupIdFormatException;
|
import static org.asamk.signal.util.ErrorUtils.handleGroupIdFormatException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleGroupNotFoundException;
|
import static org.asamk.signal.util.ErrorUtils.handleGroupNotFoundException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleIOException;
|
import static org.asamk.signal.util.ErrorUtils.handleIOException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleInvalidNumberException;
|
import static org.asamk.signal.util.ErrorUtils.handleInvalidNumberException;
|
||||||
import static org.asamk.signal.util.ErrorUtils.handleNotAGroupMemberException;
|
import static org.asamk.signal.util.ErrorUtils.handleNotAGroupMemberException;
|
||||||
|
import static org.asamk.signal.util.ErrorUtils.handleTimestampAndSendMessageResults;
|
||||||
|
|
||||||
public class SendReactionCommand implements LocalCommand {
|
public class SendReactionCommand implements LocalCommand {
|
||||||
|
|
||||||
|
@ -66,19 +68,18 @@ public class SendReactionCommand implements LocalCommand {
|
||||||
long targetTimestamp = ns.getLong("target_timestamp");
|
long targetTimestamp = ns.getLong("target_timestamp");
|
||||||
|
|
||||||
try {
|
try {
|
||||||
|
final Pair<Long, List<SendMessageResult>> results;
|
||||||
if (ns.getString("group") != null) {
|
if (ns.getString("group") != null) {
|
||||||
byte[] groupId = Util.decodeGroupId(ns.getString("group"));
|
byte[] groupId = Util.decodeGroupId(ns.getString("group"));
|
||||||
m.sendGroupMessageReaction(emoji, isRemove, targetAuthor, targetTimestamp, groupId);
|
results = m.sendGroupMessageReaction(emoji, isRemove, targetAuthor, targetTimestamp, groupId);
|
||||||
} else {
|
} else {
|
||||||
m.sendMessageReaction(emoji, isRemove, targetAuthor, targetTimestamp, ns.getList("recipient"));
|
results = m.sendMessageReaction(emoji, isRemove, targetAuthor, targetTimestamp, ns.getList("recipient"));
|
||||||
}
|
}
|
||||||
|
handleTimestampAndSendMessageResults(results.first(), results.second());
|
||||||
return 0;
|
return 0;
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
handleIOException(e);
|
handleIOException(e);
|
||||||
return 3;
|
return 3;
|
||||||
} catch (EncapsulatedExceptions e) {
|
|
||||||
handleEncapsulatedExceptions(e);
|
|
||||||
return 3;
|
|
||||||
} catch (AssertionError e) {
|
} catch (AssertionError e) {
|
||||||
handleAssertionError(e);
|
handleAssertionError(e);
|
||||||
return 1;
|
return 1;
|
||||||
|
|
|
@ -6,11 +6,10 @@ import org.asamk.signal.manager.GroupNotFoundException;
|
||||||
import org.asamk.signal.manager.Manager;
|
import org.asamk.signal.manager.Manager;
|
||||||
import org.asamk.signal.manager.NotAGroupMemberException;
|
import org.asamk.signal.manager.NotAGroupMemberException;
|
||||||
import org.asamk.signal.storage.groups.GroupInfo;
|
import org.asamk.signal.storage.groups.GroupInfo;
|
||||||
|
import org.asamk.signal.util.ErrorUtils;
|
||||||
import org.freedesktop.dbus.exceptions.DBusExecutionException;
|
import org.freedesktop.dbus.exceptions.DBusExecutionException;
|
||||||
import org.whispersystems.signalservice.api.crypto.UntrustedIdentityException;
|
import org.whispersystems.libsignal.util.Pair;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.EncapsulatedExceptions;
|
import org.whispersystems.signalservice.api.messages.SendMessageResult;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.NetworkFailureException;
|
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.UnregisteredUserException;
|
|
||||||
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
@ -43,41 +42,28 @@ public class DbusSignalImpl implements Signal {
|
||||||
return sendMessage(message, attachments, recipients);
|
return sendMessage(message, attachments, recipients);
|
||||||
}
|
}
|
||||||
|
|
||||||
private static DBusExecutionException convertEncapsulatedExceptions(EncapsulatedExceptions e) {
|
private static void checkSendMessageResults(long timestamp, List<SendMessageResult> results) throws DBusExecutionException {
|
||||||
if (e.getNetworkExceptions().size() + e.getUnregisteredUserExceptions().size() + e.getUntrustedIdentityExceptions().size() == 1) {
|
List<String> errors = ErrorUtils.getErrorMessagesFromSendMessageResults(results);
|
||||||
if (e.getNetworkExceptions().size() == 1) {
|
if (errors.size() == 0) {
|
||||||
NetworkFailureException n = e.getNetworkExceptions().get(0);
|
return;
|
||||||
return new Error.Failure("Network failure for \"" + n.getE164number() + "\": " + n.getMessage());
|
|
||||||
} else if (e.getUnregisteredUserExceptions().size() == 1) {
|
|
||||||
UnregisteredUserException n = e.getUnregisteredUserExceptions().get(0);
|
|
||||||
return new Error.UnregisteredUser("Unregistered user \"" + n.getE164Number() + "\": " + n.getMessage());
|
|
||||||
} else if (e.getUntrustedIdentityExceptions().size() == 1) {
|
|
||||||
UntrustedIdentityException n = e.getUntrustedIdentityExceptions().get(0);
|
|
||||||
return new Error.UntrustedIdentity("Untrusted Identity for \"" + n.getIdentifier() + "\": " + n.getMessage());
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
StringBuilder message = new StringBuilder();
|
StringBuilder message = new StringBuilder();
|
||||||
message.append("Failed to send (some) messages:").append('\n');
|
message.append(timestamp).append('\n');
|
||||||
for (NetworkFailureException n : e.getNetworkExceptions()) {
|
message.append("Failed to send (some) messages:\n");
|
||||||
message.append("Network failure for \"").append(n.getE164number()).append("\": ").append(n.getMessage()).append('\n');
|
for (String error : errors) {
|
||||||
}
|
message.append(error).append('\n');
|
||||||
for (UnregisteredUserException n : e.getUnregisteredUserExceptions()) {
|
|
||||||
message.append("Unregistered user \"").append(n.getE164Number()).append("\": ").append(n.getMessage()).append('\n');
|
|
||||||
}
|
|
||||||
for (UntrustedIdentityException n : e.getUntrustedIdentityExceptions()) {
|
|
||||||
message.append("Untrusted Identity for \"").append(n.getIdentifier()).append("\": ").append(n.getMessage()).append('\n');
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return new Error.Failure(message.toString());
|
throw new Error.Failure(message.toString());
|
||||||
}
|
}
|
||||||
|
|
||||||
@Override
|
@Override
|
||||||
public long sendMessage(final String message, final List<String> attachments, final List<String> recipients) {
|
public long sendMessage(final String message, final List<String> attachments, final List<String> recipients) {
|
||||||
try {
|
try {
|
||||||
return m.sendMessage(message, attachments, recipients);
|
final Pair<Long, List<SendMessageResult>> results = m.sendMessage(message, attachments, recipients);
|
||||||
} catch (EncapsulatedExceptions e) {
|
checkSendMessageResults(results.first(), results.second());
|
||||||
throw convertEncapsulatedExceptions(e);
|
return results.first();
|
||||||
} catch (InvalidNumberException e) {
|
} catch (InvalidNumberException e) {
|
||||||
throw new Error.InvalidNumber(e.getMessage());
|
throw new Error.InvalidNumber(e.getMessage());
|
||||||
} catch (AttachmentInvalidException e) {
|
} catch (AttachmentInvalidException e) {
|
||||||
|
@ -90,11 +76,10 @@ public class DbusSignalImpl implements Signal {
|
||||||
@Override
|
@Override
|
||||||
public void sendEndSessionMessage(final List<String> recipients) {
|
public void sendEndSessionMessage(final List<String> recipients) {
|
||||||
try {
|
try {
|
||||||
m.sendEndSessionMessage(recipients);
|
final Pair<Long, List<SendMessageResult>> results = m.sendEndSessionMessage(recipients);
|
||||||
|
checkSendMessageResults(results.first(), results.second());
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new Error.Failure(e.getMessage());
|
throw new Error.Failure(e.getMessage());
|
||||||
} catch (EncapsulatedExceptions e) {
|
|
||||||
throw convertEncapsulatedExceptions(e);
|
|
||||||
} catch (InvalidNumberException e) {
|
} catch (InvalidNumberException e) {
|
||||||
throw new Error.InvalidNumber(e.getMessage());
|
throw new Error.InvalidNumber(e.getMessage());
|
||||||
}
|
}
|
||||||
|
@ -103,11 +88,11 @@ public class DbusSignalImpl implements Signal {
|
||||||
@Override
|
@Override
|
||||||
public long sendGroupMessage(final String message, final List<String> attachments, final byte[] groupId) {
|
public long sendGroupMessage(final String message, final List<String> attachments, final byte[] groupId) {
|
||||||
try {
|
try {
|
||||||
return m.sendGroupMessage(message, attachments, groupId);
|
Pair<Long, List<SendMessageResult>> results = m.sendGroupMessage(message, attachments, groupId);
|
||||||
|
checkSendMessageResults(results.first(), results.second());
|
||||||
|
return results.first();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new Error.Failure(e.getMessage());
|
throw new Error.Failure(e.getMessage());
|
||||||
} catch (EncapsulatedExceptions e) {
|
|
||||||
throw convertEncapsulatedExceptions(e);
|
|
||||||
} catch (GroupNotFoundException | NotAGroupMemberException e) {
|
} catch (GroupNotFoundException | NotAGroupMemberException e) {
|
||||||
throw new Error.GroupNotFound(e.getMessage());
|
throw new Error.GroupNotFound(e.getMessage());
|
||||||
} catch (AttachmentInvalidException e) {
|
} catch (AttachmentInvalidException e) {
|
||||||
|
@ -184,11 +169,11 @@ public class DbusSignalImpl implements Signal {
|
||||||
@Override
|
@Override
|
||||||
public byte[] updateGroup(final byte[] groupId, final String name, final List<String> members, final String avatar) {
|
public byte[] updateGroup(final byte[] groupId, final String name, final List<String> members, final String avatar) {
|
||||||
try {
|
try {
|
||||||
return m.updateGroup(groupId, name, members, avatar);
|
final Pair<byte[], List<SendMessageResult>> results = m.updateGroup(groupId, name, members, avatar);
|
||||||
|
checkSendMessageResults(0, results.second());
|
||||||
|
return results.first();
|
||||||
} catch (IOException e) {
|
} catch (IOException e) {
|
||||||
throw new Error.Failure(e.getMessage());
|
throw new Error.Failure(e.getMessage());
|
||||||
} catch (EncapsulatedExceptions e) {
|
|
||||||
throw convertEncapsulatedExceptions(e);
|
|
||||||
} catch (GroupNotFoundException | NotAGroupMemberException e) {
|
} catch (GroupNotFoundException | NotAGroupMemberException e) {
|
||||||
throw new Error.GroupNotFound(e.getMessage());
|
throw new Error.GroupNotFound(e.getMessage());
|
||||||
} catch (InvalidNumberException e) {
|
} catch (InvalidNumberException e) {
|
||||||
|
|
|
@ -96,10 +96,7 @@ import org.whispersystems.signalservice.api.messages.multidevice.VerifiedMessage
|
||||||
import org.whispersystems.signalservice.api.profiles.SignalServiceProfile;
|
import org.whispersystems.signalservice.api.profiles.SignalServiceProfile;
|
||||||
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
import org.whispersystems.signalservice.api.push.ContactTokenDetails;
|
||||||
import org.whispersystems.signalservice.api.push.SignalServiceAddress;
|
import org.whispersystems.signalservice.api.push.SignalServiceAddress;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.EncapsulatedExceptions;
|
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.MissingConfigurationException;
|
import org.whispersystems.signalservice.api.push.exceptions.MissingConfigurationException;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.NetworkFailureException;
|
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.UnregisteredUserException;
|
|
||||||
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
||||||
import org.whispersystems.signalservice.api.util.SleepTimer;
|
import org.whispersystems.signalservice.api.util.SleepTimer;
|
||||||
import org.whispersystems.signalservice.api.util.StreamDetails;
|
import org.whispersystems.signalservice.api.util.StreamDetails;
|
||||||
|
@ -134,7 +131,6 @@ import java.util.Collection;
|
||||||
import java.util.Collections;
|
import java.util.Collections;
|
||||||
import java.util.Date;
|
import java.util.Date;
|
||||||
import java.util.HashSet;
|
import java.util.HashSet;
|
||||||
import java.util.LinkedList;
|
|
||||||
import java.util.List;
|
import java.util.List;
|
||||||
import java.util.Locale;
|
import java.util.Locale;
|
||||||
import java.util.Objects;
|
import java.util.Objects;
|
||||||
|
@ -161,7 +157,7 @@ public class Manager implements Closeable {
|
||||||
private SignalServiceAccountManager accountManager;
|
private SignalServiceAccountManager accountManager;
|
||||||
private SignalServiceMessagePipe messagePipe = null;
|
private SignalServiceMessagePipe messagePipe = null;
|
||||||
private SignalServiceMessagePipe unidentifiedMessagePipe = null;
|
private SignalServiceMessagePipe unidentifiedMessagePipe = null;
|
||||||
private boolean discoverableByPhoneNumber = true;
|
private final boolean discoverableByPhoneNumber = true;
|
||||||
|
|
||||||
public Manager(SignalAccount account, PathConfig pathConfig, SignalServiceConfiguration serviceConfiguration, String userAgent) {
|
public Manager(SignalAccount account, PathConfig pathConfig, SignalServiceConfiguration serviceConfiguration, String userAgent) {
|
||||||
this.account = account;
|
this.account = account;
|
||||||
|
@ -540,9 +536,12 @@ public class Manager implements Closeable {
|
||||||
return account.getGroupStore().getGroups();
|
return account.getGroupStore().getGroups();
|
||||||
}
|
}
|
||||||
|
|
||||||
public long sendGroupMessage(String messageText, List<String> attachments,
|
public Pair<Long, List<SendMessageResult>> sendGroupMessage(
|
||||||
byte[] groupId)
|
String messageText,
|
||||||
throws IOException, EncapsulatedExceptions, GroupNotFoundException, AttachmentInvalidException, NotAGroupMemberException {
|
List<String> attachments,
|
||||||
|
byte[] groupId
|
||||||
|
)
|
||||||
|
throws IOException, GroupNotFoundException, AttachmentInvalidException, NotAGroupMemberException {
|
||||||
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder().withBody(messageText);
|
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder().withBody(messageText);
|
||||||
if (attachments != null) {
|
if (attachments != null) {
|
||||||
messageBuilder.withAttachments(Utils.getSignalServiceAttachments(attachments));
|
messageBuilder.withAttachments(Utils.getSignalServiceAttachments(attachments));
|
||||||
|
@ -558,12 +557,12 @@ public class Manager implements Closeable {
|
||||||
|
|
||||||
messageBuilder.withExpiration(g.messageExpirationTime);
|
messageBuilder.withExpiration(g.messageExpirationTime);
|
||||||
|
|
||||||
return sendMessageLegacy(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
return sendMessage(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void sendGroupMessageReaction(String emoji, boolean remove, String targetAuthor,
|
public Pair<Long, List<SendMessageResult>> sendGroupMessageReaction(String emoji, boolean remove, String targetAuthor,
|
||||||
long targetSentTimestamp, byte[] groupId)
|
long targetSentTimestamp, byte[] groupId)
|
||||||
throws IOException, EncapsulatedExceptions, InvalidNumberException, NotAGroupMemberException, GroupNotFoundException {
|
throws IOException, InvalidNumberException, NotAGroupMemberException, GroupNotFoundException {
|
||||||
SignalServiceDataMessage.Reaction reaction = new SignalServiceDataMessage.Reaction(emoji, remove, canonicalizeAndResolveSignalServiceAddress(targetAuthor), targetSentTimestamp);
|
SignalServiceDataMessage.Reaction reaction = new SignalServiceDataMessage.Reaction(emoji, remove, canonicalizeAndResolveSignalServiceAddress(targetAuthor), targetSentTimestamp);
|
||||||
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
||||||
.withReaction(reaction);
|
.withReaction(reaction);
|
||||||
|
@ -574,10 +573,10 @@ public class Manager implements Closeable {
|
||||||
messageBuilder.asGroupMessage(group);
|
messageBuilder.asGroupMessage(group);
|
||||||
}
|
}
|
||||||
final GroupInfo g = getGroupForSending(groupId);
|
final GroupInfo g = getGroupForSending(groupId);
|
||||||
sendMessageLegacy(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
return sendMessage(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void sendQuitGroupMessage(byte[] groupId) throws GroupNotFoundException, IOException, EncapsulatedExceptions, NotAGroupMemberException {
|
public Pair<Long, List<SendMessageResult>> sendQuitGroupMessage(byte[] groupId) throws GroupNotFoundException, IOException, NotAGroupMemberException {
|
||||||
SignalServiceGroup group = SignalServiceGroup.newBuilder(SignalServiceGroup.Type.QUIT)
|
SignalServiceGroup group = SignalServiceGroup.newBuilder(SignalServiceGroup.Type.QUIT)
|
||||||
.withId(groupId)
|
.withId(groupId)
|
||||||
.build();
|
.build();
|
||||||
|
@ -589,10 +588,10 @@ public class Manager implements Closeable {
|
||||||
g.removeMember(account.getSelfAddress());
|
g.removeMember(account.getSelfAddress());
|
||||||
account.getGroupStore().updateGroup(g);
|
account.getGroupStore().updateGroup(g);
|
||||||
|
|
||||||
sendMessageLegacy(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
return sendMessage(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
||||||
}
|
}
|
||||||
|
|
||||||
private byte[] sendUpdateGroupMessage(byte[] groupId, String name, Collection<SignalServiceAddress> members, String avatarFile) throws IOException, EncapsulatedExceptions, GroupNotFoundException, AttachmentInvalidException, NotAGroupMemberException {
|
private Pair<byte[], List<SendMessageResult>> sendUpdateGroupMessage(byte[] groupId, String name, Collection<SignalServiceAddress> members, String avatarFile) throws IOException, GroupNotFoundException, AttachmentInvalidException, NotAGroupMemberException {
|
||||||
GroupInfo g;
|
GroupInfo g;
|
||||||
if (groupId == null) {
|
if (groupId == null) {
|
||||||
// Create new group
|
// Create new group
|
||||||
|
@ -637,24 +636,21 @@ public class Manager implements Closeable {
|
||||||
|
|
||||||
SignalServiceDataMessage.Builder messageBuilder = getGroupUpdateMessageBuilder(g);
|
SignalServiceDataMessage.Builder messageBuilder = getGroupUpdateMessageBuilder(g);
|
||||||
|
|
||||||
sendMessageLegacy(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
final Pair<Long, List<SendMessageResult>> result = sendMessage(messageBuilder, g.getMembersWithout(account.getSelfAddress()));
|
||||||
return g.groupId;
|
return new Pair<>(g.groupId, result.second());
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendUpdateGroupMessage(byte[] groupId, SignalServiceAddress recipient) throws IOException, EncapsulatedExceptions, NotAGroupMemberException, GroupNotFoundException, AttachmentInvalidException {
|
Pair<Long, List<SendMessageResult>> sendUpdateGroupMessage(byte[] groupId, SignalServiceAddress recipient) throws IOException, NotAGroupMemberException, GroupNotFoundException, AttachmentInvalidException {
|
||||||
if (groupId == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
GroupInfo g = getGroupForSending(groupId);
|
GroupInfo g = getGroupForSending(groupId);
|
||||||
|
|
||||||
if (!g.isMember(recipient)) {
|
if (!g.isMember(recipient)) {
|
||||||
return;
|
throw new NotAGroupMemberException(groupId, g.name);
|
||||||
}
|
}
|
||||||
|
|
||||||
SignalServiceDataMessage.Builder messageBuilder = getGroupUpdateMessageBuilder(g);
|
SignalServiceDataMessage.Builder messageBuilder = getGroupUpdateMessageBuilder(g);
|
||||||
|
|
||||||
// Send group message only to the recipient who requested it
|
// Send group message only to the recipient who requested it
|
||||||
sendMessageLegacy(messageBuilder, Collections.singleton(recipient));
|
return sendMessage(messageBuilder, Collections.singleton(recipient));
|
||||||
}
|
}
|
||||||
|
|
||||||
private SignalServiceDataMessage.Builder getGroupUpdateMessageBuilder(GroupInfo g) throws AttachmentInvalidException {
|
private SignalServiceDataMessage.Builder getGroupUpdateMessageBuilder(GroupInfo g) throws AttachmentInvalidException {
|
||||||
|
@ -677,11 +673,7 @@ public class Manager implements Closeable {
|
||||||
.withExpiration(g.messageExpirationTime);
|
.withExpiration(g.messageExpirationTime);
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendGroupInfoRequest(byte[] groupId, SignalServiceAddress recipient) throws IOException, EncapsulatedExceptions {
|
Pair<Long, List<SendMessageResult>> sendGroupInfoRequest(byte[] groupId, SignalServiceAddress recipient) throws IOException {
|
||||||
if (groupId == null) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
SignalServiceGroup.Builder group = SignalServiceGroup.newBuilder(SignalServiceGroup.Type.REQUEST_INFO)
|
SignalServiceGroup.Builder group = SignalServiceGroup.newBuilder(SignalServiceGroup.Type.REQUEST_INFO)
|
||||||
.withId(groupId);
|
.withId(groupId);
|
||||||
|
|
||||||
|
@ -689,7 +681,7 @@ public class Manager implements Closeable {
|
||||||
.asGroupMessage(group.build());
|
.asGroupMessage(group.build());
|
||||||
|
|
||||||
// Send group info request message to the recipient who sent us a message with this groupId
|
// Send group info request message to the recipient who sent us a message with this groupId
|
||||||
sendMessageLegacy(messageBuilder, Collections.singleton(recipient));
|
return sendMessage(messageBuilder, Collections.singleton(recipient));
|
||||||
}
|
}
|
||||||
|
|
||||||
void sendReceipt(SignalServiceAddress remoteAddress, long messageId) throws IOException, UntrustedIdentityException {
|
void sendReceipt(SignalServiceAddress remoteAddress, long messageId) throws IOException, UntrustedIdentityException {
|
||||||
|
@ -700,9 +692,9 @@ public class Manager implements Closeable {
|
||||||
getMessageSender().sendReceipt(remoteAddress, getAccessFor(remoteAddress), receiptMessage);
|
getMessageSender().sendReceipt(remoteAddress, getAccessFor(remoteAddress), receiptMessage);
|
||||||
}
|
}
|
||||||
|
|
||||||
public long sendMessage(String messageText, List<String> attachments,
|
public Pair<Long, List<SendMessageResult>> sendMessage(String messageText, List<String> attachments,
|
||||||
List<String> recipients)
|
List<String> recipients)
|
||||||
throws IOException, EncapsulatedExceptions, AttachmentInvalidException, InvalidNumberException {
|
throws IOException, AttachmentInvalidException, InvalidNumberException {
|
||||||
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder().withBody(messageText);
|
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder().withBody(messageText);
|
||||||
if (attachments != null) {
|
if (attachments != null) {
|
||||||
List<SignalServiceAttachment> attachmentStreams = Utils.getSignalServiceAttachments(attachments);
|
List<SignalServiceAttachment> attachmentStreams = Utils.getSignalServiceAttachments(attachments);
|
||||||
|
@ -720,25 +712,25 @@ public class Manager implements Closeable {
|
||||||
|
|
||||||
messageBuilder.withAttachments(attachmentPointers);
|
messageBuilder.withAttachments(attachmentPointers);
|
||||||
}
|
}
|
||||||
return sendMessageLegacy(messageBuilder, getSignalServiceAddresses(recipients));
|
return sendMessage(messageBuilder, getSignalServiceAddresses(recipients));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void sendMessageReaction(String emoji, boolean remove, String targetAuthor,
|
public Pair<Long, List<SendMessageResult>> sendMessageReaction(String emoji, boolean remove, String targetAuthor,
|
||||||
long targetSentTimestamp, List<String> recipients)
|
long targetSentTimestamp, List<String> recipients)
|
||||||
throws IOException, EncapsulatedExceptions, InvalidNumberException {
|
throws IOException, InvalidNumberException {
|
||||||
SignalServiceDataMessage.Reaction reaction = new SignalServiceDataMessage.Reaction(emoji, remove, canonicalizeAndResolveSignalServiceAddress(targetAuthor), targetSentTimestamp);
|
SignalServiceDataMessage.Reaction reaction = new SignalServiceDataMessage.Reaction(emoji, remove, canonicalizeAndResolveSignalServiceAddress(targetAuthor), targetSentTimestamp);
|
||||||
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
final SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
||||||
.withReaction(reaction);
|
.withReaction(reaction);
|
||||||
sendMessageLegacy(messageBuilder, getSignalServiceAddresses(recipients));
|
return sendMessage(messageBuilder, getSignalServiceAddresses(recipients));
|
||||||
}
|
}
|
||||||
|
|
||||||
public void sendEndSessionMessage(List<String> recipients) throws IOException, EncapsulatedExceptions, InvalidNumberException {
|
public Pair<Long, List<SendMessageResult>> sendEndSessionMessage(List<String> recipients) throws IOException, InvalidNumberException {
|
||||||
SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
SignalServiceDataMessage.Builder messageBuilder = SignalServiceDataMessage.newBuilder()
|
||||||
.asEndSessionMessage();
|
.asEndSessionMessage();
|
||||||
|
|
||||||
final Collection<SignalServiceAddress> signalServiceAddresses = getSignalServiceAddresses(recipients);
|
final Collection<SignalServiceAddress> signalServiceAddresses = getSignalServiceAddresses(recipients);
|
||||||
try {
|
try {
|
||||||
sendMessageLegacy(messageBuilder, signalServiceAddresses);
|
return sendMessage(messageBuilder, signalServiceAddresses);
|
||||||
} catch (Exception e) {
|
} catch (Exception e) {
|
||||||
for (SignalServiceAddress address : signalServiceAddresses) {
|
for (SignalServiceAddress address : signalServiceAddresses) {
|
||||||
handleEndSession(address);
|
handleEndSession(address);
|
||||||
|
@ -793,7 +785,7 @@ public class Manager implements Closeable {
|
||||||
account.save();
|
account.save();
|
||||||
}
|
}
|
||||||
|
|
||||||
public byte[] updateGroup(byte[] groupId, String name, List<String> members, String avatar) throws IOException, EncapsulatedExceptions, GroupNotFoundException, AttachmentInvalidException, InvalidNumberException, NotAGroupMemberException {
|
public Pair<byte[], List<SendMessageResult>> updateGroup(byte[] groupId, String name, List<String> members, String avatar) throws IOException, GroupNotFoundException, AttachmentInvalidException, InvalidNumberException, NotAGroupMemberException {
|
||||||
if (groupId.length == 0) {
|
if (groupId.length == 0) {
|
||||||
groupId = null;
|
groupId = null;
|
||||||
}
|
}
|
||||||
|
@ -1098,34 +1090,6 @@ public class Manager implements Closeable {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* This method throws an EncapsulatedExceptions exception instead of returning a list of SendMessageResult.
|
|
||||||
*/
|
|
||||||
private long sendMessageLegacy(SignalServiceDataMessage.Builder messageBuilder, Collection<SignalServiceAddress> recipients)
|
|
||||||
throws EncapsulatedExceptions, IOException {
|
|
||||||
final long timestamp = System.currentTimeMillis();
|
|
||||||
messageBuilder.withTimestamp(timestamp);
|
|
||||||
List<SendMessageResult> results = sendMessage(messageBuilder, recipients);
|
|
||||||
|
|
||||||
List<UntrustedIdentityException> untrustedIdentities = new LinkedList<>();
|
|
||||||
List<UnregisteredUserException> unregisteredUsers = new LinkedList<>();
|
|
||||||
List<NetworkFailureException> networkExceptions = new LinkedList<>();
|
|
||||||
|
|
||||||
for (SendMessageResult result : results) {
|
|
||||||
if (result.isUnregisteredFailure()) {
|
|
||||||
unregisteredUsers.add(new UnregisteredUserException(result.getAddress().getLegacyIdentifier(), null));
|
|
||||||
} else if (result.isNetworkFailure()) {
|
|
||||||
networkExceptions.add(new NetworkFailureException(result.getAddress().getLegacyIdentifier(), null));
|
|
||||||
} else if (result.getIdentityFailure() != null) {
|
|
||||||
untrustedIdentities.add(new UntrustedIdentityException("Untrusted", result.getAddress().getLegacyIdentifier(), result.getIdentityFailure().getIdentityKey()));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (!untrustedIdentities.isEmpty() || !unregisteredUsers.isEmpty() || !networkExceptions.isEmpty()) {
|
|
||||||
throw new EncapsulatedExceptions(untrustedIdentities, unregisteredUsers, networkExceptions);
|
|
||||||
}
|
|
||||||
return timestamp;
|
|
||||||
}
|
|
||||||
|
|
||||||
private Collection<SignalServiceAddress> getSignalServiceAddresses(Collection<String> numbers) throws InvalidNumberException {
|
private Collection<SignalServiceAddress> getSignalServiceAddresses(Collection<String> numbers) throws InvalidNumberException {
|
||||||
final Set<SignalServiceAddress> signalServiceAddresses = new HashSet<>(numbers.size());
|
final Set<SignalServiceAddress> signalServiceAddresses = new HashSet<>(numbers.size());
|
||||||
|
|
||||||
|
@ -1135,8 +1099,10 @@ public class Manager implements Closeable {
|
||||||
return signalServiceAddresses;
|
return signalServiceAddresses;
|
||||||
}
|
}
|
||||||
|
|
||||||
private List<SendMessageResult> sendMessage(SignalServiceDataMessage.Builder messageBuilder, Collection<SignalServiceAddress> recipients)
|
private Pair<Long, List<SendMessageResult>> sendMessage(SignalServiceDataMessage.Builder messageBuilder, Collection<SignalServiceAddress> recipients)
|
||||||
throws IOException {
|
throws IOException {
|
||||||
|
final long timestamp = System.currentTimeMillis();
|
||||||
|
messageBuilder.withTimestamp(timestamp);
|
||||||
if (messagePipe == null) {
|
if (messagePipe == null) {
|
||||||
messagePipe = getMessageReceiver().createMessagePipe();
|
messagePipe = getMessageReceiver().createMessagePipe();
|
||||||
}
|
}
|
||||||
|
@ -1156,10 +1122,10 @@ public class Manager implements Closeable {
|
||||||
account.getSignalProtocolStore().saveIdentity(r.getAddress(), r.getIdentityFailure().getIdentityKey(), TrustLevel.UNTRUSTED);
|
account.getSignalProtocolStore().saveIdentity(r.getAddress(), r.getIdentityFailure().getIdentityKey(), TrustLevel.UNTRUSTED);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return result;
|
return new Pair<>(timestamp, result);
|
||||||
} catch (UntrustedIdentityException e) {
|
} catch (UntrustedIdentityException e) {
|
||||||
account.getSignalProtocolStore().saveIdentity(resolveSignalServiceAddress(e.getIdentifier()), e.getIdentityKey(), TrustLevel.UNTRUSTED);
|
account.getSignalProtocolStore().saveIdentity(resolveSignalServiceAddress(e.getIdentifier()), e.getIdentityKey(), TrustLevel.UNTRUSTED);
|
||||||
return Collections.emptyList();
|
return new Pair<>(timestamp, Collections.emptyList());
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
// Send to all individually, so sync messages are sent correctly
|
// Send to all individually, so sync messages are sent correctly
|
||||||
|
@ -1180,7 +1146,7 @@ public class Manager implements Closeable {
|
||||||
results.add(sendMessage(address, message));
|
results.add(sendMessage(address, message));
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return results;
|
return new Pair<>(timestamp, results);
|
||||||
}
|
}
|
||||||
} finally {
|
} finally {
|
||||||
if (message != null && message.isEndSession()) {
|
if (message != null && message.isEndSession()) {
|
||||||
|
@ -1553,9 +1519,7 @@ public class Manager implements Closeable {
|
||||||
if (message.getGroupContext().isPresent() && message.getGroupContext().get().getGroupV1().isPresent()) {
|
if (message.getGroupContext().isPresent() && message.getGroupContext().get().getGroupV1().isPresent()) {
|
||||||
SignalServiceGroup groupInfo = message.getGroupContext().get().getGroupV1().get();
|
SignalServiceGroup groupInfo = message.getGroupContext().get().getGroupV1().get();
|
||||||
GroupInfo group = getGroup(groupInfo.getGroupId());
|
GroupInfo group = getGroup(groupInfo.getGroupId());
|
||||||
if (groupInfo.getType() == SignalServiceGroup.Type.DELIVER && group != null && group.blocked) {
|
return groupInfo.getType() == SignalServiceGroup.Type.DELIVER && group != null && group.blocked;
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
return false;
|
return false;
|
||||||
|
|
|
@ -2,13 +2,12 @@ package org.asamk.signal.util;
|
||||||
|
|
||||||
import org.asamk.signal.manager.GroupNotFoundException;
|
import org.asamk.signal.manager.GroupNotFoundException;
|
||||||
import org.asamk.signal.manager.NotAGroupMemberException;
|
import org.asamk.signal.manager.NotAGroupMemberException;
|
||||||
import org.whispersystems.signalservice.api.crypto.UntrustedIdentityException;
|
import org.whispersystems.signalservice.api.messages.SendMessageResult;
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.EncapsulatedExceptions;
|
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.NetworkFailureException;
|
|
||||||
import org.whispersystems.signalservice.api.push.exceptions.UnregisteredUserException;
|
|
||||||
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
import org.whispersystems.signalservice.api.util.InvalidNumberException;
|
||||||
|
|
||||||
import java.io.IOException;
|
import java.io.IOException;
|
||||||
|
import java.util.ArrayList;
|
||||||
|
import java.util.List;
|
||||||
|
|
||||||
public class ErrorUtils {
|
public class ErrorUtils {
|
||||||
|
|
||||||
|
@ -21,17 +20,36 @@ public class ErrorUtils {
|
||||||
System.err.println("If you use an Oracle JRE please check if you have unlimited strength crypto enabled, see README");
|
System.err.println("If you use an Oracle JRE please check if you have unlimited strength crypto enabled, see README");
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void handleEncapsulatedExceptions(EncapsulatedExceptions e) {
|
public static int handleTimestampAndSendMessageResults(long timestamp, List<SendMessageResult> results) {
|
||||||
|
System.out.println(timestamp);
|
||||||
|
List<String> errors = getErrorMessagesFromSendMessageResults(results);
|
||||||
|
return handleSendMessageResultErrors(errors);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static List<String> getErrorMessagesFromSendMessageResults(List<SendMessageResult> results) {
|
||||||
|
List<String> errors = new ArrayList<>();
|
||||||
|
for (SendMessageResult result : results) {
|
||||||
|
if (result.isNetworkFailure()) {
|
||||||
|
errors.add(String.format("Network failure for \"%s\"", result.getAddress().getLegacyIdentifier()));
|
||||||
|
} else if (result.isUnregisteredFailure()) {
|
||||||
|
errors.add(String.format("Unregistered user \"%s\"", result.getAddress().getLegacyIdentifier()));
|
||||||
|
} else if (result.getIdentityFailure() != null) {
|
||||||
|
errors.add(String.format("Untrusted Identity for \"%s\"", result.getAddress().getLegacyIdentifier()));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return errors;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static int handleSendMessageResultErrors(List<String> errors) {
|
||||||
|
if (errors.size() == 0) {
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
System.err.println("Failed to send (some) messages:");
|
System.err.println("Failed to send (some) messages:");
|
||||||
for (NetworkFailureException n : e.getNetworkExceptions()) {
|
for (String error : errors) {
|
||||||
System.err.println("Network failure for \"" + n.getE164number() + "\": " + n.getMessage());
|
System.err.println(error);
|
||||||
}
|
|
||||||
for (UnregisteredUserException n : e.getUnregisteredUserExceptions()) {
|
|
||||||
System.err.println("Unregistered user \"" + n.getE164Number() + "\": " + n.getMessage());
|
|
||||||
}
|
|
||||||
for (UntrustedIdentityException n : e.getUntrustedIdentityExceptions()) {
|
|
||||||
System.err.println("Untrusted Identity for \"" + n.getIdentifier() + "\": " + n.getMessage());
|
|
||||||
}
|
}
|
||||||
|
return 3;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static void handleIOException(IOException e) {
|
public static void handleIOException(IOException e) {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue