Appengine.Maven.Guice.Sitebricks.Objectify
I have been trying to read and use all these technology for my project but I did not find any single article/blog which includes setup instructions for all of them.Most of these technologies are from Google and optimizes for their PaaS solution.
- Google AppEngine - The platform as a service supporting several languages including Python, Java.
- Maven - Build tool like ant but lets you pull the libraries from the original repository dynamically at the build time.
- Guice - Dependency Injection tool like spring without XML configuration. Configuration is done in code itself.
- Sitebricks - Libraries for creating REST webservices and dynamic HTML page (separating HTML and Data ).
- Objectify - Library to interact with Google AppEngine datastore and automatic memcached management.
Following are the major steps I took for creating a working project.
- Since I was trying to create a web app with maven I needed to create the folder structure for web project containing WEB-INF etc. I could do this manually too. Details .
$ mvn archetype:generate -DgroupId=com.neil -DartifactId=NoteWebApp -DarchetypeArtifactId=maven-archetype-webapp -DinteractiveMode=false
- Added the Google appengine dependency in the POM.xml in the recently created Maven project.
[code language="xml"]
<dependency>
<groupId>com.google.appengine</groupId>
<artifactId>appengine-api-1.0-sdk</artifactId>
<version>1.8.1</version>
</dependency>
[/code] - Add Google app engine plugin for the maven tools to be used for running devserver and uploading the build to appengine cloud.
Here we could also mention the debug port which any idea can connect.- Local server can be started using "mvn appengine:devserver". Details.
- Local app can be deployed in appengine server at "mvn appengine:update"
Sometimes "port in use" error occurs if previous debug port or server port is not closed gracefully then we have to query for the process using the port and kill it.sudo lsof -i :8080 # checks port 8080 in mac
kill -9 2828
[code language="xml"]
<plugins>
<plugin>
<groupId>com.google.appengine</groupId>
<artifactId>appengine-maven-plugin</artifactId>
<version>1.8.1</version>
<configuration>
<!--
<jvmFlags>
<jvmFlag>-Xdebug</jvmFlag>
<jvmFlag>-agentlib:jdwp=transport=dt_socket,address=5000,server=y,suspend=n</jvmFlag>
</jvmFlags>
-->
</configuration>
</plugin>
</plugins>
[/code] - Add the appengine-web.xml in the same directory as web.xml which will hold the appengine configuration.
[code language="xml"]
<appengine-web-app xmlns="http://appengine.google.com/ns/1.0">
<!-- create this unique id in appengine console in web-->
<application>testjavaneil</application>
<!-- I only keep only one version during dev and keep overwriting it. -->
<version>1</version>
<!-- I have no idea about it -->
<threadsafe>true</threadsafe>
</appengine-web-app>
[/code] - Add dependency for Guice
[code language="xml"]
<dependency>
<groupId>com.google.inject</groupId>
<artifactId>guice</artifactId>
<version>3.0</version>
</dependency>
[/code] - Add a listener servlet in web.xml which will get executed when container comes up. Also add a filter which will redirect all the urls to the Guice filter
which in turn will have the information which class to be executed depending on the request URL.This configuration will be done in the listener.
[code language="xml"]
<filter>
<filter-name>webFilter</filter-name>
<filter-class>com.google.inject.servlet.GuiceFilter</filter-class>
</filter>
<filter-mapping>
<filter-name>webFilter</filter-name>
<url-pattern>/*</url-pattern>
</filter-mapping>
<listener>
<listener-class>com.neil.MyGuiceServletConfig</listener-class>
</listener>
[/code] - As promised in the last step I will create the listener with the information of url mappings
[code language="java"]
public class MyGuiceServletConfig extends GuiceServletContextListener {
@Override
protected Injector getInjector() {
return Guice.createInjector(
//Keep sending Guice the modules
new SitebricksModule() {
@Override
protected void configureSitebricks() {
scan(NotebookService.class.getPackage());
//Should change this to logger, this is just to proove that
//sitebrick scans the classes for annotations like @At etc.
System.out.println("****** Scan complete ******");
}
}
, new ServletModule() {
@Override
protected void configureServlets() {
//Servlet classes have to be singleton to be consistent with servlet specification
//In tranditional cases web.xml config tell the container to do so I guess.
bind(com.neil.NotebookServlet.class).in(Singleton.class);
//Analogous to typcial servlet URL mappings
serve("/servlet").with(com.neil.NotebookServlet.class);
}
}
);
}
}
[/code] - Following is a typical servlet class but will will avoid this and use templates and webservices to render data in HTML or JSON format.
[code language="java"]
public class NotebookServlet extends HttpServlet {
//Register the entity class for data persistance service
static {
ObjectifyService.register(Note.class);
}
/**
* Get requests come here
*
* @param req
* @param resp
* @throws ServletException
* @throws IOException
*/
@Override
public void doGet(
HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
resp.setContentType("text/html");
//Render the form for adding notes
resp.getWriter().println(
"<form method=post action="/servlet" >" +
"<input name="note.text" size="20" type=text/>" +
"<input type=submit value="Add Note">" +
"</form>");
resp.getWriter().println("List of notes");
//load all the data from datastore
List<Note> notes = ObjectifyService.ofy().load().type(Note.class).list();
//Render the notes in each row of the table.
resp.getWriter().println("<table><tr style="background:grey"><th>Date</th><th>Note</th></tr>");
for (Note noteEntry : notes) {
resp.getWriter().println("<tr>");
resp.getWriter().println("<td>" + noteEntry.getDate().toString() + "</td><td>" + noteEntry.getText() + "</td>");
resp.getWriter().println("</tr>");
}
resp.getWriter().println("<table>");
}
/**
* Handles the form submit post requests and redirect to the same get request to display the list of
* notes
* @param req
* @param resp
* @throws ServletException
* @throws IOException
*/
@Override
public void doPost(HttpServletRequest req, HttpServletResponse resp)
throws ServletException, IOException {
Note note = new Note();
note.setDate(new Date());
note.setText(req.getParameter("note.text"));
ObjectifyService.ofy().save().entities(note).now();
doGet(req, resp);
}
}
[/code] - Create the entity to map the database
[code language="java"]
@Entity
public class Note {
@Id
private Long id;
private Date date;
public Long getId() {
return id;
}
public void setId(Long id) {
this.id = id;
}
private String text;
//rest of the getters and setters
}
[/code] - Create the class with sitebrick and annotate as service. annotate the url mapping and get post methods
[code language="java"]
@At("/notes")
@Service
public class NotebookService {
private Note note = new Note();
//Register the entity class for the Objectify persistance service.
public NotebookService() {
ObjectifyService.register(Note.class);
}
@Get
Reply<List<Note>> showNotes() {
//Prepare the HTTP headers
Map<String, String> headers = new HashMap<String, String>();
headers.put("Content-Type", "application/json");
//Fetch data from database
List<Note> notes = ObjectifyService.ofy().load().type(Note.class).list();
//Convert the entity object to JSON and return.
return Reply.with(notes).as(Json.class).headers(headers);
}
public Note getNote() {
return note;
}
public void setNote(Note note) {
this.note = note;
}
/**
* Post request endpoint here inserts data in database
* and returns the JSON of the single entry which was newly
* created
*
* @param request the body of the request containing data is
* obtained from here.
* @return
*/
@Post
public Reply postNote(Request request) {
Map<String, String> headers = new HashMap<String, String>();
headers.put("Content-Type", "application/json");
//Read JSON data and create entity object
note = request.read(Note.class).as(Json.class);
//System generated date instead of user
note.setDate(new Date());
//Store data
ObjectifyService.ofy().save().entities(note).now();
//Just return the newly added data
return Reply.with(note).as(Json.class).headers(headers);
/*
//to redirect to a url but this class is only for
//webservice call, we don't ahve to redirect to any page
return Reply.saying().redirect("/servlet");
*/
}
}
[/code] - For using sitebricks with HTML template create another class and HTML
[code language="java"]
@At("/webnotes")
@Show("/notes.html")
public class Webnote {
//When HTML page is rendered , this instance variable would be used for the placeholders to populate
private List<Note> notes;
//Following instance variable will be populated when form is submitted from the HTML template
//The name of the input fields will be mapped to the entity components
private Note note = new Note();
@Get
public void get() {
this.notes = ObjectifyService.ofy().load().type(Note.class).list(); //load from db
}
public Note getNote() {
return note;
}
public void setNote(Note note) {
this.note = note;
}
@Post
public String post() {
//Date is not provided by the form, server date is populayted
note.setDate(new Date());
ObjectifyService.ofy().save().entities(note).now();
//Redirect to same class and render the same HTML template
return "webnotes";
}
public List<Note> getNotes() {
return notes;
}
public void setNotes(List<Note> notes) {
this.notes = notes;
}
}
[/code]
[code language="java"]
<!--
HTML Template for sitebricks. This has the html form and table to enter and display the data
However theer are placeholders for sitebricks to replace the data before serving to client
-->
<html>
<head>
<title></title>
</head>
<body>
<form method=post action="/webnotes">
<input name=note.text size=20 type=text/>
<input type=submit value="Add Note">
</form>
<br>
<table>
<tr style="background:grey">
<th>Date</th>
<th>Note</th>
</tr>
@Repeat(items=notes, var="note")
<tr>
<td>${note.date}</td>
<td>${note.text}</td>
</tr>
</table>
</body>
</html>
[/code] - Add the objectify dependencies in POM.xml
[code language="xml"]
<dependency>
<groupId>com.googlecode.objectify</groupId>
<artifactId>objectify</artifactId>
<version>4.0b3</version>
</dependency>
[/code] - Register the objectify Entity class as service whenever we create any service which will have database interaction. I do it in constructor of the service class
[code language="java"]
public NotebookService() {
ObjectifyService.register(Note.class);
}
</li>
[/code] - Read/write data using Objectify. Note.class is the entity class
[code language="java"]
List<Note> notes = ObjectifyService.ofy().load().type(Note.class).list();
[/code]
[code language="java"]
ObjectifyService.ofy().save().entities(note).now();
[/code]
I have put all the code in Github and its still evolving, however you can get hold of the particular commit.