Advanced Application Clients

This chapter introduces advanced modules of an application client.

1. Overview

A JEUS client can be executed as simply as a Java class is executed without using a client container described in the Jakarta EE specifications. This chapter discusses the types and JNDI default binding names of resources, for which a client container performs dependency injection. It also describes security and transaction services that can be used by application clients and clients that run without a client container.

If you execute a JEUS client without using a client container, the client cannot use the dependency injection service of the client container. With security settings, you can execute JEUS’s various services. With transaction functions, you can manage the resources and applications as global transactions.

2. Dependency Injection

This section describes injection details common to application clients, web applications, and EJB applications.

Injection can be performed for resources such as an environment variable, which can be mapped to an EJB object. In general, a resource’s name can be found in java:comp/env context, a JNDI context of the application component. Because the actual resource is bound to the JNDI global context, you need to know the global binding name.

Resources have their own JNDI global binding names. The JNDI global binding name of an EJB application should be specified by one of the following methods:

  • <export-name> of jeus-ejb-dd.xml

  • <mapped-name> of ejb-jar.xml in the standard

  • mappedName of an annotation specified in the EJB application

  • EJB is bound to the default JNDI name of JEUS specified in JEUS EJB Guide

To obtain an EJB through JNDI without a client container, it is necessary to understand the rules for setting the default JNDI name. Because it is hard for a developer to know to which name the EJB will be bound to, it is recommended to specify a JNDI binding name using either of the suggested ways.

To inject a resource, specify a JNDI global binding name in jeus-ejb-dd.xml, jeus-web-dd.xml, and jeus-client-dd.xml, which are JEUS DD files. You can also map the resource to the value specified by mapped-name in ejb-jar.xml, web.xml or application-client.xml, or to the mappedName in the annotation. If nothing is specified, use a name according to the basic rules of JEUS.

In the actual development, it is recommended to use XML in specifying the JNDI global binding name, rather than using the mappedName of the annotation. If the application will be executed in many places, XML should be used since it needs to use a global name according to the environment. Injection is normally done for annotated setter methods and variables. However, it is possible to perform injection for non-annotated setter methods and variables, if they are specified by XML descriptor.

  1. For more information about injection, refer to Jakarta EE 9 Platform, and see "5. Resources, Naming and Injection" section of this guide about resources that can be injected.

  2. For EJBContext injection for EJB applications, refer to JEUS EJB Guide.

The following example shows how a client injects an EJB application using a mappedName of the annotation. Since statelessEJB1 application uses 'MyEJB1' as its JNDI global binding name, the client specifies the same name for mappedName of the @EJB annotation. If the client uses JNDI Lookup instead of Injection, it is possible to directly use the JNDI global binding name for lookup. If the client runs in a client container, it is possible to use the name, java:comp/env/ejb/sless1, from the application context using the application-client.xml file.

If the client uses JNDI Lookup instead of Injection, it is possible to directly use the JNDI global binding name for lookup. If the client runs in a client container, it is possible to use the name, java:comp/env/ejb/sless1, from the application context using the application-client.xml file.

EJB Reference Injection
import ejb1.RemoteSession;

@Stateless(name="StatelessEJB1", mappedName="MyEJB1")
public class StatelessEJB1 implements RemoteSession, LocalSession {...
}
...

@EJB(name="sless1", beanName="StatelessEJB1", mappedName="MyEJB1")
private RemoteSession sless1;
...
RemoteSession session = context.lookup("MyEJB1");
...
// with client container and application-client.xml descriptor
RemoteSession session = context.lookup("java:comp/env/ejb/sless1");

2.1. EJB Injection

For EJB reference injection, the following binding names are used.

  • If the variable is a business interface:

    If a mappedName is specified, the global name used for lookup should be the following format:

    mappedName + "#" + Business_Interface_Name

    In the previous example, the name will be set to MyEJB1 or MyEJB1#RemoteSession. If deployed as EAR or EJB JAR file and if ejb-link is given to the ejb-jar.xml file or if there is a beanName in the annotation, find an EJB in the same application and use its mappedName as the global name for lookup. Otherwise, find an EJB in the application with a business interface name, and use the EJB’s mappedName as the global name.

    Finally, perform JNDI Lookup with the business interface name. In the previous example, the name, 'java:global/<module-name>/MyEJB1' or 'java:global/<module-name>/MyEJB1#ejb1.RemoteSession,' is used for Lookup. In this way, use a default binding name when there is no mappedName for EJB deployment.

  • If the variable is a sub interface of EJBHome/EJBObject interface:

    If there is a mappedName, use it as the global name. If deployed as EAR or EJB JAR file and if ejb-link is given to the ejb-jar.xml file or if there is a beanName in the annotation, find an EJB in the same application and use its mappedName as the global name for lookup. Otherwise, find an EJB in the application with a variable type interface name, and use the EJB’s mappedName as the global name.

    Finally, perform JNDI Lookup with the variable type interface name.

2.2. Resource Injection

