Monday, January 30, 2017

Java Struts2 in 10 minutes

Introduction

Apache Struts 2 is an elegant, extensible framework for creating enterprise-ready Java web applications. The framework is designed to streamline the full development cycle, from building, to deploying, to maintaining applications over time. Apache Struts 2 was originally known as WebWork 2.

The Struts 2 framework is used to develop MVC (Model View Controller) based web applications. Struts 2 is the combination of webwork framework of opensymphony and struts1.


struts2 = webwork + struts1

Architecture



Request Life Cycle




About Struts 2 Request Flow
  1. The controller receives the user request and determine which Struts 2 action to invoke.
  2. The framework creates an instance of this action and associate it with the newly created instance of the ActionInvocation.
  3. In Struts 2 the invocation of action should pass through a series of interceptors as defined in the application's XML file.
  4. The framework calls the ActionInvocations invoke() method to start the execution of the action.
  5. Each time the invoke() method is called, ActionInvocation consults its state and executes whichever interceptor comes next.
  6. ActionInvocation hands control over to the interceptor in the stack by calling the interceptors intercept() method.
  7. The intercept() method of the interceptor in turn calls the invoke() method of the ActionInvocation till all the interceptors are invoked, in the end the action itself will be called and the corresponding result will be returned back to the user.
  8. Some interceptor do work before the action is executed and some do work after the action is executed. It's not necessary that it should do something each time it is invoked.
  9. These interceptors are invoke both before and after the action.
  10. First all the interceptors are executed in the order they are defined in the stack.
  11. Then the action is invoked and the result is generated.
  12. Again all the interceptors present in the stack are invoked in the reverse order.
  13. The other important features of Struts 2 are OGNL and ValueStack.
  14. Object-Graph Navigation Language (OGNL) is a powerful expression language that is used to reference and manipulate data on the ValueStack.
  15. OGNL help in data transfer and type conversion.
  16. OGNL expression language provides simplified syntax to reference java objects.
  17. OGNL is used to bind the java-side data properties to the string-based view layer.



How Struts2  works

Struts 2 MVC is realised by three core framework components: actions, results, and the ServletFilter. The diagram below shows how these three components interact with each other.





1.1. Servlet Filter

The servlet filter acts as a controller in Struts 2. It inspects each incoming request to determine which Struts 2 action should handle the request. The framework handles all of the controller work for the application. All request URLs need to be mapped to actions with XML-based configuration files or Java annotations.

1.2. Interceptors

Interceptors execute before and after the request processing. They provide cross-cutting tasks so that they can be easily reused as well as separated from other architectural concerns. For example, validation parameters before invoking login action.

1.3. Action

Action handles the client requests in two different ways. First, the action plays an important role in the transfer of data from the request through to the view, whether its a JSP or other type of result. Second, the action must assist the framework in determining which result should render the view that will be returned in the response to the request.
Note-: In Struts 2 the model is implemented by the Action component.

1.4. Result

Result is a JSP or HTML page to create a view for client response. Struts 2 provides their own tags that we can use in JSP pages to create a response. Struts tags are great example of JSP Custom Tags.

Controller
StrutsPrepareAndExecuteFilter or FilterDispatcher(now depreciated) are used as front Controller in Struts2.  In Struts 1, ActionServlet is front controller.
Example
web.xml


