从本地触发远程服务器上的快照

Posted

技术标签:

【中文标题】从本地触发远程服务器上的快照【英文标题】:Trigger Snapshots on Remote Server from local 【发布时间】:2016-07-12 23:25:11 【问题描述】:

我正在尝试从我的测试代码中远程分析在运行 1.8 JVM 和 Apache Tomcat 7.xx 的 64 位 linux 服务器上运行的 alfresco,但无法弄清楚如何以编程方式触发快照。

我想要做的是连接到远程服务器,开始分析,然后从我用 Java 编写的测试代码中将该服务器的性能快照保存到我的本地机器上。

我已经在 linux 服务器上安装了 JProfiler 9.2,并且可以通过 JProfiler GUI 进行连接和拍摄快照。服务器还需要 SSH 连接以确保安全。我想从我的代码中执行此操作,类似于 Controller.saveSnapshot(file) 对本地 JVM 的工作方式。

这可能吗?

我知道我可以设置触发器并让远程分析器在服务器上保存快照,但这不是我想要的。

此外,我研究过使用命令行控制器,但即使在远程 VM 选项中使用正确的参数,也无法将其连接到服务器。

我也尝试使用 ConnectionFactor.createRemoteConnection(),但是没有看到允许输入密码的参数,所以失败了。

【问题讨论】:

【参考方案1】:

您可以通过编程方式访问 JProfiler MBean。以下是有关如何执行此操作的示例。我会在远程机器上运行这样一个程序并通过 SSH 启动它,因为 JMX 连接很难通过 SSH 建立隧道。

import javax.management.MBeanServerConnection;
import javax.management.ObjectName;
import javax.management.remote.JMXConnector;
import javax.management.remote.JMXConnectorFactory;
import javax.management.remote.JMXServiceURL;
import java.util.Collections;
import java.util.HashMap;
import java.util.Map;

// Shows how to connect to the the JProfiler MBean programatically
// The profiled process has to be in offline profiling mode and the JMX server
// has to be started by passing -Djprofiler.jmxServerPort=[port] to the profiled JVM.
// This will not work in nowait mode because the MBean is not registered in that case.
public class MBeanProgrammaticAccessExample 

    public static void main(String[] args) throws Exception 
        if (args.length == 0) 
            System.out.println("Specify the port as an argument that was passed to " +
                    "the profiled JVM with the VM parameter " +
                    "-Djprofiler.jmxServerPort=[port]");
        
        String port = args[0];
        // In this case the connection is made to a process on localhost, but it could
        // be on a remote system as well. Note that the connection is made via JMX which
        // does not work well with firewalls
        System.out.println("Connecting to localhost:" + port);
        JMXServiceURL jmxUrl = new JMXServiceURL(
                "service:jmx:rmi:///jndi/rmi://localhost:" + port + "/jmxrmi");
        JMXConnector connector = JMXConnectorFactory.newJMXConnector(jmxUrl, 
                Collections.<String, Object>emptyMap());

        Map<String, Object> env = new HashMap<>();

        // If you have protected the JMX server with a JMX password file by passing 
        // -Djprofiler.jmxPasswordFile=[file] to the profiled JVM, you can specify 
        // the password like this:
        //env.put(JMXConnector.CREDENTIALS, new String[] "username", "password");

        connector.connect(env);
        MBeanServerConnection connection = connector.getMBeanServerConnection();
        ObjectName objectName = new ObjectName(
                "com.jprofiler.api.agent.mbean:type=RemoteController");
        if (!connection.isRegistered(objectName)) 
            throw new RuntimeException("JProfiler MBean not found.");
        

        RemoteControllerMBean mbeanProxy = JMX.newMBeanProxy(connection, 
                objectName, RemoteControllerMBean.class, true);

        // You can look up all available operations in the javadoc of 
        // com.jprofiler.api.agent.mbean.RemoteControllerMBean
        System.out.println("Recording CPU data for 5 seconds ....");
        mbeanProxy.startCPURecording(true);
        // If you do not want a dependency on the JProfiler classes 
        // you can make the above call like this:
        //connection.invoke(objectName, "startCPURecording", new Object[] true,
        // new String[] Boolean.TYPE.getName());

        Thread.sleep(5000);
        System.out.println("Saving snapshot to the working directory " +
                "of the profiled JVM ....");
        mbeanProxy.saveSnapshot("snapshot.jps");

        connector.close();
        System.out.println("Success");

    

【讨论】:

以上是关于从本地触发远程服务器上的快照的主要内容,如果未能解决你的问题,请参考以下文章

如何将远程服务器上的文件复制到自己用的电脑上?

从服务器上拷贝文件到本地电脑

服务器上的 Git

使用python比较远程目录和本地目录上的文件

git怎么从远程仓库拉取到本地仓库

在远程服务器上的 docker 容器中设置本地 PyCharm 远程解释器