EJB 远程客户端从 JBoss AS 7.1 迁移到 Wildfly 8.1

Posted

技术标签:

【中文标题】EJB 远程客户端从 JBoss AS 7.1 迁移到 Wildfly 8.1【英文标题】:EJB remote client migration from JBoss AS 7.1 to Wildfly 8.1 【发布时间】:2015-05-21 09:53:45 【问题描述】:

我们开发了一个培训应用程序,其中包含一个与 EJB 通信的独立 Java 客户端。工作设置包括 Windows 7 上的 JBoss AS 7.1 和通过 /bin/add-user.bat 创建的应用程序用户。

客户端是这样编码的:

Properties jndiProps = new Properties();
jndiProps.put(Context.INITIAL_CONTEXT_FACTORY, "org.jboss.naming.remote.client.InitialContextFactory");
jndiProps.put(Context.URL_PKG_PREFIXES, "org.jboss.ejb.client.naming");
jndiProps.put("jboss.naming.client.ejb.context", true);
jndiProps.put(Context.PROVIDER_URL, "remote://localhost:4447");
jndiProps.put(Context.SECURITY_PRINCIPAL, "user");
jndiProps.put(Context.SECURITY_CREDENTIALS, "xxx");
Context ctx = new InitialContext(jndiProps);
MyBeanRemote myBean = (MyBeanRemote) ctx.lookup("ejb:/training//MyBean!mypackage.MyBeanRemote");
String result = myBean.greet("John");

客户端是用类路径中的 jboss-client.jar 启动的。

现在我们尝试使用成功部署的 WildFly 8.1,但客户端失败

Exception in thread "main" java.lang.IllegalStateException: EJBCLIENT000025: No EJB receiver available for handling [appName:, moduleName:syjeews, distinctName:] combination for invocation context org.jboss.ejb.client.EJBClientInvocationContext@6e7d3146

将 JNDI 查找名称更改为在部署期间打印出来的名称(例如 java:global/training/MyBean!mypackage.MyBeanRemote)导致

