1 package org
.asamk
.signal
.manager
.internal
;
3 import org
.asamk
.signal
.manager
.config
.ServiceConfig
;
4 import org
.asamk
.signal
.manager
.config
.ServiceEnvironmentConfig
;
5 import org
.asamk
.signal
.manager
.util
.Utils
;
6 import org
.signal
.libsignal
.metadata
.certificate
.CertificateValidator
;
7 import org
.signal
.libsignal
.net
.Network
;
8 import org
.signal
.libsignal
.zkgroup
.profiles
.ClientZkProfileOperations
;
9 import org
.slf4j
.Logger
;
10 import org
.slf4j
.LoggerFactory
;
11 import org
.whispersystems
.signalservice
.api
.SignalServiceAccountManager
;
12 import org
.whispersystems
.signalservice
.api
.SignalServiceDataStore
;
13 import org
.whispersystems
.signalservice
.api
.SignalServiceMessageReceiver
;
14 import org
.whispersystems
.signalservice
.api
.SignalServiceMessageSender
;
15 import org
.whispersystems
.signalservice
.api
.SignalSessionLock
;
16 import org
.whispersystems
.signalservice
.api
.account
.AccountApi
;
17 import org
.whispersystems
.signalservice
.api
.cds
.CdsApi
;
18 import org
.whispersystems
.signalservice
.api
.crypto
.SignalServiceCipher
;
19 import org
.whispersystems
.signalservice
.api
.groupsv2
.ClientZkOperations
;
20 import org
.whispersystems
.signalservice
.api
.groupsv2
.GroupsV2Api
;
21 import org
.whispersystems
.signalservice
.api
.groupsv2
.GroupsV2Operations
;
22 import org
.whispersystems
.signalservice
.api
.link
.LinkDeviceApi
;
23 import org
.whispersystems
.signalservice
.api
.push
.ServiceIdType
;
24 import org
.whispersystems
.signalservice
.api
.push
.SignalServiceAddress
;
25 import org
.whispersystems
.signalservice
.api
.ratelimit
.RateLimitChallengeApi
;
26 import org
.whispersystems
.signalservice
.api
.registration
.RegistrationApi
;
27 import org
.whispersystems
.signalservice
.api
.services
.ProfileService
;
28 import org
.whispersystems
.signalservice
.api
.storage
.StorageServiceApi
;
29 import org
.whispersystems
.signalservice
.api
.storage
.StorageServiceRepository
;
30 import org
.whispersystems
.signalservice
.api
.svr
.SecureValueRecovery
;
31 import org
.whispersystems
.signalservice
.api
.username
.UsernameApi
;
32 import org
.whispersystems
.signalservice
.api
.util
.CredentialsProvider
;
33 import org
.whispersystems
.signalservice
.api
.util
.UptimeSleepTimer
;
34 import org
.whispersystems
.signalservice
.api
.websocket
.SignalWebSocket
;
35 import org
.whispersystems
.signalservice
.internal
.push
.PushServiceSocket
;
36 import org
.whispersystems
.signalservice
.internal
.websocket
.OkHttpWebSocketConnection
;
38 import java
.io
.IOException
;
39 import java
.net
.InetSocketAddress
;
40 import java
.net
.Proxy
;
41 import java
.util
.List
;
42 import java
.util
.Optional
;
43 import java
.util
.concurrent
.ExecutorService
;
44 import java
.util
.concurrent
.TimeUnit
;
45 import java
.util
.function
.Supplier
;
47 public class SignalDependencies
{
49 private static final Logger logger
= LoggerFactory
.getLogger(SignalDependencies
.class);
51 private final Object LOCK
= new Object();
53 private final ServiceEnvironmentConfig serviceEnvironmentConfig
;
54 private final String userAgent
;
55 private final CredentialsProvider credentialsProvider
;
56 private final SignalServiceDataStore dataStore
;
57 private final ExecutorService executor
;
58 private final SignalSessionLock sessionLock
;
60 private boolean allowStories
= true;
62 private SignalServiceAccountManager accountManager
;
63 private AccountApi accountApi
;
64 private RateLimitChallengeApi rateLimitChallengeApi
;
65 private CdsApi cdsApi
;
66 private UsernameApi usernameApi
;
67 private GroupsV2Api groupsV2Api
;
68 private RegistrationApi registrationApi
;
69 private LinkDeviceApi linkDeviceApi
;
70 private StorageServiceApi storageServiceApi
;
71 private GroupsV2Operations groupsV2Operations
;
72 private ClientZkOperations clientZkOperations
;
74 private PushServiceSocket pushServiceSocket
;
75 private Network libSignalNetwork
;
76 private SignalWebSocket
.AuthenticatedWebSocket authenticatedSignalWebSocket
;
77 private SignalWebSocket
.UnauthenticatedWebSocket unauthenticatedSignalWebSocket
;
78 private SignalServiceMessageReceiver messageReceiver
;
79 private SignalServiceMessageSender messageSender
;
81 private List
<SecureValueRecovery
> secureValueRecovery
;
82 private ProfileService profileService
;
85 final ServiceEnvironmentConfig serviceEnvironmentConfig
,
86 final String userAgent
,
87 final CredentialsProvider credentialsProvider
,
88 final SignalServiceDataStore dataStore
,
89 final ExecutorService executor
,
90 final SignalSessionLock sessionLock
92 this.serviceEnvironmentConfig
= serviceEnvironmentConfig
;
93 this.userAgent
= userAgent
;
94 this.credentialsProvider
= credentialsProvider
;
95 this.dataStore
= dataStore
;
96 this.executor
= executor
;
97 this.sessionLock
= sessionLock
;
100 public void resetAfterAddressChange() {
101 if (this.pushServiceSocket
!= null) {
102 this.pushServiceSocket
.close();
103 this.pushServiceSocket
= null;
104 this.accountManager
= null;
105 this.messageReceiver
= null;
106 this.messageSender
= null;
107 this.profileService
= null;
108 this.groupsV2Api
= null;
109 this.registrationApi
= null;
110 this.secureValueRecovery
= null;
112 if (this.authenticatedSignalWebSocket
!= null) {
113 this.authenticatedSignalWebSocket
.forceNewWebSocket();
115 if (this.unauthenticatedSignalWebSocket
!= null) {
116 this.unauthenticatedSignalWebSocket
.forceNewWebSocket();
121 * This method needs to be called before the first websocket is created
123 public void setAllowStories(final boolean allowStories
) {
124 this.allowStories
= allowStories
;
127 public ServiceEnvironmentConfig
getServiceEnvironmentConfig() {
128 return serviceEnvironmentConfig
;
131 public SignalSessionLock
getSessionLock() {
135 public PushServiceSocket
getPushServiceSocket() {
136 return getOrCreate(() -> pushServiceSocket
,
137 () -> pushServiceSocket
= new PushServiceSocket(serviceEnvironmentConfig
.signalServiceConfiguration(),
140 getClientZkProfileOperations(),
141 ServiceConfig
.AUTOMATIC_NETWORK_RETRY
));
144 public Network
getLibSignalNetwork() {
145 return getOrCreate(() -> libSignalNetwork
, () -> {
146 libSignalNetwork
= new Network(serviceEnvironmentConfig
.netEnvironment(), userAgent
);
147 setSignalNetworkProxy(libSignalNetwork
);
151 private void setSignalNetworkProxy(Network libSignalNetwork
) {
152 final var proxy
= Utils
.getHttpsProxy();
153 if (proxy
.address() instanceof InetSocketAddress addr
) {
154 switch (proxy
.type()) {
155 case Proxy
.Type
.DIRECT
-> {
157 case Proxy
.Type
.HTTP
-> {
159 libSignalNetwork
.setProxy("http", addr
.getHostName(), addr
.getPort(), null, null);
160 } catch (IOException e
) {
161 logger
.warn("Failed to set http proxy", e
);
164 case Proxy
.Type
.SOCKS
-> {
166 libSignalNetwork
.setProxy("socks", addr
.getHostName(), addr
.getPort(), null, null);
167 } catch (IOException e
) {
168 logger
.warn("Failed to set socks proxy", e
);
175 public SignalServiceAccountManager
getAccountManager() {
176 return getOrCreate(() -> accountManager
,
177 () -> accountManager
= new SignalServiceAccountManager(getAccountApi(),
178 getPushServiceSocket(),
179 getGroupsV2Operations()));
182 public SignalServiceAccountManager
createUnauthenticatedAccountManager(String number
, String password
) {
183 return SignalServiceAccountManager
.createWithStaticCredentials(getServiceEnvironmentConfig().signalServiceConfiguration(),
187 SignalServiceAddress
.DEFAULT_DEVICE_ID
,
190 ServiceConfig
.AUTOMATIC_NETWORK_RETRY
,
191 ServiceConfig
.GROUP_MAX_SIZE
);
194 public AccountApi
getAccountApi() {
195 return getOrCreate(() -> accountApi
, () -> accountApi
= new AccountApi(getAuthenticatedSignalWebSocket()));
198 public RateLimitChallengeApi
getRateLimitChallengeApi() {
199 return getOrCreate(() -> rateLimitChallengeApi
,
200 () -> rateLimitChallengeApi
= new RateLimitChallengeApi(getAuthenticatedSignalWebSocket()));
203 public CdsApi
getCdsApi() {
204 return getOrCreate(() -> cdsApi
, () -> cdsApi
= new CdsApi(getAuthenticatedSignalWebSocket()));
207 public UsernameApi
getUsernameApi() {
208 return getOrCreate(() -> usernameApi
, () -> usernameApi
= new UsernameApi(getUnauthenticatedSignalWebSocket()));
211 public GroupsV2Api
getGroupsV2Api() {
212 return getOrCreate(() -> groupsV2Api
, () -> groupsV2Api
= getAccountManager().getGroupsV2Api());
215 public RegistrationApi
getRegistrationApi() {
216 return getOrCreate(() -> registrationApi
, () -> registrationApi
= getAccountManager().getRegistrationApi());
219 public LinkDeviceApi
getLinkDeviceApi() {
220 return getOrCreate(() -> linkDeviceApi
,
221 () -> linkDeviceApi
= new LinkDeviceApi(getAuthenticatedSignalWebSocket()));
224 private StorageServiceApi
getStorageServiceApi() {
225 return getOrCreate(() -> storageServiceApi
,
226 () -> storageServiceApi
= new StorageServiceApi(getAuthenticatedSignalWebSocket(),
227 getPushServiceSocket()));
230 public StorageServiceRepository
getStorageServiceRepository() {
231 return new StorageServiceRepository(getStorageServiceApi());
234 public GroupsV2Operations
getGroupsV2Operations() {
235 return getOrCreate(() -> groupsV2Operations
,
236 () -> groupsV2Operations
= new GroupsV2Operations(ClientZkOperations
.create(serviceEnvironmentConfig
.signalServiceConfiguration()),
237 ServiceConfig
.GROUP_MAX_SIZE
));
240 private ClientZkOperations
getClientZkOperations() {
241 return getOrCreate(() -> clientZkOperations
,
242 () -> clientZkOperations
= ClientZkOperations
.create(serviceEnvironmentConfig
.signalServiceConfiguration()));
245 private ClientZkProfileOperations
getClientZkProfileOperations() {
246 final var clientZkOperations
= getClientZkOperations();
247 return clientZkOperations
.getProfileOperations();
250 public SignalWebSocket
.AuthenticatedWebSocket
getAuthenticatedSignalWebSocket() {
251 return getOrCreate(() -> authenticatedSignalWebSocket
, () -> {
252 final var timer
= new UptimeSleepTimer();
253 final var healthMonitor
= new SignalWebSocketHealthMonitor(timer
);
255 authenticatedSignalWebSocket
= new SignalWebSocket
.AuthenticatedWebSocket(() -> new OkHttpWebSocketConnection(
257 serviceEnvironmentConfig
.signalServiceConfiguration(),
258 Optional
.of(credentialsProvider
),
261 allowStories
), timer
, TimeUnit
.SECONDS
.toMillis(10));
262 healthMonitor
.monitor(authenticatedSignalWebSocket
);
266 public SignalWebSocket
.UnauthenticatedWebSocket
getUnauthenticatedSignalWebSocket() {
267 return getOrCreate(() -> unauthenticatedSignalWebSocket
, () -> {
268 final var timer
= new UptimeSleepTimer();
269 final var healthMonitor
= new SignalWebSocketHealthMonitor(timer
);
271 unauthenticatedSignalWebSocket
= new SignalWebSocket
.UnauthenticatedWebSocket(() -> new OkHttpWebSocketConnection(
273 serviceEnvironmentConfig
.signalServiceConfiguration(),
277 allowStories
), timer
, TimeUnit
.SECONDS
.toMillis(10));
278 healthMonitor
.monitor(unauthenticatedSignalWebSocket
);
282 public SignalServiceMessageReceiver
getMessageReceiver() {
283 return getOrCreate(() -> messageReceiver
,
284 () -> messageReceiver
= new SignalServiceMessageReceiver(getPushServiceSocket()));
287 public SignalServiceMessageSender
getMessageSender() {
288 return getOrCreate(() -> messageSender
,
289 () -> messageSender
= new SignalServiceMessageSender(getPushServiceSocket(),
292 getAuthenticatedSignalWebSocket(),
293 getUnauthenticatedSignalWebSocket(),
296 ServiceConfig
.MAX_ENVELOPE_SIZE
));
299 public List
<SecureValueRecovery
> getSecureValueRecovery() {
300 return getOrCreate(() -> secureValueRecovery
,
301 () -> secureValueRecovery
= serviceEnvironmentConfig
.svr2Mrenclaves()
303 .map(mr
-> (SecureValueRecovery
) getAccountManager().getSecureValueRecoveryV2(mr
))
307 public ProfileService
getProfileService() {
308 return getOrCreate(() -> profileService
,
309 () -> profileService
= new ProfileService(getClientZkProfileOperations(),
310 getMessageReceiver(),
311 getAuthenticatedSignalWebSocket(),
312 getUnauthenticatedSignalWebSocket()));
315 public SignalServiceCipher
getCipher(ServiceIdType serviceIdType
) {
316 final var certificateValidator
= new CertificateValidator(serviceEnvironmentConfig
.unidentifiedSenderTrustRoot());
317 final var address
= new SignalServiceAddress(credentialsProvider
.getAci(), credentialsProvider
.getE164());
318 final var deviceId
= credentialsProvider
.getDeviceId();
319 return new SignalServiceCipher(address
,
321 serviceIdType
== ServiceIdType
.ACI ? dataStore
.aci() : dataStore
.pni(),
323 certificateValidator
);
326 private <T
> T
getOrCreate(Supplier
<T
> supplier
, Callable creator
) {
327 var value
= supplier
.get();
332 synchronized (LOCK
) {
333 value
= supplier
.get();
338 return supplier
.get();
342 private interface Callable
{