]> nmode's Git Repositories - signal-cli/blob - lib/src/main/java/org/asamk/signal/manager/helper/RecipientHelper.java
Add labels to Containerfile
[signal-cli] / lib / src / main / java / org / asamk / signal / manager / helper / RecipientHelper.java
1 package org.asamk.signal.manager.helper;
2
3 import org.asamk.signal.manager.SignalDependencies;
4 import org.asamk.signal.manager.api.RecipientIdentifier;
5 import org.asamk.signal.manager.api.UnregisteredRecipientException;
6 import org.asamk.signal.manager.config.ServiceEnvironmentConfig;
7 import org.asamk.signal.manager.storage.SignalAccount;
8 import org.asamk.signal.manager.storage.recipients.RecipientId;
9 import org.signal.libsignal.usernames.BaseUsernameException;
10 import org.signal.libsignal.usernames.Username;
11 import org.slf4j.Logger;
12 import org.slf4j.LoggerFactory;
13 import org.whispersystems.signalservice.api.push.ACI;
14 import org.whispersystems.signalservice.api.push.PNI;
15 import org.whispersystems.signalservice.api.push.ServiceId;
16 import org.whispersystems.signalservice.api.push.SignalServiceAddress;
17 import org.whispersystems.signalservice.api.services.CdsiV2Service;
18 import org.whispersystems.util.Base64UrlSafe;
19
20 import java.io.IOException;
21 import java.util.Collection;
22 import java.util.HashMap;
23 import java.util.HashSet;
24 import java.util.Map;
25 import java.util.Optional;
26 import java.util.Set;
27
28 public class RecipientHelper {
29
30 private final static Logger logger = LoggerFactory.getLogger(RecipientHelper.class);
31
32 private final SignalAccount account;
33 private final SignalDependencies dependencies;
34 private final ServiceEnvironmentConfig serviceEnvironmentConfig;
35
36 public RecipientHelper(final Context context) {
37 this.account = context.getAccount();
38 this.dependencies = context.getDependencies();
39 this.serviceEnvironmentConfig = dependencies.getServiceEnvironmentConfig();
40 }
41
42 public SignalServiceAddress resolveSignalServiceAddress(RecipientId recipientId) {
43 final var address = account.getRecipientAddressResolver().resolveRecipientAddress(recipientId);
44 if (address.number().isEmpty() || address.serviceId().isPresent()) {
45 return address.toSignalServiceAddress();
46 }
47
48 // Address in recipient store doesn't have a uuid, this shouldn't happen
49 // Try to retrieve the uuid from the server
50 final var number = address.number().get();
51 final ServiceId serviceId;
52 try {
53 serviceId = getRegisteredUserByNumber(number);
54 } catch (UnregisteredRecipientException e) {
55 logger.warn("Failed to get uuid for e164 number: {}", number);
56 // Return SignalServiceAddress with unknown UUID
57 return address.toSignalServiceAddress();
58 } catch (IOException e) {
59 logger.warn("Failed to get uuid for e164 number: {}", number, e);
60 // Return SignalServiceAddress with unknown UUID
61 return address.toSignalServiceAddress();
62 }
63 return account.getRecipientAddressResolver()
64 .resolveRecipientAddress(account.getRecipientResolver().resolveRecipient(serviceId))
65 .toSignalServiceAddress();
66 }
67
68 public RecipientId resolveRecipient(final SignalServiceAddress address) {
69 return account.getRecipientResolver().resolveRecipient(address);
70 }
71
72 public Set<RecipientId> resolveRecipients(Collection<RecipientIdentifier.Single> recipients) throws UnregisteredRecipientException {
73 final var recipientIds = new HashSet<RecipientId>(recipients.size());
74 for (var number : recipients) {
75 final var recipientId = resolveRecipient(number);
76 recipientIds.add(recipientId);
77 }
78 return recipientIds;
79 }
80
81 public RecipientId resolveRecipient(final RecipientIdentifier.Single recipient) throws UnregisteredRecipientException {
82 if (recipient instanceof RecipientIdentifier.Uuid uuidRecipient) {
83 return account.getRecipientResolver().resolveRecipient(ServiceId.from(uuidRecipient.uuid()));
84 } else if (recipient instanceof RecipientIdentifier.Number numberRecipient) {
85 final var number = numberRecipient.number();
86 return account.getRecipientStore().resolveRecipientByNumber(number, () -> {
87 try {
88 return getRegisteredUserByNumber(number);
89 } catch (Exception e) {
90 return null;
91 }
92 });
93 } else if (recipient instanceof RecipientIdentifier.Username usernameRecipient) {
94 final var username = usernameRecipient.username();
95 return account.getRecipientStore().resolveRecipientByUsername(username, () -> {
96 try {
97 return getRegisteredUserByUsername(username);
98 } catch (Exception e) {
99 return null;
100 }
101 });
102 }
103 throw new AssertionError("Unexpected RecipientIdentifier: " + recipient);
104 }
105
106 public Optional<RecipientId> resolveRecipientOptional(final RecipientIdentifier.Single recipient) {
107 try {
108 return Optional.of(resolveRecipient(recipient));
109 } catch (UnregisteredRecipientException e) {
110 return Optional.empty();
111 }
112 }
113
114 public RecipientId refreshRegisteredUser(RecipientId recipientId) throws IOException, UnregisteredRecipientException {
115 final var address = resolveSignalServiceAddress(recipientId);
116 if (address.getNumber().isEmpty()) {
117 return recipientId;
118 }
119 final var number = address.getNumber().get();
120 final var serviceId = getRegisteredUserByNumber(number);
121 return account.getRecipientTrustedResolver()
122 .resolveRecipientTrusted(new SignalServiceAddress(serviceId, number));
123 }
124
125 public Map<String, RegisteredUser> getRegisteredUsers(final Set<String> numbers) throws IOException {
126 Map<String, RegisteredUser> registeredUsers = getRegisteredUsersV2(numbers, true);
127
128 // Store numbers as recipients, so we have the number/uuid association
129 registeredUsers.forEach((number, u) -> account.getRecipientTrustedResolver()
130 .resolveRecipientTrusted(u.aci, u.pni, Optional.of(number)));
131
132 return registeredUsers;
133 }
134
135 private ServiceId getRegisteredUserByNumber(final String number) throws IOException, UnregisteredRecipientException {
136 final Map<String, RegisteredUser> aciMap;
137 try {
138 aciMap = getRegisteredUsers(Set.of(number));
139 } catch (NumberFormatException e) {
140 throw new UnregisteredRecipientException(new org.asamk.signal.manager.api.RecipientAddress(null, number));
141 }
142 final var user = aciMap.get(number);
143 if (user == null) {
144 throw new UnregisteredRecipientException(new org.asamk.signal.manager.api.RecipientAddress(null, number));
145 }
146 return user.getServiceId();
147 }
148
149 private Map<String, RegisteredUser> getRegisteredUsersV2(
150 final Set<String> numbers, boolean useCompat
151 ) throws IOException {
152 // Only partial refresh is implemented here
153 final CdsiV2Service.Response response;
154 try {
155 response = dependencies.getAccountManager()
156 .getRegisteredUsersWithCdsi(Set.of(),
157 numbers,
158 account.getRecipientStore().getServiceIdToProfileKeyMap(),
159 useCompat,
160 Optional.empty(),
161 serviceEnvironmentConfig.getCdsiMrenclave(),
162 token -> {
163 // Not storing for partial refresh
164 });
165 } catch (NumberFormatException e) {
166 throw new IOException(e);
167 }
168 logger.debug("CDSI request successful, quota used by this request: {}", response.getQuotaUsedDebugOnly());
169
170 final var registeredUsers = new HashMap<String, RegisteredUser>();
171 response.getResults()
172 .forEach((key, value) -> registeredUsers.put(key,
173 new RegisteredUser(value.getAci(), Optional.of(value.getPni()))));
174 return registeredUsers;
175 }
176
177 private ACI getRegisteredUserByUsername(String username) throws IOException, BaseUsernameException {
178 return dependencies.getAccountManager()
179 .getAciByUsernameHash(Base64UrlSafe.encodeBytesWithoutPadding(Username.hash(username)));
180 }
181
182 public record RegisteredUser(Optional<ACI> aci, Optional<PNI> pni) {
183
184 public RegisteredUser {
185 aci = aci.isPresent() && aci.get().equals(ServiceId.UNKNOWN) ? Optional.empty() : aci;
186 pni = pni.isPresent() && pni.get().equals(ServiceId.UNKNOWN) ? Optional.empty() : pni;
187 if (aci.isEmpty() && pni.isEmpty()) {
188 throw new AssertionError("Must have either a ACI or PNI!");
189 }
190 }
191
192 public ServiceId getServiceId() {
193 return aci.map(a -> (ServiceId) a).or(this::pni).orElse(null);
194 }
195 }
196 }