- Author: SĂ©bastien Mosser [email protected]
- Reviewer: Anne-Marie DĂ©ry [email protected]
- Version: 02.2016
This case study is used to illustrate the different technologies involved in the Introduction to Software Architecture course given at Polytech Nice - Sophia Antipolis at the graduate level. This demonstration code requires the following software to run properly:
- Build & J2E environment configuration: Maven 3
- J2E implementation language: Java 8
- .Net implemenyation language: Mono >3.12
To compile the demonstration (j2e, .Net and client parts), simply run the compilation script. The first compilation can take (a lot of) time, considering that Maven will have to download all the java dependencies necessary to build and run the system (the application server weights 43Mb):
mosser@azrael $ ./buildAll.sh
To run the demonstration, first start the two servers in two different terminals, then start the remote client in a third one:
# J2E terminal (^C to stop)
mosser@azrael $ cd j2e
mosser@azrael j2e$ mvn tomee:run
# .Net terminal (return to stop)
mosser@azrael $ cd dotNet
mosser@azrael dotNet$ mono server.exe
# Remote Client (bye to stop)
mosser@azrael $ cd client
mosser@azrael client$ mvn exec:java
The Cookie Factory (TCF) is a major bakery brand in the USA. The Cookie on Demand (CoD) system is an innovative service offered by TCF to its customer. They can order cookies online thanks to an application, and select when they'll pick-up their order in a given shop. The CoD system ensures to TCF's happy customers that they'll always retrieve their pre-paid warm cookies on time.
The system is defined as layers:
- A remote client (green) , that will run on each customer's device;
- A J2E kernel (blue), implementing the business logic of the CoD system;
- An interoperability layer (grey) between the client and the kernel, implemented as SOAP-based web services;
- An external partner (orange, implemented in .Net), communicating with the CoD system through a Web Service.
To deliver the expected features, the coD system defines the following interfaces:
CartModifier
: operations to handle a given customer's cart, like adding or removing cookies, retrieving the contents of the cart and validating the cart to process the associated order;CustomerFinder
: a finder interface to retrieve a customer based on her identifier (here simplified to her name);CustomerRegistration
: operations to handle customer's registration (users profile, ...)CatalogueExploration
: operations to retrieve recipes available for purchase in the CoD;OrderProcessing
: process an order (kitchen order lifecycle management);Payment
: operations related to the payment of a given cart's contents;Tracker
: order tracker to retrieve information about the current status of a given order.
The business objects are simple: Cookies
are defined as an enumerate, binding a name to a price. An Item
models the elements stored inside a cart, i.e., a given cookie and the quantity to order. A Customer
makes orders thanks to the CoD system, and an Order
stores the set of items effectively ordered by the associated customer (bidirectional association).
As the focus of the course is an Introduction to Software Architecture, we made the choice to go as lightweight as possible with respect to the tooling. As a consequence, we decided not to deploy a real set of application servers and use embedded artifacts instead. This is the very justification of using TomEE+ as J2E container (instead of a classical Tomcat or Glassfish container) and Mono as .Net implementation (instead of the classical Visual Studio technological stack). We defend that the execution details are not important when compared to the complexity of designing the right system. In addition, mapping this demonstration to existing application servers is pure engineering, with no added value.
Regarding the IDE support, this demonstration was designed using IntelliJ 15 Ultimate. The community version does not include the J2E and Web Service technological stack. However, the only constraint on the IDE is that it must support the generation of stubs code from WSDL contracts.
The J2E part of the TCF system is defined as a Maven project, in the j2e
directory. The pom.xml
file declares a set of dependency to support EJB development, as well as the configuration of the TomEE+ application server to smoothly deploy the implemented beans. As the system is implemented as a WAR artifact for deployment purpose, we need to declare an empty web.xml file in the webapp/WEB-INF
directory. The unit tests are implemented as JUnit tests (classical), combined with the Arquilian framework to support the testing of components deployed in an application server. The configuration of Arquilian for test purpose is declared in the src/test/resource
directory (file named arquilian.xml
).
Warning: Starting the backend with mvn tomee:run
will not trigger a compilation of the system if the backend was previously built. You'll have to invoke mvn package tomee:run
to force maven to recompile the system.
The client acts as an interactive command-line. We store the WSDL contracts as java resources (src/main/resources
). The package cli.framework
defines a very simple interactive shell, and the package cli.commands
implements the different commands available in the shell for the customer.
We focus here on the implementation of our first component, dedicated to handle customer's carts. The component is very basic, as it only exposes 3 operations: (i) adding an Item to a given customer's cart, (ii) removing an item in such a cart and (iii) retrieving the contents of a given cart. The definition of this component is modeled as a classical Java interface, annotated as @Local
as we are implementing a local component.
@Local
public interface Cart {
boolean add(Customer c, Item item);
boolean remove(Customer c, Item item);
Set<Item> contents(Customer c);
}
The simple way to implement this service is to rely on a Stateful bean. The semantics of such class of beans is that each artifact connected to a given instance of the bean will always talk to the same instance. It implies for the J2E container to support a session between the caller and the callee, which consumes memory, introduce a bottleneck and prevent load-balancing. However, this is only the first step, so let's go easy for this one.
The implementation of the component is straightforward, using a Map to store the binding that exists between customers and items. The state-full property ensures that we'll always talk to the same map.
@Stateful
public class CartStateFullBean implements Cart {
private Map<Customer, Set<Item>> carts = new HashMap<>();
@Override
public boolean add(Customer c, Item item) {
Set<Item> items = contents(c);
Optional<Item> existing = items.stream().filter(e -> e.getCookie().equals(item.getCookie())).findFirst();
if(existing.isPresent()) {
items.remove(existing.get());
Item toAdd = new Item(item.getCookie(), item.getQuantity() + existing.get().getQuantity());
if(toAdd.getQuantity() > 0) { items.add(toAdd); }
} else {
items.add(item);
}
carts.put(c, items);
return true;
}
@Override
public boolean remove(Customer c, Item item) {
return add(c, new Item(item.getCookie(), -item.getQuantity()));
}
@Override
public Set<Item> contents(Customer c) {
return carts.getOrDefault(c, new HashSet<Item>());
}
}
The previously implemented component should ensure the four following properties: (i) the cart of a given customer is empty by default, (ii) adding multiple items results in a cart containing such items, (iii) one can remove cookies from a cart and finally (iii) one can modify the already existing quantity for a given item. Considering a given Cart
named cart
, the test implementation is also straightforward.
@Test public void emptyCartByDefault() {
Customer c = new Customer(UUID.randomUUID().toString());
Set<Item> data = cart.contents(c);
assertArrayEquals(new Item[] {}, data.toArray());
}
@Test public void addItems() {
Customer john = new Customer("john");
cart.add(john, new Item(Cookies.CHOCOLALALA, 2));
cart.add(john, new Item(Cookies.DARK_TEMPTATION, 3));
Item[] oracle = new Item[] {new Item(Cookies.CHOCOLALALA, 2), new Item(Cookies.DARK_TEMPTATION, 3) };
assertEquals(new HashSet<>(Arrays.asList(oracle)), cart.contents(john));
}
@Test public void removeItems() {
Customer john = new Customer("john");
cart.add(john, new Item(Cookies.CHOCOLALALA, 2));
cart.remove(john, new Item(Cookies.CHOCOLALALA, 2));
assertArrayEquals(new Item[] {}, cart.contents(john).toArray());
cart.add(john, new Item(Cookies.CHOCOLALALA, 6));
cart.remove(john, new Item(Cookies.CHOCOLALALA, 5));
assertArrayEquals(new Item[] {new Item(Cookies.CHOCOLALALA, 1)}, cart.contents(john).toArray());
}
@Test public void modifyQuantities() {
Customer john = new Customer("john");
cart.add(john, new Item(Cookies.CHOCOLALALA, 2));
cart.add(john, new Item(Cookies.DARK_TEMPTATION, 3));
cart.add(john, new Item(Cookies.CHOCOLALALA, 3));
Item[] oracle = new Item[] {new Item(Cookies.CHOCOLALALA, 5), new Item(Cookies.DARK_TEMPTATION, 3) };
assertEquals(new HashSet<>(Arrays.asList(oracle)), cart.contents(john));
}
This code is purely functional, assuming a Cart
(the interface, no one cares about the concrete implementation). But as the Cart is a component, its lifecycle is handled by the J2E container. As a consequence, we need three to run this very test inside a container, on a deployed component. Additional information to make this test a working one is needed: (i) define how elements can be packaged into a deployable unit, (ii) inject a Cart inside this unit and (iii) asks for the Arquillian test runner instead of the classical JUnit one to run the test inside a deployed version of our system.
@RunWith(Arquillian.class)
public class CartTest {
// Classes to package into a deployable unit used to run the test
@Deployment public static WebArchive createDeployment() {
return ShrinkWrap.create(WebArchive.class)
.addAsWebInfResource(EmptyAsset.INSTANCE, "beans.xml")
// Business Objects
.addPackage(Customer.class.getPackage())
// Components interfaces
.addPackage(Cart.class.getPackage())
// Component implementation
.addPackage(CartStateFullBean.class.getPackage());
}
// Injecting a Cart inside the test suite
@EJB private Cart cart;
// test code goes here ...
}
You must remark that the Cart
is never initialized. This is how dependency injection works. The container analyzes the @EJB
annotation and will bind your local variable to an instance a component that respect this interface, at runtime. It is not your responsibility anymore to instantiate objects when they implement EJBs.
To run the tests from your IDE, you must configure the environment to be Arquillian-compliant (i.e., started inside a J2E container). IntelliJ users simply have to answer to the questions asked by the IDE, selecting a TomEE container when asked for. This also allows one to activate the debug mode while testing. If you decide to go for the latests versions of the different APIs (i.e., using SNAPSHOTs versions in your POM), there is no IDE integration provided, you'll only rely on Maven for test execution, and no debug mode will be available.
The previously described component is actually wrong. It works as a client will always be connected to the very same instance of the bean, but multiple beans will not share the same in memory map, as each one contains a partial information. In addition, being state-full implies to maintain a session between the caller and the callee, which is a performance-killer for the container as it prevent the bean management process to properly handle the component lifecycle.
Going stateless means to expose a bean that does not store itself customer's information. These information will be stored in an in-memory Database, shared by all the cart beans.
We first define an abstract cart bean, that will factorize the commonality existing between our two implementations.
public abstract class AbstractCartBean implements Cart {
@Override
public final boolean remove(Customer c, Item item) {
return add(c, new Item(item.getCookie(), -item.getQuantity()));
}
/**
* Protected method to update the cart of a given customer, shared by both stateful and stateless beans
*/
protected Set<Item> updateCart(Customer c, Item item) {
Set<Item> items = contents(c);
Optional<Item> existing = items.stream().filter(e -> e.getCookie().equals(item.getCookie())).findFirst();
if(existing.isPresent()) {
items.remove(existing.get());
Item toAdd = new Item(item.getCookie(), item.getQuantity() + existing.get().getQuantity());
if(toAdd.getQuantity() > 0) { items.add(toAdd); }
} else {
items.add(item);
}
return items;
}
}
We then implement a shared database, using an in-memory map. This database must be a singleton, as it will be shared by all the beans that exists in the system.
@Singleton
public class Database {
private Map<Customer, Set<Item>> carts = new HashMap<>();
public Map<Customer, Set<Item>> getCarts() { return carts; }
public void flush() { carts = new HashMap<>(); }
}
We can now implement a stateless bean, that will interact with the database (thanks to the @EJB
annotation). As the state-full and state-less beans share the same component interface, we need to name the bean (using the name
argument) so that one can explicitly refer to a given implementation if needed.
@Stateless(name = "cart-stateless")
public class CartStateLessBean extends AbstractCartBean {
@EJB Database memory;
@Override
public boolean add(Customer c, Item item) {
memory.getCarts().put(c, updateCart(c, item));
return true;
}
@Override
public Set<Item> contents(Customer c) {
return memory.getCarts().getOrDefault(c, new HashSet<Item>());
}
}
The test suite must be updated to refer to these changes. First, we have to explicitly load the stateless bean, using the name
argument of the @EJB
annotation. Then, we need to modify the WebArchive deployable unit to include the database (stored in a package named utils
). Finally, we need to flush the database before each test using the classical @Before
mechanism available in Junit.
@Deployment public static WebArchive createDeployment() {
return ShrinkWrap.create(WebArchive.class)
.addAsWebInfResource(EmptyAsset.INSTANCE, "beans.xml")
.addPackage(Database.class.getPackage())
.addPackage(Customer.class.getPackage())
.addPackage(Cart.class.getPackage())
.addPackage(CartStateFullBean.class.getPackage());
}
@EJB private Database memory;
@EJB(name = "cart-stateless") private Cart cart;
@Before public void flushDatabase() { memory.flush(); }
// test code goes here, unchanged.
Warning: When a component implements an interface, it is supposed to properly implement the contract. Thus, if the container does not find a bean with the given name, it will chose one randomly among the available one, as it should be semantically equivalent.
The Cookie Factory system must be exposed to remote clients, so that a customer can order some cookies through the service. Two options: (i) exposing the EJBs as remote ones, or (ii) expose these beans through a Web Service. The first option implies for the clients to be J2E-compliant, when the second option will allows clients to be developed in any language.
The other advantage of exposing the internal system thanks to a set of satellite web services is the decoupling ensured by this approach. The operations exposed by the web service are not the exact reflect of the associated bean, a web service can combine multiple beans to add value to the system. The Web Service layer is the public API of our architecture.
Warning: Implementing web services implies to respect a set of constraints:
- Web services are stateless, according to the WS standard. As a consequence, any beans exposed to the outside world through a Web Service must be stateless.
- Business objects exposed through the public API must be serializable, and must define an empty constructor and get/set methods to be properly populated by the WE framework
- The TomEE container must
Here, we decide to expose two operations in the public API: adding a cookie to a cart and getting cart contents. We exposed the Item
business object in these methods as it exactly models what we are using in these methods. But the Customer
business class is purely internal, it does not make any sense to expose to the outside world our complete representation of a Customer. Thus, we use a plain String to identify the customer in the public API.
A Web service contract is defined by an annotated interface. The annotations are straightforward:
- A
WebMethod
annotation tags the methods to expose as service operations; - A
WebParam
annotation tags the parameters to change their name, or handle xml namespace manually; - A
WebResult
annotation tags the returned value, like@WebParam
.
@WebService
public interface CartWebService {
@WebMethod
void addItemToCustomerCart(@WebParam(name = "customer_name") String customerName,
@WebParam(name = "item") Item it);
@WebMethod
@WebResult(name = "cart_contents")
Set<Item> getCustomerCartContents(@WebParam(name = "customer_name") String customerName);
}
The implementation of these operations is simply done by a concrete class that implements the interface and delegates business work to the bean. This concrete class is actually a stateless bean that consumes another bean.
@WebService(targetNamespace = "http://www.polytech.unice.fr/si/4a/isa/tcf")
@Stateless(name = "CartWS")
public class CartWebServiceImpl implements CartWebService {
@EJB(name="stateless-cart") Cart cart;
@Override
public void addItemToCustomerCart(String customerName, Item it) {
cart.add(new Customer(customerName), it);
}
@Override
public Set<Item> getCustomerCartContents(String customerName) {
return cart.contents(new Customer(customerName));
}
}
The application container will support the exposition of this class according the the Web Service standard. To start the container in server mode, use the mvn tomee:run
We create a remote client for the service, in a new maven project (client
) directory.
The contract of the service, exposed using the Web Service Description Language (WSDL), is automatically exposed by TomEE: http://localhost:8080/tcf-backend/webservices/CartWS?wsdl.
Copy paste the contents of this contract (use the view source option in your browser) into a local file located inside the client project, for example in src/main/resources/CartWS.wsdl
. Then, use the web services capabilities of your IDE to generate the Java code that will support the interactions with the service. For IntelliJ Ultimate users (the community edition does not contains the web service stack), right-click on the WSDL contract, then select Web Services and Generate Java from WSDL. The Java files generated from the WSDL contract will be stored inside a given package (e.g., stubs.cart
).
Warning: Obviously, each time the service contract will change, you'll have to rerun the stub generation process on the latest WSDL contract.
To instantiate the stub that will support the communication between the client and the service, we simply ask the generated stub code to do so (in the class CartWSDemo
):
private static CartWebService initialize() {
System.out.println("#### Instantiating the WS Proxy");
CartWebServiceImplService factory = new CartWebServiceImplService();
CartWebService ws = factory.getCartWebServiceImplPort();
return ws;
}
From the client perspective, the code to be used to consume the web service is pretty simple: it is a Java version of the exposed contract. One can remark that some information is lost during the Java to WSDL to Java contract generation then stub generation process. For example, the cart is modeled as a Set on the server side, and was translated into a List on the client side.
public static void main() {
CartWebService ws = initialize();
List<Item> cart = ws.getCustomerCartContents("john");
System.out.println("Cart is empty: " + cart.isEmpty());
Item i = new Item();
i.setCookie(Cookies.CHOCOLALALA); i.setQuantity(3);
ws.addItemToCustomerCart("john", i);
i.setCookie(Cookies.DARK_TEMPTATION); i.setQuantity(2);
ws.addItemToCustomerCart("john", i);
i.setCookie(Cookies.CHOCOLALALA); i.setQuantity(4);
ws.addItemToCustomerCart("john", i);
cart = ws.getCustomerCartContents("john");
System.out.println("John's cart: " +cart);
}
The previously described code works well, but rely on a very string assumption: the service will always be located at the very same location (on localhost). Moreover, the client will load at runtime the WSDL contract, so if one moves the contract elsewhere, the client code does not work anymore. And it is anyway not reasonable to package a distributed application that will only run on localhost. We need to clean our default implementation to be more aware of the server location.
First point, look at the WSDL contract. Even if stored locally as a resource, it refers to a remote file located at http://localhost:8080/tcf-backend/webservices/CartWS?wsdl=CartWebService.wsdl. We store this file as a local one, side by side with the initial contract, in a file named CartWSType.wsdl
(as it basically defines the data types associated to our contract). Then, we edit the CartWS.wsdl
file to point to this local file instead of the remote one (in the wsdlLocation
attribute). We should now edit the initialization code to refer to our local file instead of the remote one.
private static CartWebService initialize() {
System.out.println("#### Loading the WSDL contract");
URL wsdlLocation = CartWSDemo.class.getResource("/CartWS.wsdl");
System.out.println("#### Instantiating the WS Proxy");
CartWebServiceImplService factory = new CartWebServiceImplService(wsdlLocation);
CartWebService ws = factory.getCartWebServiceImplPort();
return ws;
}
It only solves half of the problem. Now, the client code will load at runtime the local WSDL contract, but it will always try to consume a service deployed on localhost. We need to configure the endpoint at runtime to point anywhere the service will be deployed. We apply the following assumption: by default it is deployed in development mode on localhost, and if command line arguments are provided, the first one is the hostname and the second one, if provided, is the port number.
private static CartWebService initialize(String host, String port) {
System.out.println("#### Loading the WSDL contract");
URL wsdlLocation = CartWSDemo.class.getResource("/CartWS.wsdl");
System.out.println("#### Instantiating the WS Proxy");
CartWebServiceImplService factory = new CartWebServiceImplService(wsdlLocation);
CartWebService ws = factory.getCartWebServiceImplPort();
System.out.println("#### Updating the endpoint address dynamically");
String address = "http://"+host+":"+port+"/tcf-backend/webservices/CartWS";
((BindingProvider) ws).getRequestContext().put(BindingProvider.ENDPOINT_ADDRESS_PROPERTY, address);
return ws;
}
The bank used by TCF to support payments in CoD implements its payment service as a REST one. This service exposes the following resources:
request
, a singleton resource that acceptPOST
requests used to postPaymentRequest
s to the Bank;payments
, a list of all payment identifiers available in the system;payments/{id}
, the description of a given payment;
Remark: Contrarily to the previous service that was SOAP-based, this one is resource-oriented. The difference between the two paradigm is essential: the previous one exposes procedures (aka Remote Procedure Call, RPC), where this one exposes resources (i.e., nouns instead of verbs).
The Web Service is implemented in the dotNet/src
directory. The compilation script generates a self-hosted server (in a file named server.exe
) that starts a web server and binds the requested URIs to the defined operations.
The description of the service interface is straightforward:
[ServiceContract]
public interface IPaymentService
{
[OperationContract]
[WebInvoke( Method = "POST", UriTemplate = "mailbox", RequestFormat = WebMessageFormat.Json, ResponseFormat = WebMessageFormat.Json)]
int ReceiveRequest(PaymentRequest request);
[OperationContract]
[WebInvoke( Method = "GET", UriTemplate = "payments/{identifier}", ResponseFormat = WebMessageFormat.Json)]
Payment FindPaymentById(int identifier);
[OperationContract]
[WebInvoke( Method = "GET", UriTemplate = "payments", ResponseFormat = WebMessageFormat.Json)]
List<int> GetAllPaymentIds();
}
The implementation is also trivial. We use a map instantiated as an instance variable to implement persistence. It makes the service stateful, which is an anti-pattern and only make sense as we are creating a Proof of Concept.
To start the service hosting server, simply run mono server.exe
. Command line parameters can be used to configure the port number for example.
As the system relies on simple HTTP requests, one can invoke the PaymentService using any HTTP client, just by specifying the right headers and body for a given request. For example using cURL, here are the different commands one can use:
azrael:~ mosser$ REQUEST='{ "CreditCard": "1234-896983", "Amount": 12.09 }'
azrael:~ mosser$ BASE_URL="http://localhost:9090"
azrael:~ mosser$ HEADERS='Content-Type: application/json'
azrael:~ mosser$ curl -i -w "\n" -H "$HEADERS" -X POST -d "$REQUEST" $BASE_URL/mailbox
HTTP/1.1 200
Content-Type: application/json; charset=utf-8
Server: Mono-HTTPAPI/1.0
Date: Thu, 25 Feb 2016 09:24:41 GMT
Content-Length: 1
Keep-Alive: timeout=15,max=100
1
azrael:~ mosser$ curl -i -w "\n" -H "$HEADERS" -X GET $BASE_URL/payments
HTTP/1.1 200
Content-Type: application/json; charset=utf-8
Server: Mono-HTTPAPI/1.0
Date: Thu, 25 Feb 2016 13:24:15 GMT
Content-Length: 7
Keep-Alive: timeout=15,max=100
[1,2,3]
azrael:~ mosser$ curl -i -w "\n" -H "$HEADERS" -X GET $BASE_URL/payments/1
HTTP/1.1 200
Content-Type: application/json; charset=utf-8
Server: Mono-HTTPAPI/1.0
Date: Thu, 25 Feb 2016 13:24:29 GMT
Content-Length: 100
Keep-Alive: timeout=15,max=100
{"Amount":12.09,"CreditCard":"1234-896983","Date":"25\/02\/2016 10:24:41","Identifier":0,"Status":0}
azrael:~ mosser$
The J2E system consumes the Bank service. As a consequence, our EJBs will act as clients of this service. We rely on the Apache CXF library to consume REST web services. We implement the methods that support the communication with the service in a utility class named BankAPI
:
private Integer pay(JSONObject body) {
String str = client().path("/mailbox")
.accept(MediaType.APPLICATION_JSON_TYPE)
.header("Content-Type", MediaType.APPLICATION_JSON)
.post(body.toString(), String.class);
return Integer.parseInt(str);
}
private JSONObject getPaymentStatus(Integer id) {
String response = client().path("/payments/" + id).get(String.class);
JSONObject payment = new JSONObject(response);
return payment;
}
private boolean isValid(JSONObject payment) {
return (payment.getInt("Status") == 0);
}
The CashierBean
class uses an instance of the BankAPI
class to interact with the remote bank service. The endpoint cannot be hardcoded in its source code. As a consequence, we define a bank.properties
file in the resources
directory, which will defined the hostname and port number to be used when interacting with the Bank. In the CashierBean
, we use a @PostConstruct
annotation to load these properties from the resource file after the bean initialization:
@PostConstruct
private void initializeRestPartnership() throws IOException {
Properties prop = new Properties();
prop.load(this.getClass().getResourceAsStream("/bank.properties"));
bank = new BankAPI( prop.getProperty("bankHostName"),
prop.getProperty("bankPortNumber"));
}
We are now facing an important issue: the J2E kernel is strongly coupled to the .Net system. One need to start the .Net server to make the J2E system available for tests purpose. To isolate the two systems for tests purpose, we need to mock the BankAPI instead of using the real one.
@Before
public void setUpContext() {
memory.flush();
items = new HashSet<>();
items.add(new Item(Cookies.CHOCOLALALA, 3));
items.add(new Item(Cookies.DARK_TEMPTATION, 2));
// Customers
john = new Customer("john", "1234-896983"); // ends with the secret YES Card number
pat = new Customer("pat", "1234-567890"); // should be rejected by the payment service
// Mocking the external partner
BankAPI mocked = mock(BankAPI.class);
cashier.useBankReference(mocked);
when(mocked.performPayment(eq(john), anyDouble())).thenReturn(true);
when(mocked.performPayment(eq(pat), anyDouble())).thenReturn(false);
}
But we also need to implement Integration Tests, that will ensure the end to end connection between our two systems. We rely on Maven to implement such a behavior:
- classical Unit tests are always run (_e.g., when invoking
mvn package
) - Integration tests will be run during the
integration-test
phase.
We will differentiate classical tests and integration ones using a file name prefix: integration tests will match *IntegrationTest
. In the pom.xml
file we rely on the following configuration to implement these specifications:
<plugin>
<groupId>org.apache.maven.plugins</groupId>
<artifactId>maven-surefire-plugin</artifactId>
<version>2.17</version>
<configuration>
<reuseForks>false</reuseForks>
<excludes>
<exclude>**/*IntegrationTest.java</exclude>
</excludes>
</configuration>
<executions>
<execution>
<id>integration-test</id>
<goals>
<goal>test</goal>
</goals>
<phase>integration-test</phase>
<configuration>
<excludes>
<exclude>**/*Test.java</exclude>
</excludes>
<includes>
<include>**/*IntegrationTest.java</include>
</includes>
</configuration>
</execution>
</executions>
</plugin>
Note: To checkout this version, be sure that you are browsing the code stored in the volatile
branch.
Interfaces are defined in the main package, as classical Java interfaces. Components are implemented in the components
sub-package, in classes with Bean
postfixed names. When a component refers to another one according to a provides/requires association, the component implementation refers to the associated Interface instead of the concrete implementation to ensure decoupling.
The Cart
component is implemented twice, first as a Stateful
bean, and then as a Stateless
one (in the cart
sub-package).
As there is no persistent backend, we mocked the persistence layer using a Singleton
bean named Database
. It stores all the necessary data in static maps. We'll se in the next section how to remove this mock and use a real persistence layer thanks to EJB Entities.
Interceptors are used inside the application server to process the messages exchanged between the different components. But the EJB framework also allows one to develop business-oriented interceptors as a support for cross-cutting features that does not fit easily inside components (e.g., code duplication).
For statistical purpose, TCF wants to count the number if processed carts. Each time the validate
operation is invoked on a Cart
, a counter will be incremented is the cart processing process went well.
One can implement this feature directly inside the validate
method. However, it will pollute the business-orientation if this implementation with a very technical concern. Thus, we decide to implement this feature as an interception. The interceptor will be located around the invocation of validate
, and will implement the following algorithm: (i) proceed to your normal behavior, (ii) if it does not throw any exception, increment the statistics counter in the database.
public class CartCounter implements Serializable {
@EJB private Database memory;
@AroundInvoke
public Object intercept(InvocationContext ctx) throws Exception {
Object result = ctx.proceed(); // do what you're supposed to do
memory.incrementCarts();
System.out.println(" #Cart processed: " + memory.howManyCarts());
return result;
}
}
To put the interceptor around our business code, we only need to annotate the expected method:
@Override
@Interceptors({CartCounter.class})
public String validate(Customer c) throws PaymentException {
return cashier.payOrder(c, contents(c));
}
One can put multiple interceptors on the very same method, they will be executed in sequence from left to right. If the bean is stateful, the interceptor must be passivation-compliant, i.e., implement the Serializable interface.
Another interesting usage of interceptors is to implement fault detection. For example, adding or removing to a given cart an Item
with a negative or null amount of cookies does not make any sense. We can then intercept the invocation of the CartWebService
operations and check the item
parameter:
public class ItemVerifier {
@AroundInvoke
public Object intercept(InvocationContext ctx) throws Exception {
Item it = (Item) ctx.getParameters()[1];
if (it.getQuantity() <= 0) {
throw new RuntimeException("Inconsistent quantity!");
}
return ctx.proceed();
}
}
In the CartWebService
interface description, we annotate the add
and remove
operations to declare the interception.
@WebMethod
@Interceptors({ItemVerifier.class})
void addItemToCustomerCart(@WebParam(name = "customer_name") String customerName,
@WebParam(name = "item") Item it)
throws UnknownCustomerException;
@WebMethod
@Interceptors({ItemVerifier.class})
void removeItemToCustomerCart(@WebParam(name = "customer_name") String customerName,
@WebParam(name = "item") Item it)
throws UnknownCustomerException;
Remark: The invocation context can be modified by an interceptor, e.g., parameters can be modified.
We consider here a tracer that will log each operation invoked inside the system. This logger is very simple to implement.
public class Logger implements Serializable {
@AroundInvoke
public Object methodLogger(InvocationContext ctx) throws Exception {
String id = ctx.getTarget().getClass().getSimpleName() + "::" + ctx.getMethod().getName();
System.out.println("*** Logger intercepts " + id);
try {
return ctx.proceed();
} finally {
System.out.println("*** End of interception for " + id);
}
}
}
It does not make any sense to manually annotate all the operations designed inside our system. In a file named ejb-jar.xml
(in the resources
directory), we simply define a regular expression associated to this very interceptor. The container will map the interceptor to any bean that match the given regular expression. In our case, we want to catch all the method, and the regular expression is quite simple: *
:
<?xml version="1.0" encoding="UTF-8"?>
<!DOCTYPE ejb-jar PUBLIC '-//Sun Microsystems, Inc.//DTD Enterprise JavaBeans2.0//EN' 'http://java.sun.com/dtd/ejb-jar_2_0.dtd'>
<ejb-jar>
<assembly-descriptor>
<interceptor-binding>
<ejb-name>*</ejb-name>
<interceptor-class>fr.unice.polytech.isa.tcf.interceptors.Logger</interceptor-class>
</interceptor-binding>
</assembly-descriptor>
</ejb-jar>
This reference implementation demonstrates the following points with respect to the Introduction to Software Architecture course objectives:
- Modeling a component-based architecture focused on offered and required functional interfaces;
- Implementing such components using (stateless) EJB Sessions with J2E;
- Using SOAP-based Web Services as an interoperable layer to integrate heterogeneous technologies through Remote Procedure Call (RPC):
remote client <--> J2E
; - Using REST-based Web Services as an interoperable layer to integrate heterogeneous technologies through Resource exposition:
J2E <--> .Net
; - Consuming web services (SOAP & Rest) from remote clients (B2C or B2B);
- Using EJB entities to support the implementation of a persistence layer;
- Using interceptors to work at the message (invocation context) level;
- Differentiating Unit tests and Integration tests using Maven.
- Use "real" application servers to support TCF deployment (Tomcat, IIS)
- See the DevOps course contents
- Use a "real" database server (e.g. postgres, MySQL)
- See the DevOps course contents
- Use an Enterprise Service Bus (ESB) to decouple the J2E system from the .Net one
- Consider to attend the Service Integration course next year (id: SOA-1).