Exception in thread "main" javax.naming.CommunicationException: Failed to connect to any server. Servers tried: [remote://localhost:4447 (java.net.ConnectException: Connection refused: no further information)]

在谷歌上搜索了一段时间后,我们偶然发现了几篇关于 SO(例如 this 或 that)或 samples 或 Wildfly Developer Guide 的文章,但所有替代方案,可能是最小的 JNDI 属性或扩展的通过 ClientContext 配置并没有使其工作。

所以我的问题是,需要做些什么来迁移上面的代码/配置以在 WildFly 下运行它?

注意:这不是生产代码,因此安全性不是问题 - 如果我可以简化整个配置,那很好 - 它应该只演示如何从独立的 Java 程序使用 EJB 远程接口。

【问题讨论】:

【参考方案1】:

你需要做两处改变

不要使用“remote://localhost:4447”,而是使用“http-remoting://localhost:8080”

jndiProps.put(Context.PROVIDER_URL, "http-remoting://localhost:8080");

在代码中配置 jndi 属性时,查找名称不应包含ejb:

MyBeanRemote myBean = (MyBeanRemote) ctx.lookup("/training//MyBean!mypackage.MyBeanRemote");

这个解决方案已经过测试并且可以工作

【讨论】:

嘿,非常感谢您迟到但仍然非常感谢的答案 - 我立即尝试了它并且它有效!不过,我仍然想知道,为什么不能使用 WildFly 在部署期间输出的确切字符串来完成查找。【参考方案2】:

提供者 URL 应该是 http-remoting://localhost:8080 而不是 remote://localhost:4447

【讨论】:

好的,我试过了,但它告诉我它找不到 JNDI 名称。除了上面带有ejb:/ 的原始版本之外,我还尝试了带有java:/ 的那些以及在部署EJB 时在控制台中列出的其他内容。知道还缺少什么吗? @Geziefer 尝试在日志中作为查找字符串找到的关于您的 ejb 的 java:global/... 条目。 @Francesco:以防万一:您的建议是正确的,但是必须省略命名空间(尽管我不知道为什么)。【参考方案3】:

我正在使用:

wildfly-10.0.0.Final(JBoss 应用服务器)。

这对我有用:

Properties jndiProperties = new Properties();
           jndiProperties.put(Context.INITIAL_CONTEXT_FACTORY, "org.jboss.naming.remote.client.InitialContextFactory");
           jndiProperties.put(Context.URL_PKG_PREFIXES, "org.jboss.ejb.client.naming");
           jndiProperties.put("jboss.naming.client.ejb.context", true);
           jndiProperties.put(Context.PROVIDER_URL, "http-remoting://localhost:8080");   //System.getProperty(Context.PROVIDER_URL, "http-remoting://localhost:8080"));

        /* jndiProperties.put(Context.SECURITY_PRINCIPAL, "user");
           jndiProperties.put(Context.SECURITY_CREDENTIALS, "xxx");*/

        InitialContext context = new InitialContext(jndiProperties);

        MyFirstEJBRemote ejb = (MyFirstEJBRemote) context.lookup("/EJBProjectName/MyFirstEJB!src.MyFirstEJBRemote");

希望对你有帮助!

【讨论】:

Hallo Davijr,我也得到了和你上面提到的相同的配置,但我仍然得到错误:线程“main”中的异常 javax.naming.CommunicationException:无法连接到任何服务器。服务器尝试:[http-remoting://localhost:8080 (java.io.IOException: JBREM000202: Abrupt close on Remoting connection 329100e4 to localhost/127.0.0.1:8080 of endpoint "config-based-naming-client-endpoint" )]。你有什么想法吗? 我尝试在不启动服务器的情况下运行该项目,但收到了类似的错误。所以我认为你忘记启动服务器了。 我不得不“添加另一个答案”并且它上升了。【参考方案4】:

这是我的名为“MyFirstEJBProject”的项目。 代码内部有一些注释可以提供帮助。

package src.ejb;

import javax.ejb.Stateless;

/**
 * Session Bean implementation class MyFirstEJB
 */
@Stateless
public class MyFirstEJB implements MyFirstEJBRemote 

    public MyFirstEJB() 
    

    @Override
    public String helloWorld() 
        return "Hello World EJB";
    




package src.ejb;

import javax.ejb.Remote;

@Remote
public interface MyFirstEJBRemote 
    public String helloWorld();



package src.clientTest;

import java.util.Properties;

import javax.naming.Context;
import javax.naming.InitialContext;
import javax.naming.NamingException;

import src.ejb.MyFirstEJB;
import src.ejb.MyFirstEJBRemote;

public class EJBClient 

    public static void main(String[] args) throws NamingException 

        /**JNDI or Java Naming and Directory Interface.
         * When JNDI constructs an initial context, the context's environment
         * is initialized with properties defined in the environment parameter
         * passed to the constructor, the system properties, the applet parameters,
         * and the application resource files.
         * 
         * JNDI applications need a way to communicate various preferences
         * and properties that define the environment in which naming and
         * directory services are accessed.
         * */

        Properties jndiProperties = new Properties();
        jndiProperties.put(Context.INITIAL_CONTEXT_FACTORY, "org.jboss.naming.remote.client.InitialContextFactory");
        jndiProperties.put(Context.URL_PKG_PREFIXES, "org.jboss.ejb.client.naming");
        jndiProperties.put("jboss.naming.client.ejb.context", true);
        jndiProperties.put(Context.PROVIDER_URL, "http-remoting://localhost:8080");

        /* jndiProperties.put(Context.SECURITY_PRINCIPAL, "user");
           jndiProperties.put(Context.SECURITY_CREDENTIALS, "xxx");*/

        /**Context For JNDI**/
        InitialContext context = new InitialContext(jndiProperties);

        /**The nameOfEJB appears on the console when the server is starting up.**/
        String nameOfEJB = "/MyFirstEJBProject/MyFirstEJB!src.ejb.MyFirstEJBRemote";

                                                  /**The Method .lookup("") search for EJB by name**/
        MyFirstEJBRemote ejb = (MyFirstEJBRemote) context.lookup(nameOfEJB);
        System.out.println(ejb.helloWorld());


        System.out.println("/** =============== 2º TEST ===============**/"); 

        /**getting a EJB name by a private Method.**/
        String nameOfEJB_2 = getEJBName("", "MyFirstEJBProject", "", 
                MyFirstEJB.class.getSimpleName(), MyFirstEJBRemote.class.getName());

        MyFirstEJBRemote ejb_2 = (MyFirstEJBRemote) context.lookup(nameOfEJB_2);
        System.out.println(ejb_2.helloWorld());
    

    private static String getEJBName(String nameofAppEAR, String nameOfProjectModulo, 
            String distinctNameOfProject, String classBeanSimpleName, String classInterfaceName)
        /**Return a object name for search on JNDI */

        String finalNameOfEJB = nameofAppEAR + "/" + nameOfProjectModulo + "/" + distinctNameOfProject + 
                "/" + classBeanSimpleName + "!" + classInterfaceName;  

        return finalNameOfEJB; 

        /**Ex:                 
           String nameofAppEAR= "";                            // EAR  (if there is)
           String nameOfProjectModulo= "MyFirstEJBProject";
           String distinctNameOfProject= "";                   // Alias (if there is)
           String classBeanSimpleName= MyFirstEJB.class.getSimpleName(); 
           String classInterfaceName=  MyFirstEJBRemote.class.getName();

           String finalNameOfEJB = "" + "/" + "MyFirstEJBProject" + "/" + "" + 
                "/" + MyFirstEJB.class.getSimpleName() + "!" + MyFirstEJBRemote.class.getName();

           The nameOfEJB appears on the console when the server is starting up:
           String nameOfEJB = "/MyFirstEJBProject/MyFirstEJB!src.ejb.MyFirstEJBRemote";   
         */
    

注意:我使用的是Wildfly(JBoss),所以需要在wildfly-10.0.0.Final\bin\client\文件夹下导入jboss-client.jar。

【讨论】:

以上是关于EJB 远程客户端从 JBoss AS 7.1 迁移到 Wildfly 8.1的主要内容,如果未能解决你的问题,请参考以下文章

远程客户端访问部署在 JBOSS AS 中的 EJB 的 JNDI 循环(7.1.1 最终版)

JBOSS EAP 7.1:NoSuchMethodError:org.jboss.as.ejb3.logging.EjbLogger.tracef(Ljava/lang/String;I)V

无法在 JBoss 的上下文中从客户端应用程序获取远程 ejb

我无法设置我的 jndi.properties 来访问 Jboss 5 上的远程 EJB

Wildfly 远程 EJB 调用

无法在 JBoss 6.4 上查找远程 EJB3