Google App Engine for Java Demo Application
April 30th, 2009
Google App Engine for Java was released a couple of weeks ago and I finally found some time to finish up my first demo application. My original idea was to do a Java version of the Salesforce.com demo that I did in Python on Google App Engine, but I quickly found out that Google App Engine for Java does not support Web services. Hopefully there will be a workaround from Salesforce.com sometime in the near future to make this possible.
This demo has the same functionality as the Python version except that it uses BigTable as the datastore instead of Salesforce.com.
You can run this demo at:
A couple of observations from this early look:
- The documentation is pretty good and really gets you up and going quickly.
- If you already have Eclipse running, the Google Plugin for Eclipse is a breeze to install.
- The development environment is really slick. The development server (Jetty) runs your application on your local computer for development and testing. The server simulates the App Engine datastore, services and sandbox restrictions.
- You can access BigTable with JDO or JPA. I choose JDO as most of the documentation uses this implementation. If you are familiar with Hibernate then JDO is fairly simple to grok.
- You can use Google Accounts for account creation and authentication. I didn’t use this service but it looks pretty slick.
- Google has included a simple Guestbook app to get you started.
The Google App Engine Dashboard is very feature-rich and intuitive. I think it gives the Amazon’s EC2 Console a run for its money.
The Google Plugin for Eclipse makes deploying the application to Google’s servers simple-stupid. If you’ve ever had to configure Tomcat, JBoss or Websphere, you’ll really enjoy the simplicity of the process. However, when creating a new application, make sure you do so through the Dashboard first before deploying it.
You can download the source for the application here, but here is the real meat of the application, the Servlet:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 | package com.jeffdouglas; import java.io.IOException; import javax.servlet.http.*; import java.util.Date; import java.util.List; import java.text.DateFormat; import javax.servlet.*; import javax.jdo.PersistenceManager; import com.jeffdouglas.entity.*; import com.google.appengine.api.datastore.Key; import com.google.appengine.api.datastore.KeyFactory; @SuppressWarnings("serial") public class TelesalesServlet extends HttpServlet { public void doGet(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { // create the persistence manager instance PersistenceManager pm = PMF.get().getPersistenceManager(); // display the lookup form if(request.getParameter("action").equals("accountLookup")) { // query for the entities by name String query = "select from " + Account.class.getName() + " where name == '"+request.getParameter("accountName")+"'"; List accounts = (List) pm.newQuery(query).execute(); // pass the list to the jsp request.setAttribute("accounts", accounts); // forward the request to the jsp RequestDispatcher dispatcher = getServletContext().getRequestDispatcher("/accountLookup.jsp"); dispatcher.forward(request, response); // display the create new account form } else if(request.getParameter("action").equals("accountCreate")) { response.sendRedirect("/accountCreate.jsp"); // process the new account creation and send them to the account display page } else if(request.getParameter("action").equals("accountCreateDo")) { // create the new account Account a = new Account( request.getParameter("name"), request.getParameter("billingCity"), request.getParameter("billingState"), request.getParameter("phone"), request.getParameter("website") ); // persist the entity try { pm.makePersistent(a); } finally { pm.close(); } response.sendRedirect("telesales?action=accountDisplay&accountId="+a.getId()); // display the account details and opportunities } else if(request.getParameter("action").equals("accountDisplay")) { // fetch the account Key k = KeyFactory.createKey(Account.class.getSimpleName(), new Integer(request.getParameter("accountId")).intValue()); Account a = pm.getObjectById(Account.class, k); // query for the opportunities String query = "select from " + Opportunity.class.getName() + " where accountId == "+request.getParameter("accountId"); List opportunities = (List) pm.newQuery(query).execute(); // pass the list to the jsp request.setAttribute("account", a); // pass the list to the jsp request.setAttribute("opportunities", opportunities); // forward the request to the jsp RequestDispatcher dispatcher = getServletContext().getRequestDispatcher("/accountDisplay.jsp"); dispatcher.forward(request, response); // display the create new opportunity form } else if(request.getParameter("action").equals("opportunityCreate")) { Key k = KeyFactory.createKey(Account.class.getSimpleName(), new Integer(request.getParameter("accountId")).intValue()); Account a = pm.getObjectById(Account.class, k); // pass the account name to the jsp request.setAttribute("accountName", a.getName()); // forward the request to the jsp RequestDispatcher dispatcher = getServletContext().getRequestDispatcher("/opportunityCreate.jsp"); dispatcher.forward(request, response); // process the new opportunity creation and send them to the account display page } else if(request.getParameter("action").equals("opportunityCreateDo")) { Date closeDate = new Date(); // try and parse the date try { DateFormat df = DateFormat.getDateInstance(3); closeDate = df.parse(request.getParameter("closeDate")); } catch(java.text.ParseException pe) { System.out.println("Exception " + pe); } // create the new opportunity Opportunity opp = new Opportunity( request.getParameter("name"), new Double(request.getParameter("amount")).doubleValue(), request.getParameter("stageName"), new Integer(request.getParameter("probability")).intValue(), closeDate, new Integer(request.getParameter("orderNumber")).intValue(), new Long(request.getParameter("accountId")) ); // persist the entity try { pm.makePersistent(opp); } finally { pm.close(); } response.sendRedirect("telesales?action=accountDisplay&accountId="+request.getParameter("accountId")); } } public void doPost(HttpServletRequest request, HttpServletResponse response) throws ServletException, IOException { doGet(request, response); } } |
Related posts:
- Google App Engine for Java and Force.com Java Toolkit Demo
- Google App Engine Now Supports Java
- Force.com Demo App on Google App Engine (Python)
- Rich Internet Applications Using Flex, Salesforce.com and Google App Engine
- Salesforce.com Sites and Google App Engine for Java
Categories: Google App Engine, Java, Salesforce






[...] http://blog.jeffdouglas.com/2009/04/30/google-app-engine-for-java-demo-app/getRequestDispatcher(”/opportunityCreate.jsp”); dispatcher.forward(request, response); // process the new opportunity creation and send them to the account display page } else if(request. … [...]
Hi Jeff
What a great little project – this is just what I needed to get me going.
Thanks – you’re a live saver, great blog btw!
James
Thank goodness. I’ll probably be able to put something up today or tomorrow after playing with your app! I wish some of this was covered in GAE-J’s reference docs. I really appreciate the time you put in to share your app.
I’ve had an issue with some of my code that recently stopped working (a simple query does not return any data) since upgrading to the latest GAE SDK and Eclipse plugin. In an attempt to troubleshoot the problem I ran your Telesales example on 2 different workstations – one where my code works and one where it has stopped working. Your Telesales example seems to exhibit the same behavior, working on the earlier SDK and not working with the latest one. Have you experienced anything like this?
I haven’t noticed anything yet but I’ll definitely take a look. Thanks for the heads up.