© The WildFly Authors.
This guide has moved to https://github.com/wildfly/quickstart/blob/main/guide/
1. Introduction
This guide will walk you through installing and starting up JBoss WildFly. It will then introduce key features of the Jakarta EE (Web Profile) programming model, of which JBoss WildFly is a certified implementation.
Jakarta EE
The Jakarta EE platform offers developers the ability to write distributed, transactional and portable applications quickly and easily. We class applications that require these capabilities "enterprise applications". These applications must be fast, secure and reliable. Jakarta EE has always offered strong messaging (JMS), transactional (JTA) and resource (JCA) capabilities as well as exposing web services via SOAP (JAX-WS). |
JBoss WildFly depart from the familiar structure of previous JBoss AS versions, so we recommend all developers follow the steps in Getting Started with JBoss WildFly to install and start up the application server for the first time.
JBoss WildFly come with a series of quickstarts aimed to get you up to writing applications with minimal fuss. We recommend to start by working through the quickstarts in this guide, in the order they are presented. If you have previous experience with Jakarta EE 6, you may wish to skip some or all of the quickstarts.
Core
-
Helloworld Quickstart. If you have previously developed applications using technologies such as JSF or Wicket, and EJB or Spring, you may wish to skip this quickstart.
-
Numberguess Quickstart. If you have previously developed applications using technologies such as JSF or Wicket, EJB or Spring, and JPA or Hibernate you may wish to skip this quickstart.
-
Greeter Quickstart. If you are a Jakarta EE wizard you may wish to skip this quickstart.
-
Kitchensink Quickstart. A great starting point for your project.
1.1. Downloading the quickstarts
The quickstarts are are available for download from JBoss Developer Framework. Make sure you download the latest zip!
2. Getting started with WildFly
To run the quickstarts with the provided build scripts, you’ll need:
If you already have any of these pieces of software, there is no need to install them again! |
- Java 8, to run WildFly and Maven
-
Choose your Java runtime, and follow their installation instructions. For example, you could choose one of:
- Maven 3, to build and deploy the quickstarts
-
Follow the official Maven installation guide if you don’t already have Maven 3 installed. You can check which version of Maven you have installed (if any) by running
mvn --version
. If you see a version newer than 3.0.0, you are ready to go. - The JBoss WildFly runtime
-
Download JBoss WildFly from the WildFly download page
- The WildFly quickstarts
-
Available from WildFly Quickstarts
If you wish to use the examples from an IDE, we recommend using CodeReady Studio, or Eclipse with JBoss Tools.
- Red Hat CodeReady Studio
-
Download CodeReady Studio from https://developers.redhat.com/products/codeready-studio/download.
- Eclipse, with JBoss Tools
-
Download JBoss Tools from http://jboss.org/tools. Make sure you install m2eclipse as well.
JBoss WildFly offer the ability to manage multiple AS instances from a single control point. A collection of such servers are referred to as members of a "domain", with a single Domain Controller process acting as the management control point. Domains can span multiple physical (or virtual) machines, with all AS instances on a given host under the control of a Host Controller process. The Host Controllers interact with the Domain Controller to control the lifecycle of the AS instances running on that host and to assist the Domain Controller in managing them. JBoss WildFly also offers a standalone mode, which is perfect for a single server. We use this throughout the quickstarts. |
2.1. Installing and starting the JBoss server on Linux, Unix or Mac OS X
First, let’s verify that both Java and Maven are correctly installed. In a console, type:
java -version
You should see a version string (at least 1.8.0
) printed. If not, contact your provider of Java for assistance. Next, type:
mvn --version
You should see a version string (at least 3.3.0
) printed. If not, contact the Maven community for assistance.
Next, we need to choose a location for WildFly to live. By default, WildFly will be extracted into wildfly-11.x.x.x
(where 11.x.x.x
matches the version you downloaded):
unzip wildfly-11.x.x.x.zip
Now, let’s start WildFly in standalone mode:
wildfly-11.x.x.x/bin/standalone.sh
If you want to stop WildFly, simply press Crtl-C whilst the terminal has focus. |
That’s it, WildFly is installed and running! Visit http://localhost:8080/ to check the server has started properly.
You can find the server log for standalone instances in
|
2.2. Installing and starting the JBoss server on Windows
First, let’s verify that both Java and Maven are correctly installed. In a Command Prompt, type:
java -version
You should see a version string (at least 1.8.0
) printed. If not, contact your provider of Java for assistance. Next, type:
mvn --version
You should see a version string (at least 3.3.0
) printed. If not, contact the Maven community for assistance.
Next, we need to choose a location for JBoss WildFly to live. By default, JBoss WildFly will be extracted into wildfly-11.x.x.x
(where 11.x.x.x
matches the version you downloaded). Unzip JBoss Enterprise Application Platform or JBoss WildFly using your tool of choice.
Finally, let’s start JBoss WildFly in standalone mode. Locate your installation and run standalone.bat
located in bin
.
If you want to stop the server, simply press Crtl-C whilst the terminal has focus. |
That’s it, JBoss WildFly is installed and running! Visit http://localhost:8080/ to check the server has started properly.
You can find the server log for standalone instances in
|
2.3. Starting the JBoss server from CodeReady Studio or Eclipse with JBoss Tools
You may choose to use CodeReady Studio, or Eclipse with JBoss Tools, rather than the command line to run JBoss WildFly, and to deploy the quickstarts. If you don’t wish to use Eclipse, you should skip this section.
Make sure you have installed and started CodeReady Studio or Eclipse. First, we need to add our WildFly instance to it. First, navigate to Preferences:
Now, locate the JBoss Tools Runtime Detection preferences:
Click Add and locate where you put servers on your disk:
Any available servers will be located, now all you need to do is click OK, and then OK on the preferences dialog:
Now, let’s start the server from Eclipse. If you previously started a server from the command line, you should stop it there first.
First, we need to make sure the Server tab is on view. Open the Window → Show View → Other… dialog:
And select the Server view:
You should see the Server View appear with the detected servers:
Now, we can start the server. Right click on the server in the Server view, and select Start :
If you want to debug your application, you can simply select Debug rather than Start . This will start the server in debug mode, and automatically attach the Eclipse debugger. |
You’ll see the server output in the Console :
That’s it, we now have the server up and running in Eclipse!
2.4. Importing the quickstarts into Eclipse
In order to import the quickstarts into Eclipse, you will need m2eclipse installed. If you have CodeReady Studio, then m2eclipse is already installed.
First, choose File → Import…:
Select Existing Maven Projects:
Click on Browse, and navigate to the quickstarts/
directory:
Finally, make sure all 4 quickstarts are found and selected, and click Finish:
Eclipse should now successfully import 4 projects:
It will take a short time to import the projects, as Maven needs to download the project’s dependencies from remote repositories.
2.5. Managing JBoss WildFly
Here we will quickly outline how you can access both the command line interface and the web management interface for managing JBoss WildFly. Detailed information for both can be found in the Administration and Configuration Guide for JBoss Enterprise Application Platform 6 or the Admin Guide for JBoss WildFly.
When the server is running, the web management interface can be accessed at http://localhost:9990/console. You can use the web management interface to create datasources, manage deployments and configure the server.
JBoss WildFly also comes with a command line interface. To run it on Linux, Unix or Mac, execute:
wildfly-11.x.x.x/bin/jboss-admin.sh --connect
Or, on Windows:
wildfly-11.x.x.x/bin/jboss-admin.bat --connect
Once started, type help to discover the commands available to you.
Throughout this guide we use the wildfly
maven plugin to deploy and undeploy applications. This plugin uses the Native Java Detyped Management API to communicate with the server. The Detyped API is used by management tools to control an entire domain of servers, and exposes only a small number of types, allowing for backwards and forwards compatibility.
3. CDI + Servlet: Helloworld quickstart
This quickstart shows you how to deploy a simple servlet to JBoss WildFly. The business logic is encapsulated in a service, which is provided as a CDI bean, and injected into the Servlet.
Contexts and Dependency Injection for Jakarta EE
CDI is a specification in Jakarta EE, inspired by JBoss Seam and Google Guice, and also drawing on lessons learned from frameworks such as Spring. It allows application developers to concentrate on developing their application logic by providing the ability to wire services together, and abstract out orthogonal concerns, all in a type safe manner. |
Switch to the quickstarts/helloworld
directory and instruct Maven to build and deploy the application:
mvn package wildfly:deploy
The quickstart uses a Maven plugin to deploy the application. The plugin requires JBoss WildFly to be running (you can find out how to start the server in Installing and starting the JBoss server on Linux, Unix or Mac OS X or Installing and starting the JBoss server on Windows).
Now, check if the application has deployed properly by clicking http://localhost:8080/wildfly-helloworld/HelloWorld. If you see a "Hello World" message it’s all working!
Should you wish to undeploy the quickstart, or redeploy after making some changes, it’s pretty easy:
|
It’s time to pull the covers back and dive into the internals of the quickstart.
3.1. Deploying the Helloworld quickstart using CodeReady Studio, or Eclipse with JBoss Tools
You may choose to deploy the quickstart using CodeReady Studio, or Eclipse with JBoss Tools. You’ll need to have JBoss WildFly started in the IDE (as described in Starting the JBoss server from JBDS or Eclipse with JBoss Tools) and to have imported the quickstarts into Eclipse (as described in Importing the quickstarts into Eclipse).
With the quickstarts imported, you can deploy the quickstart by right clicking on the wildfly-helloworld
project, and choosing Run As → Run On Server:
Make sure the correct server is selected, and hit Finish:
You should see the server start up (unless you already started it in Starting the JBoss server from JBDS or Eclipse with JBoss Tools) and the application deploy in the Console log
3.2. The helloworld quickstart in depth
The quickstart is very simple - all it does is print "Hello World" onto a web page.
The helloworld quickstart is comprised of a servlet and a CDI bean. We also include an empty beans.xml
file, which tells JBoss WildFly to look for beans in this application and to activate the CDI. beans.xml
is located in WEB-INF/
, which can be found in the src/main/webapp
directory. Also in this directory we include index.html
which uses a simple meta refresh to send the users browser to the Servlet, which is located at http://localhost:8080/wildfly-helloworld/HelloWorld.
All the configuration files for this quickstart are located in WEB-INF/
, which can be found in the src/main/webapp
directory.
Notice that we don’t even need a web.xml
!
Let’s start by taking a look at the servlet:
@SuppressWarnings("serial")
@WebServlet("/HelloWorld") (1)
public class HelloWorldServlet extends HttpServlet {
static String PAGE_HEADER =
"<html><head><title>helloworld</title></head><body>"; (2)
static String PAGE_FOOTER = "</body></html>";
@Inject
HelloService helloService; (3)
@Override
protected void doGet(HttpServletRequest req,
HttpServletResponse resp)
throws ServletException, IOException {
resp.setContentType("text/html");
PrintWriter writer = resp.getWriter();
writer.println(PAGE_HEADER);
writer.println("<h1>" +
helloService.createHelloMessage("World") + (4)
"</h1>");
writer.println(PAGE_FOOTER);
writer.close();
}
}
1 | If you’ve used Servlet before, then you’ll remember having to use xml to register your servlets. Fortunately, this is a thing of the past. Now all you need to do is add the @WebServlet annotation, and provide a mapping to a URL used to access the servlet. Much cleaner! |
2 | Every web page needs to be correctly formed HTML. We’ve created static Strings to hold the minimum header and footer to write out. |
3 | We inject the HelloService (a CDI bean) which generates the actual message. This allows to alter the implementation of HelloService at a later date without changing the view layer at all (assuming we don’t alter the API of HelloService ). |
4 | We call into the service to generate the message "Hello World", and write it out to the HTTP request. |
The package declaration and imports have been excluded from these listings. The complete listing is available in the quickstart source. |
Now we understand how the information is sent to the browser, let’s take a look at the service.
public class HelloService { String createHelloMessage(String name) { return "Hello " + name + "!"; } }
The service is very simple - no registration (XML or annotation) is required!
4. CDI + JSF: Numberguess quickstart
This quickstart shows you how to create and deploy a simple application to JBoss WildFly; the application does not persist any information. Information is displayed using a JSF view, and business logic is encapsulated in two CDI beans.
Switch to the quickstarts/numberguess
directory and instruct Maven to build and deploy the application:
mvn package wildfly:deploy
The quickstart uses a Maven plugin to deploy the application. The plugin requires JBoss WildFly to be running (you can find out how to start the server in Installing and starting the JBoss server on Linux, Unix or Mac OS X or Installing and starting the JBoss server on Windows).
Or you can start the server using an IDE, like CodeReady Studio.
Now, see if you can determine the most efficient approach to pinpoint the random number at the URL http://localhost:8080/wildfly-numberguess.
Should you wish to undeploy the quickstart, or redeploy after making some changes, it’s pretty easy:
|
It’s time to pull the covers back and dive into the internals of the quickstart.
4.1. Deploying the Numberguess quickstart using Eclipse
You may choose to deploy the quickstart using Eclipse. You’ll need to have JBoss WildFly started in Eclipse as described in Starting the JBoss server from JBDS or Eclipse with JBoss Tools) and to have imported the quickstarts into Eclipse (as described in Importing the quickstarts into Eclipse).
With the quickstarts imported, you can deploy the quickstart by right clicking on the wildfly-numberguess
project, and choosing Run As → Run On Server:
Make sure the correct server is selected, and hit Finish:
You should see the server start up (unless you already started it in Starting the JBoss server from JBDS or Eclipse with JBoss Tools) and the application deploy in the Console log:
4.2. The numberguess quickstart in depth
In the numberguess application you get 10 attempts to guess a number between 1 and 100. After each attempt, you’re told whether your guess was too high or too low.
The quickstart is comprised of a number of beans, configuration files and Facelets (JSF) views, packaged as a war module. Let’s start by examining the configuration files.
All the configuration files for this quickstart are located in WEB-INF/
, which can be found in the src/main/webapp
directory. First, we have the JSF 2.0 version of faces-config.xml
. A standardized version of Facelets is the default view handler in JSF 2.0, so there’s really nothing that we have to configure. WildFly goes above and beyond Jakarta EE here, and will automatically configure JSF for you if you include this file. Thus, the configuration consists of only the root element.
<faces-config version="2.0"
xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="
http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/web-facesconfig_2_0.xsd">
</faces-config>
There’s also an empty beans.xml
file, which tells WildFly to look for beans in this application and to activate the CDI.
Notice that we don’t even need a web.xml
!
Let’s take a look at the main JSF view, src/main/webapp/home.xhtml
.
JSF uses the .xhtml extension for source files, but serves up the rendered views with the .jsf extension. |
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<head>
<meta http-equiv="Content-Type" content="text/html;
charset=iso-8859-1" />
<title>numberguess</title>
</head>
<body>
<div id="content">
<h1>Guess a number...</h1>
<h:form id="numberGuess">
<!-- Feedback for the user on their guess -->
<div style="color: red"> (1)
<h:messages id="messages" globalOnly="false" />
<h:outputText id="Higher" value="Higher!"
rendered="#{game.number gt game.guess and game.guess ne 0}" />
<h:outputText id="Lower" value="Lower!"
rendered="#{game.number lt game.guess and game.guess ne 0}" />
</div>
<!-- Instructions for the user -->
<div> (2)
I'm thinking of a number between <span
id="numberGuess:smallest">#{game.smallest}</span> and <span
id="numberGuess:biggest">#{game.biggest}</span>. You have
#{game.remainingGuesses} guesses remaining.
</div>
<!-- Input box for the users guess, plus a button to submit, and reset -->
<!-- These are bound using EL to our CDI beans -->
<div>
Your guess: (3)
<h:inputText id="inputGuess" value="#{game.guess}"
required="true" size="3"
disabled="#{game.number eq game.guess}"
validator="#{game.validateNumberRange}" /> (4)
<h:commandButton id="guessButton" value="Guess"
action="#{game.check}"
disabled="#{game.number eq game.guess}" />
</div>
<div> (5)
<h:commandButton id="restartButton" value="Reset"
action="#{game.reset}" immediate="true" />
</div>
</h:form>
</div>
<br style="clear: both" />
</body>
</html>
1 | There are a number of messages which can be sent to the user, "Higher!" and "Lower!" |
2 | As the user guesses, the range of numbers they can guess gets smaller - this sentence changes to make sure they know the number range of a valid guess. |
3 | This input field is bound to a bean property using a value expression. |
4 | A validator binding is used to make sure the user doesn’t accidentally input a number outside of the range in which they can guess - if the validator wasn’t here, the user might use up a guess on an out of bounds number. |
5 | There must be a way for the user to send their guess to the server. Here we bind to an action method on the bean. |
The quickstart consists of 4 classes, the first two of which are qualifiers. First, there is the @Random
qualifier, used for injecting a random number:
A qualifier is used to disambiguate between two beans both of which are eligible for injection based on their type. For more, see the Weld Reference Guide. |
@Target({ TYPE, METHOD, PARAMETER, FIELD })
@Retention(RUNTIME)
@Documented
@Qualifier
public @interface Random {
}
There is also the @MaxNumber
qualifier, used for injecting the maximum number that can be injected:
@Retention(RUNTIME)
@Documented
@Qualifier
public @interface MaxNumber {
}
The application-scoped Generator
class is responsible for creating the random number, via a producer method. It also exposes the maximum possible number via a producer method:
@SuppressWarnings("serial")
@ApplicationScoped
public class Generator implements Serializable {
private java.util.Random random = new java.util.Random(System.currentTimeMillis());
private int maxNumber = 100;
java.util.Random getRandom() {
return random;
}
@Produces
@Random
int next() {
// a number between 1 and 100
return getRandom().nextInt(maxNumber - 1) + 1;
}
@Produces
@MaxNumber
int getMaxNumber() {
return maxNumber;
}
}
The Generator
is application scoped, so we don’t get a different random each time.
The final bean in the application is the session-scoped Game
class. This is the primary entry point of the application. It’s responsible for setting up or resetting the game, capturing and validating the user’s guess and providing feedback to the user with a FacesMessage
. We’ve used the post-construct lifecycle method to initialize the game by retrieving a random number from the @RandomInstance<Integer>
bean.
You’ll notice that we’ve also added the @Named
annotation to this class. This annotation is only required when you want to make the bean accessible to a JSF view via EL (i.e. #{game}
)
@SuppressWarnings("serial")
@Named
@SessionScoped
public class Game implements Serializable {
/**
* The number that the user needs to guess
*/
private int number;
/**
* The users latest guess
*/
private int guess;
/**
* The smallest number guessed so far (so we can track the valid guess range).
*/
private int smallest;
/**
* The largest number guessed so far
*/
private int biggest;
/**
* The number of guesses remaining
*/
private int remainingGuesses;
/**
* The maximum number we should ask them to guess
*/
@Inject
@MaxNumber
private int maxNumber;
/**
* The random number to guess
*/
@Inject
@Random
Instance<Integer> randomNumber;
public Game() {
}
public int getNumber() {
return number;
}
public int getGuess() {
return guess;
}
public void setGuess(int guess) {
this.guess = guess;
}
public int getSmallest() {
return smallest;
}
public int getBiggest() {
return biggest;
}
public int getRemainingGuesses() {
return remainingGuesses;
}
/**
* Check whether the current guess is correct, and update the biggest/smallest guesses as needed.
* Give feedback to the user if they are correct.
*/
public void check() {
if (guess > number) {
biggest = guess - 1;
} else if (guess < number) {
smallest = guess + 1;
} else if (guess == number) {
FacesContext.getCurrentInstance().addMessage(null, new FacesMessage("Correct!"));
}
remainingGuesses--;
}
/**
* Reset the game, by putting all values back to their defaults, and getting a new random number.
* We also call this method when the user starts playing for the first time using
* {@linkplain PostConstruct @PostConstruct} to set the initial values.
*/
@PostConstruct
public void reset() {
this.smallest = 0;
this.guess = 0;
this.remainingGuesses = 10;
this.biggest = maxNumber;
this.number = randomNumber.get();
}
/**
* A JSF validation method which checks whether the guess is valid. It might not be valid because
* there are no guesses left, or because the guess is not in range.
*
*/
public void validateNumberRange(FacesContext context, UIComponent toValidate, Object value) {
if (remainingGuesses <= 0) {
FacesMessage message = new FacesMessage("No guesses left!");
context.addMessage(toValidate.getClientId(context), message);
((UIInput) toValidate).setValid(false);
return;
}
int input = (Integer) value;
if (input < smallest || input > biggest) {
((UIInput) toValidate).setValid(false);
FacesMessage message = new FacesMessage("Invalid guess");
context.addMessage(toValidate.getClientId(context), message);
}
}
}
5. CDI + JPA + EJB + JTA + JSF: Greeter quickstart
This quickstart shows you how to create and deploy an application which persists information to a database to JBoss WildFly. Information is displayed using JSF views, business logic is encapsulated in CDI beans, information is persisted using JPA, and transactions can be controlled manually or using EJB.
Switch to the quickstarts/greeter
directory and instruct Maven to build and deploy the application:
mvn package wildfly:deploy
The quickstart uses a Maven plugin to deploy the application. The plugin requires JBoss WildFly to be running (you can find out how to start the server in Installing and starting the JBoss server on Linux, Unix or Mac OS X or Installing and starting the JBoss server on Windows).
Or you can start the server using an IDE, like CodeReady Studio. If you are using CodeReady Studio, you can deploy the quickstart by right clicking on the greeter
project, and choosing Run As → Run On Server:
Make sure the server is selected, and hit Finish:
You should see the server start up (unless you already started it in Starting the JBoss server from JBDS or Eclipse with JBoss Tools) and the application deploy in the Console log:
To use the application, visit http://localhost:8080/greeter/.
5.1. The greeter quickstart in depth
In the quickstart, all users are stored in an H2 database (an in-memory, embedded database provided out of the box in JBoss WildFly). Each user is stored as an entity, and entities are mapped to the database using JPA. By default, transactions are managed manually, using the JTA API. Optionally, you can use EJB to manage transactions (we’ll look at how to enable that later). We need a transaction in progress in order to read and write any entities.
The quickstart is comprised of two JSF views, an entity, and a number of CDI beans. Additionally, there are the usual configuration files in WEB-INF/
(which can be found in the src/main/webapp
directory). Here we find beans.xml
and faces-config.xml
which tell JBoss WildFly to enable CDI and JSF for the application. Notice that we don’t need a web.xml
. There are two new configuration files in WEB-INF/classes/META-INF
(which can be found in the src/main/resources
directory of the quickstart) — persistence.xml
, which sets up JPA, and import.sql
which Hibernate, the JPA provider in JBoss WildFly, will use to load the initial users into the application when the application starts.
persistence.xml
is pretty straight forward, and links JPA to a datasource:
<persistence version="2.0"
xmlns="http://java.sun.com/xml/ns/persistence"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="
http://java.sun.com/xml/ns/persistence
http://java.sun.com/xml/ns/persistence/persistence_2_0.xsd">
<persistence-unit name="primary"> (1)
<!-- If you are running in a production environment, add a
managed data source, this example data source is just
for development and testing! -->
<!-- The datasource is deployed as
WEB-INF/greeter-quickstart-ds.xml, you can find it in
the source at
src/main/webapp/WEB-INF/greeter-quickstart-ds.xml -->
<jta-data-source>java:jboss/datasources/GreeterQuickstartDS</jta-data-source> (2)
<properties>
<!-- Properties for Hibernate --> (3)
<property name="hibernate.hbm2ddl.auto"
value="create-drop" />
<property name="hibernate.show_sql"
value="false" />
</properties>
</persistence-unit>
</persistence>
1 | The persistence unit is given a name, so that the application can use multiple if needed. If only one is defined, JPA will automatically use it. |
2 | The persistence unit references a data source. Here we are using the built in, sample, data source. |
3 | JPA allows us to configure the JPA provider specific properties. Here we tell Hibernate to automatically create any needed tables when the application starts (and drop them when the application is stopped). |
JBoss WildFly ships with a
sample datasource |
Let’s take a look at the JSF views. First up is src/main/webapp/greet.xhtml
:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
"http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<ui:composition template="template.xhtml"> (1)
<ui:define name="content">
<h:messages /> (2)
<h:form id="greetForm">
<h:panelGrid columns="3">
<h:outputLabel for="username">Enter username:</h:outputLabel>
<h:inputText id="username"
value="#{greetController.username}" />
<h:message for="username" />
</h:panelGrid>
<h:commandButton id="greet" value="Greet!"
action="#{greetController.greet}" />
</h:form>
<h:outputText value="#{greetController.greeting}"
rendered="#{not empty greetController.greeting}" /> (3)
<br />
<h:link outcome="/create.xhtml" value="Add a new user" /> (4)
</ui:define>
</ui:composition>
</html>
1 | As we have multiple views in this application, we’ve created a template that defines the common elements. We’ll examine this next. Here we define the "content" section of the page, which will be inserted into the template. |
2 | We output any messages for the user at the top of the form, e.g. when a new user is created. |
3 | The greeting message is only rendered if there is a message. |
4 | We also a link to the page which allows a user to be added. |
Now let’s take a look at the template. It defines common elements for the page, and allows pages which use it to insert content in various places.
<!-- The template for our app, defines some regions -->
<head>
<meta http-equiv="Content-Type" content="text/html; charset=iso-8859-1" />
<title>greeter</title>
<ui:insert name="head" /> (1)
</head>
<body>
<div id="container">
<div id="header"></div>
<div id="sidebar"></div>
<div id="content">
<ui:insert name="content" /> (2)
</div>
<br style="clear: both" />
</div>
</body>
</html>
1 | The head, defined in case a page wants to add some content to the head of the page. |
2 | The content, defined by a page using this template, will be inserted here |
Finally, let’s take a look at the user management page. It provides a form to add users:
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml"
xmlns:ui="http://java.sun.com/jsf/facelets"
xmlns:h="http://java.sun.com/jsf/html"
xmlns:f="http://java.sun.com/jsf/core">
<ui:composition template="template.xhtml">
<ui:define name="content">
<h:messages />
<h:form>
<h:panelGrid columns="3">
<h:outputLabel for="username">Enter username:</h:outputLabel>
<h:inputText id="username" value="#{newUser.username}" />
<h:message for="username" />
<h:outputLabel for="firstName">Enter first name:</h:outputLabel>
<h:inputText id="firstName" value="#{newUser.firstName}" />
<h:message for="firstName" />
<h:outputLabel for="lastName">Enter last name:</h:outputLabel>
<h:inputText id="lastName" value="#{newUser.lastName}" />
<h:message for="lastName" />
</h:panelGrid>
<h:commandButton action="#{createController.create}"
value="Add User" />
</h:form>
<h:link outcome="/greet.xhtml">Greet a user!</h:link>
</ui:define>
</ui:composition>
</html>
The quickstart has one entity, which is mapped via JPA to the relational database:
@Entity (1)
public class User {
@Id (2)
@GeneratedValue
private Long id;
@Column(unique = true)
private String username;
private String firstName; (3)
private String lastName;
public Long getId() { (4)
return id;
}
public String getUsername() {
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getFirstName() {
return firstName;
}
public void setFirstName(String firstName) {
this.firstName = firstName;
}
public String getLastName() {
return lastName;
}
public void setLastName(String lastName) {
this.lastName = lastName;
}
}
1 | The @Entity annotation used on the class tells JPA that this class should be mapped as a table in the database. |
2 | Every entity requires an id, the @Id annotation placed on a field (or a JavaBean mutator/accessor) tells JPA that this property is the id. You can use a synthetic id, or a natural id (as we do here). |
3 | The entity also stores the real name of the user |
4 | As this is Java, every property needs an accessor/mutator! |
We use a couple of controller classes to back the JSF pages. First up is GreetController
which is responsible for getting the user’s real name from persistence layer, and then constructing the message.
@Named (1)
@RequestScoped (2)
public class GreetController {
@Inject
private UserDao userDao; (3)
private String username;
private String greeting;
public void greet() {
User user = userDao.getForUsername(username);
if (user != null) {
greeting = "Hello, " +
user.getFirstName() +
" " +
user.getLastName() +
"!";
} else {
greeting =
"No such user exists! Use 'emuster' or 'jdoe'";
}
}
public String getUsername() { (4)
return username;
}
public void setUsername(String username) {
this.username = username;
}
public String getGreeting() {
return greeting;
}
}
1 | The bean is given a name, so we can access it from JSF |
2 | The bean is request scoped, a different greeting can be made in each request |
3 | We inject the UserDao , which handles database abstraction |
4 | We need to expose JavaBean style mutators and accessors for every property the JSF page needs to access to |
The second controller class is responsible for adding a new user:
@Named (1)
@RequestScoped (2)
public class CreateController {
@Inject (3)
private FacesContext facesContext;
@Inject (4)
private UserDao userDao;
@Named (5)
@Produces
@RequestScoped
private User newUser = new User();
public void create() {
try {
userDao.createUser(newUser);
String message = "A new user with id " +
newUser.getId() +
" has been created successfully";
facesContext.addMessage(null, new FacesMessage(message));
} catch (Exception e) {
String message = "An error has occured while creating" +
" the user (see log for details)";
facesContext.addMessage(null, new FacesMessage(message));
}
}
}
1 | The bean is given a name, so we can access it from JSF |
2 | The bean is request scoped, a different user can be added in each request |
3 | We inject the FacesContext , to allow us to send messages to the user when the user is created, or if an error occurs |
4 | We inject the UserDao , which handles database abstraction |
5 | We expose a prototype user using a named producer, which allows us to access it from a JSF page |
Now that we have the controllers in place, let’s look at the most interesting part of the application, how we interact with the database. As we mentioned earlier, by default the application uses the JTA API to manually control transactions. To implement both approaches, we’ve defined a UserDao
interface, with two implementations, one of which (the EJB variant) is as an alternative which can be enabled via a deployment descriptor. If you were wondering why we "hid" the persistence logic in the UserDao
, rather than placing it directly in the controller classes, this is why!
Let’s first look at the interface, and the manual transaction control variant.
public interface UserDao {
User getForUsername(String username);
void createUser(User user);
}
The methods are fairly self explanatory, so let’s move on quickly to the implementation, ManagedBeanUserDao
:
public class ManagedBeanUserDao implements UserDao {
@Inject
private EntityManager entityManager; (1)
@Inject
private UserTransaction utx; (2)
public User getForUsername(String username) { (3)
try {
User user;
try {
utx.begin();
Query query = entityManager.createQuery("select u from User u where u.username = :username");
query.setParameter("username", username);
user = (User) query.getSingleResult();
} catch (NoResultException e) {
user = null;
}
utx.commit();
return user;
} catch (Exception e) {
try {
utx.rollback();
} catch (SystemException se) {
throw new RuntimeException(se);
}
throw new RuntimeException(e);
}
}
public void createUser(User user) { (4)
try {
try {
utx.begin();
entityManager.persist(user);
} finally {
utx.commit();
}
} catch (Exception e) {
try {
utx.rollback();
} catch (SystemException se) {
throw new RuntimeException(se);
}
throw new RuntimeException(e);
}
}
}
1 | We inject the entity manager. This was set up in persistence.xml . |
2 | We inject the UserTransaction , to allow us to programmatically control the transaction |
3 | The getUserForUsername method can check whether a user with a matching username and password exists, and return it if it does. |
4 | createUser persists a new user to the database. |
You’ve probably noticed two things as you’ve read through this. Firstly, that manually managing transactions is a real pain. Secondly, you may be wondering how the entity manager and the logger are injected. First, let’s tidy up the transaction manager, and use EJB to provide us with declarative transaction support.
The class EJBUserDao
provides this, and is defined as an alternative. Alternatives are disabled by default, and when enabled replace the original implementation. In order to enable this variant of UserDao
, edit beans.xml
and uncomment the alternative. Your beans.xml
should now look like:
<beans xmlns="http://java.sun.com/xml/ns/javaee"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="
http://java.sun.com/xml/ns/javaee
http://java.sun.com/xml/ns/javaee/beans_1_0.xsd">
<!-- Uncomment this alternative to see EJB declarative transactions in use -->
<alternatives>
<class>org.jboss.as.quickstarts.greeter.domain.EJBUserDao</class>
</alternatives>
</beans>
Now, let’s look at EJBUserDao
:
@Stateful
@Alternative
public class EJBUserDao implements UserDao {
@Inject
private EntityManager entityManager;
public User getForUsername(String username) {
try {
Query query = entityManager.createQuery("select u from User u where u.username = ?");
query.setParameter(1, username);
return (User) query.getSingleResult();
} catch (NoResultException e) {
return null;
}
}
public void createUser(User user) {
entityManager.persist(user);
}
}
Using declarative transaction management has allowed us to remove a third of the lines of code from the class, but more importantly emphasizes the functionality of the class. Much better!
Sharp eyed developers who are used to Jakarta EE will have noticed that we have added this EJB to a war. This is the key improvement offered in EJB 3.1. |
Finally, let’s take a look at the Resources
class, which provides resources such as the entity manager. CDI recommends using "resource producers", as we do in this quickstart, to alias resources to CDI beans, allowing for a consistent style throughout our application:
public class Resources {
// Expose an entity manager using the resource producer pattern
@SuppressWarnings("unused")
@PersistenceContext
@Produces
private EntityManager em; (1)
@Produces
Logger getLogger(InjectionPoint ip) { (2)
String category = ip.getMember()
.getDeclaringClass()
.getName();
return Logger.getLogger(category);
}
@Produces
FacesContext getFacesContext() { (3)
return FacesContext.getCurrentInstance();
}
}
1 | We use the "resource producer" pattern, from CDI, to "alias" the old fashioned @PersistenceContext injection of the entity manager to a CDI style injection. This allows us to use a consistent injection style (@Inject ) throughout the application. |
2 | We expose a JDK logger for injection. In order to save a bit more boiler plate, we automatically set the logger category as the class name! |
3 | We expose the FacesContext via a producer method, which allows it to be injected. If we were adding tests, we could also then mock it out. |
That concludes our tour of the greeter application!
CDI + JSF + EJB + JTA + Bean Validation + Jakarta REST : Kitchensink quickstart
This quickstart shows off all the new features of Jakarta EE, and makes a great starting point for your project.
Bean Validation
Bean Validation is a specification in Jakarta EE, inspired by Hibernate Validator. It allows application developers to specify constraints once (often in their domain model), and have them applied in all layers of the application, protecting data and giving useful feedback to users. |
Jakarta REST: The Java API for RESTful Web Services
Jakarta REST is a specification in Jakarta EE. It allows application developers to easily expose Java services as RESTful web services. |
Switch to the quickstarts/kitchensink
directory and instruct Maven to build and deploy the application:
mvn package wildfly:deploy
The quickstart uses a Maven plugin to deploy the application. The plugin requires JBoss WildFly to be running (you can find out how to start the server in Installing and starting the JBoss server on Linux, Unix or Mac OS X or Installing and starting the JBoss server on Windows).
Or you can start the server using an IDE, like Eclipse.
Now, check if the application has deployed properly by clicking http://localhost:8080/wildfly-kitchensink. If you see a splash page it’s all working!
Should you wish to undeploy the quickstart, or redeploy after making some changes, it’s pretty easy:
|
It’s time to pull the covers back and dive into the internals of the application.
Deploying the Kitchensink quickstart using CodeReady Studio, or Eclipse with JBoss Tools
You may choose to deploy the quickstart using CodeReady Studio, or Eclipse with JBoss Tools. You'll need to have JBoss WildFly started in the IDE (as described in <<GettingStarted-with_jboss_tools, Starting the JBoss server from JBDS or Eclipse with JBoss Tools>>) and to have imported the quickstarts into Eclipse (as described in <<GettingStarted-importing_quickstarts_into_eclipse,Importing the quickstarts into Eclipse>>). With the quickstarts imported, you can deploy the quickstart by right clicking on the `wildfly-kitchensink` project, and choosing _Run As -> Run On Server_: image:gfx/Eclipse_KitchenSink_Deploy_1.jpg[] Make sure the server is selected, and hit Finish: image:gfx/Eclipse_Deploy_2.jpg[] You should see the server start up (unless you already started it in <<GettingStarted-with_jboss_tools, Starting the JBoss server from JBDS or Eclipse with JBoss Tools>>) and the application deploy in the Console log: image:gfx/Eclipse_KitchenSink_Deploy_3.jpg[] The kitchensink quickstart in depth ----------------------------------- The kitchensink application shows off a number of Jakarta EE technologies such as CDI, JSF, EJB, JTA, Jakarta REST. It does this by providing a member registration database, available via JSF and Jakarta REST. As usual, let's start by looking at the necessary deployment descriptors. By now, we're very used to seeing `beans.xml` and `faces-config.xml` in `WEB-INF/` (which can be found in the `src/main/webapp` directory). Notice that, once again, we don't need a `web.xml`. There are two configuration files in `WEB-INF/classes/META-INF` (which can be found in the `src/main/resources` directory) — `persistence.xml`, which sets up JPA, and `import.sql` which Hibernate, the JPA provider in JBoss WildFly, will use to load the initial users into the application when the application starts. We discussed both of these files in detail in <<GreeterQuickstart-,the Greeter Quickstart>>, and these are largely the same. Next, let's take a look at the JSF view the user sees. As usual, we use a template to provide the sidebar and footer. This one lives in `src/main/webapp/WEB-INF/templates/default.xhtml`: .src/main/webapp/WEB-INF/templates/default.xhtml [source,html] ------------------------------------------------------------------------ <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Strict//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-strict.dtd"> <html xmlns="http://www.w3.org/1999/xhtml" xmlns:h="http://java.sun.com/jsf/html" xmlns:ui="http://java.sun.com/jsf/facelets"> <h:head> (1) <title>kitchensink</title> <meta http-equiv="Content-Type" content="text/html; charset=utf-8" /> <h:outputStylesheet name="css/screen.css" /> </h:head> <h:body> <div id="container"> <div class="dualbrand"> <img src="resources/gfx/dualbrand_logo.png" /> </div> <div id="content"> <ui:insert name="content"> (2) [Template content will be inserted here] </ui:insert> </div> <div id="aside"> (3) <p>Learn more about JBoss Enterprise Application Platform 6.</p> <ul> <li> <a href="http://red.ht/jbeap-6-docs"> Documentation </a> </li> <li> <a href="http://red.ht/jbeap-6"> Product Information </a> </li> </ul> <p>Learn more about JBoss WildFly.</p> <ul> <li> <a href="http://jboss.org/jdf/quickstarts/wildfly-quickstart/guide"> Getting Started Developing Applications Guide </a> </li> <li> <a href="http://jboss.org/jbossas"> Community Project Information </a> </li> </ul> </div> <div id="footer"> <p> This project was generated from a Maven archetype from JBoss.<br /> </p> </div> </div> </h:body> </html> ------------------------------------------------------------------------ <1> We have a common `<head>` element, where we define styles and more <2> The content is inserted here, and defined by views using this template <3> This application defines a common sidebar and footer, putting them in the template means we only have to define them once That leaves the main page, index.xhtml , in which we place the content unique to the main page: .src/main/webapp/index.xhtml [source,html] ------------------------------------------------------------------------ <?xml version="1.0" encoding="UTF-8"?> <ui:composition xmlns="http://www.w3.org/1999/xhtml" xmlns:ui="http://java.sun.com/jsf/facelets" xmlns:f="http://java.sun.com/jsf/core" xmlns:h="http://java.sun.com/jsf/html" template="/WEB-INF/templates/default.xhtml"> <ui:define name="content"> <h1>Welcome to JBoss!</h1> <h:form id="reg"> (1) <h2>Member Registration</h2> <p>Enforces annotation-based constraints defined on the model class.</p> <h:panelGrid columns="3" columnClasses="titleCell"> <h:outputLabel for="name" value="Name:" /> <h:inputText id="name" value="#{newMember.name}" /> (2) <h:message for="name" errorClass="invalid" /> <h:outputLabel for="email" value="Email:" /> <h:inputText id="email" value="#{newMember.email}" /> (2) <h:message for="email" errorClass="invalid" /> <h:outputLabel for="phoneNumber" value="Phone #:" /> <h:inputText id="phoneNumber" value="#{newMember.phoneNumber}" /> (2) <h:message for="phoneNumber" errorClass="invalid" /> </h:panelGrid> <p> <h:panelGrid columns="2"> <h:commandButton id="register" action="#{memberController.register}" value="Register" styleClass="register" /> <h:messages styleClass="messages" errorClass="invalid" infoClass="valid" warnClass="warning" globalOnly="true" /> </h:panelGrid> </p> </h:form> <h2>Members</h2> <h:panelGroup rendered="#{empty members}"> <em>No registered members.</em> </h:panelGroup> <h:dataTable var="_member" value="#{members}" rendered="#{not empty members}" styleClass="simpletablestyle"> (3) <h:column> <f:facet name="header">Id</f:facet> #{_member.id} </h:column> <h:column> <f:facet name="header">Name</f:facet> #{_member.name} </h:column> <h:column> <f:facet name="header">Email</f:facet> #{_member.email} </h:column> <h:column> <f:facet name="header">Phone #</f:facet> #{_member.phoneNumber} </h:column> <h:column> <f:facet name="header">REST URL</f:facet> <a href="#{request.contextPath}/rest/members/#{_member.id}"> /rest/members/#{_member.id} </a> </h:column> <f:facet name="footer"> REST URL for all members: <a href="#{request.contextPath}/rest/members"> /rest/members </a> </f:facet> </h:dataTable> </ui:define> </ui:composition> ------------------------------------------------------------------------ <1> The JSF form allows us to register new users. There should be one already created when the application started. <2> The application uses Bean Validation to validate data entry. The error messages from Bean Validation are automatically attached to the relevant field by JSF, and adding a messages JSF component will display them. <3> This application exposes REST endpoints for each registered member. The application helpfully displays the URL to the REST endpoint on this page. Next, let's take a look at the Member entity, before we look at how the application is wired together: .src/main/java/org/jboss/as/quickstarts/kitchensink/model/Member.java [source,java] ------------------------------------------------------------------------ SuppressWarnings("serial") @Entity (1) @XmlRootElement (2) @Table(uniqueConstraints = @UniqueConstraint(columnNames = "email")) public class Member implements Serializable { @Id @GeneratedValue private Long id; @NotNull @Size(min = 1, max = 25) @Pattern(regexp = "[A-Za-z ]*", message = "must contain only letters and spaces") (3) private String name; @NotNull @NotEmpty @Email (4) private String email; @NotNull @Size(min = 10, max = 12) @Digits(fraction = 0, integer = 12) (5) @Column(name = "phone_number") private String phoneNumber; public Long getId() { return id; } public void setId(Long id) { this.id = id; } public String getName() { return name; } public void setName(String name) { this.name = name; } public String getEmail() { return email; } public void setEmail(String email) { this.email = email; } public String getPhoneNumber() { return phoneNumber; } public void setPhoneNumber(String phoneNumber) { this.phoneNumber = phoneNumber; } } ------------------------------------------------------------------------ <1> As usual with JPA, we define that the class is an entity by adding @Entity <2> Members are exposed as a RESTful service using Jakarta REST. We can use JAXB to map the object to XML and to do this we need to add @XmlRootElement <3> Bean Validation allows constraints to be defined once (on the entity) and applied everywhere. Here we constrain the person's name to a certain size and regular expression <4> Hibernate Validator also offers some extra validations such as @Email <5> @Digits , @NotNull and @Size are further examples of constraints Let's take a look at `MemberRepository`, which is responsible for interactions with the persistence layer: .src/main/java/org/jboss/as/quickstarts/kitchensink/data/MemberRepository.java [source,java] ------------------------------------------------------------------------ @ApplicationScoped (1) public class MemberRepository { @Inject (2) private EntityManager em; public Member findById(Long id) { return em.find(Member.class, id); } public Member findByEmail(String email) { CriteriaBuilder cb = em.getCriteriaBuilder(); (3) CriteriaQuery<Member> c = cb.createQuery(Member.class); Root<Member> member = c.from(Member.class); c.select(member).where(cb.equal(member.get("email"), email)); return em.createQuery(c).getSingleResult(); } public List<Member> findAllOrderedByName() { CriteriaBuilder cb = em.getCriteriaBuilder(); CriteriaQuery<Member> criteria = cb.createQuery(Member.class); Root<Member> member = criteria.from(Member.class); criteria.select(member).orderBy(cb.asc(member.get("name"))); return em.createQuery(criteria).getResultList(); (4) } } ------------------------------------------------------------------------ <1> The bean is application scoped, as it is a singleton <2> The entity manager is injected, to allow interaction with JPA <3> The JPA criteria api is used to load a member by their unique identifier, their email address <4> The criteria api can also be used to load lists of entities Let's take a look at `MemberListProducer`, which is responsible for managing the list of registered members. .src/main/java/org/jboss/as/quickstarts/kitchensink/data/MemberListProducer.java [source,java] ------------------------------------------------------------------------ @RequestScoped (1) public class MemberListProducer { @Inject (2) private MemberRepository memberRepository; private List<Member> members; // @Named provides access the return value via the EL variable // name "members" in the UI (e.g. Facelets or JSP view) @Produces (3) @Named public List<Member> getMembers() { return members; } public void onMemberListChanged( (4) @Observes(notifyObserver = Reception.IF_EXISTS) final Member member) { retrieveAllMembersOrderedByName(); } @PostConstruct public void retrieveAllMembersOrderedByName() { members = memberRepository.findAllOrderedByName(); } } ------------------------------------------------------------------------ <1> This bean is request scoped, meaning that any fields (such as members ) will be stored for the entire request <2> The `MemberRepository` is responsible or interactions with the persistence layer <3> The list of members is exposed as a producer method, it's also available via EL <4> The observer method is notified whenever a member is created, removed, or updated. This allows us to refresh the list of members whenever they are needed. This is a good approach as it allows us to cache the list of members, but keep it up to date at the same time Let's now look at MemberRegistration, the service that allows us to create new members: .src/main/java/org/jboss/as/quickstarts/kitchensink/service/MemberRegistration.java [source,java] ------------------------------------------------------------------------ @Stateless (1) public class MemberRegistration { @Inject (2) private Logger log; @Inject private EntityManager em; @Inject private Event<Member> memberEventSrc; public void register(Member member) throws Exception { log.info("Registering " + member.getName()); em.persist(member); memberEventSrc.fire(member); (3) } } ------------------------------------------------------------------------ <1> This bean requires transactions as it needs to write to the database. Making this an EJB gives us access to declarative transactions - much simpler than manual transaction control! <2> Here we inject a JDK logger, defined in the `Resources` class <3> An event is sent every time a member is updated. This allows other pieces of code (in this quickstart the member list is refreshed) to react to changes in the member list without any coupling to this class. Now, let's take a look at the `Resources` class, which provides resources such as the entity manager. CDI recommends using "resource producers", as we do in this quickstart, to alias resources to CDI beans, allowing for a consistent style throughout our application: .src/main/java/org/jboss/as/quickstarts/kitchensink/util/Resources.java [source,java] ------------------------------------------------------------------------ public class Resources { // use @SuppressWarnings to tell IDE to ignore warnings about // field not being referenced directly @SuppressWarnings("unused") (1) @Produces @PersistenceContext private EntityManager em; @Produces (2) public Logger produceLog(InjectionPoint injectionPoint) { return Logger.getLogger(injectionPoint.getMember() .getDeclaringClass() .getName()); } @Produces (3) @RequestScoped public FacesContext produceFacesContext() { return FacesContext.getCurrentInstance(); } } ------------------------------------------------------------------------ <1> We use the "resource producer" pattern, from CDI, to "alias" the old fashioned `@PersistenceContext` injection of the entity manager to a CDI style injection. This allows us to use a consistent injection style (`@Inject`) throughout the application. <2> We expose a JDK logger for injection. In order to save a bit more boiler plate, we automatically set the logger category as the class name! <3> We expose the `FacesContext` via a producer method, which allows it to be injected. If we were adding tests, we could also then mock it out. If you want to define your own datasource, take a look at the link:http://docs.redhat.com/docs/en-US/JBoss_Enterprise_Application_Platform/6/html/Administration_and_Configuration_Guide/index.html[Administration and Configuration Guide for JBoss Enterprise Application Platform 6] or the link:https://docs.jboss.org/author/display/AS71/Getting+Started+Guide[Getting Started Guide]. Of course, we need to allow JSF to interact with the services. The `MemberController` class is responsible for this: .src/main/java/org/jboss/as/quickstarts/kitchensink/controller/MemberController.java [source,java] ------------------------------------------------------------------------ @Model (1) public class MemberController { @Inject (2) private FacesContext facesContext; @Inject (3) private MemberRegistration memberRegistration; @Produces (4) @Named private Member newMember; @PostConstruct (5) public void initNewMember() { newMember = new Member(); } public void register() throws Exception { try { memberRegistration.register(newMember); (6) FacesMessage m = new FacesMessage(FacesMessage.SEVERITY_INFO, "Registered!", "Registration successful"); facesContext.addMessage(null, m); (7) initNewMember(); (8) } catch (Exception e) { String errorMessage = getRootErrorMessage(e); FacesMessage m = new FacesMessage(FacesMessage.SEVERITY_ERROR, errorMessage, "Registration unsuccessful"); facesContext.addMessage(null, m); } } private String getRootErrorMessage(Exception e) { // Default to general error message that registration failed. String errorMessage = "Registration failed. See server log for more information"; if (e == null) { // This shouldn't happen, but return the default messages return errorMessage; } // Start with the exception and recurse to find the root cause Throwable t = e; while (t != null) { // Get the message from the Throwable class instance errorMessage = t.getLocalizedMessage(); t = t.getCause(); } // This is the root cause message return errorMessage; } } ------------------------------------------------------------------------ <1> The `MemberController` class uses the `@Model` stereotype, which adds `@Named` and `@RequestScoped` to the class <2> The `FacesContext` is injected, so that messages can be sent to the user <3> The `MemberRegistration` bean is injected, to allow the controller to interact with the database <4> The `Member` class is exposed using a named producer field, which allows access from JSF. Note that that the named producer field has dependent scope, so every time it is injected, the field will be read <5> The `@PostConstruct` annotation causes a new member object to be placed in the `newMember` field when the bean is instantiated <6> When the register method is called, the `newMember` object is passed to the persistence service <7> We also send a message to the user, to give them feedback on their actions <8> Finally, we replace the `newMember` with a new object, thus blanking out the data the user has added so far. This works as the producer field is dependent scoped Before we wrap up our tour of the kitchensink application, let's take a look at how the Jakarta REST endpoints are created. Firstly, `JaxRSActivator`, which extends `Application` and is annotated with `@ApplicationPath`, is the Jakarta EE "no XML" approach to activating Jakarta REST. .src/main/java/org/jboss/as/quickstarts/kitchensink/rest/JaxRsActivator.java [source,java] ------------------------------------------------------------------------ @ApplicationPath("/rest") public class JaxRsActivator extends Application { /* class body intentionally left blank */ } ------------------------------------------------------------------------ The real work goes in `MemberResourceRESTService`, which produces the endpoint: .src/main/java/org/jboss/as/quickstarts/kitchensink/rest/MemberResourceRESTService.java [source,java] ------------------------------------------------------------------------ @Path("/members") (1) @RequestScoped (2) public class MemberResourceRESTService { @Inject (3) private Logger log; @Inject (4) private Validator validator; @Inject (5) private MemberRepository repository; @Inject (6) private MemberRegistration registration; @GET (7) @Produces(MediaType.APPLICATION_JSON) public List<Member> listAllMembers() { return repository.findAllOrderedByName(); } @GET (8) @Path("/{id:[0-9][0-9]*}") @Produces(MediaType.APPLICATION_JSON) public Member lookupMemberById(@PathParam("id") long id) { Member member = repository.findById(id); if (member == null) { throw new WebApplicationException(Response.Status.NOT_FOUND); } return member; } /** * Creates a new member from the values provided. Performs * validation, and will return a Jakarta REST response with either * 200 ok, or with a map of fields, and related errors. */ @POST @Consumes(MediaType.APPLICATION_JSON) @Produces(MediaType.APPLICATION_JSON) public Response createMember(Member member) { (9) Response.ResponseBuilder builder = null; try { // Validates member using bean validation validateMember(member); (10) registration.register(member); (11) //Create an "ok" response builder = Response.ok(); } catch (ConstraintViolationException ce) { (12) //Handle bean validation issues builder = createViolationResponse( ce.getConstraintViolations()); } catch (ValidationException e) { //Handle the unique constrain violation Map<String, String> responseObj = new HashMap<String, String>(); responseObj.put("email", "Email taken"); builder = Response.status(Response.Status.CONFLICT) .entity(responseObj); } catch (Exception e) { // Handle generic exceptions Map<String, String> responseObj = new HashMap<String, String>(); responseObj.put("error", e.getMessage()); builder = Response.status(Response.Status.BAD_REQUEST) .entity(responseObj); } return builder.build(); } /** * <p> * Validates the given Member variable and throws validation * exceptions based on the type of error. If the error is * standard bean validation errors then it will throw a * ConstraintValidationException with the set of the * constraints violated. * </p> * <p> * If the error is caused because an existing member with the * same email is registered it throws a regular validation * exception so that it can be interpreted separately. * </p> * * @param member Member to be validated * @throws ConstraintViolationException * If Bean Validation errors exist * @throws ValidationException * If member with the same email already exists */ private void validateMember(Member member) throws ConstraintViolationException, ValidationException { //Create a bean validator and check for issues. Set<ConstraintViolation<Member>> violations = validator.validate(member); if (!violations.isEmpty()) { throw new ConstraintViolationException( new HashSet<ConstraintViolation<?>>(violations)); } //Check the uniqueness of the email address if (emailAlreadyExists(member.getEmail())) { throw new ValidationException("Unique Email Violation"); } } /** * Creates a Jakarta REST "Bad Request" response including a map of * all violation fields, and their message. This can then be * used by clients to show violations. * * @param violations A set of violations that needs to be * reported * @return Jakarta REST response containing all violations */ private Response.ResponseBuilder createViolationResponse (Set<ConstraintViolation<?>> violations) { log.fine("Validation completed. violations found: " + violations.size()); Map<String, String> responseObj = new HashMap<String, String>(); for (ConstraintViolation<?> violation : violations) { responseObj.put(violation.getPropertyPath().toString(), violation.getMessage()); } return Response.status(Response.Status.BAD_REQUEST) .entity(responseObj); } /** * Checks if a member with the same email address is already * registered. This is the only way to easily capture the * "@UniqueConstraint(columnNames = "email")" constraint from * the Member class. * * @param email The email to check * @return True if the email already exists, and false otherwise */ public boolean emailAlreadyExists(String email) { Member member = null; try { member = repository.findByEmail(email); } catch (NoResultException e) { // ignore } return member != null; } } ------------------------------------------------------------------------ <1> The `@Path` annotation tells Jakarta REST that this class provides a REST endpoint mapped to `rest/members` (concatenating the path from the activator with the path for this endpoint). <2> The bean is request scoped, as Jakarta REST interactions typically don't hold state between requests <3> Jakarta REST endpoints are CDI enabled, and can use CDI-style injection. <4> CDI allows us to inject a Bean Validation `Validator` instance, which is used to validate the POSTed member before it is persisted <5> `MemberRegistration` is injected to allow us to alter the member database <6> `MemberRepository` is injected to allow us to query the member database <7> The `listAllMembers()` method is called when the raw endpoint is accessed and offers up a list of endpoints. Notice that the object is automatically marshalled to JSON by RESTEasy (the Jakarta REST implementation included in JBoss WildFly). <8> The `lookupMemberById()` method is called when the endpoint is accessed with a member id parameter appended (for example `rest/members/1)`. Again, the object is automatically marshalled to JSON by RESTEasy. <9> `createMember()` is called when a POST is performed on the URL. Once again, the object is automatically unmarshalled from JSON. <10> In order to ensure that the member is valid, we call the `validateMember` method, which validates the object, and adds any constraint violations to the response. These can then be handled on the client side, and displayed to the user <11> The object is then passed to the `MemberRegistration` service to be persisted <12> We then handle any remaining issues with validating the object, which are raised when the object is persisted That concludes our tour of the kitchensink quickstart. If you would like to use this project as a basis for your own application, you can of course copy this application sources and modify it. = Creating your own application :Author: Pete Muir [[Archetype-]] What we didn't tell you about the kitchensink quickstart is that it is generated from a Maven archetype. Using this archetype offers you the perfect opportunity to generate your own project. You can create a project from the archetype using Red Hat CodeReady Studio, or Eclipse with JBoss Tools. First, open up _JBoss Central_, if it isn't already open. Hit _Cmd-3_ (Mac) or _Ctrl-3_ (Windows, Linux) and type _JBoss Central_: image:gfx/Eclipse_JBoss_Central_1.png[] You will now be shown _JBoss Central_, an excellent place to find about all things JBoss! image:gfx/Eclipse_JBoss_Central_2.png[] To create a new project, based on the kitchensink quickstart, click on _Create Projects | Jakarta EE Web Project_: image:gfx/Eclipse_JavaEEWebProject_1.png[] Red Hat CodeReady Studio will then check that you have the necessary pre-requisites to create the project. If you are using CodeReady, then you should, otherwise, JBoss Tools will help you install the necessary pre-requisites. See link:http://jboss.org/tools[JBoss Tools] for more information. Hit _Next >_. On the next screen you can enter a project name, package for sample code, and finally select a target runtime: image:gfx/Eclipse_JavaEEWebProject_2.png[] Finally, hit _Finish_. You'll be presented with the _New Project Example_ dialog, in which you can simply hit _Finish_: image:gfx/Eclipse_JavaEEWebProject_3.png[] You should now have a brand new project: image:gfx/Eclipse_JavaEEWebProject_4.png[] Enjoy! To use the archetype to generate a new project, you should run: mvn archetype:generate \ -DarchetypeArtifactId=jboss-javaee7-webapp-archetype \ -DarchetypeGroupId=org.jboss.spec.archetypes \ -DarchetypeVersion=7.1.1.CR2 \ Maven will download the archetype and it's dependencies, and ask you some questions: ------------------------------------------------------------------------ $ > mvn archetype:generate \ -DarchetypeArtifactId=jboss-javaee7-webapp-archetype \ -DarchetypeGroupId=org.jboss.spec.archetypes \ -DarchetypeVersion=7.1.1.CR2 [INFO] Scanning for projects... [INFO] [INFO] ------------------------------------------------------------------------ [INFO] Building Maven Stub Project (No POM) 1 [INFO] ------------------------------------------------------------------------ [INFO] ......... Define value for property 'groupId': : com.acme.corp (1) Define value for property 'artifactId': : acme-sales (2) Define value for property 'version': 1.0-SNAPSHOT: : (3) Define value for property 'package': com.acme.corp: : (4) [INFO] Using property: name = Jakarta EE webapp project (5) Confirm properties configuration: groupId: com.acme.corp artifactId: acme-sales version: 1.0-SNAPSHOT package: com.acme.corp name: Jakarta EE webapp project Y: : [WARNING] CP Don't override file /Users/pmuir/tmp/acme-sales/.settings/org.eclipse.jdt.apt.core.prefs [INFO] ------------------------------------------------------------------------ [INFO] BUILD SUCCESS [INFO] ------------------------------------------------------------------------ [INFO] Total time: 14.774s [INFO] Finished at: Mon Jun 06 18:53:38 BST 2011 [INFO] Final Memory: 7M/125M [INFO] ------------------------------------------------------------------------ $ > ------------------------------------------------------------------------ <1> Enter the groupId you wish to use <2> Enter the artifactId you wish to use <3> Enter the version you wish to use, or just hit Enter if you wish to accept the default 1.0-SNAPSHOT <4> Enter the java package you wish to use, or just hit Enter if you wish to accept the default (which is copied from groupId ). <5> Finally, if you are happy with your choices, hit Enter and Maven will generate the project for you. And that's it, you now have a brand new project with the same functionality as `kitchensink`, but customized with your details. [IMPORTANT] ======================================================================== The archetype contains some sample code to get you started. If you would prefer a blank canvas, with only a project skeleton, then use `jboss-javaee7-webapp-blank-archetype` as your archetype id. ======================================================================== [IMPORTANT] .Prefer Enterprise Applications (EARs)? ======================================================================== The archetype generates a WAR project. With Jakarta EE, you can include EJBs in your WAR, meaning you won't need an EAR until you need to divide your code into modules. If you would like to create an EAR based project then use `jboss-javaee7-webapp-ear-archetype` as your archetype id (or if you want a blank EAR, then `jboss-javaee7-webapp-ear-blank-archetype`). ======================================================================== = More Resources [cols=","] |======================================================================= | link:Getting_Started_Guide{outfilesuffix}[Getting Started Guide] |The Getting Started Guide covers topics such as server layout (what you can configure where), data source definition, and using the web management interface. |======================================================================= :leveloffset: +1 [[Developing_Jakarta_Faces_Project_Using,_Maven_and_IntelliJ]] = Developing Jakarta Faces Project Using JBoss AS7, Maven and IntelliJ JBoss AS7 is a very 'modern' application server that has very fast startup speed. So it's an excellent container to test your Jakarta Faces project. In this article, I'd like to show you how to use AS7, maven and IntelliJ together to develop your Jakarta Faces project. In this article I'd like to introduce the following things: * Create a project using Maven * Add Jakarta Faces into project * Writing Code * Add JBoss AS 7 deploy plugin into project * Deploy project to JBoss AS 7 * Import project into IntelliJ * Add IntelliJ Jakarta Faces support to project * Add JBoss AS7 to IntelliJ * Debugging project with IntelliJ and AS7 I won't explain many basic concepts about AS7, maven and IntelliJ in this article because there are already many good introductions on these topics. So before doing the real work, there some preparations should be done firstly: *Download JBoss AS7* It could be downloaded from here: https://www.jboss.org/jbossas/downloads/ Using the latest release would be fine. When I'm writing this article the latest version is 7.1.1.Final. *Install Maven* Please make sure you have maven installed on your machine. Here is my environment: [source,options="nowrap"] ---- weli@power:~$ mvn -version Apache Maven 3.0.3 (r1075438; 2011-03-01 01:31:09+0800) Maven home: /usr/share/maven Java version: 1.6.0_33, vendor: Apple Inc. Java home: /System/Library/Java/JavaVirtualMachines/1.6.0.jdk/Contents/Home Default locale: en_US, platform encoding: MacRoman OS name: "mac os x", version: "10.8", arch: "x86_64", family: "mac" ---- *Get IntelliJ* In this article I'd like to use IntelliJ Ultimate Edition as the IDE for development, it's a commercial software and can be downloaded from: https://www.jetbrains.com/idea/ The version I'm using is IntelliJ IDEA Ultimate 11.1 After all of these prepared, we can dive into the real work: [[create-a-project-using-maven]] == Create a project using Maven Use the following maven command to create a web project: [source,options="nowrap"] ---- mvn archetype:create -DarchetypeGroupId=org.apache.maven.archetypes \ -DarchetypeArtifactId=maven-archetype-webapp \ -DarchetypeVersion=1.0 \ -DgroupId=net.bluedash \ -DartifactId=jsfdemo \ -Dversion=1.0-SNAPSHOT ---- If everything goes fine maven will generate the project for us: image:images/jsf/8108c4f111aab2c3465472eb84cf1d9b7cf912d0.jpg[alt=jsf/8108c4f111aab2c3465472eb84cf1d9b7cf912d0.jpg] The contents of the project is shown as above. [[add-Jakarta-Faces-into-project]] == Add Jakarta Faces into project The Jakarta Faces library is now included in maven repo, so we can let maven to manage the download for us. First is to add repository into our pom.xml: [source,java,options="nowrap"] ---- <repository> <id>jvnet-nexus-releases</id> <name>jvnet-nexus-releases</name> <url>https://maven.java.net/content/repositories/releases/</url> </repository> ---- Then we add Jakarta Faces dependency into pom.xml: [source,xml,options="nowrap"] ---- <dependency> <groupId>jakarta.faces</groupId> <artifactId>jakarta.faces-api</artifactId> <version>4.0.1</version> <scope>provided</scope> </dependency> ---- Please note the 'scope' is 'provided', because we don't want to bundle the jsf.jar into the war produced by our project later, as JBoss AS7 already have Jakarta Faces bundled in. Then we run 'mvn install' to update the project, and maven will download jsf-api for us automatically. [[writing-code]] == Writing Code Writing Jakarta Faces code in this article is trivial, so I've put written a project called 'jsfdemo' onto github: https://github.com/liweinan/jsfdemo Please clone this project into your local machine, and import it into IntelliJ following the steps described as above. [[add-jboss-as-7-deploy-plugin-into-project]] == Add JBoss AS 7 deploy plugin into project JBoss AS7 has provide a set of convenient maven plugins to perform daily tasks such as deploying project into AS7. In this step let's see how to use it in our project. We should put AS7's repository into pom.xml: [source,xml,options="nowrap"] ---- <repository> <id>jboss-public-repository-group</id> <name>JBoss Public Repository Group</name> <url>https://repository.jboss.org/nexus/content/groups/public/</url> <layout>default</layout> <releases> <enabled>true</enabled> <updatePolicy>never</updatePolicy> </releases> <snapshots> <enabled>true</enabled> <updatePolicy>never</updatePolicy> </snapshots> </repository> ---- And also the plugin repository: [source,java,options="nowrap"] ---- <pluginRepository> <id>jboss-public-repository-group</id> <name>JBoss Public Repository Group</name> <url>https://repository.jboss.org/nexus/content/groups/public/</url> <releases> <enabled>true</enabled> </releases> <snapshots> <enabled>true</enabled> </snapshots> </pluginRepository> ---- And put jboss deploy plugin into 'build' section: [source,java,options="nowrap"] ---- <plugin> <groupId>org.jboss.as.plugins</groupId> <artifactId>jboss-as-maven-plugin</artifactId> <executions> <execution> <phase>package</phase> <goals> <goal>deploy</goal> </goals> </execution> </executions> </plugin> ---- I've put the final version pom.xml here to check whether your modification is correct: https://github.com/liweinan/jsfdemo/blob/master/pom.xml Now we have finished the setup work for maven. [[deploy-project-to-jboss-as-7]] == Deploy project to JBoss AS 7 To deploy the project to JBoss AS7, we should start AS7 firstly. In JBoss AS7 directory, run following command: [source,java,options="nowrap"] ---- bin/standalone.sh ---- AS7 should start in a short time. Then let's go back to our project directory and run maven command: [source,java,options="nowrap"] ---- mvn -q jboss-as:deploy ---- Maven will use some time to download necessary components for a while, so please wait patiently. After a while, we can see the result: image:images/jsf/97d781c6be9db755aef80a110f1d9b29590610d6.jpg[alt=jsf/97d781c6be9db755aef80a110f1d9b29590610d6.jpg] And if you check the console output of AS7, you can see the project is deployed: image:images/jsf/2._java.jpg[alt=jsf/2._java.jpg] Now we have learnt how to create a Jakarta Faces project and deploy it to AS7 without any help from graphical tools. Next let's see how to use IntelliJ IDEA to go on developing/debugging our project. [[import-project-into-intellij]] == Import project into IntelliJ Now it's time to import the project into IntelliJ. Now let's open IntelliJ, and choose 'New Project...': image:images/jsf/05222f3059e387df96ce04d2aea156c82af15096.jpg[alt=jsf/05222f3059e387df96ce04d2aea156c82af15096.jpg] The we choose 'Import project from external model': image:images/jsf/d68a0cdbc8c90db3db8af998f34616f73c7fe809.jpg[alt=jsf/d68a0cdbc8c90db3db8af998f34616f73c7fe809.jpg] Next step is choosing 'Maven': image:images/jsf/0b3d1cb5794fb54a2465da93648b5a0d1a6643f3.jpg[alt=jsf/0b3d1cb5794fb54a2465da93648b5a0d1a6643f3.jpg] Then IntelliJ will ask you the position of the project you want to import. In 'Root directory' input your project's directory and leave other options as default: image:images/jsf/2f192d02993248c97e2ac42ea8f3105d855e5cdf.jpg[alt=jsf/2f192d02993248c97e2ac42ea8f3105d855e5cdf.jpg] For next step, just click 'Next': image:images/jsf/3a3ee36eb581930822c4a66362795345f5d2f9a7.jpg[alt=jsf/3a3ee36eb581930822c4a66362795345f5d2f9a7.jpg] Finally click 'Finish': image:images/jsf/91e40cd0b1545cff4622857d6dc9959f96faf056.jpg[alt=jsf/91e40cd0b1545cff4622857d6dc9959f96faf056.jpg] Hooray! We've imported the project into IntelliJ now icon:smile-o[role="yellow"] [[adding-intellij-Jakarta-Faces-support-to-project]] == Adding IntelliJ Jakarta Faces support to project Let's see how to use IntelliJ and AS7 to debug the project. First we need to add 'Jakarta Faces' facet into project. Open project setting: image:images/jsf/8b8d0051f4f15033f17cb859c65f2d8481914678.jpg[alt=jsf/8b8d0051f4f15033f17cb859c65f2d8481914678.jpg] Click on 'Facets' section on left; Select 'Web' facet that we already have, and click the '+' on top, choose 'Jakarta Faces': image:images/jsf/e6947b84a56a698ca1392a440081bddfb5cae284.jpg[alt=jsf/e6947b84a56a698ca1392a440081bddfb5cae284.jpg] Select 'Web' as parent facet: image:images/jsf/6b2296be1bb2d8a81952caef0f025a139a39b381.jpg[alt=jsf/6b2296be1bb2d8a81952caef0f025a139a39b381.jpg] Click 'Ok': image:images/jsf/9988c572bad281146f405e9287f645a3da201885.jpg[alt=jsf/9988c572bad281146f405e9287f645a3da201885.jpg] Now we have enabled IntelliJ's Jakarta Faces support for project. [[add-jboss-as7-to-intellij]] == Add JBoss AS7 to IntelliJ Let's add JBoss AS7 into IntelliJ and use it to debug our project. First please choose 'Edit Configuration' in menu tab: image:images/jsf/dc0550785aae11f9d3eb439fdc0c51069affd25d.jpg[alt=jsf/dc0550785aae11f9d3eb439fdc0c51069affd25d.jpg] Click '+' and choose 'JBoss Server' -> 'Local': image:images/jsf/1231420c938f087030cb3dcd37237b5585beb154.jpg[alt=jsf/1231420c938f087030cb3dcd37237b5585beb154.jpg] Click 'configure': image:images/jsf/d7e6ab58230b2d31fdcd8fd5f14cd4eb47b05f64.jpg[alt=jsf/d7e6ab58230b2d31fdcd8fd5f14cd4eb47b05f64.jpg] and choose your JBoss AS7: image:images/jsf/f7b29ac8009f04fc7f209222ced0bcf54f4b8d9a.jpg[alt=jsf/f7b29ac8009f04fc7f209222ced0bcf54f4b8d9a.jpg] Now we need to add our project into deployment. Click the 'Deployment' tab: image:images/jsf/6802fb7e29283d0e064a7cc4466b918995ba5645.jpg[alt=jsf/6802fb7e29283d0e064a7cc4466b918995ba5645.jpg] Choose 'Artifact', and add our project: image:images/jsf/359484b8f6f2c655d94132e9cb6f9dbe5a058656.jpg[alt=jsf/359484b8f6f2c655d94132e9cb6f9dbe5a058656.jpg] Leave everything as default and click 'Ok', now we've added JBoss AS7 into IntelliJ [[debugging-project-with-intellij-and-as7]] == Debugging project with IntelliJ and AS7 Now comes the fun part. To debug our project, we cannot directly use the 'debug' feature provided by IntelliJ right now(maybe in the future version this problem could be fixed). So now we should use the debugging config provided by AS7 itself to enable JPDA feature, and then use the remote debug function provided by IntelliJ to get things done. Let's dive into the details now: First we need to enable JPDA config inside AS7, open 'bin/standalone.conf' and find following lines: [source,java,options="nowrap"] ---- # Sample JPDA settings for remote socket debugging #JAVA_OPTS="$JAVA_OPTS -Xrunjdwp:transport=dt_socket,address=8787,server=y,suspend=n" ---- Enable the above config by removing the leading hash sign: [source,java,options="nowrap"] ---- # Sample JPDA settings for remote socket debugging JAVA_OPTS="$JAVA_OPTS -Xrunjdwp:transport=dt_socket,address=8787,server=y,suspend=n" ---- [IMPORTANT] With WildFly you can directly start the server in debug mode: [source,java,options="nowrap"] ---- bin/standalone.sh --debug --server-config=standalone.xml ---- Now we start AS7 in IntelliJ: image:images/jsf/52369d67f9117c924213de24dd6642b48e47a436.png[alt=jsf/52369d67f9117c924213de24dd6642b48e47a436.png] Please note we should undeploy the existing 'jsfdemo' project in AS7 as we've added by maven jboss deploy plugin before. Or AS7 will tell us there is already existing project with same name so IntelliJ could not deploy the project anymore. If the project start correctly we can see from the IntelliJ console window, and please check the debug option is enabled: image:images/jsf/eaac5cb1a836809ab29513346b527fe051b7c7ac.png[alt=jsf/eaac5cb1a836809ab29513346b527fe051b7c7ac.png] Now we will setup the debug configuration, click 'debug' option on menu: image:images/jsf/b8323caf6980c40c3d635db5e308b03847618d06.jpg[alt=jsf/b8323caf6980c40c3d635db5e308b03847618d06.jpg] Choose 'Edit Configurations': image:images/jsf/8327bbe0e83cb7170dd84767631c98956e91c42c.jpg[alt=jsf/8327bbe0e83cb7170dd84767631c98956e91c42c.jpg] Then we click 'Add' and choose Remote: image:images/jsf/7103da6b6323e515a03a04cafe111aa7c6b3169d.jpg[alt=jsf/7103da6b6323e515a03a04cafe111aa7c6b3169d.jpg] Set the 'port' to the one you used in AS7 config file 'standalone.conf': image:images/jsf/30bbef45137c7d45ae300ba8d551423d1feefc96.png[alt=jsf/30bbef45137c7d45ae300ba8d551423d1feefc96.png] Leave other configurations as default and click 'Ok'. Now we need to set breakpoints in project, let's choose TimeBean.java and set a breakpoint on 'getNow()' method by clicking the left side of that line of code: image:images/jsf/a96b7d32e04aa67956bd00a187f09b75a5af241e.jpg[alt=jsf/a96b7d32e04aa67956bd00a187f09b75a5af241e.jpg] Now we can use the profile to do debug: image:images/jsf/5ea6987d1635c2c58d3ccdb1f5718f29d6a0fac3.png[alt=jsf/5ea6987d1635c2c58d3ccdb1f5718f29d6a0fac3.png] If everything goes fine we can see the console output: image:images/jsf/1096ebbbf2b29e694e300e02a48d0fa4207cb746.jpg[alt=jsf/1096ebbbf2b29e694e300e02a48d0fa4207cb746.jpg] Now we go to web browser and see our project's main page, try to click on 'Get current time': image:images/jsf/5ad5d0216d3326e9bc29705042db59f11c3c1e70.png[alt=jsf/5ad5d0216d3326e9bc29705042db59f11c3c1e70.png] Then IntelliJ will popup and the code is pausing on break point: image:images/jsf/2499d43c0dce2cab72ba472c8452a2b57999ac84.jpg[alt=jsf/2499d43c0dce2cab72ba472c8452a2b57999ac84.jpg] And we could inspect our project now. [[conclusion]] == Conclusion In this article I've shown to you how to use maven to create a project using Jakarta Faces and deploy it in JBoss AS7, and I've also talked about the usage of IntelliJ during project development phase. Hope the contents are practical and helpful to you icon:smile-o[role="yellow"] [[references]] == References * _https://developer.jboss.org/wiki/JBossAS7UsingJPDAToDebugTheASSourceCode[JBoss AS7: Using JPDA to debug the AS source code]_ * _https://developer.jboss.org/wiki/MavenGettingStarted-Developers[Maven Getting Started - Developers]_ * _https://blog.v-s-f.co.uk/2010/09/jsf-2-1-project-using-eclipse-and-maven-2/[JSF 2.1 project using Eclipse and Maven 2:http]_ * _https://www.amazon.com/Practical-RichFaces-Max-Katz/dp/1430234490/ref=dp_ob_title_bk[Practical RichFaces]_ * _https://javaserverfaces.java.net/download.html[Oracle Mojarra JavaServer Faces]_ * _https://github.com/jbossas/jboss-as-maven-plugin[JBoss AS7 Maven Plugin]_ [[Getting_Started_Developing_Applications_Presentation_Demo]] = Getting Started Developing Applications Presentation & Demo This document is a "script" for use with the quickstarts associated with the link:Getting_Started_Developing_Applications_Guide.html[Getting Started Developing Applications Guide]. It can be used as the basis for demoing/explaining the Jakarta EE programming model with JBoss AS 7. There is an associated presentation – JBoss AS - Getting Started Developing Applications – which can be used to introduce the Jakarta EE ecosystem. The emphasis here is on the programming model, not on OAM/dev-ops, performance etc. [[prerequisites-for-using-the-script]] == Prerequisites for using the script * JBoss AS 7 downloaded and installed * Eclipse Indigo with m2eclipse and JBoss Tools installed * The quickstarts downloaded and imported into Eclipse * Make sure `$JBOSS_HOME` is set. * Make sure `src/test/resources/arquillian.xml` has the correct path to your JBoss AS install for kitchensink * Make sure your font size is set in Eclipse so everyone can read the text! [[import-examples-into-eclipse-and-set-up-jboss-as]] == Import examples into Eclipse and set up JBoss AS TODO [[the-helloworld-quickstart]] == The Helloworld Quickstart [[introduction-1]] === Introduction This quickstart is extremely basic, and is really useful for nothing more than showing than the app server is working properly, and our deployment mechanism is working. We recommend you use this quickstart to demonstrate the various ways you can deploy apps to JBoss AS 7. [[using-maven]] === Using Maven . Start JBoss AS 7 from the console + [source,options="nowrap"] ---- $ JBOSS_HOME/bin/standalone.sh ---- . Deploy the app using Maven + [source,options="nowrap"] ---- $ mvn clean package jboss-as:deploy ---- + [IMPORTANT] ==== The quickstarts use the jboss-as maven plugin to deploy and undeploy applications. This plugin uses the JBoss AS Native Java Detyped Management API to communicate with the server. The Detyped API is used by management tools to control an entire domain of servers, and exposes only a small number of types, allowing for backwards and forwards compatibility. ==== . Show the app has deployed in the terminal. + Visit http://localhost:8080/jboss-as-helloworld . Undeploy the app using Maven + [source,options="nowrap"] ---- $ mvn jboss-as:undeploy ---- [[using-the-command-line-interface-cli]] === Using the Command Line Interface (CLI) . Start JBoss AS 7 from the console (if not already running) + [source,options="nowrap"] ---- $ JBOSS_HOME/bin/standalone.sh ---- . Build the war + [source,options="nowrap"] ---- $ mvn clean package ---- . Start the CLI + [source,options="nowrap"] ---- $ JBOSS_HOME/bin/jboss-admin.sh --connect ---- + [IMPORTANT] + The command line also uses the Deptyped Management API to communicate with the server. It's designed to be as "unixy" as possible, allowing you to "cd" into nodes, with full tab completion etc. The CLI allows you to deploy and undeploy applications, create Jakarta Messaging queues, topics etc., create datasources (normal and XA). It also fully supports the domain node. . Deploy the app + [source,options="nowrap"] ---- $ deploy target/jboss-as-helloworld.war ---- . Show the app has deployed + [source,java,options="nowrap"] ---- $ undeploy jboss-as-helloworld.war ---- [[using-the-web-management-interface]] === Using the web management interface . Start JBoss AS 7 from the console (if not already running) + [source,options="nowrap"] ---- $ JBOSS_HOME/bin/standalone.sh ---- . Build the war + [source,options="nowrap"] ---- $ mvn clean package ---- . Open up the web management interface http://localhost:9990/console + [IMPORTANT] ==== The web management interface offers the same functionality as the CLI (and again uses the Detyped Management API), but does so using a pretty GWT interface! You can set up virtual servers, interrogate sub systems and more. ==== . Navigate `Manage Deployments -> Add content`. Click on choose file and locate `helloworld/target/jboss-as-helloworld.war`. . Click `Next` and `Finish` to upload the war to the server. . Now click `Enable` and `Ok` to start the application . Switch to the console to show it deployed . Now click `Remove` [[using-the-filesystem]] === Using the filesystem . Start JBoss AS 7 from the console (if not already running) + [source,options="nowrap"] ---- $ JBOSS_HOME/bin/standalone.sh ---- . Build the war + [source,options="nowrap"] ---- $ mvn clean package ---- + [IMPORTANT] ==== Of course, you can still use the good ol' file system to deploy. Just copy the file to `$JBOSS_HOME/standalone/deployments`. ==== . Copy the war + [source,options="nowrap"] ---- $ cp target/jboss-as-helloworld.war $JBOSS_HOME/standalone/deployments ---- . Show the war deployed + [IMPORTANT] ==== The filesystem deployment uses marker files to indicate the status of a deployment. As this deployment succeeded we get a `$JBOSS_HOME/standalone/deployments/jboss-as-helloworld.war.deployed` file. If the deployment failed, you would get a `.failed` file etc. ==== . Undeploy the war + [source,options="nowrap"] ---- rm $JBOSS_HOME/standalone/deployments/jboss-as-helloworld.war.deployed ---- . Show the deployment stopping! . Start and stop the app server, show that the deployment really is gone! + [IMPORTANT] ==== This gives you much more precise control over deployments than before ==== [[using-eclipse]] === Using Eclipse . Add a JBoss AS server .. Bring up the Server view .. Right click in it, and choose `New -> Server` . Choose JBoss AS 7.0 and hit Next .. Locate the server on your disc .. Hit Finish . Start JBoss AS in Eclipse .. Select the server .. Click the Run button .. Deploy the app . right click on the app, choose `Run As -> Run On Server` .. Select the AS 7 instance you want to use .. Hit finish . Load the app at http://localhost:8080/jboss-as-helloworld [[digging-into-the-app]] === Digging into the app . Open up the helloworld quickstart in Eclipse, and open up `src/main/webapp`. . Point out that we don't require a `web.xml` anymore! . Show `beans.xml` and explain it's a marker file used to JBoss AS to enable CDI (open it, show that it is empty) . Show `index.html`, and explain it is just used to kick the user into the app (open it, show the meta-refresh) . Open up the `pom.xm` - and emphasise that it's pretty simple. .. There is no parent pom, everything for the build is *here* .. Show that we are enabling the JBoss Maven repo - explain you can do this in your POM or in system wide ( `settings.xml`) .. Show the `dependencyManagement` section. Here we import the JBoss AS 7 Web Profile API. Explain that this gives you all the versions for all of the JBoss AS 7 APIs that are in the web profile. Explain we could also depend on this directly, which would give us the whole set of APIs, but that here we've decided to go for slightly tighter control and specify each dependency ourselves .. Show the import for CDI, JSR-250 and Servlet API. Show that these are all provided - we are depending on build in server implementations, not packaging this stuff! .. Show the plugin sections - nothing that exciting here, the war plugin is out of date and requires you to provide `web.xml` icon:smile-o[role="yellow"] , configure the JBoss AS Maven Plugin, set the Java version to 6. . Open up `src/main/java` and open up the `HelloWorldServlet`. .. Point out the `@WebServlet` - explain this one annotation removes about 8 lines of XML - no need to separately map a path either. This is much more refactor safe .. Show that we can inject services into a Servlet .. Show that we use the service (line 41) + #Cmd-click on `HelloService` .. This is a CDI bean - very simple, no annotations required! .. Explain injection ... Probably used to string based bean resolution ... This is typesafe (refactor safe, take advantage of the compiler and the IDE - we just saw that!) ... When CDI needs to inject something, the first thing it looks at is the type - and if the type of the injection point is assignable from a bean, CDI will inject that bean [[the-numberguess-quickstart]] == The numberguess quickstart [[introduction-2]] === Introduction This quickstart adds in a "complete" view layer into the mix. Jakarta EE ships with a Jakarta Faces. Jakarta Faces is a server side rendering, component orientated framework, where you write markup using an HTML like language, adding in dynamic behavior by binding components to beans in the back end. The quickstart also makes more use of CDI to wire the application together. [[run-the-app]] === Run the app . Start JBoss AS in Eclipse . Deploy it using Eclipse - just right click on the app, choose `Run As -> Run On Server` . Select the AS 7 instance you want to use . Hit finish . Load the app at http://localhost:8080/jboss-as-numberguess . Make a few guesses [[deployment-descriptors-srcmainwebappweb-inf]] === Deployment descriptors src/main/webapp/WEB-INF Emphasize the lack of them! No need to open any of them, just point them out . `web.xml` - don't need it! . `beans.xml` - as before, marker file . `faces-config.xml` - nice feature from AS7 - we can just put `faces-config.xml` into the WEB-INF and it enables Jakarta Faces (inspiration from CDI) . `pom.xml` we saw this before, this time it's the same but adds in Jakarta Faces API [[views]] === Views . `index.html` - same as before, just kicks us into the app . `home.xhtml` .. Lines 19 - 25 – these are messages output depending on state of beans (minimise coupling between controller and view layer by interrogating state, not pushing) . Line 20 – output any messages pushed out by the controller . Line 39 - 42 – the input field is bound to the guess field on the game bean. We validate the input by calling a method on the game bean. . Line 43 - 45 – the command button is used to submit the form, and calls a method on the game bean . Line 48, 49, The reset button again calls a method on the game bean [[beans]] === Beans . `Game.java` – this is the main controller for the game. App has no persistence etc. .. `@Named` – As we discussed CDI is typesafe, (beans are injected by type) but sometimes need to access in a non-typesafe fashion. @Named exposes the Bean in EL - and allows us to access it from Jakarta Faces .. `@SessionScoped` – really simple app, we keep the game data in the session - to play two concurrent games, need two sessions. This is not a limitation of CDI, but simply keeps this demo very simple. CDI will create a bean instance the first time the game bean is accessed, and then always load that for you .. `@Inject maxNumber` – here we inject the maximum number we can guess. This allows us to externalize the config of the game .. `@Inject rnadomNumber` – here we inject the random number we need to guess. Two things to discuss here .. Instance - normally we can inject the object itself, but sometimes it's useful to inject a "provider" of the object (in this case so that we can get a new random number when the game is reset!). Instance allows us to `get()` a new instance when needed .. Qualifiers - now we have two types of Integer (CDI auto-boxes types when doing injection) so we need to disambiguate. Explain qualifiers and development time approach to disambiguation. You will want to open up `@MaxNumber` and `@Random` here. .. `@PostConstruct` – here is our reset method - we also call it on startup to set up initial values. Show use of `Instance.get()`. . `Generator.java` This bean acts as our random number generator. . `@ApplicationScoped` explain about other scopes available in CDI + extensibility. .. `next()` Explain about producers being useful for determining bean instance at runtime .. `getMaxNumber()` Explain about producers allowing for loose coupling [[the-login-quickstart]] == The login quickstart [[introduction-3]] === Introduction The login quickstart builds on the knowledge of CDI and Jakarta Faces we have got from numberguess. New stuff we will learn about is how to use Jakarta Persistence to store data in a database, how to use Jakarta Transactions to control transactions, and how to use Jakarta Enterprise Beans for declarative TX control. [[run-the-app-1]] === Run the app . Start JBoss AS in Eclipse . Deploy it using Eclipse - just right click on the app, choose `Run As -> Run On Server` . Select the AS 7 instance you want to use . Hit finish . Load the app at http://localhost:8080/jboss-as-login . Login as admin/admin . Create a new user [[deployment-descriptors]] === Deployment Descriptors . Show that we have the same ones we are used in `src/main/webapp` – `beans.xml`, `faces-config.xml` . We have a couple of new ones in `src/main/resources` .. `persistence.xml`. Not too exciting. We are using a datasource that AS7 ships with. It's backed by the H2 database and is purely a sample datasource to use in sample applications. We also tell Hibernate to auto-create tables - as you always have. .. `import.sql` Again, the same old thing you are used to in Hibernate - auto-import data when the app starts. . `pom.xml` is the same again, but just adds in dependencies for Jakarta Persistence, Jakarta Transactions and Jakarta Enterprise Beans [[views-1]] === Views . `template.xhtml` One of the updates added to Jakarta Faces was templating ability. We take advantage of that in this app, as we have multiple views .. Actually nothing too major here, we define the app "title" and we could easily define a common footer etc. (we can see this done in the kitchensink app) .. The `ui:insert` command inserts the actual content from the templated page. + # `home.xhtml` .. Uses the template .. Has some input fields for the login form, button to login and logout, link to add users. .. Binds fields to credentials bean}} .. Buttons link to login bean which is the controller . `users.xhtml` .. Uses the template .. Displays all users using a table .. Has a form with input fields to add users. .. Binds fields to the newUser bean .. Methods call on userManager bean [[beans-1]] === Beans . `Credentials.java` Backing bean for the login form field, pretty trivial. It's request scoped (natural for a login field) and named so we can get it from Jakarta Faces. . `Login.java` .. Is session scoped (a user is logged in for the length of their session or until they log out}} .. Is accessible from EL .. Injects the current credentials .. Uses the userManager service to load the user, and sends any messages to Jakarta Faces as needed .. Uses a producer method to expose the @LoggedIn user (producer methods used as we don't know which user at development time) . `User.java` Is a pretty straightforward Jakarta Persistence entity. Mapped with `@Entity`, has an natural id. . `UserManager.java` This is an interface, and by default we use the ManagedBean version, which requires manual TX control . `ManagedBeanUserManager.java` - accessible from EL, request scoped. .. Injects a logger (we'll see how that is produced in a minute) .. Injects the entity manager (again, just a min) .. Inject the UserTransaction (this is provided by CDI) .. `getUsers()` standard Jakarta Persistence-QL that we know and love - but lots of ugly TX handling code. .. Same for `addUser()` and `findUser()` methods - very simple Jakarta Persistence but... .. Got a couple of producer methods. ... `getUsers()` is obvious - loads all the users in the database. No ambiguity - CDI takes into account generic types when injecting. Also note that CDI names respect JavaBean naming conventions ... `getNewUser()` is used to bind the new user form to from the view layer - very nice as it decreases coupling - we could completely change the wiring on the server side (different approach to creating the newUser bean) and no need to change the view layer. . `EJBUserManager.java` .. It's an alternative – explain alternatives, and that they allow selection of beans at deployment time .. Much simple now we have declarative TX control. .. Start to see how we can introduce Jakarta Enterprise Beans to get useful enterprise services such as declarative TX control . `Resources.java` .. `{EntityManager}` - explain resource producer pattern [[the-kitchensink-quickstart]] == The kitchensink quickstart [[introduction-4]] === Introduction The kitchensink quickstart is generated from an archetype available for JBoss AS (tell people to check the link:/pages/createpage.action?spaceKey=WFLY&title=Getting+Started+Developing+Applications&linkCreation=true&fromPageId=557131[Getting Started Developing Applications] Guide for details). It demonstrates CDI, Jakarta Faces, Jakarta Enterprise Beans, Jakarta Persistence (which we've seen before) and Jakarta RESTful Web Services and Bean Validation as well. We add in Arquillian for testing. [[run-the-app-2]] === Run the app . Start JBoss AS in Eclipse . Deploy it using Eclipse - just right click on the app, choose `Run As -> Run On Server` . Select the AS 7 instance you want to use . Hit finish . Load the app at http://localhost:8080/jboss-as-kitchensink . Register a member - make sure to enter an invalid email and phone - show Jakarta Bean Validation at work . Click on the member URL and show the output from Jakarta RESTful Web Services [[Jakarta-Bean-Validation]] === Jakarta Bean Validation . Explain the benefits of Jakarta Bean Validation - need your data always valid (protect your data) AND good errors for your user. BV allows you to express once, apply often. . `index.xhtml` .. Show the input fields – no validators attached .. Show the message output . `Member.java` ... Hightlight the various validation annotations . Jakarta EE automatically applies the validators in both the persistence layer and in your views [[Jakarta-RESTful-Web-Services]] === Jakarta RESTful Web Services . `index.xhtml` - Show that URL generation is just manual . `JaxRsActivator.java` - simply activates Jakarta RESTful Web Services . `Member.java` - add Jakarta XML Binding annotation to make Jakarta XML Binding process the class properly . `MemberResourceRESTService.java` .. `@Path` sets the Jakarta RESTful Web Services resource .. Jakarta RESTful Web Services services can use injection .. `@GET` methods are auto transformed to XML using Jakarta XML Binding . And that is it! [[arquillian-getting-started]] === Arquillian . Make sure JBoss AS is running + [source,options="nowrap"] ---- mvn clean test -Parq-jbossas-remote ---- . Explain the difference between managed and remote . Make sure JBoss AS is stopped + [source,options="nowrap"] ---- mvn clean test -Parq-jbossas-managed ---- . Start JBoss AS in Eclipse . Update the project to use the `arq-jbossas-remote` profile . Run the test from Eclipse + Right click on test, `Run As -> JUnit Test` + `MemberRegistrationTest.java` . Discuss micro deployments . Explain Arquilian allows you to use injection . Explain that Arquillian allows you to concentrate just on your test logic NOTE: References in this document to CDI refer to Jakarta Contexts and Dependency Injection unless otherwise noted.