Skip to contentSkip to navigationSkip to topbar
On this page

Respond to Incoming Phone Calls in Java


In this guide, we'll show you how to use Programmable Voice(link takes you to an external page) to respond to incoming phone calls in your Java web application. Code on your server can decide what a caller hears when they dial the number you've bought or ported to Twilio. The code snippets in this guide are written using the Java servlet API(link takes you to an external page), and should be runnable in any servlet container, such as Tomcat(link takes you to an external page).


Respond to incoming calls in your web application

respond-to-incoming-calls-in-your-web-application page anchor
Incoming Voice.

When a Twilio phone number receives an incoming call, Twilio will send an HTTP request to your web application, asking for instructions on how to handle the call. Your web application will respond with an XML document containing TwiML. That TwiML contains the instruction that Twilio will follow to say some arbitrary text, play an MP3 file, make a recording and much more.

To start answering phone calls, you must:

  • Buy and configure a Twilio-powered phone number(link takes you to an external page) capable of making and receiving phone calls, and point it at your web application
  • Write web application code to tell Twilio how to handle the incoming call using TwiML
  • Make your web application accessible on the Internet so Twilio can send you a webhook request when you receive a call
(warning)

Warning

If you are sending SMS messages to the U.S. or Canada, before proceeding further, be aware of updated restrictions on the use of Toll-Free numbers for messaging, including TF numbers obtained by purchasing them. These restrictions do not apply to Voice or other uses outside of SMS messaging. See this support article(link takes you to an external page) for details.


Buy and configure a phone number

buy-and-configure-a-phone-number page anchor

In the Twilio Console, you can search for and buy phone numbers in countries around the world. Numbers that have the Voice capability can make and receive voice phone calls from just about anywhere on the planet.

Search for voice capable numbers.

Once you purchase a number, you'll need to configure that number to send a request to your web application. This callback mechanism is called a webhook(link takes you to an external page). This can be done in the number's configuration page.

configure an incoming phone number URL.

Webhooks are user-defined HTTP(link takes you to an external page) callbacks. They are usually triggered by some event, such as receiving an SMS message or an incoming phone call. When that event occurs, Twilio makes an HTTP request (usually a POST or a GET(link takes you to an external page)) to the URL configured for the webhook.

To handle a webhook, you only need to build a small web application that can accept the HTTP requests. Almost all server-side programming languages offer some framework for you to do this. Examples across languages include ASP.NET MVC(link takes you to an external page) for C#, Servlets(link takes you to an external page) and Spark(link takes you to an external page) for Java, Express(link takes you to an external page) for Node.js, Django(link takes you to an external page) and Flask(link takes you to an external page) for Python, and Rails(link takes you to an external page) and Sinatra(link takes you to an external page) for Ruby. PHP(link takes you to an external page) has its own web app framework built in, although frameworks like Laravel(link takes you to an external page), Symfony(link takes you to an external page) and Yii(link takes you to an external page) are also popular.

Whichever framework and language you choose, webhooks function the same for every Twilio application. They will make an HTTP request to a URI that you provide to Twilio. Your application performs whatever logic you feel necessary - read/write from a database, integrate with another API or perform some computation - then replies to Twilio with a TwiML response with the instructions you want Twilio to perform.

What is TwiML?

what-is-twiml page anchor

TwiML is the Twilio Markup Language, which is just to say that it's an XML(link takes you to an external page) document with special tags defined by Twilio to help you build your SMS and voice applications. TwiML is easier shown than explained. Here's some TwiML you might use to respond to an incoming phone call:

1
<?xml version="1.0" encoding="UTF-8"?>
2
<Response>
3
<Say>Thanks for calling!</Say>
4
</Response>

And here's some TwiML you might use to respond to an incoming SMS message:

1
<?xml version="1.0" encoding="UTF-8"?>
2
<Response>
3
<Message>We got your message, thank you!</Message>
4
</Response>

Every TwiML document will have the root <Response> element and within that can contain one or more verbs. Verbs are actions you'd like Twilio to take, such as <Say> a greeting to a caller, or send an SMS <Message> in reply to an incoming message. For a full reference on everything you can do with TwiML, refer to our TwiML API Reference.


Write Java to handle the incoming phone call

write-java-to-handle-the-incoming-phone-call page anchor

Now comes the fun part - writing code that will handle an incoming HTTP request from Twilio! Our code will dictate what happens when our phone number receives a call by responding with TwiML.

First, you need to install the Twilio Java Helper Library(link takes you to an external page) in your project.

