Spring Boot and Microservices
Spring Boot and Microservices
1. Microservices – A Primer
11.References
1. Microservices – A Primer
What is Microservices?
But there is no official definition of Microservices by industry standards. It’s recent phenomenon
in software industry to architect the new softwares which should be light weight, easier to deploy
To understand in details, you can definitely read Martin Fowler’s Microservices or Chris
Richardson’s Microservices.
We will not be covering this post in detail as compared to link I have posted here. Microservices
are small services that can run independently, but can also easily communicate with other services.
In traditional monolithic architecture style, there is a single application with single code base. An
application contains number of modules which are interrelated and can have external
dependencies. It’s a multi-tier enterprise application and has been used to build software for long.
Microservice architecture style was born out of need to build an application that could easily be
supported for mobile applications. Older style was not easy to support for mobile and new
generation way to handling of data. Any large enterprise application can be easily built using
A simple ground rule of microservice architecture pattern is to build a standalone service that can
be run without depending on any other service. That means for a large application can have more
than one services talking to each other, communicating with their own databases, but still
performing the business logic. Databases are used to ensure loose coupling of services.
1. Account Service
2. Shipment Service
3. Inventory Service
2. Runtime service to handle runtime and backend data to process business logic
3. Logging service
5. Session service
UI for the e-commerce application can be built independently to use backend services to show/edit
data.
Pros
2. Reliability – A fault in the service can only bring down that service, depending on handling
and grids.
4. Availability – Dispatching the patch or newer version of service requires less downtime
6. Design and Development – Each service can be developed independently and helps
developer to manage the service easily without worrying about other services.
Cons
1. Performance – All services involved in application have to communicate with each other
3. Memory usage – Possible duplicate data across services and lot of duplication in cache.
2. Spring Boot REST CRUD API
In this chapter, we will learn how to write a CRUD Rest API using Spring Boot. Spring boot
provides some cool features to create a production ready Spring application that can be deployed
as a war file on any environment. Part of this book, we will create a microservice using Spring
Boot and deploy on docker. Let’s start with creation of a simple REST API.
As first step, let’s create a maven project in eclipse. You can create this by going into File > New
We will be using spring-boot and all the required dependencies including spring-data. Spring data
JPA provides lot of useful enhancements that can be seamlessly used with spring-boot project.
Spring-data will cover the data access layer which is basically implements persistence. Once we
use spring-data, we don’t have to add any external hibernate or eclipselink JPA APIs. Also some
of the data access repositories provided by spring-data makes implementing data access layer code
less worrisome.
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.betterjavacode</groupId>
<artifactId>Benefits</artifactId>
<packaging>war</packaging>
<version>0.0.1-SNAPSHOT</version>
<name>Benefits Maven Webapp</name>
<url>http://maven.apache.org</url>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.4.2.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
<plugin>
<artifactId>maven-war-plugin</artifactId>
<version>2.6</version>
<configuration>
<warSourceDirectory>WebContent</warSourceDirectory>
<failOnMissingWebXml>false</failOnMissingWebXml>
</configuration>
</plugin>
</plugins>
<finalName>Benefits</finalName>
</build>
</project>
We will be creating a rest api for Benefits service which will have companies and users as main
objects. We are only covering basic data model classes at the moment, but as part of the series
we will develop a web application. Each company will company profile and each user will
userprofile. So we will have four basic entities Company, CompanyProfile, User, UserProfile.
package com.betterjavacode.benefits.entities;
import java.io.Serializable;
import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.OneToOne;
import javax.persistence.Table;
@Entity(name = "Company")
@Table(name = "company")
public class Company implements Serializable {
/**
*
*/
private static final long serialVersionUID = 1L;
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int id;
@Column
private String name;
@Column
private int statusid;
@Column
private String type;
@Column
private String ein;
package com.betterjavacode.benefits.entities;
import java.io.Serializable;
import java.util.Date;
import java.util.List;
import javax.persistence.CascadeType;
import javax.persistence.Column;
import javax.persistence.Entity;
import javax.persistence.FetchType;
import javax.persistence.GeneratedValue;
import javax.persistence.GenerationType;
import javax.persistence.Id;
import javax.persistence.JoinColumn;
import javax.persistence.JoinTable;
import javax.persistence.ManyToMany;
import javax.persistence.OneToOne;
import javax.persistence.Table;
import com.fasterxml.jackson.annotation.JsonFormat;
import com.fasterxml.jackson.annotation.JsonFormat.Feature;
@Entity(name = "User")
@Table(name = "user")
public class User implements Serializable {
/**
*
*/
private static final long serialVersionUID = 1L;
public User() {
@Id
@GeneratedValue(strategy = GenerationType.IDENTITY)
private int id;
@Column
private Date createdate;
@Column(name = "enabled")
private int enabled;
@JsonFormat(with = Feature.ACCEPT_SINGLE_VALUE_AS_ARRAY)
@ManyToMany(cascade = CascadeType.ALL)
@JoinTable(name = "user_role", joinColumns = @JoinColumn(name =
"user_id"), inverseJoinColumns = @JoinColumn(name = "role_id"))
private List<Role> roles;
Part of our architecture for REST API, we will have following three layers
1. Rest layer
So in Business object layer, we will implement all the managers which will handle processing of
rest requests to create, update, read or delete the data. In subsequent posts, we will enhance this
package com.betterjavacode.benefits.managers;
import java.util.List;
import org.springframework.beans.factory.annotation.Autowired;
import com.betterjavacode.benefits.entities.User;
import com.betterjavacode.benefits.interfaces.UserManager;
import com.betterjavacode.benefits.repositories.UserRepository;
public class UserManagerImpl implements UserManager {
@Autowired
public void setUserRepository(UserRepository userRepository) {
this.userRepository = userRepository;
}
@Override
public User createUser(User u) {
if (u != null) {
User user = userRepository.save(u);
return user;
} else {
return null;
}
}
@Override
public User updateUser(User u) {
// TODO Auto-generated method stub
return null;
}
@Override
public User getUser(int id) {
User user = userRepository.findOne(id);
if (user == null) {
return null;
}
return user;
}
@Override
public List getAllUsers() {
List userList = (List) userRepository.findAll();
return userList;
}
@Override
public void deleteUser(int guid) {
// TODO Auto-generated method stub
User user = userRepository.findOne(guid);
if (user == null) {
return;
}
userRepository.delete(user);
}
}
Create a REST controller
One of the best uses of Spring boot is to create rest API and the feature it offers for the same is to
use REST controller. Spring-boot offers an annotation for the same as @RestController.
package com.betterjavacode.benefits.controller;
import java.util.List;
import javax.ws.rs.core.Response;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import com.betterjavacode.benefits.entities.User;
import com.betterjavacode.benefits.interfaces.UserManager;
@RestController
@RequestMapping("benefits/v1")
public class UserService {
@Autowired
UserManager userMgr;
package com.betterjavacode.benefits.controller;
import java.util.List;
import javax.ws.rs.core.Response;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.web.bind.annotation.PathVariable;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import org.springframework.web.bind.annotation.RestController;
import com.betterjavacode.benefits.entities.Company;
import com.betterjavacode.benefits.interfaces.CompanyManager;
@RestController
@RequestMapping("benefits/v1")
public class CompanyService {
@Autowired
CompanyManager compMgr;
We are using embedded tomcat in this Spring-boot project. So once we are done building and
installing the code through maven, we can run the project through eclipse or standalone war file in
tomcat. For our demo purposes, we will run this application through eclipse, which will start
embedded tomcat.
If we execute the url http://localhost:8080/benefits/v1/users/1 – it will display json for user data as
below
3. Swagger Documentation for REST API
In this lesson, we will show how to add swagger documentation to Spring boot rest API. We
learned how to create a Spring Boot REST API in previous chapter. In Microservices’ world, these
days documenting your API is a standard norm. Swagger provides a handy interface and a simple
way to build these documentations that any client can test any moment. They don’t need to have
What is Swagger?
Swagger was intended to provide a standard, language-agnostic interface to REST APIs which
allow anyone to understand the capabilities of a service without any source code, documentation
of source code. You can find more details about Swagger here.
In our previous lesson, we added Spring boot REST API. We will add swagger documentation to
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger2</artifactId>
<version>2.6.1</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger-ui</artifactId>
<version>2.6.1</version>
<scope>compile</scope>
</dependency>
In our main starting Application class, we will add few configurations for setting up a bean which
will handle swagger documentation. In below code, I show what I have added in Application.java
file. Basically, we have created a new bean of type Docket which takes care of swagger
configuration.
@EnableSwagger2
@SpringBootApplication(scanBasePackages = { "com.betterjavacode.benefits" })
public class Application {
@Bean
public Docket benefitsApi() {
return new Docket(DocumentationType.SWAGGER_2).groupName("Benefits")
.apiInfo(apiInfo())
.select()
.apis(RequestHandlerSelectors.any())
.paths(PathSelectors.any())
.build()
.pathMapping("/");
}
Now once we have added the configuration, we can build our project with maven clean install.
After successful build, run the project from eclipse as a Java application. We will access swagger
swagger documentation for REST API chapter 3. In this lesson, we will add error handling and
logging to our REST API. Error handling and Logging are two different ideas, so I will divide this
1. Logging
In most production applications, logging is critical and it is used for multiple purposes. Few of
those uses are debugging the production issues or auditing for the application. Over the years,
different logging libraries have evolved to use in java based applications. slf4j is the most popular
In our tutorial for this application, we will be using log4j2 which is the most recent and advance
logging library out there. It provides lot of useful features for performance, support for multiple
APIs, advance filtering, automatic reloading of configurations etc. We will not cover any of these
in this article, if you are interested to read about log4j2 libraries, read here.
To use log4j2, we will add the maven dependency to our project’s pom file. This should look like
below
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
</dependency>
To enable logging, we will have to add a configuration file in our application. This configuration
file can be XML, JSON or YAML file. We will be using a XML file log4j2.xml which will look
like below
So we are using Console and BenefitsFile as two loggers which will log into a console and file
respectively. We are setting log level to DEBUG. If you log any messages with a level lower than
DEBUG, they will be logged into console or file. We will have to add a file benefits.log in
classpath to achieve this logging in file. Log pattern is with date time, log level, class from which
Once we have required logging libraries and logging configuration adjusted, we can add logging
in our code to capture this logging during runtime execution. In one of the managers
@Override
public List<Company> getAllCompanies()
{
LOGGER.info(" Enter >> getAllCompanies() ");
List<Company> cList = (List<Company>) companyRepository.findAll();
LOGGER.info(" Exit << getAllCompanies() ");
return cList;
}
Now once we execute our spring boot application, we can capture the logs in console or file. The
2. Error Handling
We will not write about exceptions in detail as it has been covered in this post Exceptions. We
will create our own custom exception which will be extended from WebApplicationException
package com.betterjavacode.benefits.utilities;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.Response;
/**
*
*/
private static final long serialVersionUID = 1L;
private int errorcode = 00; // 00 indicates - no error
public InvalidRequestException() {
Now we can use this custom exception in our managers when we want to throw an error message
to indicate if there is anything wrong with client request. Similarly we can build another exception
to show if there is anything wrong on server side. Following snippet shows from
@Override
public Company getCompany(int guid)
{
LOGGER.info(" Enter >> getCompany() ");
Company company = companyRepository.findOne(guid);
if (company == null) {
LOGGER.info(" Exit << createCompany() ");
throw new InvalidRequestException(400, "Company not found");
}
LOGGER.info(" Exit << getCompany() ");
return company;
}
In this lesson, we showed how to handle logging and errors in a REST API.
5. Consuming RESTFul Webservice
Continuing the series of lessons on Spring Boot, in this post, we will investigate how to consume
a REST API service we built previously. This is a short lesson, but we will show how to read the
data and how to post the data with some of the features Spring Boot offers to consume a REST
service for client side. Eventual goal is to use this feature to call our rest service during runtime to
use the data from database to display on views which a user will be able to see.
Purpose
Purpose of this post is to read company data from Company REST API and also to create a
Build a client
RestTemplate is easiest way for client to interact with server side code with just one line of code.
In our client code, we will need a RestTemplate object, REST service URL. Since this is a sample
we are building, we will be adding a main method in this class to run this client side of the code.
In real life scenarios, during runtime, a client code will call rest template to get server side data,
Similarly, we will have another request to post the data on server side to create a company. The
In this post, we showed how to use RestTemplate a feature that spring boot provides to consume
a REST service.
6. Producing and Consuming SOAP Webservice
In this lesson, we will describe how to create a SOAP webservice from our existing Spring Boot
REST API. This SOAP webservice will provide us user data from the database which is we have
1. Requirements
1. Eclipse Mars2
4. Java 7
5. Tomcat 8
We will use our existing Spring Boot REST API to build an application that will act as a SOAP
web service to provide users data. For a given user id, web service will return user data.
Let’s create a schema file in src/main/resources directory and maven will create java classes based
<xs:schema xmlns:xs="http://www.w3.org/2001/XMLSchema"
xmlns:tns="https://betterjavacode.com/benefits/soap"
targetNamespace="https://betterjavacode.com/benefits/soap"
elementFormDefault="qualified">
<xs:element name="getUserRequest">
<xs:complexType>
<xs:sequence>
<xs:element name="id" type="xs:int"/>
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:element name="getUserResponse">
<xs:complexType>
<xs:sequence>
<xs:element name="user" type="tns:user"/>
</xs:sequence>
</xs:complexType>
</xs:element>
<xs:complexType name="user">
<xs:sequence>
<xs:element name="id" type="xs:int"/>
<xs:element name="firstname" type="xs:string"/>
<xs:element name="middlename" type="xs:string"/>
<xs:element name="lastname" type="xs:string"/>
<xs:element name="username" type="xs:string"/>
<xs:element name="createdate" type="xs:date"/>
<xs:element name="jobtitle" type="xs:string"/>
<xs:element name="email" type="xs:string"/>
</xs:sequence>
</xs:complexType>
</xs:schema>
Now to generate classes from schema, we have to make sure we have all the right dependencies in
our pom.xml. We will also add spring boot service dependency to create a SOAP web service.
<project xmlns="http://maven.apache.org/POM/4.0.0"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://maven.apache.org/POM/4.0.0
http://maven.apache.org/maven-v4_0_0.xsd">
<modelVersion>4.0.0</modelVersion>
<groupId>com.betterjavacode</groupId>
<artifactId>Benefits</artifactId>
<packaging>war</packaging>
<version>0.0.1-SNAPSHOT</version>
<name>Benefits Maven Webapp</name>
<url>http://maven.apache.org</url>
<parent>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-parent</artifactId>
<version>1.4.2.RELEASE</version>
</parent>
<dependencies>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-data-jpa</artifactId>
</dependency>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<scope>runtime</scope>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-jdbc</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-test</artifactId>
</dependency>
<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
</dependency>
<dependency>
<groupId>io.swagger</groupId>
<artifactId>swagger-jersey2-jaxrs</artifactId>
<version>1.5.12</version>
</dependency>
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger2</artifactId>
<version>2.6.1</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>io.springfox</groupId>
<artifactId>springfox-swagger-ui</artifactId>
<version>2.6.1</version>
<scope>compile</scope>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-api</artifactId>
</dependency>
<dependency>
<groupId>org.apache.logging.log4j</groupId>
<artifactId>log4j-core</artifactId>
</dependency>
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-web-services</artifactId>
</dependency>
<dependency>
<groupId>wsdl4j</groupId>
<artifactId>wsdl4j</artifactId>
</dependency>
<dependency>
<groupId>javax.xml.bind</groupId>
<artifactId>jaxb-api</artifactId>
<version>2.1</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<scope>test</scope>
</dependency>
</dependencies>
<build>
<plugins>
<plugin>
<artifactId>maven-compiler-plugin</artifactId>
<version>3.3</version>
<configuration>
<source>1.8</source>
<target>1.8</target>
</configuration>
</plugin>
<plugin>
<artifactId>maven-war-plugin</artifactId>
<version>2.6</version>
<configuration>
<warSourceDirectory>WebContent</warSourceDirectory>
<failOnMissingWebXml>false</failOnMissingWebXml>
</configuration>
</plugin>
<plugin>
<groupId>org.codehaus.mojo</groupId>
<artifactId>jaxb2-maven-plugin</artifactId>
<version>1.6</version>
<executions>
<execution>
<id>xjc</id>
<goals>
<goal>xjc</goal>
</goals>
</execution>
</executions>
<configuration>
<schemaDirectory>${project.basedir}/src/main/resources/</schemaDirectory>
<outputDirectory>${project.basedir}/src/main/java</outputDirectory>
<clearOutputDir>false</clearOutputDir>
<schemaLanguage>WSDL</schemaLanguage>
<generatePackage>com.betterjavacode.benefits.soap</generatePackage>
<forceRegenerate>true</forceRegenerate>
<scehmas>
<schema>
<url>http://localhost:8080/benefits/endpoints/users.wsdl</url>
</schema>
</scehmas>
</configuration>
</plugin>
</plugins>
<finalName>Benefits</finalName>
</build>
</project>
If we run the project with maven build now, the plugin jaxb2-maven-plugin will generate classes
under com.betterjavacode.benefits.soap directory. It will also enable our wsdl SOAP url for users.
GetUserRequest
GetUserResponse
ObjectFactory
package-info
User
Next, we will define an interface for our service. This will look like below
package com.betterjavacode.benefits.services;
Implementation of this service will be mapping out entity class User to generated class for soap
service User. Using the id as a key to get user data from repository, we will map to soap service
user. For post purposes, we will not show the implementation of this interface.
What is a service endpoint? When a SOAP request for defined URL is handled by Spring servlet,
Spring servlet redirects that request to service endpoint. Service endpoint then processes that
request to create a response. Our spring-boot-starter-web-services dependency will bring all the
package com.betterjavacode.benefits.services.endpoints;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.ws.server.endpoint.annotation.Endpoint;
import org.springframework.ws.server.endpoint.annotation.PayloadRoot;
import org.springframework.ws.server.endpoint.annotation.RequestPayload;
import org.springframework.ws.server.endpoint.annotation.ResponsePayload;
import com.betterjavacode.benefits.services.UserAccountService;
import com.betterjavacode.benefits.soap.GetUserRequest;
import com.betterjavacode.benefits.soap.GetUserResponse;
import com.betterjavacode.benefits.soap.User;
@Endpoint
public class UserAccountServiceEndpoint
{
@Endpoint annotation allows the class to be defined as service endpoint and included in
@Component annotation for scanning. Make sure the namespace defined in this class matches
with XSD schema definition. Otherwise, you can run into error for “No Endpoint defined for“.
6. Configuration
Next, we will configure our configuration class to generate wsdl endpoint. This configuration class
package com.betterjavacode.benefits;
import org.springframework.boot.web.servlet.ServletRegistrationBean;
import org.springframework.context.ApplicationContext;
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.io.ClassPathResource;
import org.springframework.ws.config.annotation.EnableWs;
import org.springframework.ws.config.annotation.WsConfigurerAdapter;
import org.springframework.ws.transport.http.MessageDispatcherServlet;
import org.springframework.ws.wsdl.wsdl11.DefaultWsdl11Definition;
import org.springframework.xml.xsd.SimpleXsdSchema;
import org.springframework.xml.xsd.XsdSchema;
@Configuration
@EnableWs
@ComponentScan("com.betterjavacode")
public class AppConfig extends WsConfigurerAdapter
{
@Bean
public ServletRegistrationBean
messageDispatcherServlet(ApplicationContext applicationContext)
{
MessageDispatcherServlet servlet = new MessageDispatcherServlet();
servlet.setApplicationContext(applicationContext);
return new ServletRegistrationBean(servlet, "/benefits/endpoints/*");
}
@Bean(name = "users")
public DefaultWsdl11Definition defaultWsdl11Definition(XsdSchema
usersSchema)
{
DefaultWsdl11Definition wsdl11definition = new
DefaultWsdl11Definition();
wsdl11definition.setPortTypeName("UserAccountService");
wsdl11definition.setLocationUri("/endpoints");
wsdl11definition.setTargetNamespace("http://com/betterjavacode/benefits/webse
rvices/useraccountservice");
wsdl11definition.setSchema(usersSchema);
return wsdl11definition;
}
@Bean
public XsdSchema usersSchema()
{
return new SimpleXsdSchema(new ClassPathResource("employees.xsd"));
}
}
this servlet with a bean to handle the URL from which request will be coming.
url http://localhost:8080/benefits/endpoints/users.wsdl
Here we showed how to create a simple SOAP webservice which we have combined with Spring
Boot REST API service. We can also test this SOAP webservice using Soap UI, as shown in below
screenshot
8. Consuming the SOAP web service
In previous steps, we showed how to produce a SOAP web service, now we will show how to
methods.
package com.betterjavacode.benefits.views;
import org.springframework.ws.client.core.support.WebServiceGatewaySupport;
import org.springframework.ws.soap.client.core.SoapActionCallback;
import com.betterjavacode.benefits.soap.GetUserRequest;
import com.betterjavacode.benefits.soap.GetUserResponse;
We will configure Jaxb2Marshaller to support JAXB to set context path. This will help us marshal
import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.oxm.jaxb.Jaxb2Marshaller;
@Configuration
public class ClientAppConfig
{
@Bean
public Jaxb2Marshaller marshaller()
{
Jaxb2Marshaller marshaller = new Jaxb2Marshaller();
marshaller.setContextPath("com.betterjavacode.benefits.soap");
return marshaller;
}
@Bean
public UserClient userClient(Jaxb2Marshaller marshaller)
{
// WSDL URL - http://localhost:8080/benefits/endpoints/users.wsdl
UserClient uc = new UserClient();
uc.setDefaultUri("http://localhost:8080/benefits/endpoints/users.wsdl");
uc.setMarshaller(marshaller);
uc.setUnmarshaller(marshaller);
return uc;
}
}
We will define a class with main method to pass an argument of user id. Our client will call the
web service with passed argument to return us the data if that user id existed in the database.
}
}
9. Conclusion
In this lesson, we showed how to create a SOAP web service and how to build a client to consume
exception. A non-technical user will not be able to understand most of the time. Also stack trace
is not the best option when we can show the same exception error in nice json format with the root
cause. In this lesson, we will show how to handle unhandled exceptions from our previous Spring
REST API we built here. This particular feature will show how to handle most HTTP 500 errors
which happen because of server side issues. Any errors with request or client side, those are HTTP
400 errors and they have been handled in previous post Error Handling in Spring Boot Rest API.
Problem
What happens when there is a database connection issue OR columns in your REST API are
different from specified in database tables? Your API will throw a 500 error and if you don’t have
any mechanism, this will display error in an html format which will not give much information to
Solution
JAX-RS provides a way to handle uncaught exceptions. This can be done by my implementing an
interface for ExtendedExceptionMapper. Basically this is a contract for a provider that takes java
exceptions and maps them to a response object which can be transformed into a JSON. This can
be implemented as below:
package com.betterjavacode.benefits.utilities;
import javax.ws.rs.WebApplicationException;
import javax.ws.rs.core.Response;
import javax.ws.rs.core.Response.Status;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.glassfish.jersey.spi.ExtendedExceptionMapper;
@Provider
public class UncaughtExceptionMapper implements
ExtendedExceptionMapper<Throwable>
{
private static final Logger LOGGER =
LogManager.getLogger(UncaughtExceptionMapper.class);
@Override
public Response toResponse(Throwable exception)
{
LOGGER.info("Enter >> toResponse ");
LOGGER.debug("Exception Caught: " + exception.getMessage());
LOGGER.info("Exit << toResponse");
return
Response.status(Status.BAD_REQUEST).entity(exception.getMessage()).build();
}
@Override
public boolean isMappable(Throwable arg0)
{
return !(arg0 instanceof WebApplicationException);
}
}
interface ExtendedExceptionMapper which provides a way to map all exceptions which are not of
toResponse method will help to log all the exceptions and convert exceptions into a Response
object.
Conclusion
In this lesson, we showed how to map all uncaught exceptions into json format response.
8. Angular JS User Interface for Spring Boot REST API
In this lesson, we will add a user interface using AngularJS to the REST API we created chapter
2.
First we will create a controller in Spring Boot rest api to call our home page. All the requests that
will come to web server, will go through this controller and controller will return a home page for
package com.betterjavacode.benefits.controller;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.stereotype.Controller;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
@Controller
public class MainController
{
public static final Logger LOGGER =
LogManager.getLogger(MainController.class);
Now, we will create an index.html page. We will also be using angular JS framework as part of
this home page so that we can build a single page application. If you are not aware of Angular JS
or new to this framework, you can read about it AngularJS. One thing to remember while creating
this page is a directory structure. Lot of issues that arise to create html pages are because of
The home page index.html page is under main/resources/templates/ directory and it looks like
below
<html ng-app="benefitApp">
<script src="https://code.angularjs.org/1.6.1/angular.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.6.1/angular-
route.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.6.1/angular-
resource.js"></script>
</head>
<body ng-controller="MainCtrl">
Hello {{name}}!
<div>
<ul class="menu">
<li><a href="listUser">user-list</a></li>
<li><a href="listCompany">company-list</a></li>
</ul>
<div ng-view="ng-view"></div>
</div>
</body>
</html>
Home page shows that this is an angular app with name “benefitApp”. This also declares a
controller “MainCtrl” with an angular view. Important to see we are importing angular.js, angular-
route.js and angular-resource.js modules. Click on user-list or company-list, will show list of users
Now to handle the controller (MainCtrl), we added in index.html, we will add app.js which will
declare the controller. This javascript file also contains config data to handle all the routing of
pages. That’s why we will be importing “ngRoute” and “ngResource” angular modules.
$scope.$routeParams = $routeParams;
})
Throughout the interaction on web pages, we will be using different controllers for editing user or
company information and creating user or company. The same file app.js will also handle routing
app.config(function($routeProvider,$locationProvider) {
$locationProvider.html5Mode(true);
$routeProvider.when('/listUser',
$routeProvider.when('/listCompany',
$routeProvider .when('/editUser/:userId',
{ templateUrl : 'views/editUser.html' }) ;
$routeProvider .when('/editCompany/:companyId',
{ templateUrl : 'views/editCompany.html' }) ;
$routeProvider.when('/createUser',
{templateUrl:'views/createUser.html'});
$routeProvider.when('/createCompany',
{templateUrl:'views/createCompany.html'});
});
Rest of the code showing all controllers’ logic has been skipped for post purposes. UserController
or CompanyController are calling rest apis which we have built using Spring boot.
Demo
Now build the code and run our embedded tomcat webserver. Fire up the
Click on user-list and it will show list of users inside the same page as below:
Click on edit button and we will see a form to update user information:
9. Spring Boot Security in web application
In this lesson, we will show how to use Spring Boot Security to login, authorization based on user
7. If user is of role USER, he is redirected to user profile page. He can only edit or save his
information.
9. If user clicks on logout, session is deleted and user is redirected to login page.
10. If user (of any role) tries to login after logout, user should be redirected to appropriate page
To completely understand this lesson, make sure you have gone through my other posts on Spring
Boot series.
2. Swagger Documentation
3. User Interface using AngularJS
Database changes
Since this post involves authorization for users, we have to do some database changes. We will
add couple of tables and respective model classes in our REST api modification.
Table role
Table user_role
create table role (id int(11) auto_increment primary key not null, role
varchar(255) )
create table user_role (user_id int(11) primary key not null, role_id int(11)
primary key not null))
user_role table helps to maintain many-to-many relationship between user and role table. We will
have only two roles for demo purposes USER and ADMIN.
Another change we have done in table user is that we have added field called password_hash to
store password set by user/administrator for user to login. We will be storing a hash password
Dependencies
Since we will be using Spring-security for authentication and authorization purposes, we will add
Other than those changes mentioned about, we will demonstrate this post in top-down fashion
So for web layer, we will define a new controller LoginController and modify our
existing MainController.
package com.betterjavacode.benefits.controller;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.stereotype.Controller;
import org.springframework.ui.Model;
import org.springframework.web.bind.annotation.RequestMapping;
import org.springframework.web.bind.annotation.RequestMethod;
import com.betterjavacode.benefits.entities.User;
import com.betterjavacode.benefits.interfaces.UserManager;
/**
*
* @author Yogesh Mali
*
*/
@Controller
public class LoginController
{
public static final Logger LOGGER =
LogManager.getLogger(LoginController.class);
@Autowired
UserManager userManager;
/**
*
* @param model
* @return
*/
@RequestMapping(value = "/user", method = RequestMethod.GET)
public String userpage(Model model)
{
LOGGER.info(" Enter >> userpage() ");
Authentication auth = SecurityContextHolder.getContext()
.getAuthentication();
String name = auth.getName();
User user = userManager.findUserByEmail(name);
model.addAttribute("name", user.getFirstname());
model.addAttribute("userid", user.getId());
LOGGER.info(" Exit << userpage() ");
return "user";
}
/**
*
* @return
*/
@RequestMapping(value = { "/login" })
public String login()
{
return "login";
}
/**
*
* @return
*/
@RequestMapping(value = "/403", method = RequestMethod.GET)
public String Error403()
{
return "403";
}
}
As shown in this controller, we have defined a user page, a login page and an error page (403). A
user with role of either USER or ADMIN or both can access a user page which shows that logged
in user’s profile.
Every user irrespective of roles, will be redirected to login page for authentication. If there are any
errors during authentication or authorization, user will be redirected to access denied page (403).
</div>
<button class="btn btn-lg btn-primary btn-block" name="Submit" value="Login"
type="Submit" th:text="Login" style="width:350px"></button> </form></div>
</body>
</html>
This login page shows a simple form to input username (email) and password and process that
to display in html page. UserManager in service layer has been enhanced to return an user based
on username (which is email) . We have also added email to be unique as a constraint in database.
User page for a user with role USER is nothing but a user profile information which he can edit
<html ng-app="benefitApp">
<head><meta http-equiv="Content-Type" content="text/html; charset=ISO-8859-1"
/>
<title>Benefit Application</title><script>document.write('<base href="' +
document.location + '" />');</script> <link rel="stylesheet"
href="/css/bootstrap.css" /><script
src="https://code.angularjs.org/1.6.1/angular.js"></script><script
src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.6.1/angular-
route.js"></script><script
src="https://cdnjs.cloudflare.com/ajax/libs/angular.js/1.6.1/angular-
resource.js"></script><script type="text/javascript"
src="./js/app.js"></script></head><body ng-controller="UserCtrl">Hello
<p th:text="${name}"></p>
<div>
<ul class="menu">
<li><a th:href="@{'userProfile/' + ${userid}}">Profile</a></li>
</ul>
<div ng-view="ng-view"></div>
</div>
<div class="input-group">
<div class="controls"> <a ng-click="logout()" class="btn btn-
small">Logout</a></div>
</div>
</body>
</html>
Authentication
Now we have the application ready with all the required backend details for adding authentication
application.
package com.betterjavacode.benefits;
import javax.sql.DataSource;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.beans.factory.annotation.Value;
import org.springframework.boot.autoconfigure.security.SecurityProperties;
import org.springframework.context.annotation.ComponentScan;
import org.springframework.context.annotation.Configuration;
import org.springframework.core.annotation.Order;
import
org.springframework.security.config.annotation.authentication.builders.Authen
ticationManagerBuilder;
import
org.springframework.security.config.annotation.web.builders.HttpSecurity;
import
org.springframework.security.config.annotation.web.builders.WebSecurity;
import
org.springframework.security.config.annotation.web.configuration.EnableWebSec
urity;
import
org.springframework.security.config.annotation.web.configuration.WebSecurityC
onfigurerAdapter;
import org.springframework.security.crypto.bcrypt.BCryptPasswordEncoder;
@Configuration
@ComponentScan("com.betterjavacode.benefits.services")
@EnableWebSecurity
@Order(SecurityProperties.ACCESS_OVERRIDE_ORDER)
public class SecurityConfig extends WebSecurityConfigurerAdapter
{
public static final Logger LOGGER =
LogManager.getLogger(SecurityConfig.class);
@Autowired
private SimpleAuthenticationSuccessHandler loginSuccess;
@Autowired
private LogoutSuccess logoutSuccess;
@Autowired
private BCryptPasswordEncoder bCryptPasswordEncoder;
@Autowired
private DataSource dataSource;
@Value("${spring.queries.users-query}")
private String usersQuery;
@Value("${spring.queries.roles-query}")
private String rolesQuery;
@Autowired
protected void configureGlobal(AuthenticationManagerBuilder auth) throws
Exception
{
LOGGER.info(" Enter >> configureGlobal() ");
auth.jdbcAuthentication()
.usersByUsernameQuery("select email,password_hash,enabled from user
where email=?")
.authoritiesByUsernameQuery("select u.email,r.role from user u inner
join user_role ur on(u.id=ur.user_id) inner join role r on(r.id=ur.role_id)
where u.email=?")
.dataSource(dataSource)
.passwordEncoder(bCryptPasswordEncoder);
LOGGER.info(" Exit << configureGlobal() ");
}
/**
* Handle Login - Authentication and Redirection
*/
@Override
protected void configure(HttpSecurity http) throws Exception
{
http.csrf()
.disable()
.authorizeRequests()
.antMatchers("/home")
.hasAuthority("ADMIN")
.antMatchers("/user")
.hasAnyAuthority("USER", "ADMIN")
.and()
.formLogin()
.loginPage("/login")
.successHandler(loginSuccess)
.permitAll()
.and()
.logout()
.logoutSuccessHandler(logoutSuccess)
.deleteCookies("JSESSIONID")
.invalidateHttpSession(false)
.permitAll()
.and()
.exceptionHandling()
.accessDeniedPage("/403");
/**
* Exclude resources from user-access
*/
@Override
public void configure(WebSecurity web) throws Exception
{
web.ignoring()
.antMatchers("/resources/**", "/static/**", "/css/**", "/js/**",
"/images/**");
}
}
What’s happening in this code?
When a user with role ADMIN or USER calls either /home or /user pages respectively, the
Once the user inputs credentials, they are validated against JDBC database authentication
If user of role USER tries to access ADMIN home page, user will be redirected to 403
If user clicks LOGOUT button on the page he is on, he will be logged out of application
and redirected back to login page. All the cookies will be deleted. Redirection will be
As shown in user.html page, once the user with role USER is logged in, he sees url for his profile
information. If user clicks this url, user sees his or her profile information. This page has a
controller called UserCtrl which basically handles the logout on this initial page. User Profile is
handles updating user profile information or logout. Rest of the code will be updated in github
There are two ways Cross-Site Request Forgery token can be handled in Spring application. First
way is by disabling this token generation. This is not a recommended approach as this put your
application to possible CSRF security attacks for hackers. If you are just doing this for demo
As spring points out, CSRF Protection should be used for any request that could be processed by
We will be using spring security to handle CSRF token on server side rather than on client side. So
every request that will come to server will be intercepted to add a CSRF token and then verified.
Angular JS verifies the cookie for CSRF token before a user can post any request.
We will add a filter that will handle setting of CSRF token in a cookie. Angular JS expects a cookie
@Override
protected void doFilterInternal(HttpServletRequest request,
HttpServletResponse response, FilterChain filterChain) throws
ServletException, IOException {
CsrfToken csrf = (CsrfToken)
request.getAttribute(CsrfToken.class.getName());
if (csrf != null) {
Cookie cookie = WebUtils.getCookie(request, "XSRF-TOKEN");
String token = csrf.getToken();
if (cookie == null || token != null &&
!token.equals(cookie.getValue())) {
cookie = new Cookie("XSRF-TOKEN", token);
cookie.setPath("/");
response.addCookie(cookie);
}
}
filterChain.doFilter(request, response);
}
Now we will enable csrf token in SecurityConfig as shown below
.and()
.csrf()
.csrfTokenRepository(csrfTokenRepository())
.and()
.addFilterAfter(new CSRFHeaderFilter(), CsrfFilter.class);
What is csrfTokenRepository?
We tell spring-security to expect CSRF token in the format that Angular wants to send it back, a
Demo
In this post, we showed how to use spring security for authentication and authorization. Now we
will show how to run the application. Once the application is built and run from eclipse, access the
Docker is a container platform delivered by a company named “Docker Inc.” Docker can be used
by developers, operators and enterprise users to deliver and use packaged software. Docker has
something called a container. Container can be a virtual machine (VM) in lay man’s terms, but
still little different from VM. Container contains packaged software delivered in a way that it can
be run isolated on a shared operating system. As per official definition – Unlike VMs, container
does not bundle a full operating system – only libraries and settings required to make the software
In this demo, we will use our spring boot application built throughout from Part I to Part VIII.
We will build a container with mysql database deployed and another container where we will
deploy spring boot application. This spring boot application container will connect to mysql
database container at runtime. The process is little complicated, but once you get your hands on
docker container, it becomes very straight forward to understand. Also for this post, I will not
explain anything about spring boot application. You should review all the previous posts I have
written explaining how to build and deploy spring boot application on an embedded tomcat.
container.
2. If your application contains user administration and password, make sure you have a super
administrator whose password you can insert in database with password_hash. This is
For most standard applications (like mysql, java 8, spring-boot), there are number of images
available in docker hub. When we will run our docker container for database, docker shell will
pull the version of that application from the hub to build a container. We will not be creating any
new or blank docker image. To run a docker container with mysql version 5.6, we will use below
command.
3308:3306 -d mysql:5.6
We are not using any password. This is not recommended for production code, I am just
Also this database is running at port 3308 to 3306 of local host machine.
mysql:5.6 to download MySQL 5.6 Server image from Docker public repo
Once this is started, there are couple of ways you can verify if we are able to connect to this
database or not.
Get the ip address of this container host with command docker-machine ip . Now in mysql
administrator workbench, access the mysql server with ip address and port 3308 and see if you can
Another way on docker shell terminal – use this command docker exec -it benefitsmysql -l , this
will connect you as a root to the shell where mysql is installed. And then you can use mysql as
To run our Spring boot application successfully, once you access mysql, create following tables:
use benefitsmysql;
create table userprofile(id int not null auto_increment, dob date, doh date,
maritalstatus varchar(50),sex varchar(50),ssn varchar(50),weight varchar(50),
height varchar(50),employmentstatus varchar(50), terminationdate date,
primary key(id));
create table user(id int not null auto_increment, createdate date, email
varchar(255),firstname varchar(255), middlename varchar(255), lastname
varchar(255),username varchar(100),jobtitle varchar(255),password_hash
text,enabled tinyint(4), userprofileid int, primary key(id), foreign
key(userprofileid) references userprofile(id));
create table role(id int not null auto_increment, role varchar(255), primary
key(id));
create table user_role(user_id int not null, role_id int not null, primary
key(user_id, role_id));
Building a docker image for Spring Boot Application along with mysql
To dockerize my spring boot application, we will use a maven plugin to build a docker image.
<plugin>
<groupId>com.spotify</groupId>
<artifactId>docker-maven-plugin</artifactId>
<version>1.0.0</version>
<configuration>
<imageName>${docker.image.prefix}/benefits</imageName>
<dockerHost>https://192.168.99.100:2376</dockerHost>
<dockerCertPath>C:\Users\Yogesh
Mali\.docker\machine\machines\default</dockerCertPath>
<dockerDirectory>src/main/docker</dockerDirectory>
<resources>
<resource>
<targetPath>/</targetPath>
<directory>${project.build.directory}</directory>
<include>${project.build.finalName}.jar</include>
</resource>
</resources>
</configuration>
</plugin>
I am passing dockerDirectory where Dockerfile will be stored to build our image. Also another
change I have made to my original pom file, is that i have added packaging as jar.
<groupId>com.betterjavacode</groupId>
<artifactId>Benefits</artifactId>
<packaging>jar</packaging>
<version>0.0.1-SNAPSHOT</version>
.................
<plugin>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-maven-plugin</artifactId>
<configuration>
<mainClass>com.betterjavacode.benefits.Application</mainClass>
</configuration>
<executions>
<execution>
<goals>
<goal>repackage</goal>
</goals>
</execution>
</executions>
</plugin>
I have also changed in my application.properties to point to mysql database container by updating
spring.datasource.url=jdbc:mysql://192.168.99.100:3308/benefitsmysql
FROM java:8
VOLUME /tmp
EXPOSE 8443
ENTRYPOINT ["java","-Djava.security.egd=file:/dev/./urandom","-jar","/Benefits.jar"]
Basically this will build a Benefits.jar using java 8 and will expose port 8443 that I can use to
access my application.
This will execute the jar built within that container. Important to note here is --link as it links other
container where we have mysql server deployed. So we are linking two containers and we will call
the database from our spring boot application container. The same command can be used little
Once the application starts successfully, we will access our application with
In this post, we showed how we can deploy Spring boot application connected to mysql on a docker
container.
11.References
1. https://www.docker.com/
2. https://stackoverflow.com/questions/45952291/connection-
refused-accessing-a-spring-boot-application-running-in-
docker-contai
3. https://github.com/jiwhiz/spring-boot-docker-mysql/wiki
4. https://kielczewski.eu/2014/12/spring-boot-security-
application/
5. https://spring.io/blog/2015/01/12/the-login-page-angular-js-
and-spring-security-part-ii
6. http://microservices.io/patterns/microservices.html
7. https://martinfowler.com/articles/microservices.html
8. https://stackoverflow.com/questions/33041733/microservices
-vs-monolithic-architecture
9. http://briansjavablog.blogspot.ca/2013/01/spring-web-
services-tutorial.html
10. http://www.concretepage.com/spring-4/spring-4-soap-
web-service-producer-consumer-example-with-tomcat
11. https://stackoverflow.com/questions/20210993/consumi
ng-a-web-service-in-java-and-soapui