It is possible to use the @Resource annotation for resources.

  • If a mappedName is specified, use it as the resource’s JNDI global binding name for lookup.

  • If not specified, use the name value of @Resource as the JNDI global binding name.

If no name value is specified, the following format is used according to the specifications.

Application class name + / + Variable or setter method property name

In the following example, the name 'jdbc/DB2' is used for JNDI Lookup. If no name is specified, the name 'test.Client/myDataSource3' is used for lookup.

Resource injection
package test;

class Client {
    @Resource(name="jdbc/DB2") // default mapping if no mapped-name private
    javax.sql.DataSource myDataSource3;
    ...
}

When there is a name specified or a default value is specified for a name attribute that is not set, the value is mapped to the application context. However, for the actual JNDI global binding name different rules apply for each vendor. Thus for compatibility reasons, the mappedName should be used.

2.3. Other Injections

In addition, it is possible to obtain a web service object, an EntityManager object, and an EntityManagerFactory object through Injection, by using annotations like @WebServiceRef, @PersistenceUnit and @PersistenceContext.

For further details, refer to Jakarta EE Platform Specification.

3. Clients Not Using Dependency Injection

You can use JEUS resources and applications through JEUS JNDI, without using an application container. To do so, execute a client program whose class path is set to the jclient.jar file in the JEUS_HOME\lib\client directory.

Because it is impossible to use dependency injection in this case, you need to modify the sources as shown in the following example. In the example, a binding name of the EJB application should follow JEUS’s default name binding rule. The client can run on a client container. This means that all clients which do not use a client container can run on a client container.

Clients Not Using Dependency Injection: <HelloClient.java>
package helloejb;

import java.io.*;
import jakarta.ejb.EJB;
import javax.naming.Context;
import javax.naming.InitialContext;
import java.util.Hashtable;

/**
 * HelloEJB application client
 */
public class HelloClient {
    private static Hello hello;

    public static void main(String[] args) {
        try {
            Hashtable env = new Hashtable();
            env.put(Context.INITIAL_CONTEXT_FACTORY,"jeus.jndi.JNSContextFactory");
            Context context = new InitialContext(env);
            hello = (Hello) context.lookup("helloejb.Hello");

            System.out.println("EJB output : " + hello.sayHello());
        } catch (Exception ex) {
            ex.printStackTrace();
        }
    }
}

Client packaging is not performed in a client container that processes XML files. Thus, you need to create a JAR file without the standard or JEUS XML file. For deployment, copy the JAR file to any location. For execution, execute the previous client class just like executing a general Java class.

The result is as follows:

$ java -cp /jeus/lib/client/jclient.jar;./hello-client.jar helloejb.HelloClient
[2012.05.23 22:45:51][2] [t-1] [Network-0405] [Endpoint] exporting Endpoint (0:192.168.0.16:9756:-1:0x79F24F28)
[2012.05.23 22:45:51][2] [t-1] [Network-0002] [Acceptor] start to listen NonBlockingChannelAcceptor: /192.168.0.16:9756
EJB output : Hello EJB!

4. Security Setting

To use JEUS’s Jakarta EE services, a client has to provide its username and password to the client runtime in order to verify authorization for using the services. The services include EJB applications and JMS resources. The following are three ways to specify a username and password:

  • Using jeus-client-dd.xml

    If <security-info> in the jeus-client-dd.xml file is specified, a user has to sign in with the specified username and password before starting an application in the client container. After this, the application attempts authentication using this username when using JEUS services. For more information about the jeus-client-dd.xml settings, refer to JEUS XML Reference.

    Security Setting: <jeus-client-dd.xml>
    <jeus-client-dd>
        ...
       <security-info>
            <provider-node-name>jeusNode</provider-node-name>
            <user>user1</user>
            <passwd>password1</passwd>
       </security-info>
       ...
    </jeus-client-dd>
  • Using the JNDI context

    When a client creates the JNDI context for an application, the client can use the JNDI properties to sign in with a username and password, which are used for authentication. This method of providing account information is available even without a client container. For the detailed information about the JNDI settings, refer to JNDI Naming Server.

    Security Setting: <Client.java>
    Hashtable env = new Hashtable();
    env.put(Context.INITIAL_CONTEXT_FACTORY, "jeus.jndi.JNSContextFactory");
    env.put(Context.PROVIDER_URL, "192.168.0.16:9736");
    env.put(Context.SECURITY_PRINCIPAL, "user1");
    env.put(Context.SECURITY_CREDENTIALS, "password1");
    Context context = new InitialContext(env);
  • Using JEUS Security API

    You can log in using JEUS’s Security API. For more information about the Security API, refer to JEUS Security Guide.

5. Transaction

UserTransaction is used to utilize EJB client applications, JDBC DataSource, JMS connection factory and destination, and more as one global transaction or an XA transaction.

The transaction manager used by a client is either a server transaction manager or client transaction manager depending on whether it has direct control over resources.

  1. For detailed description of UserTransaction, refer to Jakarta Transaction 2.0 Specification.

  2. For detailed explanation of the transaction manager, refer to Transaction Manager.