]> nmode's Git Repositories - signal-cli/blob - src/main/java/org/asamk/signal/commands/SendReactionCommand.java
Extract ContactHelper and IncomingMessageHandler
[signal-cli] / src / main / java / org / asamk / signal / commands / SendReactionCommand.java
1 package org.asamk.signal.commands;
2
3 import net.sourceforge.argparse4j.impl.Arguments;
4 import net.sourceforge.argparse4j.inf.Namespace;
5 import net.sourceforge.argparse4j.inf.Subparser;
6
7 import org.asamk.Signal;
8 import org.asamk.signal.JsonWriter;
9 import org.asamk.signal.OutputWriter;
10 import org.asamk.signal.PlainTextWriter;
11 import org.asamk.signal.commands.exceptions.CommandException;
12 import org.asamk.signal.commands.exceptions.UnexpectedErrorException;
13 import org.asamk.signal.commands.exceptions.UserErrorException;
14 import org.asamk.signal.manager.Manager;
15 import org.asamk.signal.manager.groups.GroupNotFoundException;
16 import org.asamk.signal.manager.groups.GroupSendingNotAllowedException;
17 import org.asamk.signal.manager.groups.NotAGroupMemberException;
18 import org.asamk.signal.util.CommandUtil;
19 import org.asamk.signal.util.ErrorUtils;
20 import org.freedesktop.dbus.errors.UnknownObject;
21 import org.freedesktop.dbus.exceptions.DBusExecutionException;
22
23 import java.io.IOException;
24 import java.util.Map;
25
26 public class SendReactionCommand implements DbusCommand, JsonRpcLocalCommand {
27
28 @Override
29 public String getName() {
30 return "sendReaction";
31 }
32
33 @Override
34 public void attachToSubparser(final Subparser subparser) {
35 subparser.help("Send reaction to a previously received or sent message.");
36 subparser.addArgument("-g", "--group-id", "--group").help("Specify the recipient group ID.").nargs("*");
37 subparser.addArgument("recipient").help("Specify the recipients' phone number.").nargs("*");
38 subparser.addArgument("--note-to-self")
39 .help("Send the reaction to self without notification.")
40 .action(Arguments.storeTrue());
41 subparser.addArgument("-e", "--emoji")
42 .required(true)
43 .help("Specify the emoji, should be a single unicode grapheme cluster.");
44 subparser.addArgument("-a", "--target-author")
45 .required(true)
46 .help("Specify the number of the author of the message to which to react.");
47 subparser.addArgument("-t", "--target-timestamp")
48 .required(true)
49 .type(long.class)
50 .help("Specify the timestamp of the message to which to react.");
51 subparser.addArgument("-r", "--remove").help("Remove a reaction.").action(Arguments.storeTrue());
52 }
53
54 @Override
55 public void handleCommand(
56 final Namespace ns, final Manager m, final OutputWriter outputWriter
57 ) throws CommandException {
58 final var isNoteToSelf = ns.getBoolean("note-to-self");
59 final var recipientStrings = ns.<String>getList("recipient");
60 final var groupIdStrings = ns.<String>getList("group-id");
61
62 final var recipientIdentifiers = CommandUtil.getRecipientIdentifiers(m,
63 isNoteToSelf,
64 recipientStrings,
65 groupIdStrings);
66
67 final var emoji = ns.getString("emoji");
68 final var isRemove = ns.getBoolean("remove");
69 final var targetAuthor = ns.getString("target-author");
70 final var targetTimestamp = ns.getLong("target-timestamp");
71
72 try {
73 final var results = m.sendMessageReaction(emoji,
74 isRemove,
75 CommandUtil.getSingleRecipientIdentifier(targetAuthor, m.getUsername()),
76 targetTimestamp,
77 recipientIdentifiers);
78 outputResult(outputWriter, results.getTimestamp());
79 ErrorUtils.handleSendMessageResults(results.getResults());
80 } catch (GroupNotFoundException | NotAGroupMemberException | GroupSendingNotAllowedException e) {
81 throw new UserErrorException(e.getMessage());
82 } catch (IOException e) {
83 throw new UnexpectedErrorException("Failed to send message: " + e.getMessage());
84 }
85 }
86
87 @Override
88 public void handleCommand(
89 final Namespace ns, final Signal signal, final OutputWriter outputWriter
90 ) throws CommandException {
91 final var recipients = ns.<String>getList("recipient");
92 final var groupIdStrings = ns.<String>getList("group-id");
93
94 final var noRecipients = recipients == null || recipients.isEmpty();
95 final var noGroups = groupIdStrings == null || groupIdStrings.isEmpty();
96 if (noRecipients && noGroups) {
97 throw new UserErrorException("No recipients given");
98 }
99 if (!noRecipients && !noGroups) {
100 throw new UserErrorException("You cannot specify recipients by phone number and groups at the same time");
101 }
102
103 final var emoji = ns.getString("emoji");
104 final var isRemove = ns.getBoolean("remove");
105 final var targetAuthor = ns.getString("target-author");
106 final var targetTimestamp = ns.getLong("target-timestamp");
107
108 try {
109 long timestamp = 0;
110 if (!noGroups) {
111 final var groupIds = CommandUtil.getGroupIds(groupIdStrings);
112 for (final var groupId : groupIds) {
113 timestamp = signal.sendGroupMessageReaction(emoji,
114 isRemove,
115 targetAuthor,
116 targetTimestamp,
117 groupId.serialize());
118 }
119 } else {
120 timestamp = signal.sendMessageReaction(emoji, isRemove, targetAuthor, targetTimestamp, recipients);
121 }
122 outputResult(outputWriter, timestamp);
123 } catch (UnknownObject e) {
124 throw new UserErrorException("Failed to find dbus object, maybe missing the -u flag: " + e.getMessage());
125 } catch (Signal.Error.InvalidNumber e) {
126 throw new UserErrorException("Invalid number: " + e.getMessage());
127 } catch (Signal.Error.GroupNotFound e) {
128 throw new UserErrorException("Failed to send to group: " + e.getMessage());
129 } catch (DBusExecutionException e) {
130 throw new UnexpectedErrorException("Failed to send message: " + e.getMessage());
131 }
132 }
133
134 private void outputResult(final OutputWriter outputWriter, final long timestamp) {
135 if (outputWriter instanceof PlainTextWriter) {
136 final var writer = (PlainTextWriter) outputWriter;
137 writer.println("{}", timestamp);
138 } else {
139 final var writer = (JsonWriter) outputWriter;
140 writer.write(Map.of("timestamp", timestamp));
141 }
142 }
143 }