01<?xml version="1.0" encoding="UTF-8"?>
03  <display-name>Struts2tutorial</display-name>
04
05  <filter>
06    <filter-name>Struts2</filter-name>
07<filter-class>org.apache.Struts2.dispatcher.ng.filter.StrutsPrepareAndExecuteFilter </filter-class>
08  </filter>
09
10  <filter-mapping>
11    <filter-name>Struts2</filter-name>
12    <url-pattern>/*</url-pattern>
13  </filter-mapping>
14
15  <welcome-file-list>
16    <welcome-file>/login.jsp</welcome-file>
17  </welcome-file-list>
18</web-app>


LoginAction.java

We only have one action class which implements Action interface{This is optional in Struts 2.Read end of article for more detail}. The LoginAction needs to implement the executemethod to handle client requests and return proper result.

package com.java.code.geeks.action;
02
03import com.opensymphony.xwork2.Action;
04
05public class LoginAction implements Action {
06
07    @Override
08    public String execute() throws Exception {
09        if (validateString(getUsername()) && validateString(getPassword()))
10            return "SUCCESS";
11        else return "ERROR";
12    }
13
14    //Java Bean to hold the form parameters
15    private String username;
16    private String password;
17
18    public String getUsername() {
19        return username;
20    }
21
22    public void setUsername(String username) {
23        this.username = username;
24    }
25
26    public String getPassword() {
27        return password;
28    }
29
30    public void setPassword(String password) {
31        this.password = password;
32    }
33
34    private boolean validateString(String str) {
35        if (str != null && !str.equals(""))
36            return true;
37        return false;
38    }
39}

login.jsp

01<%@ page language="java" contentType="text/html; charset=US-ASCII"
02    pageEncoding="US-ASCII"%>
03<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
04<%@ taglib uri="/Struts-tags" prefix="s"%>
05<html>
06    <head>
07        <meta http-equiv="Content-Type" content="text/html; charset=US-ASCII">
08        <title>Login Page</title>
09    </head>
10    <body>
11        <h3>Welcome to Struts 2 Login Tutorial</h3>
12    <s:form action="home">
13        <s:textfield name="username" label="User Name"></s:textfield>
14        <s:textfield name="password" label="Password" type="password"></s:textfield>
15        <s:submit value="Login"></s:submit>
16    </s:form>
17    </body>
18</html>

welcome.jsp

01<%@ page language="java" contentType="text/html; charset=US-ASCII"
02    pageEncoding="US-ASCII"%>
03<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
04<%@ taglib uri="/Struts-tags" prefix="s"%>
05<html>
06    <head>
07        <meta http-equiv="Content-Type" content="text/html; charset=US-ASCII">
08        <title>Welcome To Struts 2 Tutorial</title>
09    </head>
10    <body>
11        <h3>Welcome <s:property value="username"></s:property>!</h3>
12    </body>
13</html>

error.jsp
01<%@ page language="java" contentType="text/html; charset=US-ASCII"
02    pageEncoding="US-ASCII"%>
03<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
04<%@ taglib uri="/Struts-tags" prefix="s"%>
05<html>
06    <head>
07    <meta http-equiv="Content-Type" content="text/html; charset=US-ASCII">
08        <title>Error Page</title>
09    </head>
10    <body>
11        <h4>User Name or Password is wrong</h4>
12        <s:include value="login.jsp"></s:include>
13    </body>
14</html>


Struts.xml
The Struts.xml is created in resources folder in src. The Struts configs are responsible for mapping action and result in the web application.
01<?xml version="1.0" encoding="UTF-8"?>
02
03<!DOCTYPE Struts PUBLIC
04        "-//Apache Software Foundation//DTD Struts Configuration 2.3//EN"
06<Struts>
07    <package name="default" namespace="/" extends="Struts-default">
08        <action name="login">
09            <result>/login.jsp</result>
10        </action>
11        <action name="home" class="com.java.code.geeks.action.LoginAction">
12            <result name="SUCCESS">/welcome.jsp</result>
13            <result name="ERROR">/error.jsp</result>
14        </action>
15    </package>
16</Struts>
For action login, there is no Action class and only one result. The client request will be forwarded to the login.jsp page when the login action is requested.
For action home, LoginAction is the action class and if execute() method returns SUCCESS the request will be processed by welcome.jsp and for ERROR it will be forwarded to error.jsp page. The namespace="/" is root namespace in Struts.

Now, we can access our application with URL http://localhost:8080/Struts2tutorial/login.action. Notice that URL is ending with .action that is the default suffix for Struts 2 action like it is .do for Struts 1.


Point to Remember-:

Struts 2 Action

In Struts 2 action class is a POJO. POJO means you are not forced to implement any interface or extend any class.In struts 2, action class is POJO (Plain Old Java Object). Generally, execute method should be specified that represents the business logic. The simple action class may look like:

Welcome.java
  1. package com.javatpoint;  
  2. public class Welcome {  
  3. public String execute(){  
  4.     return "success";  
  5. }  
  6. }  

Action Interface

A convenient approach is to implement the com.opensymphony.xwork2.Action interface that defines 5 constants and one execute method.

5 Constants of Action Interface

Action interface provides 5 constants that can be returned form the action class. They are:
  1. SUCCESS indicates that action execution is successful and a success result should be shown to the user.
  2. ERROR indicates that action execution is failed and a error result should be shown to the user.
  3. LOGIN indicates that user is not logged-in and a login result should be shown to the user.
  4. INPUT indicates that validation is failed and a input result should be shown to the user again.
  5. NONE indicates that action execution is successful but no result should be shown to the user.
Let's see what values are assigned to these constants:
  1. public static final String SUCCESS = "success";  
  2. public static final String ERROR = "error";  
  3. public static final String LOGIN  = "login";  
  4. public static final String INPUT = "input";  
  5. public static final String NONE = "none";  

Method of Action Interface

Action interface contains only one method execute that should be implemented overridden by the action class even if you are not forced.
  1. public String execute();  

Example of Struts Action that implements Action interface

If we implement the Action interface, we can directly use the constants instead of values.
Welcome.java
  1. package com.javatpoint;  
  2. import com.opensymphony.xwork2.Action;  
  3. public class Welcome implements Action{  
  4. public String execute(){  
  5.     return SUCCESS;  
  6. }  
  7. }  

ActionSupport class

It is a convenient class that implements many interfaces such as Action, Validateable, ValidationAware, TextProvider, LocaleProvider and Serializable . So it is mostly used instead of Action.

Example of Struts Action that extends ActionSupport class

Let's see the example of Action class that extends the ActionSupport class.
Welcome.java
  1. package com.javatpoint;  
  2. import com.opensymphony.xwork2.ActionSupport;  
  3. public class Welcome extends ActionSupport{  
  4. public String execute(){  
  5.     return SUCCESS;  
  6. }  
  7. }  


Also Read-: ValueStack and OGNL differences.

No comments:

Post a Comment

System Design :: Performace Tuning: Scaling, Resiliency, persistence

Netflix System Deisgn