1 package org
.asamk
.signal
.commands
;
3 import net
.sourceforge
.argparse4j
.impl
.Arguments
;
4 import net
.sourceforge
.argparse4j
.inf
.Namespace
;
5 import net
.sourceforge
.argparse4j
.inf
.Subparser
;
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
.AttachmentInvalidException
;
15 import org
.asamk
.signal
.manager
.Manager
;
16 import org
.asamk
.signal
.manager
.groups
.GroupId
;
17 import org
.asamk
.signal
.manager
.groups
.GroupLinkState
;
18 import org
.asamk
.signal
.manager
.groups
.GroupNotFoundException
;
19 import org
.asamk
.signal
.manager
.groups
.GroupPermission
;
20 import org
.asamk
.signal
.manager
.groups
.NotAGroupMemberException
;
21 import org
.asamk
.signal
.util
.CommandUtil
;
22 import org
.asamk
.signal
.util
.ErrorUtils
;
23 import org
.freedesktop
.dbus
.exceptions
.DBusExecutionException
;
24 import org
.slf4j
.Logger
;
25 import org
.slf4j
.LoggerFactory
;
28 import java
.io
.IOException
;
29 import java
.util
.ArrayList
;
30 import java
.util
.HashMap
;
31 import java
.util
.List
;
33 public class UpdateGroupCommand
implements DbusCommand
, JsonRpcLocalCommand
{
35 private final static Logger logger
= LoggerFactory
.getLogger(UpdateGroupCommand
.class);
38 public String
getName() {
43 public void attachToSubparser(final Subparser subparser
) {
44 subparser
.help("Create or update a group.");
45 subparser
.addArgument("-g", "--group-id", "--group").help("Specify the group ID.");
46 subparser
.addArgument("-n", "--name").help("Specify the new group name.");
47 subparser
.addArgument("-d", "--description").help("Specify the new group description.");
48 subparser
.addArgument("-a", "--avatar").help("Specify a new group avatar image file");
49 subparser
.addArgument("-m", "--member").nargs("*").help("Specify one or more members to add to the group");
50 subparser
.addArgument("-r", "--remove-member")
52 .help("Specify one or more members to remove from the group");
53 subparser
.addArgument("--admin").nargs("*").help("Specify one or more members to make a group admin");
54 subparser
.addArgument("--remove-admin")
56 .help("Specify one or more members to remove group admin privileges");
58 subparser
.addArgument("--reset-link")
59 .action(Arguments
.storeTrue())
60 .help("Reset group link and create new link password");
61 subparser
.addArgument("--link")
62 .help("Set group link state, with or without admin approval")
63 .choices("enabled", "enabled-with-approval", "disabled");
65 subparser
.addArgument("--set-permission-add-member")
66 .help("Set permission to add new group members")
67 .choices("every-member", "only-admins");
68 subparser
.addArgument("--set-permission-edit-details")
69 .help("Set permission to edit group details")
70 .choices("every-member", "only-admins");
71 subparser
.addArgument("--set-permission-send-messages")
72 .help("Set permission to send messages")
73 .choices("every-member", "only-admins");
75 subparser
.addArgument("-e", "--expiration").type(int.class).help("Set expiration time of messages (seconds)");
78 GroupLinkState
getGroupLinkState(String value
) throws UserErrorException
{
84 return GroupLinkState
.ENABLED
;
85 case "enabled-with-approval":
86 case "enabledWithApproval":
87 return GroupLinkState
.ENABLED_WITH_APPROVAL
;
89 return GroupLinkState
.DISABLED
;
91 throw new UserErrorException("Invalid group link state: " + value
);
95 GroupPermission
getGroupPermission(String value
) throws UserErrorException
{
102 return GroupPermission
.EVERY_MEMBER
;
105 return GroupPermission
.ONLY_ADMINS
;
107 throw new UserErrorException("Invalid group permission: " + value
);
112 public void handleCommand(
113 final Namespace ns
, final Manager m
, final OutputWriter outputWriter
114 ) throws CommandException
{
115 final var groupIdString
= ns
.getString("group-id");
116 var groupId
= CommandUtil
.getGroupId(groupIdString
);
118 final var localNumber
= m
.getUsername();
120 var groupName
= ns
.getString("name");
121 var groupDescription
= ns
.getString("description");
122 var groupMembers
= CommandUtil
.getSingleRecipientIdentifiers(ns
.getList("member"), localNumber
);
123 var groupRemoveMembers
= CommandUtil
.getSingleRecipientIdentifiers(ns
.getList("remove-member"), localNumber
);
124 var groupAdmins
= CommandUtil
.getSingleRecipientIdentifiers(ns
.getList("admin"), localNumber
);
125 var groupRemoveAdmins
= CommandUtil
.getSingleRecipientIdentifiers(ns
.getList("remove-admin"), localNumber
);
126 var groupAvatar
= ns
.getString("avatar");
127 var groupResetLink
= ns
.getBoolean("reset-link");
128 var groupLinkState
= getGroupLinkState(ns
.getString("link"));
129 var groupExpiration
= ns
.getInt("expiration");
130 var groupAddMemberPermission
= getGroupPermission(ns
.getString("set-permission-add-member"));
131 var groupEditDetailsPermission
= getGroupPermission(ns
.getString("set-permission-edit-details"));
132 var groupSendMessagesPermission
= getGroupPermission(ns
.getString("set-permission-send-messages"));
135 boolean isNewGroup
= false;
136 Long timestamp
= null;
137 if (groupId
== null) {
139 var results
= m
.createGroup(groupName
,
141 groupAvatar
== null ?
null : new File(groupAvatar
));
142 timestamp
= results
.second().getTimestamp();
143 ErrorUtils
.handleSendMessageResults(results
.second().getResults());
144 groupId
= results
.first();
150 var results
= m
.updateGroup(groupId
,
159 groupAddMemberPermission
,
160 groupEditDetailsPermission
,
161 groupAvatar
== null ?
null : new File(groupAvatar
),
163 groupSendMessagesPermission
== null
165 : groupSendMessagesPermission
== GroupPermission
.ONLY_ADMINS
);
166 if (results
!= null) {
167 timestamp
= results
.getTimestamp();
168 ErrorUtils
.handleSendMessageResults(results
.getResults());
170 outputResult(outputWriter
, timestamp
, isNewGroup ? groupId
: null);
171 } catch (AttachmentInvalidException e
) {
172 throw new UserErrorException("Failed to add avatar attachment for group\": " + e
.getMessage());
173 } catch (GroupNotFoundException
| NotAGroupMemberException e
) {
174 throw new UserErrorException(e
.getMessage());
175 } catch (IOException e
) {
176 throw new UnexpectedErrorException("Failed to send message: " + e
.getMessage());
181 public void handleCommand(
182 final Namespace ns
, final Signal signal
, final OutputWriter outputWriter
183 ) throws CommandException
{
184 var groupId
= CommandUtil
.getGroupId(ns
.getString("group-id"));
186 var groupName
= ns
.getString("name");
187 if (groupName
== null) {
191 List
<String
> groupMembers
= ns
.getList("member");
192 if (groupMembers
== null) {
193 groupMembers
= new ArrayList
<>();
196 var groupAvatar
= ns
.getString("avatar");
197 if (groupAvatar
== null) {
202 var newGroupId
= signal
.updateGroup(groupId
== null ?
new byte[0] : groupId
.serialize(),
206 if (groupId
== null) {
207 outputResult(outputWriter
, null, GroupId
.unknownVersion(newGroupId
));
209 } catch (Signal
.Error
.AttachmentInvalid e
) {
210 throw new UserErrorException("Failed to add avatar attachment for group\": " + e
.getMessage());
211 } catch (DBusExecutionException e
) {
212 throw new UnexpectedErrorException("Failed to send message: " + e
.getMessage());
216 private void outputResult(final OutputWriter outputWriter
, final Long timestamp
, final GroupId groupId
) {
217 if (outputWriter
instanceof PlainTextWriter
) {
218 final var writer
= (PlainTextWriter
) outputWriter
;
219 if (groupId
!= null) {
220 writer
.println("Created new group: \"{}\"", groupId
.toBase64());
222 if (timestamp
!= null) {
223 writer
.println("{}", timestamp
);
226 final var writer
= (JsonWriter
) outputWriter
;
227 final var result
= new HashMap
<>();
228 if (timestamp
!= null) {
229 result
.put("timestamp", timestamp
);
231 if (groupId
!= null) {
232 result
.put("groupId", groupId
.toBase64());
234 writer
.write(result
);