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
.GroupIdFormatException
;
18 import org
.asamk
.signal
.manager
.groups
.GroupLinkState
;
19 import org
.asamk
.signal
.manager
.groups
.GroupNotFoundException
;
20 import org
.asamk
.signal
.manager
.groups
.GroupPermission
;
21 import org
.asamk
.signal
.manager
.groups
.NotAGroupMemberException
;
22 import org
.asamk
.signal
.util
.ErrorUtils
;
23 import org
.asamk
.signal
.util
.Util
;
24 import org
.freedesktop
.dbus
.exceptions
.DBusExecutionException
;
25 import org
.slf4j
.Logger
;
26 import org
.slf4j
.LoggerFactory
;
27 import org
.whispersystems
.signalservice
.api
.util
.InvalidNumberException
;
30 import java
.io
.IOException
;
31 import java
.util
.ArrayList
;
32 import java
.util
.HashMap
;
33 import java
.util
.List
;
35 public class UpdateGroupCommand
implements DbusCommand
, JsonRpcLocalCommand
{
37 private final static Logger logger
= LoggerFactory
.getLogger(UpdateGroupCommand
.class);
40 public String
getName() {
45 public void attachToSubparser(final Subparser subparser
) {
46 subparser
.help("Create or update a group.");
47 subparser
.addArgument("-g", "--group-id", "--group").help("Specify the group ID.");
48 subparser
.addArgument("-n", "--name").help("Specify the new group name.");
49 subparser
.addArgument("-d", "--description").help("Specify the new group description.");
50 subparser
.addArgument("-a", "--avatar").help("Specify a new group avatar image file");
51 subparser
.addArgument("-m", "--member").nargs("*").help("Specify one or more members to add to the group");
52 subparser
.addArgument("-r", "--remove-member")
54 .help("Specify one or more members to remove from the group");
55 subparser
.addArgument("--admin").nargs("*").help("Specify one or more members to make a group admin");
56 subparser
.addArgument("--remove-admin")
58 .help("Specify one or more members to remove group admin privileges");
60 subparser
.addArgument("--reset-link")
61 .action(Arguments
.storeTrue())
62 .help("Reset group link and create new link password");
63 subparser
.addArgument("--link")
64 .help("Set group link state, with or without admin approval")
65 .choices("enabled", "enabled-with-approval", "disabled");
67 subparser
.addArgument("--set-permission-add-member")
68 .help("Set permission to add new group members")
69 .choices("every-member", "only-admins");
70 subparser
.addArgument("--set-permission-edit-details")
71 .help("Set permission to edit group details")
72 .choices("every-member", "only-admins");
74 subparser
.addArgument("-e", "--expiration").type(int.class).help("Set expiration time of messages (seconds)");
77 GroupLinkState
getGroupLinkState(String value
) throws UserErrorException
{
83 return GroupLinkState
.ENABLED
;
84 case "enabled-with-approval":
85 case "enabledWithApproval":
86 return GroupLinkState
.ENABLED_WITH_APPROVAL
;
88 return GroupLinkState
.DISABLED
;
90 throw new UserErrorException("Invalid group link state: " + value
);
94 GroupPermission
getGroupPermission(String value
) throws UserErrorException
{
101 return GroupPermission
.EVERY_MEMBER
;
104 return GroupPermission
.ONLY_ADMINS
;
106 throw new UserErrorException("Invalid group permission: " + value
);
111 public void handleCommand(
112 final Namespace ns
, final Manager m
, final OutputWriter outputWriter
113 ) throws CommandException
{
114 GroupId groupId
= null;
115 final var groupIdString
= ns
.getString("group-id");
116 if (groupIdString
!= null) {
118 groupId
= Util
.decodeGroupId(groupIdString
);
119 } catch (GroupIdFormatException e
) {
120 throw new UserErrorException("Invalid group id: " + e
.getMessage());
124 var groupName
= ns
.getString("name");
125 var groupDescription
= ns
.getString("description");
126 var groupMembers
= ns
.<String
>getList("member");
127 var groupRemoveMembers
= ns
.<String
>getList("remove-member");
128 var groupAdmins
= ns
.<String
>getList("admin");
129 var groupRemoveAdmins
= ns
.<String
>getList("remove-admin");
130 var groupAvatar
= ns
.getString("avatar");
131 var groupResetLink
= ns
.getBoolean("reset-link");
132 var groupLinkState
= getGroupLinkState(ns
.getString("link"));
133 var groupExpiration
= ns
.getInt("expiration");
134 var groupAddMemberPermission
= getGroupPermission(ns
.getString("set-permission-add-member"));
135 var groupEditDetailsPermission
= getGroupPermission(ns
.getString("set-permission-edit-details"));
138 boolean isNewGroup
= false;
139 if (groupId
== null) {
141 var results
= m
.createGroup(groupName
,
143 groupAvatar
== null ?
null : new File(groupAvatar
));
144 ErrorUtils
.handleSendMessageResults(results
.second());
145 groupId
= results
.first();
151 var results
= m
.updateGroup(groupId
,
160 groupAddMemberPermission
,
161 groupEditDetailsPermission
,
162 groupAvatar
== null ?
null : new File(groupAvatar
),
164 Long timestamp
= null;
165 if (results
!= null) {
166 timestamp
= results
.first();
167 ErrorUtils
.handleSendMessageResults(results
.second());
169 outputResult(outputWriter
, timestamp
, isNewGroup ? groupId
: null);
170 } catch (AttachmentInvalidException e
) {
171 throw new UserErrorException("Failed to add avatar attachment for group\": " + e
.getMessage());
172 } catch (GroupNotFoundException e
) {
173 logger
.warn("Unknown group id: {}", groupIdString
);
174 } catch (NotAGroupMemberException e
) {
175 logger
.warn("You're not a group member");
176 } catch (InvalidNumberException e
) {
177 throw new UserErrorException("Failed to parse member number: " + e
.getMessage());
178 } catch (IOException e
) {
179 throw new UnexpectedErrorException("Failed to send message: " + e
.getMessage());
184 public void handleCommand(
185 final Namespace ns
, final Signal signal
, final OutputWriter outputWriter
186 ) throws CommandException
{
187 byte[] groupId
= null;
188 if (ns
.getString("group-id") != null) {
190 groupId
= Util
.decodeGroupId(ns
.getString("group-id")).serialize();
191 } catch (GroupIdFormatException e
) {
192 throw new UserErrorException("Invalid group id: " + e
.getMessage());
195 if (groupId
== null) {
196 groupId
= new byte[0];
199 var groupName
= ns
.getString("name");
200 if (groupName
== null) {
204 List
<String
> groupMembers
= ns
.getList("member");
205 if (groupMembers
== null) {
206 groupMembers
= new ArrayList
<>();
209 var groupAvatar
= ns
.getString("avatar");
210 if (groupAvatar
== null) {
215 var newGroupId
= signal
.updateGroup(groupId
, groupName
, groupMembers
, groupAvatar
);
216 if (groupId
.length
!= newGroupId
.length
) {
217 outputResult(outputWriter
, null, GroupId
.unknownVersion(newGroupId
));
219 } catch (Signal
.Error
.AttachmentInvalid e
) {
220 throw new UserErrorException("Failed to add avatar attachment for group\": " + e
.getMessage());
221 } catch (DBusExecutionException e
) {
222 throw new UnexpectedErrorException("Failed to send message: " + e
.getMessage());
226 private void outputResult(final OutputWriter outputWriter
, final Long timestamp
, final GroupId groupId
) {
227 if (outputWriter
instanceof PlainTextWriter
) {
228 final var writer
= (PlainTextWriter
) outputWriter
;
229 if (groupId
!= null) {
230 writer
.println("Created new group: \"{}\"", groupId
.toBase64());
232 if (timestamp
!= null) {
233 writer
.println("{}", timestamp
);
236 final var writer
= (JsonWriter
) outputWriter
;
237 final var result
= new HashMap
<>();
238 if (timestamp
!= null) {
239 result
.put("timestamp", timestamp
);
241 if (groupId
!= null) {
242 result
.put("groupId", groupId
.toBase64());
244 writer
.write(result
);