1 package org
.asamk
.signal
.manager
.helper
;
3 import org
.asamk
.signal
.manager
.api
.AttachmentInvalidException
;
4 import org
.asamk
.signal
.manager
.config
.ServiceConfig
;
5 import org
.asamk
.signal
.manager
.internal
.SignalDependencies
;
6 import org
.asamk
.signal
.manager
.storage
.AttachmentStore
;
7 import org
.asamk
.signal
.manager
.util
.AttachmentUtils
;
8 import org
.asamk
.signal
.manager
.util
.IOUtils
;
9 import org
.signal
.libsignal
.protocol
.InvalidMessageException
;
10 import org
.slf4j
.Logger
;
11 import org
.slf4j
.LoggerFactory
;
12 import org
.whispersystems
.signalservice
.api
.messages
.SignalServiceAttachment
;
13 import org
.whispersystems
.signalservice
.api
.messages
.SignalServiceAttachmentPointer
;
14 import org
.whispersystems
.signalservice
.api
.messages
.SignalServiceAttachmentStream
;
15 import org
.whispersystems
.signalservice
.api
.push
.exceptions
.MissingConfigurationException
;
16 import org
.whispersystems
.signalservice
.api
.util
.StreamDetails
;
19 import java
.io
.IOException
;
20 import java
.io
.InputStream
;
21 import java
.io
.OutputStream
;
22 import java
.nio
.file
.Files
;
23 import java
.util
.ArrayList
;
24 import java
.util
.List
;
26 public class AttachmentHelper
{
28 private static final Logger logger
= LoggerFactory
.getLogger(AttachmentHelper
.class);
30 private final SignalDependencies dependencies
;
31 private final AttachmentStore attachmentStore
;
33 public AttachmentHelper(final Context context
) {
34 this.dependencies
= context
.getDependencies();
35 this.attachmentStore
= context
.getAttachmentStore();
38 public File
getAttachmentFile(SignalServiceAttachmentPointer pointer
) {
39 return attachmentStore
.getAttachmentFile(pointer
);
42 public StreamDetails
retrieveAttachment(final String id
) throws IOException
{
43 return attachmentStore
.retrieveAttachment(id
);
46 public List
<SignalServiceAttachment
> uploadAttachments(final List
<String
> attachments
) throws AttachmentInvalidException
, IOException
{
47 var attachmentStreams
= createAttachmentStreams(attachments
);
49 // Upload attachments here, so we only upload once even for multiple recipients
50 var attachmentPointers
= new ArrayList
<SignalServiceAttachment
>(attachmentStreams
.size());
51 for (var attachmentStream
: attachmentStreams
) {
52 attachmentPointers
.add(uploadAttachment(attachmentStream
));
54 return attachmentPointers
;
57 private List
<SignalServiceAttachmentStream
> createAttachmentStreams(List
<String
> attachments
) throws AttachmentInvalidException
, IOException
{
58 if (attachments
== null) {
61 final var signalServiceAttachments
= new ArrayList
<SignalServiceAttachmentStream
>(attachments
.size());
62 for (var attachment
: attachments
) {
63 final var uploadSpec
= dependencies
.getMessageSender().getResumableUploadSpec().toProto();
64 signalServiceAttachments
.add(AttachmentUtils
.createAttachmentStream(attachment
, uploadSpec
));
66 return signalServiceAttachments
;
69 public SignalServiceAttachmentPointer
uploadAttachment(String attachment
) throws IOException
, AttachmentInvalidException
{
70 final var uploadSpec
= dependencies
.getMessageSender().getResumableUploadSpec().toProto();
71 var attachmentStream
= AttachmentUtils
.createAttachmentStream(attachment
, uploadSpec
);
72 return uploadAttachment(attachmentStream
);
75 public SignalServiceAttachmentPointer
uploadAttachment(SignalServiceAttachmentStream attachment
) throws IOException
{
76 var messageSender
= dependencies
.getMessageSender();
77 return messageSender
.uploadAttachment(attachment
);
80 public void downloadAttachment(final SignalServiceAttachment attachment
) {
81 if (!attachment
.isPointer()) {
82 logger
.warn("Invalid state, can't store an attachment stream.");
85 var pointer
= attachment
.asPointer();
86 if (pointer
.getPreview().isPresent()) {
87 final var preview
= pointer
.getPreview().get();
89 attachmentStore
.storeAttachmentPreview(pointer
,
90 outputStream
-> outputStream
.write(preview
, 0, preview
.length
));
91 } catch (IOException e
) {
92 logger
.warn("Failed to download attachment preview, ignoring: {}", e
.getMessage());
97 attachmentStore
.storeAttachment(pointer
, outputStream
-> this.retrieveAttachment(pointer
, outputStream
));
98 } catch (IOException e
) {
99 logger
.warn("Failed to download attachment ({}), ignoring: {}", pointer
.getRemoteId(), e
.getMessage());
103 void retrieveAttachment(SignalServiceAttachment attachment
, OutputStream outputStream
) throws IOException
{
104 retrieveAttachment(attachment
, input
-> IOUtils
.copyStream(input
, outputStream
));
107 public void retrieveAttachment(
108 SignalServiceAttachment attachment
, AttachmentHandler consumer
109 ) throws IOException
{
110 if (attachment
.isStream()) {
111 var input
= attachment
.asStream().getInputStream();
112 // don't close input stream here, it might be reused later (e.g. with contact sync messages ...)
113 consumer
.handle(input
);
117 final var pointer
= attachment
.asPointer();
118 logger
.debug("Retrieving attachment {} with size {}", pointer
.getRemoteId(), pointer
.getSize());
119 var tmpFile
= IOUtils
.createTempFile();
120 try (var input
= retrieveAttachmentAsStream(pointer
, tmpFile
)) {
121 consumer
.handle(input
);
124 Files
.delete(tmpFile
.toPath());
125 } catch (IOException e
) {
126 logger
.warn("Failed to delete received attachment temp file “{}”, ignoring: {}",
133 private InputStream
retrieveAttachmentAsStream(
134 SignalServiceAttachmentPointer pointer
, File tmpFile
135 ) throws IOException
{
137 return dependencies
.getMessageReceiver()
138 .retrieveAttachment(pointer
, tmpFile
, ServiceConfig
.MAX_ATTACHMENT_SIZE
);
139 } catch (MissingConfigurationException
| InvalidMessageException e
) {
140 throw new IOException(e
);
145 public interface AttachmentHandler
{
147 void handle(InputStream inputStream
) throws IOException
;