从Shell脚本调用JMX MBean方法


98

是否有任何库允许我从Shell脚本调用JMX MBean方法。我们通过JMX公开了一些操作/管理命令,我们可以让管理员使用JConsole或VisualVM,但最好将某些任务留给自动化。在这种自动化中,我们希望能够在运行的服务器上调用JMX MBean方法,最好是从Shell脚本中调用。

Answers:


106

可以使用以下命令行JMX实用程序:

  1. jmxterm-似乎是功能最全的实用程序。
  2. cmdline-jmxclient-在WebArchive项目中使用似乎很简单(并且自2006年以来没有任何发展)
  3. Groovy脚本和JMX-提供了一些真正强大的JMX功能,但需要使用Groovy和其他库设置。
  4. JManage命令行功能 -(缺点是它需要运行的JManage服务器来代理命令)

Groovy JMX示例:

import java.lang.management.*
import javax.management.ObjectName
import javax.management.remote.JMXConnectorFactory as JmxFactory
import javax.management.remote.JMXServiceURL as JmxUrl

def serverUrl = 'service:jmx:rmi:///jndi/rmi://localhost:9003/jmxrmi'
String beanName = "com.webwars.gameplatform.data:type=udmdataloadsystem,id=0"
def server = JmxFactory.connect(new JmxUrl(serverUrl)).MBeanServerConnection
def dataSystem = new GroovyMBean(server, beanName)

println "Connected to:\n$dataSystem\n"

println "Executing jmxForceRefresh()"
dataSystem.jmxForceRefresh();

cmdline-jmxclient示例:

如果您有

  • MBean:com.company.data:type=datasystem,id=0

使用一个称为:

  • jmxForceRefresh()

然后,您可以编写一个简单的bash脚本(假设您下载了cmdline-jmxclient-0.10.3.jar并放置在与脚本相同的目录中):

#!/bin/bash

cmdLineJMXJar=./cmdline-jmxclient-0.10.3.jar
user=yourUser
password=yourPassword
jmxHost=localhost
port=9003

#No User and password so pass '-'
echo "Available Operations for com.company.data:type=datasystem,id=0"
java -jar ${cmdLineJMXJar} ${user}:${password} ${jmxHost}:${port} com.company.data:type=datasystem,id=0

echo "Executing XML update..."
java -jar ${cmdLineJMXJar} - ${jmxHost}:${port} com.company.data:type=datasystem,id=0 jmxForceRefresh

jmxterm似乎不适用于Java 7 bugs.launchpad.net/jmxterm/+bug/942693
artbristol 2013年

19

我已经开发了jmxfuse,它将JMX Mbeans公开为具有与/ proc fs类似功能的Linux FUSE文件系统。它依靠Jolokia作为通往JMX的桥梁。公开属性和操作以供读取和写入。

http://code.google.com/p/jmxfuse/

例如,读取属性:

me@oddjob:jmx$ cd log4j/root/attributes
me@oddjob:jmx$ cat priority

编写属性:

me@oddjob:jmx$ echo "WARN" > priority

调用操作:

me@oddjob:jmx$ cd Catalina/none/none/WebModule/localhost/helloworld/operations/addParameter
me@oddjob:jmx$ echo "myParam myValue" > invoke

12

Syabru Nagios的JMX插件是为了从Nagios的使用,但不要求Nagios的,是命令行使用非常方便:

~$ ./check_jmx -U service:jmx:rmi:///jndi/rmi://localhost:1099/JMXConnector --username myuser --password mypass -O java.lang:type=Memory -A HeapMemoryUsage -K used 
JMX OK - HeapMemoryUsage.used = 445012360 | 'HeapMemoryUsage used'=445012360;;;;

这很棒,而且非常快。大约0.3秒返回一个值,而jmxterm为3秒
sivann

9

用Java编写此代码可能是最简单的

import javax.management.*;
import javax.management.remote.*;

public class JmxInvoke {

    public static void main(String... args) throws Exception {

        JMXConnectorFactory.connect(new JMXServiceURL(args[0]))
            .getMBeanServerConnection().invoke(new ObjectName(args[1]), args[2], new Object[]{}, new String[]{})


    }

}

这将编译为单个.class,并且不需要服务器或任何复杂的maven打包中的依赖项。

用它来称呼它

javac JmxInvoke.java
java -cp . JmxInvoke [url] [beanName] [method]

4

有点冒险,但是您可以使用JMX控制台表单中的值,URL和http身份验证(如果需要)运行curl POST命令:

curl -s -X POST --user 'myuser:mypass'
  --data "action=invokeOp&name=App:service=ThisServiceOp&methodIndex=3&arg0=value1&arg1=value1&submit=Invoke"
  http://yourhost.domain.com/jmx-console/HtmlAdaptor

当心:方法索引可能会随软件的更改而改变。网络表单的实现可能会发生变化。

上面的内容基于您要执行的操作的JMX服务页面的来源:

http://yourhost.domain.com/jmx-console/HtmlAdaptor?action=inspectMBean&name=YourJMXServiceName

表格来源:

