Can I have an Action without a form > Yes If your Action does not need any data and it does not need to make any data available to the view or controller component that it forwards to,it doesn't need a
Struts 2 Validation Example
Struts 2 Validation Example
Validate a login page using Struts 2. Let's first create the login page. use Struts UI tags to create the login page. The <s:head /> tag should be placed in the head section of the HTML page. The s:head tag automatically generates links to the css and javascript libraries that are necessary to render the form elements.
The s:form tag contains all the form elements. The action attribute contains the action name to wich the form should be submitted. This action name should be same as the one specified in the XML declarative architecture. In this example we use struts.xml file to do the configuration.
The textfield tag is used create a text box. The label attribute of the textfield tag contains the name to be displayed on the page and the name attribute contains the name of the property in the action class to be mapped. The password tag is same as the textfield tag except that the input value is masked. The submit tag is used to create a submit button, the value "Login" represents the label of the button. <%@taglib uri="/struts-tags" prefix="s" %>
<html> <head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Login Page</title>
<s:head /> </head> <body>
<s:form action="Login">
<s:textfield name="userName" label="User Name" /> <s:password name="password" label="Password" /> <s:submit value="Login" />
</s:form> </body> </html>
When the user clicks the Login button the request will be forwarded to the Login action.
We do the action mapping using the struts.xml file. First we need to create a package for our action. <struts>
<package name="default" extends="struts-default"> <action name="Login" class="Login">
<result name="input">/login.jsp</result> <result name="success">/success.jsp</result> </action>
</struts>
Here our "default" package extends "struts-default" package. By extending the "struts-default" package the action will by default inherit the set of interceptors defined in the defaultstack. The "struts-default" package is defined in the struts-default.xml file.
All the common tasks done by the Actions are seperated and placed in different interceptors. You can define an interceptor stack for each action. Most commonly used interceptors are grouped in
defaultstack of the struts-default package. The defaultstack will be sufficient in most cases. The inteceptors will be fired in the order in which they are declared in the stack both before and after the action is executed.
Here the "Login" action is mapped to the "Login" class in the "vaannila" package. The results are defined using the "<result>" element. If any validation errors occur the user will be forwarded to the login.jsp page. If the login is successfull then the user will be forwarded to the success.jsp page.
public class Login extends ActionSupport { private String userName;
private String password; public Login() { } public String execute() { return SUCCESS; }
public void validate() {
if (getUserName().length() == 0) {
addFieldError("userName", "User Name is required"); } else if (!getUserName().equals("Eswar")) {
addFieldError("userName", "Invalid User"); } if (getPassword().length() == 0) { addFieldError("password", getText("password.required")); } } //….Settersandgettersforusernameandpassword }
The ActionSupport class implements Action interface which exposes the execute() method.
The following constants are declared in the Action interface which can be used as return values in the execute() method.
public static final String ERROR = "error" public static final String INPUT = "input" public static final String LOGIN = "login" public static final String NONE = "none" public static final String SUCCESS = "success" ERROR is returned when the action execution fails.
INPUT is returned when the action requires more input from the user. LOGIN is returned when the user is not logged into the system.
NONE is returned when the action execution is successfull and there are no views to display.
SUCCESS is returned when the action executed successfully and the corresponding result is displayed to the user.
Now lets see the roles played by the different interceptors.
The params interceptor helps in transfering the request data onto the action object. The workflow interceptor controls the flow of cotrol.
The workflow interceptor checks whether the action implements the Validateable interface , if it does, the workflow interceptor will invoke the validate() method of the Action class.
In the validate() method we validate the user name and the password. If the validation fails an error is added using the addFiledError() method.
The validate() method doesn't return any errors, instead it stores all the errors with the help of the ValidationAware interface.
Now the workflow interceptor will check any validation errors has occured. If any error has occured the workflow interceptor will stop the request processing and transfer the control to the input page with the appropriate error messages.
Login.proerties
password.required = Password is required. Validation using XML
The naming convention of the XML validation file should be ActionClass-Validation.xml. Here our Action Class name is "Login.java" and the XML validation file name is "Login-Validation.xml".
The Login-Validation.xml file contains the following code.
<!DOCTYPE validators PUBLIC "-//OpenSymphony Group//XWork Validator 1.0.2//EN" "http://www.opensymphony.com/xwork/xwork-validator-1.0.2.dtd">
<validators>
<field name="userName">
<field-validator type="requiredstring">
<message>User Name is required.</message> </field-validator> </field> <field name="password"> <field-validator type="requiredstring"> <message key="password.required" /> </field-validator> </field> </validators>
The field element contains the name of the form property that needs to be validated. The filed-validator element inside the field element contains the type of validation that needs to be performed.
specify the error message directly using the message element or you can use the properties file to define all the errror messages and use the key attribute to specify the error key.
Note the properties file should also have the same name as the Action class. The Login Action class contains the following code.
public class Login extends ActionSupport { private String userName;
private String password; public Login() { } public String execute() { return SUCCESS; }
….//setterandgetter Login.jsp
<%@page contentType="text/html" pageEncoding="UTF-8"%> <%@taglib uri="/struts-tags" prefix="s" %>
<html> <head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>Login Page</title> <s:head /> </head> <body> <s:form action="LoginAction">
<s:textfield name="userName" label="User Name" /> <s:password name="password" label="Password" /> <s:submit value="Login" />
</s:form> </body> </html>
The <s:head /> tag is used to include the required css and js file for the selected theme. By default the xhtml theme is used.
Domain Object as Java Bean property
Transferring the data to the domain object is automatically done by the params interceptor. getUser().getAge();
InJSPitsreferredas… <body>
<s:form action="UserAction" >
<s:textfield name="user.name" label="User Name" /> <s:textfield name="user.age" label="Age" />
<s:radio name="user.sex" label="Sex" list="{'M','F'}" /> …….
In Result.jsp page
User Name :<s:property value="user.name" /><br> Age :<s:property value="user.age" /><br> Hobbies :<s:property value="user.hobby" /><br> Country :<s:property value="user.country" /><br> Model Driven Inferface.
To create a ModelDriven Action our Action class should implement the ModelDriven interface and should include the modelDriven interceptor. The modelDriven interceptor is already included in the default stack. The next step is to implement the getModel() method in such a way that it returns the application domain object, in our example we return the User object.
When using the ModelDriven method,need to initialize the User object ourselves. The framework will automatically transfers the form data into the User object.
public class UserAction extends ActionSupport implements ModelDriven { private User user = new User();
public UserAction() { }
public Object getModel() { return user; } public String execute() { return SUCCESS; } public User getUser() { return user; } public void setUser(User user) { this.user = user; } }
user.getXXX();
The User class contains the following attributes and the corresponding getter and setter methods. public class User {
private String name; private int age; private String sex; private String[] hobby; private String country;
public String getName() { return name; }
public void setName(String name) { this.name = name; } public int getAge() { return age; }
public void setAge(int age) { this.age = age; } public String getSex() { return sex; }
public void setSex(String sex) { this.sex = sex; } public String[] getHobby() { return hobby; }
public void setHobby(String[] hobby) { this.hobby = hobby; } public String getCountry() { return country; }
public void setCountry(String country) { this.country = country; } }
In the jsp page the user attributes can be accessed directly. To refer the user's age, the value of the name attribute should be
name=―age‖; name = "age" index.jsp
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<%@taglib uri="/struts-tags" prefix="s" %> <html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>User Details</title>
</head> <body>
<s:form action="UserAction" >
<s:textfield name="name" label="User Name" /> <s:textfield name="age" label="Age" />
<s:radio name="sex" label="Sex" list="{'M','F'}" />
<s:checkboxlist name="hobby" label="Hobby" list="{'Music','Art','Dance'}"/> <s:select name="country" label="Country"
list="{'Select','India','USA','France','Spain'}" /> <s:submit /> </s:form> </body> </html> The result.jsp
<!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<%@taglib uri="/struts-tags" prefix="s" %> <html>
<head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>User Details</title>
</head> <body>
<h2>User Details</h2> <hr>
User Name :<s:property value="name" /><br> Age :<s:property value="age" /><br>
Hobbies :<s:property value="hobby" /><br> Country :<s:property value="country" /><br> </body>
</html>
ModelDriven is like an Action Form in Struts 1.x FileUploadInterpreter – Inbuilt
Input Form.jsp
<s:form action="fileUpload" method="post" enctype="multipart/form-data" > <s:file name="userImage" label="User Image" />
<s:submit /> </s:form>
PublicClassFileUploadActionextendsActionSupport{….}
<%@ page language="java" contentType="text/html; charset=ISO-8859-1" pageEncoding="ISO-8859-1"%>
<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html> <head>
<meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"> <title>File Upload</title>
</head> <body>
You have uploaded the following file. <hr>
File Name : ${userImageFileName} <br> Content Type : ${userImageContentType} </body>
</html>
Now let's run the example.
validate either programmatically or declaratively. Validate the file in the validate() method of the FileUploadAction class. You can get the file using the getX() method and the file type type using the getXContentType() method.
The only change we need to do is to add few parameters to the fileUpload interceptor. So we copy the defaultStack from the struts-default.xml and paste it in our struts.xml file and rename the stack to fileUploadStack. Our struts.xml file contains the following code.
We can validate either programmatically or declaratively. Let's see how to do programmatically first. Validate the file in the validate() method of the FileUploadAction class. You can get the file using the getX() method and the file type type using the getXContentType() method.
In this example we validate declaratively. To do this we need to create our own interceptor stack. The only change we need to do is to add few parameters to the fileUpload interceptor. So we copy the defaultStack from the struts-default.xml and paste it in our struts.xml file and rename the stack to fileUploadStack. Our struts.xml file contains the following code.
<!DOCTYPE struts PUBLIC 02."-//Apache Software Foundation//DTD Struts Configuration 2.0//EN""http://struts.apache.org/dtds/struts-2.0.dtd">
<struts>
<package name="fileUploadPackage" extends="struts-default"> <interceptors> <interceptor-stack name="fileUploadStack"> <interceptor-ref name="exception" /> <interceptor-ref name="alias" /> <interceptor-ref name="servletConfig" /> <interceptor-ref name="prepare" /> <interceptor-ref name="i18n" /> <interceptor-ref name="chain" /> <interceptor-ref name="debugging" /> <interceptor-ref name="profiling" /> <interceptor-ref name="scopedModelDriven" /> <interceptor-ref name="modelDriven" /> <interceptor-ref name="fileUpload"> <param name="maximumSize">10240</param>
<param name="allowedTypes"> image/jpeg,image/gif,image/png</param> </interceptor-ref>
<interceptor-ref name="checkbox" /> <interceptor-ref name="staticParams" />
<interceptor-ref name="actionMappingParams" /> <interceptor-ref name="params">
<param name="excludeParams"> dojo\..*,^struts\..*</param> </interceptor-ref>
<interceptor-ref name="conversionError" /> <interceptor-ref name="validation">
<param name="excludeMethods"> input,back,cancel,browse</param> </interceptor-ref>
<interceptor-ref name="workflow">
<param name="excludeMethods"> input,back,cancel,browse</param> </interceptor-ref>
</interceptor-stack> </interceptors>
<action name="fileUpload" class="FileUploadAction"> <interceptor-ref name="fileUploadStack" /> <result name="input">/index.jsp</result> <result name="success">/success.jsp</result> </action> </package> </struts>
The maximumSize value is set in bytes. Here we set the maximumSize to 10kb. The allowedTypes indicate the file types that can be uploaded. Here we set it to only image files like
image/jpeg,image/gif,image/png. Struts 2 – spring Integration .
In Web.xml - Add the ContextLoaderListener <listener>
<listener-class>org.springframework.web.context.ContextLoaderListener</listener-class> </listener>
1. Add the Applicationcontext.xml file <context-param>
<param-name>contextConfigLocation</param-name>
<param-value>/WEB-INF/applicationContext*.xml</param-value> </context-param>
you don't specify the contextConfigLocation context parameter, the ContextLoaderListener will look for a file called /WEB-INF/applicationContext.xml to load. Once the context files are loaded, Spring creates a WebApplicationContext object based on the bean definitions and stores it in the ServletContext of one's web application.
All Java web frameworks are built on top of the Servlet API, and so one can use the following code snippet to get access to this 'business context' ApplicationContext created by the ContextLoaderListener. WebApplicationContext ctx = WebApplicationContextUtils.getWebApplicationContext(servletContext); 1. The WebApplicationContextUtils class is for convenience, so you don't have to remember the name of the
ServletContext attribute. Its getWebApplicationContext() method will return null if an object doesn't exist under the WebApplicationContext.ROOT_WEB_APPLICATION_CONTEXT_ATTRIBUTE key. Rather than risk getting NullPointerExceptions in your application, it's better to use the
getRequiredWebApplicationContext() method. This method throws an exception when the ApplicationContext is missing.
Struts2 with Tiles integration Web.xml
<context-param>
<param-name> org.apache.tiles.impl.BasicTilesContainer.DEFINITIONS_CONFIG </param-name> <param-value>/WEB-INF/tiles.xml</param-value> </context-param> Listner <listener> <listener-class>org.apache.struts2.tiles.StrutsTilesListener</listener-class> </listener> Web.xml - filter <filter> <filter-name>struts2</filter-name> <filter-class> org.apache.struts2.dispatcher.ng.filter. StrutsPrepareAndExecuteFilter </filter-class> </filter>
<filter-mapping>
<filter-name>struts2</filter-name> <url-pattern>/*</url-pattern> </filter-mapping>
The tiles.xml file contains the following tile definitions .<?xml version="1.0" encoding="UTF-8" ?>
<!DOCTYPE tiles-definitions PUBLIC
"-//Apache Software Foundation//DTD Tiles Configuration 2.0//EN" "http://tiles.apache.org/dtds/tiles-config_2_0.dtd">
<tiles-definitions>
<definition name="baseLayout" template="/baseLayout.jsp"> <put-attribute name="title" value="Template"/>
<put-attribute name="header" value="/header.jsp"/> <put-attribute name="menu" value="/menu.jsp"/> <put-attribute name="body" value="/body.jsp"/> <put-attribute name="footer" value="/footer.jsp"/> </definition>
<definition name="welcome" extends="baseLayout"> <put-attribute name="title" value="Welcome"/>
<put-attribute name="body" value="/welcome.jsp"/> </definition>
<definition name="friends" extends="baseLayout"> <put-attribute name="title" value="Friends"/>
<put-attribute name="body" value="/friends.jsp"/> </definition>
<definition name="office" extends="baseLayout"> <put-attribute name="title" value="Office"/>
<put-attribute name="body" value="/office.jsp"/> </definition>
</tiles-definitions>
Define a "baseLayout" that contains a title, header, menu, body and footer regions. The header, menu and footer region remains the same for all the layouts only the title and body content changes. In the baseLayout.jsp page we create a classic tiles layout as shown below.
<%@ taglib uri="http://tiles.apache.org/tags-tiles" prefix="tiles" %> <!DOCTYPE HTML PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">
<html> <head>
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <title>
<tiles:insertAttribute name="title" ignore="true" /> </title>
</head> <body>
<table border="1" cellpadding="2" cellspacing="2" align="center"> <tr>
<tiles:insertAttribute name="header" /> </td> </tr> <tr>