Skip to contentSkip to navigationSkip to topbar
Rate this page:
On this page

Receive and Download Images on Incoming Media Messages with Java


You know how to receive and reply to incoming SMS messages. What if you receive an MMS message containing an image you'd like to download? Let's learn how we can grab that image and any other incoming MMS media using Java.


Create MMS processing project

create-mms-processing-project page anchor

When Twilio receives a message for your phone number, it can make an HTTP call to a webhook that you create. The easiest way to handle HTTP requests in Java is to use Spark web framework(link takes you to an external page).

Twilio expects, at the very least, for your webhook to return a 200 OK response if everything is peachy. Often, however, you will return some TwiML in your response as well. TwiML is just a set of XML commands telling Twilio how you'd like it to respond to your message. Rather than manually generating the XML, we'll use the twilio helper library that can make generating the TwiML and the rest of the webhook plumbing easy peasy.


Receive MMS message and images

receive-mms-message-and-images page anchor

Get incoming message details

get-incoming-message-details page anchor

When Twilio calls your webhook, it sends a number of parameters about the message you just received.

Most of these, such as the To phone number, the From phone number, and the Body of the message are available as properties of the request parameter to the Spark views.

Get number of attachments

get-number-of-attachments page anchor

We may receive more than one media per message, this parameter informs us how many we received. We used a custom class parseBody to get the value and cast it to an Integer, to be used in a following loop.


_10
Map<String, String> parameters = parseBody(req.body());
_10
String numMediaStr = parameters.get("NumMedia");
_10
int numMedia = Integer.parseInt(numMediaStr);

Since an MMS message can have multiple attachments, Twilio will send us form variables named MediaUrlX, where X is a zero-based index. So, for example, the URL for the first media attachment will be in the MediaUrl0 parameter, the second in MediaUrl1, and so on.

In order to handle a dynamic number of attachments, we loop through all the available URLs:


_10
while (numMedia > 0) {
_10
numMedia = numMedia - 1;
_10
String mediaUrl = parameters.get(String.format("MediaUrl%d", numMedia));
_10
}

Determine content type of media

determine-content-type-of-media page anchor

Attachments to MMS messages can be of many different file types. JPG(link takes you to an external page) and GIF(link takes you to an external page) images as well as MP4(link takes you to an external page) and 3GP(link takes you to an external page) files are all common. Twilio handles the determination of the file type for you and you can get the standard mime type from the MediaContentTypeX parameter. If you are expecting photos, then you will likely see a lot of attachments with the mime type of image/jpeg.


_10
while (numMedia > 0) {
_10
numMedia = numMedia - 1;
_10
String mediaUrl = parameters.get(String.format("MediaUrl%d", numMedia));
_10
String contentType = parameters.get(String.format("MediaContentType%d", numMedia));
_10
}


Depending on your use case, storing the URLs to the images (or videos or whatever) may be all you need. There's two key features to these URLs that make them very pliable for your use in your apps:

  1. They are publicly accessible without any need for authentication to make sharing easy.
  2. They are permanent (unless you explicitly delete the media, see later).

For example, if you are building a browser-based app that needs to display the images, all you need to do is drop an <img src="twilio url to your image"> tag into the page. If this works for you, then perhaps all you need is to store the URL in a database character field.

Save media to local file system

save-media-to-local-file-system page anchor

If you want to save the media attachments to a file, then you will need to make an HTTP request to the media URL and write the response stream to a file. If you need a unique filename, you can use the last part of the media URL. For example, suppose your media URL is the following:


_10
https://api.twilio.com/2010-04-01/Accounts/ACxxxx/Messages/MMxxxx/Media/ME27be8a708784242c0daee207ff73db67

You can use that last part of the URL as a unique filename. Figuring out a good extension to use is a little tricker. If you are only expecting images, you could just assume a ".jpg" extension. For a little more flexibility, you can lookup the mime type and determine a good extension to use based on that.

Here's the complete code for our controller that saves each MMS attachment to the App_Data folder:

Saves MMS attachment with Java

saves-mms-attachment-with-java page anchor