Respond to an incoming call with TwiMLink to code sample: Respond to an incoming call with TwiM
1
import java.io.IOException;
2
3
import javax.servlet.ServletException;
4
import javax.servlet.annotation.WebServlet;
5
import javax.servlet.http.HttpServlet;
6
import javax.servlet.http.HttpServletRequest;
7
import javax.servlet.http.HttpServletResponse;
8
9
import com.twilio.twiml.voice.Say;
10
import com.twilio.twiml.TwiMLException;
11
import com.twilio.twiml.VoiceResponse;
12
13
@SuppressWarnings("serial")
14
@WebServlet("/voice")
15
public class IncomingCallServlet extends HttpServlet {
16
// Handle HTTP POST to /voice
17
protected void doPost(HttpServletRequest request, HttpServletResponse response)
18
throws ServletException, IOException {
19
20
Say say = new Say.Builder("Hello world!").build();
21
VoiceResponse twiml = new VoiceResponse.Builder().say(say).build();
22
23
// Render TwiML as XML
24
response.setContentType("text/xml");
25
26
try {
27
response.getWriter().print(twiml.toXml());
28
} catch (TwiMLException e) {
29
e.printStackTrace();
30
}
31
32
}
33
}

The TwiML generated by our server code

the-twiml-generated-by-our-server-code page anchor

The Helper Libraries help you generate an XML string that looks like this.

1
<?xml version="1.0" encoding="UTF-8"?>
2
<Response>
3
<Say>Hello world!</Say>
4
</Response>

In order for the webhooks in this code sample to work, Twilio must be able to send your web application an HTTP request over the Internet. Of course, that means your application needs to have a URL or IP address that Twilio can reach.

In production you probably have a public URL, but you probably don't during development. That's where ngrok(link takes you to an external page) comes in. ngrok gives you a public URL for a local port on your development machine, which you can use to configure your Twilio webhooks as described above.

Once ngrok is installed, you can use it at the command line to create a tunnel to whatever port your web application is running on. For example, this will create a public URL for a web application listening on port 3000.

ngrok http 3000

After executing that command, you will see that ngrok has given your application a public URL that you can use in your webhook configuration in the Twilio console.

ngrok screen.

Grab your ngrok public URL and head back to the phone number you configured earlier. Now let's switch it from using a TwiML Bin to use your new ngrok URL. Don't forget to append the URL path to your actual TwiML logic! ("http://<your ngrok subdomain>.ngrok.io/voice" for example)

configure an incoming phone number URL.

Create Java responses to incoming calls

create-java-responses-to-incoming-calls page anchor

In the example above, we returned pre-defined TwiML in response to the incoming call. The real power of using webhooks like this is executing dynamic code (based on the information Twilio sends to your application) to change what you present to the user on the other end of the phone call. You could query your database, reference a customer's phone number in your CRM, or execute any kind of custom logic before determining how to respond to your user.

Create a dynamic response to an incoming callLink to code sample: Create a dynamic response to an incoming call
1
import java.io.IOException;
2
3
import javax.servlet.ServletException;
4
import javax.servlet.annotation.WebServlet;
5
import javax.servlet.http.HttpServlet;
6
import javax.servlet.http.HttpServletRequest;
7
import javax.servlet.http.HttpServletResponse;
8
9
import com.twilio.twiml.voice.Play;
10
import com.twilio.twiml.voice.Say;
11
import com.twilio.twiml.TwiMLException;
12
import com.twilio.twiml.VoiceResponse;
13
14
@SuppressWarnings("serial")
15
@WebServlet("/voice")
16
public class IncomingCallServlet extends HttpServlet {
17
// Handle HTTP POST to /voice
18
protected void doPost(HttpServletRequest request, HttpServletResponse response)
19
throws ServletException, IOException {
20
// Get the city from the incoming call (if available)
21
String fromCity = request.getParameter("FromCity");
22
if (fromCity == null) {
23
fromCity = "home slice";
24
}
25
26
Say say = new Say.Builder(String.format("Never gonna give you up, %s!", fromCity)).build();
27
Play play = new Play.Builder("https://demo.twilio.com/docs/classic.mp3").build();
28
VoiceResponse response = new VoiceResponse.Builder().say(say).play(play).build();
29
30
// Render TwiML as XML
31
response.setContentType("text/xml");
32
33
try {
34
response.getWriter().print(twiml.toXml());
35
} catch (TwiMLException e) {
36
e.printStackTrace();
37
}
38
39
40
41
}
42
43
}
(warning)

Protect your webhooks

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 here, and check out our full guide to securing your Servlet application by validating incoming Twilio requests.


Great work! In a few lines of code, you've started answering phone calls in your Java servlet application. If you're interested in learning more about building voice applications in Java, perhaps we could tempt you with a few tutorials? Tutorials walk through full sample applications that implement production Twilio use cases, like these:

Happy hacking!

Need some help?

Terms of service

Copyright © 2025 Twilio Inc.