集群环境下通过JMX访问JMSQueue

发布于 2024-12-08 03:44:03 字数 3349 浏览 1 评论 0原文

配置: WLS 集群 (10.3) 具有两个节点 #1 和 #2。当前在 #1 上提供了一个可迁移的 JMSServer。一个可迁移的 JMSQueue。

问题: 某些 EJB 使用 timeToDeliever 设置为 60 秒的消息填充 JSMQueue。 (60 秒内不可见),另一个 EJB 将使用 JMX 在该(不可见)消息变得可见之前获取该消息。如果另一个 EJB 在 #2 上执行,它无法找到 JMSServer,因此不会弹出消息。该代码在非集群环境中运行良好:(



    public class PurgeWLSQueue {

        private static final String WLS_USERNAME = "weblogic";
        private static final String WLS_PASSWORD = "weblogic";
        private static final String WLS_HOST = "localhost";
        private static final int WLS_PORT = 7001;
        private static final String JMS_SERVER = "wlsbJMSServer";
        private static final String JMS_DESTINATION = "test.q";

        private static JMXConnector getMBeanServerConnector(String jndiName) throws Exception {
            Hashtable h = new Hashtable();
            JMXServiceURL serviceURL = new JMXServiceURL("t3", WLS_HOST, WLS_PORT, jndiName);
            h.put(Context.SECURITY_PRINCIPAL, WLS_USERNAME);
            h.put(Context.SECURITY_CREDENTIALS, WLS_PASSWORD);
            h.put(JMXConnectorFactory.PROTOCOL_PROVIDER_PACKAGES, "weblogic.management.remote");
            JMXConnector connector = JMXConnectorFactory.connect(serviceURL, h);
            return connector;
        }

       public static void main(String[] args) {
            try {
                JMXConnector connector = 
                  getMBeanServerConnector("/jndi/"+RuntimeServiceMBean.MBEANSERVER_JNDI_NAME);
                MBeanServerConnection mbeanServerConnection = 
                  connector.getMBeanServerConnection();

                ObjectName service = new ObjectName("com.bea:Name=RuntimeService,Type=weblogic.management.mbeanservers.runtime.RuntimeServiceMBean");
                ObjectName serverRuntime = (ObjectName) mbeanServerConnection.getAttribute(service, "ServerRuntime");
                ObjectName jmsRuntime = (ObjectName) mbeanServerConnection.getAttribute(serverRuntime, "JMSRuntime");
                ObjectName[] jmsServers = (ObjectName[]) mbeanServerConnection.getAttribute(jmsRuntime, "JMSServers");
                for (ObjectName jmsServer: jmsServers) {
                    if (JMS_SERVER.equals(jmsServer.getKeyProperty("Name"))) {
                        ObjectName[] destinations = (ObjectName[]) mbeanServerConnection.getAttribute(jmsServer, "Destinations");
                        for (ObjectName destination: destinations) {
                            if (destination.getKeyProperty("Name").endsWith("!"+JMS_DESTINATION)) {
                                Object o = mbeanServerConnection.invoke(
                                    destination,
                                    "deleteMessages",
                                    new Object[] {""},        // selector expression
                                    new String[] {"java.lang.String"});
                                System.out.println("Result: "+o);
                                break;
                            }
                        }
                        break;
                    }
                }
                connector.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

该代码是从本论坛上的 Miklos Csuka 借来的)

是否有其他方法可以在不指定 JMSServer 的情况下获取该消息,即我可以直接寻址 JMSQueue 吗?还有其他想法吗?

Configuration: WLS-cluster (10.3) with two nodes #1 and #2. One migratable JMSServer currently available on #1. One migratable JMSQueue.

Problem:
Some EJB is populating the JSMQueue with a message with timeToDeliever set to 60 sec. (not visible in 60 sec.) and another EJB is going to use JMX to get that (not visible) message before it gets visible. If this other EJB executes on #2, it can't find the JMSServer and therefore not pop the message. The code works fine on a non clustered environment:



    public class PurgeWLSQueue {

        private static final String WLS_USERNAME = "weblogic";
        private static final String WLS_PASSWORD = "weblogic";
        private static final String WLS_HOST = "localhost";
        private static final int WLS_PORT = 7001;
        private static final String JMS_SERVER = "wlsbJMSServer";
        private static final String JMS_DESTINATION = "test.q";

        private static JMXConnector getMBeanServerConnector(String jndiName) throws Exception {
            Hashtable h = new Hashtable();
            JMXServiceURL serviceURL = new JMXServiceURL("t3", WLS_HOST, WLS_PORT, jndiName);
            h.put(Context.SECURITY_PRINCIPAL, WLS_USERNAME);
            h.put(Context.SECURITY_CREDENTIALS, WLS_PASSWORD);
            h.put(JMXConnectorFactory.PROTOCOL_PROVIDER_PACKAGES, "weblogic.management.remote");
            JMXConnector connector = JMXConnectorFactory.connect(serviceURL, h);
            return connector;
        }

       public static void main(String[] args) {
            try {
                JMXConnector connector = 
                  getMBeanServerConnector("/jndi/"+RuntimeServiceMBean.MBEANSERVER_JNDI_NAME);
                MBeanServerConnection mbeanServerConnection = 
                  connector.getMBeanServerConnection();

                ObjectName service = new ObjectName("com.bea:Name=RuntimeService,Type=weblogic.management.mbeanservers.runtime.RuntimeServiceMBean");
                ObjectName serverRuntime = (ObjectName) mbeanServerConnection.getAttribute(service, "ServerRuntime");
                ObjectName jmsRuntime = (ObjectName) mbeanServerConnection.getAttribute(serverRuntime, "JMSRuntime");
                ObjectName[] jmsServers = (ObjectName[]) mbeanServerConnection.getAttribute(jmsRuntime, "JMSServers");
                for (ObjectName jmsServer: jmsServers) {
                    if (JMS_SERVER.equals(jmsServer.getKeyProperty("Name"))) {
                        ObjectName[] destinations = (ObjectName[]) mbeanServerConnection.getAttribute(jmsServer, "Destinations");
                        for (ObjectName destination: destinations) {
                            if (destination.getKeyProperty("Name").endsWith("!"+JMS_DESTINATION)) {
                                Object o = mbeanServerConnection.invoke(
                                    destination,
                                    "deleteMessages",
                                    new Object[] {""},        // selector expression
                                    new String[] {"java.lang.String"});
                                System.out.println("Result: "+o);
                                break;
                            }
                        }
                        break;
                    }
                }
                connector.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

(This code is borrowed from Miklos Csuka on this forum)

Are there any other ways on getting that messages without specifying JMSServer, i.e can I directly address the JMSQueue? Any other ideas?

如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。

扫码二维码加入Web技术交流群

发布评论

需要 登录 才能够评论, 你可以免费 注册 一个本站的账号。

评论(1

入怼 2024-12-15 03:44:03

啊,解决了!
对于面临同样问题的其他人,请改用域运行时服务:

ObjectName service = new ObjectName("com.bea:Name=DomainRuntimeService,Type=weblogic.management.mbeanservers.domainruntime.DomainRuntimeServiceMBean");

并确保访问 WLS 集群上的admin端口。

Ah, solved it!
For others facing the same problem, use the domain runtime service instead:

ObjectName service = new ObjectName("com.bea:Name=DomainRuntimeService,Type=weblogic.management.mbeanservers.domainruntime.DomainRuntimeServiceMBean");

and be sure to access the admin port on the WLS-cluster.

~没有更多了~
我们使用 Cookies 和其他技术来定制您的体验包括您的登录状态等。通过阅读我们的 隐私政策 了解更多相关信息。 单击 接受 或继续使用网站,即表示您同意使用 Cookies 和您的相关数据。
原文