_96
package com.twilio.app;
_96
_96
import static spark.Spark.*;
_96
_96
import java.util.Map;
_96
import java.util.HashMap;
_96
import java.io.File;
_96
import java.io.InputStream;
_96
import java.io.UnsupportedEncodingException;
_96
import java.net.URL;
_96
import java.net.URLDecoder;
_96
_96
import org.apache.tika.mime.MimeType;
_96
import org.apache.tika.mime.MimeTypes;
_96
_96
import org.apache.commons.io.FileUtils;
_96
import org.apache.http.HttpResponse;
_96
import org.apache.http.client.methods.HttpGet;
_96
import org.apache.http.impl.client.CloseableHttpClient;
_96
import org.apache.http.impl.client.HttpClients;
_96
import org.apache.http.impl.client.LaxRedirectStrategy;
_96
_96
import com.twilio.twiml.MessagingResponse;
_96
import com.twilio.twiml.messaging.Body;
_96
import com.twilio.twiml.messaging.Message;
_96
_96
public class App {
_96
public static void main(String[] args) {
_96
post("/sms", (req, res) -> {
_96
Map<String, String> parameters = parseBody(req.body());
_96
String numMediaStr = parameters.get("NumMedia");
_96
int numMedia = Integer.parseInt(numMediaStr);
_96
_96
if (numMedia > 0) {
_96
while (numMedia > 0) {
_96
numMedia = numMedia - 1;
_96
_96
// Get all info
_96
String mediaUrl = parameters.get(String.format("MediaUrl%d", numMedia));
_96
String contentType = parameters.get(String.format("MediaContentType%d", numMedia));
_96
String fileName = mediaUrl.substring(mediaUrl.lastIndexOf("/") + 1);
_96
MimeTypes allTypes = MimeTypes.getDefaultMimeTypes();
_96
MimeType mimeType = allTypes.forName(contentType);
_96
String fileExtension = mimeType.getExtension();
_96
File file = new File(fileName + fileExtension);
_96
_96
// Download file
_96
URL url = new URL(mediaUrl);
_96
CloseableHttpClient httpclient = HttpClients.custom()
_96
.setRedirectStrategy(new LaxRedirectStrategy())
_96
.build();
_96
HttpGet get = new HttpGet(url.toURI());
_96
HttpResponse response = httpclient.execute(get);
_96
InputStream source = response.getEntity().getContent();
_96
FileUtils.copyInputStreamToFile(source, file);
_96
}
_96
}
_96
_96
// Send message back
_96
String message = (numMedia > 0) ? String.format("Thanks for sending us %s file(s)!", numMedia) : "Send us an image!";
_96
res.type("application/xml");
_96
Body body = new Body
_96
.Builder(message)
_96
.build();
_96
Message sms = new Message
_96
.Builder()
_96
.body(body)
_96
.build();
_96
MessagingResponse twiml = new MessagingResponse
_96
.Builder()
_96
.message(sms)
_96
.build();
_96
return twiml.toXml();
_96
_96
});
_96
}
_96
_96
// Body parser help
_96
public static Map<String, String> parseBody(String body) throws UnsupportedEncodingException {
_96
String[] unparsedParams = body.split("&");
_96
Map<String, String> parsedParams = new HashMap<String, String>();
_96
for (int i = 0; i < unparsedParams.length; i++) {
_96
String[] param = unparsedParams[i].split("=");
_96
if (param.length == 2) {
_96
parsedParams.put(urlDecode(param[0]), urlDecode(param[1]));
_96
} else if (param.length == 1) {
_96
parsedParams.put(urlDecode(param[0]), "");
_96
}
_96
}
_96
return parsedParams;
_96
}
_96
_96
public static String urlDecode(String s) throws UnsupportedEncodingException {
_96
return URLDecoder.decode(s, "utf-8");
_96
}
_96
}

Another idea for these image files could be uploading them to a cloud storage service like Azure Blob Storage(link takes you to an external page) or Amazon S3(link takes you to an external page). You could also save them to a database, if necessary. They're just regular files at this point. Go crazy.

Delete media from Twilio

delete-media-from-twilio page anchor

If you are downloading the attachments and no longer need them to be stored by Twilio, you can easily delete them. You can send an HTTP DELETE request to the media URL and it will be deleted, but you will need to be authenticated to do this.

Java

_19
// Install the Java helper library from twilio.com/docs/java/install
_19
_19
import com.twilio.Twilio;
_19
import com.twilio.rest.api.v2010.account.message.Media;
_19
_19
public class Example {
_19
// Find your Account SID and Auth Token at twilio.com/console
_19
// and set the environment variables. See http://twil.io/secure
_19
public static final String ACCOUNT_SID = System.getenv("TWILIO_ACCOUNT_SID");
_19
public static final String AUTH_TOKEN = System.getenv("TWILIO_AUTH_TOKEN");
_19
_19
public static void main(String[] args) {
_19
Twilio.init(ACCOUNT_SID, AUTH_TOKEN);
_19
Media.deleter(
_19
"MM800f449d0399ed014aae2bcc0cc2f2ec",
_19
"ME557ce644e5ab84fa21cc21112e22c485")
_19
.delete();
_19
}
_19
}

(warning)

Warning

Twilio supports HTTP Basic and Digest Authentication. Authentication allows you to password protect your TwiML URLs on your web server so that only you and Twilio can access them. Learn more about HTTP authentication and validating incoming requests here.


All the code, in a complete working project, is available on GitHub(link takes you to an external page). If you need to dig a bit deeper, you can head over to our API Reference and learn more about the Twilio webhook request and the Media resource. Also, you will want to be aware of the pricing(link takes you to an external page) for storage of all the media files that you keep on Twilio's servers.

We'd love to hear what you build with this.


Rate this page: