]> nmode's Git Repositories - signal-cli/blob - lib/src/main/java/org/asamk/signal/manager/api/MessageEnvelope.java
Improve exit code for message sending
[signal-cli] / lib / src / main / java / org / asamk / signal / manager / api / MessageEnvelope.java
1 package org.asamk.signal.manager.api;
2
3 import org.asamk.signal.manager.groups.GroupId;
4 import org.asamk.signal.manager.groups.GroupUtils;
5 import org.asamk.signal.manager.helper.RecipientAddressResolver;
6 import org.asamk.signal.manager.storage.recipients.RecipientAddress;
7 import org.asamk.signal.manager.storage.recipients.RecipientResolver;
8 import org.whispersystems.signalservice.api.messages.SignalServiceAttachment;
9 import org.whispersystems.signalservice.api.messages.SignalServiceAttachmentRemoteId;
10 import org.whispersystems.signalservice.api.messages.SignalServiceContent;
11 import org.whispersystems.signalservice.api.messages.SignalServiceDataMessage;
12 import org.whispersystems.signalservice.api.messages.SignalServiceEnvelope;
13 import org.whispersystems.signalservice.api.messages.SignalServiceGroup;
14 import org.whispersystems.signalservice.api.messages.SignalServiceGroupContext;
15 import org.whispersystems.signalservice.api.messages.SignalServiceReceiptMessage;
16 import org.whispersystems.signalservice.api.messages.SignalServiceTypingMessage;
17 import org.whispersystems.signalservice.api.messages.calls.AnswerMessage;
18 import org.whispersystems.signalservice.api.messages.calls.BusyMessage;
19 import org.whispersystems.signalservice.api.messages.calls.HangupMessage;
20 import org.whispersystems.signalservice.api.messages.calls.IceUpdateMessage;
21 import org.whispersystems.signalservice.api.messages.calls.OfferMessage;
22 import org.whispersystems.signalservice.api.messages.calls.OpaqueMessage;
23 import org.whispersystems.signalservice.api.messages.calls.SignalServiceCallMessage;
24 import org.whispersystems.signalservice.api.messages.multidevice.BlockedListMessage;
25 import org.whispersystems.signalservice.api.messages.multidevice.ContactsMessage;
26 import org.whispersystems.signalservice.api.messages.multidevice.MessageRequestResponseMessage;
27 import org.whispersystems.signalservice.api.messages.multidevice.ReadMessage;
28 import org.whispersystems.signalservice.api.messages.multidevice.SentTranscriptMessage;
29 import org.whispersystems.signalservice.api.messages.multidevice.SignalServiceSyncMessage;
30 import org.whispersystems.signalservice.api.messages.multidevice.ViewOnceOpenMessage;
31 import org.whispersystems.signalservice.api.messages.multidevice.ViewedMessage;
32
33 import java.io.File;
34 import java.util.List;
35 import java.util.Optional;
36 import java.util.Set;
37 import java.util.stream.Collectors;
38
39 public record MessageEnvelope(
40 Optional<RecipientAddress> sourceAddress,
41 int sourceDevice,
42 long timestamp,
43 long serverReceivedTimestamp,
44 long serverDeliveredTimestamp,
45 boolean isUnidentifiedSender,
46 Optional<Receipt> receipt,
47 Optional<Typing> typing,
48 Optional<Data> data,
49 Optional<Sync> sync,
50 Optional<Call> call
51 ) {
52
53 public record Receipt(long when, Type type, List<Long> timestamps) {
54
55 static Receipt from(final SignalServiceReceiptMessage receiptMessage) {
56 return new Receipt(receiptMessage.getWhen(),
57 Type.from(receiptMessage.getType()),
58 receiptMessage.getTimestamps());
59 }
60
61 public enum Type {
62 DELIVERY,
63 READ,
64 VIEWED,
65 UNKNOWN;
66
67 static Type from(SignalServiceReceiptMessage.Type type) {
68 return switch (type) {
69 case DELIVERY -> DELIVERY;
70 case READ -> READ;
71 case VIEWED -> VIEWED;
72 case UNKNOWN -> UNKNOWN;
73 };
74 }
75 }
76 }
77
78 public record Typing(long timestamp, Type type, Optional<GroupId> groupId) {
79
80 public static Typing from(final SignalServiceTypingMessage typingMessage) {
81 return new Typing(typingMessage.getTimestamp(),
82 typingMessage.isTypingStarted() ? Type.STARTED : Type.STOPPED,
83 Optional.ofNullable(typingMessage.getGroupId().transform(GroupId::unknownVersion).orNull()));
84 }
85
86 public enum Type {
87 STARTED,
88 STOPPED,
89 }
90 }
91
92 public record Data(
93 long timestamp,
94 Optional<GroupContext> groupContext,
95 Optional<GroupCallUpdate> groupCallUpdate,
96 Optional<String> body,
97 int expiresInSeconds,
98 boolean isExpirationUpdate,
99 boolean isViewOnce,
100 boolean isEndSession,
101 boolean hasProfileKey,
102 Optional<Reaction> reaction,
103 Optional<Quote> quote,
104 Optional<Payment> payment,
105 List<Attachment> attachments,
106 Optional<Long> remoteDeleteId,
107 Optional<Sticker> sticker,
108 List<SharedContact> sharedContacts,
109 List<Mention> mentions,
110 List<Preview> previews
111 ) {
112
113 static Data from(
114 final SignalServiceDataMessage dataMessage,
115 RecipientResolver recipientResolver,
116 RecipientAddressResolver addressResolver,
117 final AttachmentFileProvider fileProvider
118 ) {
119 return new Data(dataMessage.getTimestamp(),
120 Optional.ofNullable(dataMessage.getGroupContext().transform(GroupContext::from).orNull()),
121 Optional.ofNullable(dataMessage.getGroupCallUpdate().transform(GroupCallUpdate::from).orNull()),
122 Optional.ofNullable(dataMessage.getBody().orNull()),
123 dataMessage.getExpiresInSeconds(),
124 dataMessage.isExpirationUpdate(),
125 dataMessage.isViewOnce(),
126 dataMessage.isEndSession(),
127 dataMessage.getProfileKey().isPresent(),
128 Optional.ofNullable(dataMessage.getReaction()
129 .transform(r -> Reaction.from(r, recipientResolver, addressResolver))
130 .orNull()),
131 Optional.ofNullable(dataMessage.getQuote()
132 .transform(q -> Quote.from(q, recipientResolver, addressResolver, fileProvider))
133 .orNull()),
134 Optional.ofNullable(dataMessage.getPayment()
135 .transform(p -> p.getPaymentNotification().isPresent() ? Payment.from(p) : null)
136 .orNull()),
137 dataMessage.getAttachments()
138 .transform(a -> a.stream().map(as -> Attachment.from(as, fileProvider)).toList())
139 .or(List.of()),
140 Optional.ofNullable(dataMessage.getRemoteDelete()
141 .transform(SignalServiceDataMessage.RemoteDelete::getTargetSentTimestamp)
142 .orNull()),
143 Optional.ofNullable(dataMessage.getSticker().transform(Sticker::from).orNull()),
144 dataMessage.getSharedContacts()
145 .transform(a -> a.stream()
146 .map(sharedContact -> SharedContact.from(sharedContact, fileProvider))
147 .toList())
148 .or(List.of()),
149 dataMessage.getMentions()
150 .transform(a -> a.stream()
151 .map(m -> Mention.from(m, recipientResolver, addressResolver))
152 .toList())
153 .or(List.of()),
154 dataMessage.getPreviews()
155 .transform(a -> a.stream().map(preview -> Preview.from(preview, fileProvider)).toList())
156 .or(List.of()));
157 }
158
159 public record GroupContext(GroupId groupId, boolean isGroupUpdate, int revision) {
160
161 static GroupContext from(SignalServiceGroupContext groupContext) {
162 if (groupContext.getGroupV1().isPresent()) {
163 return new GroupContext(GroupId.v1(groupContext.getGroupV1().get().getGroupId()),
164 groupContext.getGroupV1Type() == SignalServiceGroup.Type.UPDATE,
165 0);
166 } else if (groupContext.getGroupV2().isPresent()) {
167 final var groupV2 = groupContext.getGroupV2().get();
168 return new GroupContext(GroupUtils.getGroupIdV2(groupV2.getMasterKey()),
169 groupV2.hasSignedGroupChange(),
170 groupV2.getRevision());
171 } else {
172 throw new RuntimeException("Invalid group context state");
173 }
174 }
175 }
176
177 public record GroupCallUpdate(String eraId) {
178
179 static GroupCallUpdate from(SignalServiceDataMessage.GroupCallUpdate groupCallUpdate) {
180 return new GroupCallUpdate(groupCallUpdate.getEraId());
181 }
182 }
183
184 public record Reaction(
185 long targetSentTimestamp, RecipientAddress targetAuthor, String emoji, boolean isRemove
186 ) {
187
188 static Reaction from(
189 SignalServiceDataMessage.Reaction reaction,
190 RecipientResolver recipientResolver,
191 RecipientAddressResolver addressResolver
192 ) {
193 return new Reaction(reaction.getTargetSentTimestamp(),
194 addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(reaction.getTargetAuthor())),
195 reaction.getEmoji(),
196 reaction.isRemove());
197 }
198 }
199
200 public record Quote(
201 long id,
202 RecipientAddress author,
203 Optional<String> text,
204 List<Mention> mentions,
205 List<Attachment> attachments
206 ) {
207
208 static Quote from(
209 SignalServiceDataMessage.Quote quote,
210 RecipientResolver recipientResolver,
211 RecipientAddressResolver addressResolver,
212 final AttachmentFileProvider fileProvider
213 ) {
214 return new Quote(quote.getId(),
215 addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(quote.getAuthor())),
216 Optional.ofNullable(quote.getText()),
217 quote.getMentions() == null
218 ? List.of()
219 : quote.getMentions()
220 .stream()
221 .map(m -> Mention.from(m, recipientResolver, addressResolver))
222 .toList(),
223 quote.getAttachments() == null
224 ? List.of()
225 : quote.getAttachments().stream().map(a -> Attachment.from(a, fileProvider)).toList());
226 }
227 }
228
229 public record Payment(String note, byte[] receipt) {
230
231 static Payment from(SignalServiceDataMessage.Payment payment) {
232 return new Payment(payment.getPaymentNotification().get().getNote(),
233 payment.getPaymentNotification().get().getReceipt());
234 }
235 }
236
237 public record Mention(RecipientAddress recipient, int start, int length) {
238
239 static Mention from(
240 SignalServiceDataMessage.Mention mention,
241 RecipientResolver recipientResolver,
242 RecipientAddressResolver addressResolver
243 ) {
244 return new Mention(addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(mention.getAci())),
245 mention.getStart(),
246 mention.getLength());
247 }
248 }
249
250 public record Attachment(
251 Optional<String> id,
252 Optional<File> file,
253 Optional<String> fileName,
254 String contentType,
255 Optional<Long> uploadTimestamp,
256 Optional<Long> size,
257 Optional<byte[]> preview,
258 Optional<Attachment> thumbnail,
259 Optional<String> caption,
260 Optional<Integer> width,
261 Optional<Integer> height,
262 boolean isVoiceNote,
263 boolean isGif,
264 boolean isBorderless
265 ) {
266
267 static Attachment from(SignalServiceAttachment attachment, AttachmentFileProvider fileProvider) {
268 if (attachment.isPointer()) {
269 final var a = attachment.asPointer();
270 return new Attachment(Optional.of(a.getRemoteId().toString()),
271 Optional.of(fileProvider.getFile(a.getRemoteId())),
272 Optional.ofNullable(a.getFileName().orNull()),
273 a.getContentType(),
274 a.getUploadTimestamp() == 0 ? Optional.empty() : Optional.of(a.getUploadTimestamp()),
275 Optional.ofNullable(a.getSize().transform(Integer::longValue).orNull()),
276 Optional.ofNullable(a.getPreview().orNull()),
277 Optional.empty(),
278 Optional.ofNullable(a.getCaption().orNull()),
279 a.getWidth() == 0 ? Optional.empty() : Optional.of(a.getWidth()),
280 a.getHeight() == 0 ? Optional.empty() : Optional.of(a.getHeight()),
281 a.getVoiceNote(),
282 a.isGif(),
283 a.isBorderless());
284 } else {
285 final var a = attachment.asStream();
286 return new Attachment(Optional.empty(),
287 Optional.empty(),
288 Optional.ofNullable(a.getFileName().orNull()),
289 a.getContentType(),
290 a.getUploadTimestamp() == 0 ? Optional.empty() : Optional.of(a.getUploadTimestamp()),
291 Optional.of(a.getLength()),
292 Optional.ofNullable(a.getPreview().orNull()),
293 Optional.empty(),
294 Optional.ofNullable(a.getCaption().orNull()),
295 a.getWidth() == 0 ? Optional.empty() : Optional.of(a.getWidth()),
296 a.getHeight() == 0 ? Optional.empty() : Optional.of(a.getHeight()),
297 a.getVoiceNote(),
298 a.isGif(),
299 a.isBorderless());
300 }
301 }
302
303 static Attachment from(
304 SignalServiceDataMessage.Quote.QuotedAttachment a, final AttachmentFileProvider fileProvider
305 ) {
306 return new Attachment(Optional.empty(),
307 Optional.empty(),
308 Optional.ofNullable(a.getFileName()),
309 a.getContentType(),
310 Optional.empty(),
311 Optional.empty(),
312 Optional.empty(),
313 a.getThumbnail() == null
314 ? Optional.empty()
315 : Optional.of(Attachment.from(a.getThumbnail(), fileProvider)),
316 Optional.empty(),
317 Optional.empty(),
318 Optional.empty(),
319 false,
320 false,
321 false);
322 }
323 }
324
325 public record Sticker(byte[] packId, byte[] packKey, int stickerId) {
326
327 static Sticker from(SignalServiceDataMessage.Sticker sticker) {
328 return new Sticker(sticker.getPackId(), sticker.getPackKey(), sticker.getStickerId());
329 }
330 }
331
332 public record SharedContact(
333 Name name,
334 Optional<Avatar> avatar,
335 List<Phone> phone,
336 List<Email> email,
337 List<Address> address,
338 Optional<String> organization
339 ) {
340
341 static SharedContact from(
342 org.whispersystems.signalservice.api.messages.shared.SharedContact sharedContact,
343 final AttachmentFileProvider fileProvider
344 ) {
345 return new SharedContact(Name.from(sharedContact.getName()),
346 Optional.ofNullable(sharedContact.getAvatar()
347 .transform(avatar1 -> Avatar.from(avatar1, fileProvider))
348 .orNull()),
349 sharedContact.getPhone().transform(p -> p.stream().map(Phone::from).toList()).or(List.of()),
350 sharedContact.getEmail().transform(p -> p.stream().map(Email::from).toList()).or(List.of()),
351 sharedContact.getAddress().transform(p -> p.stream().map(Address::from).toList()).or(List.of()),
352 Optional.ofNullable(sharedContact.getOrganization().orNull()));
353 }
354
355 public record Name(
356 Optional<String> display,
357 Optional<String> given,
358 Optional<String> family,
359 Optional<String> prefix,
360 Optional<String> suffix,
361 Optional<String> middle
362 ) {
363
364 static Name from(org.whispersystems.signalservice.api.messages.shared.SharedContact.Name name) {
365 return new Name(Optional.ofNullable(name.getDisplay().orNull()),
366 Optional.ofNullable(name.getGiven().orNull()),
367 Optional.ofNullable(name.getFamily().orNull()),
368 Optional.ofNullable(name.getPrefix().orNull()),
369 Optional.ofNullable(name.getSuffix().orNull()),
370 Optional.ofNullable(name.getMiddle().orNull()));
371 }
372 }
373
374 public record Avatar(Attachment attachment, boolean isProfile) {
375
376 static Avatar from(
377 org.whispersystems.signalservice.api.messages.shared.SharedContact.Avatar avatar,
378 final AttachmentFileProvider fileProvider
379 ) {
380 return new Avatar(Attachment.from(avatar.getAttachment(), fileProvider), avatar.isProfile());
381 }
382 }
383
384 public record Phone(
385 String value, Type type, Optional<String> label
386 ) {
387
388 static Phone from(org.whispersystems.signalservice.api.messages.shared.SharedContact.Phone phone) {
389 return new Phone(phone.getValue(),
390 Type.from(phone.getType()),
391 Optional.ofNullable(phone.getLabel().orNull()));
392 }
393
394 public enum Type {
395 HOME,
396 WORK,
397 MOBILE,
398 CUSTOM;
399
400 static Type from(org.whispersystems.signalservice.api.messages.shared.SharedContact.Phone.Type type) {
401 return switch (type) {
402 case HOME -> HOME;
403 case WORK -> WORK;
404 case MOBILE -> MOBILE;
405 case CUSTOM -> CUSTOM;
406 };
407 }
408 }
409 }
410
411 public record Email(
412 String value, Type type, Optional<String> label
413 ) {
414
415 static Email from(org.whispersystems.signalservice.api.messages.shared.SharedContact.Email email) {
416 return new Email(email.getValue(),
417 Type.from(email.getType()),
418 Optional.ofNullable(email.getLabel().orNull()));
419 }
420
421 public enum Type {
422 HOME,
423 WORK,
424 MOBILE,
425 CUSTOM;
426
427 static Type from(org.whispersystems.signalservice.api.messages.shared.SharedContact.Email.Type type) {
428 return switch (type) {
429 case HOME -> HOME;
430 case WORK -> WORK;
431 case MOBILE -> MOBILE;
432 case CUSTOM -> CUSTOM;
433 };
434 }
435 }
436 }
437
438 public record Address(
439 Type type,
440 Optional<String> label,
441 Optional<String> street,
442 Optional<String> pobox,
443 Optional<String> neighborhood,
444 Optional<String> city,
445 Optional<String> region,
446 Optional<String> postcode,
447 Optional<String> country
448 ) {
449
450 static Address from(org.whispersystems.signalservice.api.messages.shared.SharedContact.PostalAddress address) {
451 return new Address(Address.Type.from(address.getType()),
452 Optional.ofNullable(address.getLabel().orNull()),
453 Optional.ofNullable(address.getLabel().orNull()),
454 Optional.ofNullable(address.getLabel().orNull()),
455 Optional.ofNullable(address.getLabel().orNull()),
456 Optional.ofNullable(address.getLabel().orNull()),
457 Optional.ofNullable(address.getLabel().orNull()),
458 Optional.ofNullable(address.getLabel().orNull()),
459 Optional.ofNullable(address.getLabel().orNull()));
460 }
461
462 public enum Type {
463 HOME,
464 WORK,
465 CUSTOM;
466
467 static Type from(org.whispersystems.signalservice.api.messages.shared.SharedContact.PostalAddress.Type type) {
468 return switch (type) {
469 case HOME -> HOME;
470 case WORK -> WORK;
471 case CUSTOM -> CUSTOM;
472 };
473 }
474 }
475 }
476 }
477
478 public record Preview(String title, String description, long date, String url, Optional<Attachment> image) {
479
480 static Preview from(
481 SignalServiceDataMessage.Preview preview, final AttachmentFileProvider fileProvider
482 ) {
483 return new Preview(preview.getTitle(),
484 preview.getDescription(),
485 preview.getDate(),
486 preview.getUrl(),
487 Optional.ofNullable(preview.getImage()
488 .transform(as -> Attachment.from(as, fileProvider))
489 .orNull()));
490 }
491 }
492 }
493
494 public record Sync(
495 Optional<Sent> sent,
496 Optional<Blocked> blocked,
497 List<Read> read,
498 List<Viewed> viewed,
499 Optional<ViewOnceOpen> viewOnceOpen,
500 Optional<Contacts> contacts,
501 Optional<Groups> groups,
502 Optional<MessageRequestResponse> messageRequestResponse
503 ) {
504
505 public static Sync from(
506 final SignalServiceSyncMessage syncMessage,
507 RecipientResolver recipientResolver,
508 RecipientAddressResolver addressResolver,
509 final AttachmentFileProvider fileProvider
510 ) {
511 return new Sync(Optional.ofNullable(syncMessage.getSent()
512 .transform(s -> Sent.from(s, recipientResolver, addressResolver, fileProvider))
513 .orNull()),
514 Optional.ofNullable(syncMessage.getBlockedList()
515 .transform(b -> Blocked.from(b, recipientResolver, addressResolver))
516 .orNull()),
517 syncMessage.getRead()
518 .transform(r -> r.stream()
519 .map(rm -> Read.from(rm, recipientResolver, addressResolver))
520 .toList())
521 .or(List.of()),
522 syncMessage.getViewed()
523 .transform(r -> r.stream()
524 .map(rm -> Viewed.from(rm, recipientResolver, addressResolver))
525 .toList())
526 .or(List.of()),
527 Optional.ofNullable(syncMessage.getViewOnceOpen()
528 .transform(rm -> ViewOnceOpen.from(rm, recipientResolver, addressResolver))
529 .orNull()),
530 Optional.ofNullable(syncMessage.getContacts().transform(Contacts::from).orNull()),
531 Optional.ofNullable(syncMessage.getGroups().transform(Groups::from).orNull()),
532 Optional.ofNullable(syncMessage.getMessageRequestResponse()
533 .transform(m -> MessageRequestResponse.from(m, recipientResolver, addressResolver))
534 .orNull()));
535 }
536
537 public record Sent(
538 long timestamp,
539 long expirationStartTimestamp,
540 Optional<RecipientAddress> destination,
541 Set<RecipientAddress> recipients,
542 Data message
543 ) {
544
545 static Sent from(
546 SentTranscriptMessage sentMessage,
547 RecipientResolver recipientResolver,
548 RecipientAddressResolver addressResolver,
549 final AttachmentFileProvider fileProvider
550 ) {
551 return new Sent(sentMessage.getTimestamp(),
552 sentMessage.getExpirationStartTimestamp(),
553 Optional.ofNullable(sentMessage.getDestination()
554 .transform(d -> addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(
555 d)))
556 .orNull()),
557 sentMessage.getRecipients()
558 .stream()
559 .map(d -> addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(d)))
560 .collect(Collectors.toSet()),
561 Data.from(sentMessage.getMessage(), recipientResolver, addressResolver, fileProvider));
562 }
563 }
564
565 public record Blocked(List<RecipientAddress> recipients, List<GroupId> groupIds) {
566
567 static Blocked from(
568 BlockedListMessage blockedListMessage,
569 RecipientResolver recipientResolver,
570 RecipientAddressResolver addressResolver
571 ) {
572 return new Blocked(blockedListMessage.getAddresses()
573 .stream()
574 .map(d -> addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(d)))
575 .toList(), blockedListMessage.getGroupIds().stream().map(GroupId::unknownVersion).toList());
576 }
577 }
578
579 public record Read(RecipientAddress sender, long timestamp) {
580
581 static Read from(
582 ReadMessage readMessage,
583 RecipientResolver recipientResolver,
584 RecipientAddressResolver addressResolver
585 ) {
586 return new Read(addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(readMessage.getSender())),
587 readMessage.getTimestamp());
588 }
589 }
590
591 public record Viewed(RecipientAddress sender, long timestamp) {
592
593 static Viewed from(
594 ViewedMessage readMessage,
595 RecipientResolver recipientResolver,
596 RecipientAddressResolver addressResolver
597 ) {
598 return new Viewed(addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(readMessage.getSender())),
599 readMessage.getTimestamp());
600 }
601 }
602
603 public record ViewOnceOpen(RecipientAddress sender, long timestamp) {
604
605 static ViewOnceOpen from(
606 ViewOnceOpenMessage readMessage,
607 RecipientResolver recipientResolver,
608 RecipientAddressResolver addressResolver
609 ) {
610 return new ViewOnceOpen(addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(
611 readMessage.getSender())), readMessage.getTimestamp());
612 }
613 }
614
615 public record Contacts(boolean isComplete) {
616
617 static Contacts from(ContactsMessage contactsMessage) {
618 return new Contacts(contactsMessage.isComplete());
619 }
620 }
621
622 public record Groups() {
623
624 static Groups from(SignalServiceAttachment groupsMessage) {
625 return new Groups();
626 }
627 }
628
629 public record MessageRequestResponse(Type type, Optional<GroupId> groupId, Optional<RecipientAddress> person) {
630
631 static MessageRequestResponse from(
632 MessageRequestResponseMessage messageRequestResponse,
633 RecipientResolver recipientResolver,
634 RecipientAddressResolver addressResolver
635 ) {
636 return new MessageRequestResponse(Type.from(messageRequestResponse.getType()),
637 Optional.ofNullable(messageRequestResponse.getGroupId()
638 .transform(GroupId::unknownVersion)
639 .orNull()),
640 Optional.ofNullable(messageRequestResponse.getPerson()
641 .transform(p -> addressResolver.resolveRecipientAddress(recipientResolver.resolveRecipient(
642 p)))
643 .orNull()));
644 }
645
646 public enum Type {
647 UNKNOWN,
648 ACCEPT,
649 DELETE,
650 BLOCK,
651 BLOCK_AND_DELETE,
652 UNBLOCK_AND_ACCEPT;
653
654 static Type from(MessageRequestResponseMessage.Type type) {
655 return switch (type) {
656 case UNKNOWN -> UNKNOWN;
657 case ACCEPT -> ACCEPT;
658 case DELETE -> DELETE;
659 case BLOCK -> BLOCK;
660 case BLOCK_AND_DELETE -> BLOCK_AND_DELETE;
661 case UNBLOCK_AND_ACCEPT -> UNBLOCK_AND_ACCEPT;
662 };
663 }
664 }
665 }
666 }
667
668 public record Call(
669 Optional<Integer> destinationDeviceId,
670 Optional<GroupId> groupId,
671 Optional<Long> timestamp,
672 Optional<Offer> offer,
673 Optional<Answer> answer,
674 Optional<Hangup> hangup,
675 Optional<Busy> busy,
676 List<IceUpdate> iceUpdate,
677 Optional<Opaque> opaque
678 ) {
679
680 public static Call from(final SignalServiceCallMessage callMessage) {
681 return new Call(Optional.ofNullable(callMessage.getDestinationDeviceId().orNull()),
682 Optional.ofNullable(callMessage.getGroupId().transform(GroupId::unknownVersion).orNull()),
683 Optional.ofNullable(callMessage.getTimestamp().orNull()),
684 Optional.ofNullable(callMessage.getOfferMessage().transform(Offer::from).orNull()),
685 Optional.ofNullable(callMessage.getAnswerMessage().transform(Answer::from).orNull()),
686 Optional.ofNullable(callMessage.getHangupMessage().transform(Hangup::from).orNull()),
687 Optional.ofNullable(callMessage.getBusyMessage().transform(Busy::from).orNull()),
688 callMessage.getIceUpdateMessages()
689 .transform(m -> m.stream().map(IceUpdate::from).toList())
690 .or(List.of()),
691 Optional.ofNullable(callMessage.getOpaqueMessage().transform(Opaque::from).orNull()));
692 }
693
694 public record Offer(long id, String sdp, Type type, byte[] opaque) {
695
696 static Offer from(OfferMessage offerMessage) {
697 return new Offer(offerMessage.getId(),
698 offerMessage.getSdp(),
699 Type.from(offerMessage.getType()),
700 offerMessage.getOpaque());
701 }
702
703 public enum Type {
704 AUDIO_CALL,
705 VIDEO_CALL;
706
707 static Type from(OfferMessage.Type type) {
708 return switch (type) {
709 case AUDIO_CALL -> AUDIO_CALL;
710 case VIDEO_CALL -> VIDEO_CALL;
711 };
712 }
713 }
714 }
715
716 public record Answer(long id, String sdp, byte[] opaque) {
717
718 static Answer from(AnswerMessage answerMessage) {
719 return new Answer(answerMessage.getId(), answerMessage.getSdp(), answerMessage.getOpaque());
720 }
721 }
722
723 public record Busy(long id) {
724
725 static Busy from(BusyMessage busyMessage) {
726 return new Busy(busyMessage.getId());
727 }
728 }
729
730 public record Hangup(long id, Type type, int deviceId, boolean isLegacy) {
731
732 static Hangup from(HangupMessage hangupMessage) {
733 return new Hangup(hangupMessage.getId(),
734 Type.from(hangupMessage.getType()),
735 hangupMessage.getDeviceId(),
736 hangupMessage.isLegacy());
737 }
738
739 public enum Type {
740 NORMAL,
741 ACCEPTED,
742 DECLINED,
743 BUSY,
744 NEED_PERMISSION;
745
746 static Type from(HangupMessage.Type type) {
747 return switch (type) {
748 case NORMAL -> NORMAL;
749 case ACCEPTED -> ACCEPTED;
750 case DECLINED -> DECLINED;
751 case BUSY -> BUSY;
752 case NEED_PERMISSION -> NEED_PERMISSION;
753 };
754 }
755 }
756 }
757
758 public record IceUpdate(long id, String sdp, byte[] opaque) {
759
760 static IceUpdate from(IceUpdateMessage iceUpdateMessage) {
761 return new IceUpdate(iceUpdateMessage.getId(), iceUpdateMessage.getSdp(), iceUpdateMessage.getOpaque());
762 }
763 }
764
765 public record Opaque(byte[] opaque, Urgency urgency) {
766
767 static Opaque from(OpaqueMessage opaqueMessage) {
768 return new Opaque(opaqueMessage.getOpaque(), Urgency.from(opaqueMessage.getUrgency()));
769 }
770
771 public enum Urgency {
772 DROPPABLE,
773 HANDLE_IMMEDIATELY;
774
775 static Urgency from(OpaqueMessage.Urgency urgency) {
776 return switch (urgency) {
777 case DROPPABLE -> DROPPABLE;
778 case HANDLE_IMMEDIATELY -> HANDLE_IMMEDIATELY;
779 };
780 }
781 }
782 }
783 }
784
785 public static MessageEnvelope from(
786 SignalServiceEnvelope envelope,
787 SignalServiceContent content,
788 RecipientResolver recipientResolver,
789 RecipientAddressResolver addressResolver,
790 final AttachmentFileProvider fileProvider
791 ) {
792 final var source = !envelope.isUnidentifiedSender() && envelope.hasSourceUuid()
793 ? recipientResolver.resolveRecipient(envelope.getSourceAddress())
794 : envelope.isUnidentifiedSender() && content != null
795 ? recipientResolver.resolveRecipient(content.getSender())
796 : null;
797 final var sourceDevice = envelope.hasSourceDevice()
798 ? envelope.getSourceDevice()
799 : content != null ? content.getSenderDevice() : 0;
800
801 Optional<Receipt> receipt;
802 Optional<Typing> typing;
803 Optional<Data> data;
804 Optional<Sync> sync;
805 Optional<Call> call;
806 if (content != null) {
807 receipt = Optional.ofNullable(content.getReceiptMessage().transform(Receipt::from).orNull());
808 typing = Optional.ofNullable(content.getTypingMessage().transform(Typing::from).orNull());
809 data = Optional.ofNullable(content.getDataMessage()
810 .transform(dataMessage -> Data.from(dataMessage, recipientResolver, addressResolver, fileProvider))
811 .orNull());
812 sync = Optional.ofNullable(content.getSyncMessage()
813 .transform(s -> Sync.from(s, recipientResolver, addressResolver, fileProvider))
814 .orNull());
815 call = Optional.ofNullable(content.getCallMessage().transform(Call::from).orNull());
816 } else {
817 receipt = Optional.empty();
818 typing = Optional.empty();
819 data = Optional.empty();
820 sync = Optional.empty();
821 call = Optional.empty();
822 }
823
824 return new MessageEnvelope(source == null
825 ? Optional.empty()
826 : Optional.of(addressResolver.resolveRecipientAddress(source)),
827 sourceDevice,
828 envelope.getTimestamp(),
829 envelope.getServerReceivedTimestamp(),
830 envelope.getServerDeliveredTimestamp(),
831 envelope.isUnidentifiedSender(),
832 receipt,
833 typing,
834 data,
835 sync,
836 call);
837 }
838
839 public interface AttachmentFileProvider {
840
841 File getFile(SignalServiceAttachmentRemoteId attachmentRemoteId);
842 }
843 }