1 package org
.asamk
.signal
.manager
.storage
.senderKeys
;
3 import org
.asamk
.signal
.manager
.api
.Pair
;
4 import org
.asamk
.signal
.manager
.helper
.RecipientAddressResolver
;
5 import org
.asamk
.signal
.manager
.storage
.recipients
.RecipientId
;
6 import org
.asamk
.signal
.manager
.storage
.recipients
.RecipientResolver
;
7 import org
.signal
.libsignal
.protocol
.InvalidMessageException
;
8 import org
.signal
.libsignal
.protocol
.groups
.state
.SenderKeyRecord
;
9 import org
.slf4j
.Logger
;
10 import org
.slf4j
.LoggerFactory
;
13 import java
.io
.FileInputStream
;
14 import java
.io
.IOException
;
15 import java
.nio
.file
.Files
;
16 import java
.util
.Arrays
;
17 import java
.util
.List
;
18 import java
.util
.Objects
;
19 import java
.util
.UUID
;
20 import java
.util
.regex
.Matcher
;
21 import java
.util
.regex
.Pattern
;
23 public class LegacySenderKeyRecordStore
{
25 private final static Logger logger
= LoggerFactory
.getLogger(LegacySenderKeyRecordStore
.class);
27 public static void migrate(
28 final File senderKeysPath
,
29 final RecipientResolver resolver
,
30 final RecipientAddressResolver addressResolver
,
31 final SenderKeyStore senderKeyStore
33 final var files
= senderKeysPath
.listFiles();
38 final var senderKeys
= parseFileNames(files
, resolver
).stream().map(key
-> {
39 final var record = loadSenderKeyLocked(key
, senderKeysPath
);
40 final var serviceId
= addressResolver
.resolveRecipientAddress(key
.recipientId
).serviceId();
41 if (record == null || serviceId
.isEmpty()) {
44 return new Pair
<>(new SenderKeyRecordStore
.Key(serviceId
.get().toString(),
46 key
.distributionId
), record);
47 }).filter(Objects
::nonNull
).toList();
49 senderKeyStore
.addLegacySenderKeys(senderKeys
);
50 deleteAllSenderKeys(senderKeysPath
);
53 private static void deleteAllSenderKeys(File senderKeysPath
) {
54 final var files
= senderKeysPath
.listFiles();
59 for (var file
: files
) {
61 Files
.delete(file
.toPath());
62 } catch (IOException e
) {
63 logger
.error("Failed to delete sender key file {}: {}", file
, e
.getMessage());
67 Files
.delete(senderKeysPath
.toPath());
68 } catch (IOException e
) {
69 logger
.error("Failed to delete sender keys directory {}: {}", senderKeysPath
, e
.getMessage());
73 final static Pattern senderKeyFileNamePattern
= Pattern
.compile("(\\d+)_(\\d+)_([\\da-z\\-]+)");
75 private static List
<Key
> parseFileNames(final File
[] files
, final RecipientResolver resolver
) {
76 return Arrays
.stream(files
)
77 .map(f
-> senderKeyFileNamePattern
.matcher(f
.getName()))
78 .filter(Matcher
::matches
)
80 final var recipientId
= resolver
.resolveRecipient(Long
.parseLong(matcher
.group(1)));
81 if (recipientId
== null) {
84 return new Key(recipientId
, Integer
.parseInt(matcher
.group(2)), UUID
.fromString(matcher
.group(3)));
86 .filter(Objects
::nonNull
)
90 private static File
getSenderKeyFile(Key key
, final File senderKeysPath
) {
91 return new File(senderKeysPath
,
92 key
.recipientId().id() + "_" + key
.deviceId() + "_" + key
.distributionId().toString());
95 private static SenderKeyRecord
loadSenderKeyLocked(final Key key
, final File senderKeysPath
) {
96 final var file
= getSenderKeyFile(key
, senderKeysPath
);
100 try (var inputStream
= new FileInputStream(file
)) {
101 return new SenderKeyRecord(inputStream
.readAllBytes());
102 } catch (IOException
| InvalidMessageException e
) {
103 logger
.warn("Failed to load sender key, resetting sender key: {}", e
.getMessage());
108 record Key(RecipientId recipientId
, int deviceId
, UUID distributionId
) {}