form method="post" action="HtmlAdaptor">
   <input type="hidden" name="action" value="invokeOp">
   <input type="hidden" name="name" value="App:service=ThisServiceOp">
   <input type="hidden" name="methodIndex" value="3">
   <hr align='left' width='80'>
   <h4>void ThisOperation()</h4>
   <p>Operation exposed for management</p>
    <table cellspacing="2" cellpadding="2" border="1">
        <tr class="OperationHeader">
            <th>Param</th>
            <th>ParamType</th>
            <th>ParamValue</th>
            <th>ParamDescription</th>
        </tr>
        <tr>
            <td>p1</td>
           <td>java.lang.String</td>
         <td> 
            <input type="text" name="arg0">
         </td>
         <td>(no description)</td>
        </tr>
        <tr>
            <td>p2</td>
           <td>arg1Type</td>
         <td> 
            <input type="text" name="arg1">
         </td>
         <td>(no description)</td>
        </tr>
    </table>
    <input type="submit" value="Invoke">
</form>

我使用a从Java中以这种方式实现了这一点,HttpURLConnection并且可以确认它是否有效。(顺便说一句submit=Invoke是不必要的)
汤姆(Tom

有可能描述它是如何工作的吗?我的意思是,默认情况下jmx使用rmi,在那里我看到了http。这是否意味着服务器必须配置为支持jmx请求http
Psychozoic

3

看一下JManage。它能够执行MBean方法并从命令行获取/设置属性。


唯一的缺点是使用命令行实用程序,它需要运行JManage才能将命令代理到您的JMX服务器。我宁愿直接对JMX服务器本身使用更轻量级的方法。
Dougnukem

3

您可能还想看看jmx4perl。它提供了对远程Java EE Server的MBean的无Java访问。但是,需要在目标平台上安装一个小型代理servlet,该servlet可以通过HTTP提供带有JSON负载的静态JMX访问。(版本0.50将通过实现JSR-160代理添加无代理模式)。

与启动本地Java JVM相比,优点是启动时间短并且易于使用。jmx4perl带有一整套Perl模块,可以在您自己的脚本中轻松使用:

use JMX::Jmx4Perl;
use JMX::Jmx4Perl::Alias;   # Import certains aliases for MBeans

print "Memory Used: ",
      JMX::Jmx4Perl
          ->new(url => "http://localhost:8080/j4p")
          ->get_attribute(MEMORY_HEAP_USED);

您也可以将别名用于常见的MBean / Attribute / Operation组合(例如,对于大多数MXBean)。有关其他功能(Nagios-Plugin,对复杂属性类型的类似XPath的访问,...),请参阅jmx4perl的文档。


1

@Dougnukem的回答对我很有帮助。我采用了Groovy方法(使用groovy 2.3.3)。

我对Dougnukem代码进行了一些更改。这将与Java 7一起使用,并且每10秒将两个属性输出到stdout。

        package com.my.company.jmx
        import groovy.util.GroovyMBean;
        import javax.management.remote.JMXServiceURL
        import javax.management.remote.JMXConnectorFactory
        import java.lang.management.*

            class Monitor {
                static main(args) {
                    def serverUrl = 'service:jmx:rmi:///jndi/rmi://localhost:5019/jmxrmi'
                    String beanName = "Catalina:type=DataSource,class=javax.sql.DataSource,name=\"jdbc/CommonDB\""
                    println  "numIdle,numActive"

                    while(1){
                        def server = JMXConnectorFactory.connect(new JMXServiceURL(serverUrl))
                       //make sure to reconnect in case the jvm was restrated 
                        server.connect()
                        GroovyMBean mbean = new GroovyMBean(server.MBeanServerConnection, beanName)
                        println  "${mbean.numIdle},${mbean.numActive}"
                        server.close()
                        sleep(10000)
                    }

                }
            }

使用maven-compiler-plugin将此代码编译到jar中,因此您仅需要groovy-all.jar即可进行groovy安装。以下是相关的插件定义和依赖关系。

   <build>
        <plugins>
            <plugin>
                <groupId>org.apache.maven.plugins</groupId>
                <artifactId>maven-compiler-plugin</artifactId>
                <configuration>
                    <compilerId>groovy-eclipse-compiler</compilerId>
                    <source>1.7</source>
                    <target>1.7</target>
                </configuration>
                <dependencies>
                    <dependency>
                        <groupId>org.codehaus.groovy</groupId>
                        <artifactId>groovy-eclipse-compiler</artifactId>
                        <version>2.8.0-01</version>
                    </dependency>
                    <dependency>
                        <groupId>org.codehaus.groovy</groupId>
                        <artifactId>groovy-eclipse-batch</artifactId>
                        <version>2.3.4-01</version>
                    </dependency>
                </dependencies>
            </plugin>
        </plugins>
    </build>
    <dependencies>
        <dependency>
            <groupId>org.codehaus.groovy</groupId>
            <artifactId>groovy-all</artifactId>
            <version>2.4.3</version>
        </dependency>
    </dependencies>

用蝙蝠或贝壳包裹它,它将把数据打印到stdout。


0

我不确定类似bash的环境。您可以尝试使用Java中的一些简单包装程序(带有程序参数)来调用远程服务器上的MBean。然后,您可以从shell脚本中调用这些包装器

如果可以使用Python或Perl之类的东西,您可能会对JSR-262感兴趣,它允许您通过Web服务公开JMX操作。它计划包含在Java 7中,但是您可能可以使用参考实现的候选版本

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.