]> nmode's Git Repositories - signal-cli/blob - lib/src/main/java/org/asamk/signal/manager/MultiAccountManagerImpl.java
74c4c20ac5f57646157f73230a4c64c2399c1ea4
[signal-cli] / lib / src / main / java / org / asamk / signal / manager / MultiAccountManagerImpl.java
1 package org.asamk.signal.manager;
2
3 import org.asamk.signal.manager.config.ServiceEnvironment;
4 import org.slf4j.Logger;
5 import org.slf4j.LoggerFactory;
6
7 import java.io.File;
8 import java.io.IOException;
9 import java.net.URI;
10 import java.util.ArrayList;
11 import java.util.Collection;
12 import java.util.HashMap;
13 import java.util.HashSet;
14 import java.util.List;
15 import java.util.Map;
16 import java.util.Set;
17 import java.util.concurrent.TimeoutException;
18 import java.util.function.Consumer;
19 import java.util.stream.Collectors;
20
21 public class MultiAccountManagerImpl implements MultiAccountManager {
22
23 private final static Logger logger = LoggerFactory.getLogger(MultiAccountManagerImpl.class);
24
25 private final Set<Consumer<Manager>> onManagerAddedHandlers = new HashSet<>();
26 private final Set<Consumer<Manager>> onManagerRemovedHandlers = new HashSet<>();
27 private final Set<Manager> managers = new HashSet<>();
28 private final Map<URI, ProvisioningManager> provisioningManagers = new HashMap<>();
29 private final File dataPath;
30 private final ServiceEnvironment serviceEnvironment;
31 private final String userAgent;
32
33 public MultiAccountManagerImpl(
34 final Collection<Manager> managers,
35 final File dataPath,
36 final ServiceEnvironment serviceEnvironment,
37 final String userAgent
38 ) {
39 this.managers.addAll(managers);
40 managers.forEach(m -> m.addClosedListener(() -> this.removeManager(m)));
41 this.dataPath = dataPath;
42 this.serviceEnvironment = serviceEnvironment;
43 this.userAgent = userAgent;
44 }
45
46 @Override
47 public List<String> getAccountNumbers() {
48 synchronized (managers) {
49 return managers.stream().map(Manager::getSelfNumber).collect(Collectors.toList());
50 }
51 }
52
53 void addManager(final Manager m) {
54 synchronized (managers) {
55 if (managers.contains(m)) {
56 return;
57 }
58 managers.add(m);
59 m.addClosedListener(() -> this.removeManager(m));
60 }
61 synchronized (onManagerAddedHandlers) {
62 for (final var handler : onManagerAddedHandlers) {
63 handler.accept(m);
64 }
65 }
66 }
67
68 @Override
69 public void addOnManagerAddedHandler(final Consumer<Manager> handler) {
70 synchronized (onManagerAddedHandlers) {
71 onManagerAddedHandlers.add(handler);
72 }
73 }
74
75 void removeManager(final Manager m) {
76 synchronized (managers) {
77 if (!managers.remove(m)) {
78 return;
79 }
80 }
81 synchronized (onManagerRemovedHandlers) {
82 for (final var handler : onManagerRemovedHandlers) {
83 handler.accept(m);
84 }
85 }
86 }
87
88 @Override
89 public void addOnManagerRemovedHandler(final Consumer<Manager> handler) {
90 synchronized (onManagerRemovedHandlers) {
91 onManagerRemovedHandlers.add(handler);
92 }
93 }
94
95 @Override
96 public Manager getManager(final String account) {
97 synchronized (managers) {
98 return managers.stream().filter(m -> m.getSelfNumber().equals(account)).findFirst().orElse(null);
99 }
100 }
101
102 @Override
103 public URI getNewProvisioningDeviceLinkUri() throws TimeoutException, IOException {
104 final var provisioningManager = getNewProvisioningManager();
105 final var deviceLinkUri = provisioningManager.getDeviceLinkUri();
106 provisioningManagers.put(deviceLinkUri, provisioningManager);
107 return deviceLinkUri;
108 }
109
110 @Override
111 public ProvisioningManager getProvisioningManagerFor(final URI deviceLinkUri) {
112 return provisioningManagers.remove(deviceLinkUri);
113 }
114
115 @Override
116 public ProvisioningManager getNewProvisioningManager() {
117 return ProvisioningManager.init(dataPath, serviceEnvironment, userAgent, this::addManager);
118 }
119
120 @Override
121 public RegistrationManager getNewRegistrationManager(String account) throws IOException {
122 return RegistrationManager.init(account, dataPath, serviceEnvironment, userAgent, this::addManager);
123 }
124
125 @Override
126 public void close() {
127 synchronized (managers) {
128 for (var m : new ArrayList<>(managers)) {
129 try {
130 m.close();
131 } catch (IOException e) {
132 logger.warn("Cleanup failed", e);
133 }
134 }
135 managers.clear();
136 }
137 }
138 }