All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.oracle.bedrock.runtime.remote.ssh.JSchRemoteTerminal Maven / Gradle / Ivy

/*
 * File: JSchRemoteTerminal.java
 *
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS HEADER.
 *
 * The contents of this file are subject to the terms and conditions of 
 * the Common Development and Distribution License 1.0 (the "License").
 *
 * You may not use this file except in compliance with the License.
 *
 * You can obtain a copy of the License by consulting the LICENSE.txt file
 * distributed with this file, or by consulting https://oss.oracle.com/licenses/CDDL
 *
 * See the License for the specific language governing permissions
 * and limitations under the License.
 *
 * When distributing the software, include this License Header Notice in each
 * file and include the License file LICENSE.txt.
 *
 * MODIFICATIONS:
 * If applicable, add the following below the License Header, with the fields
 * enclosed by brackets [] replaced by your own identifying information:
 * "Portions Copyright [year] [name of copyright owner]"
 */

package com.oracle.bedrock.runtime.remote.ssh;

import com.jcraft.jsch.ChannelExec;
import com.jcraft.jsch.JSch;
import com.jcraft.jsch.JSchException;
import com.jcraft.jsch.Session;
import com.oracle.bedrock.OptionsByType;
import com.oracle.bedrock.Bedrock;
import com.oracle.bedrock.lang.StringHelper;
import com.oracle.bedrock.options.LaunchLogging;
import com.oracle.bedrock.options.Variable;
import com.oracle.bedrock.runtime.Application;
import com.oracle.bedrock.runtime.options.CommandInterceptor;
import com.oracle.bedrock.runtime.options.Shell;
import com.oracle.bedrock.runtime.options.WorkingDirectory;
import com.oracle.bedrock.runtime.remote.AbstractRemoteTerminal;
import com.oracle.bedrock.runtime.remote.RemoteApplicationProcess;
import com.oracle.bedrock.runtime.remote.RemotePlatform;
import com.oracle.bedrock.runtime.remote.RemoteTerminal;
import com.oracle.bedrock.table.Table;

import java.io.File;
import java.util.List;
import java.util.Properties;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * A {@link RemoteTerminal} based on SSH (uses JSch) for a {@link RemotePlatform}.
 * 

* Copyright (c) 2015. All Rights Reserved. Oracle Corporation.
* Oracle is a registered trademark of Oracle Corporation and/or its affiliates. * * @author Jonathan Knight */ public class JSchRemoteTerminal extends AbstractRemoteTerminal { /** * The {@link Logger} for this class. */ private static Logger LOGGER = Logger.getLogger(JSchRemoteTerminal.class.getName()); /** * The {@link JSch} framework. */ protected JSchSessionFactory sessionFactory; /** * Create a {@link JSchRemoteTerminal} that will connect to a remote * platform with the specified connection details. * * @param platform the {@link RemotePlatform} */ public JSchRemoteTerminal(RemotePlatform platform) { this(platform, new JSchSessionFactory()); } /** * Create a {@link JSchRemoteTerminal} that will connect to a remote * platform with the specified connection details. * * @param platform the {@link RemotePlatform} * @param sessionFactory the {@link JSchSessionFactory} to use to obtain a JSch {@link Session} */ public JSchRemoteTerminal(RemotePlatform platform, JSchSessionFactory sessionFactory) { super(platform); this.sessionFactory = sessionFactory; } @Override public RemoteApplicationProcess launch(Launchable launchable, Class applicationClass, OptionsByType optionsByType) { // acquire the remote platform on which to launch the application RemotePlatform platform = getRemotePlatform(); // establish a specialized SocketFactory for JSch JSchSocketFactory socketFactory = new JSchSocketFactory(); // initially there's no session Session session = null; try { // create the remote session session = sessionFactory.createSession(platform.getAddress().getHostName(), platform.getPort(), platform.getUserName(), platform.getAuthentication(), socketFactory, optionsByType); ChannelExec execChannel = (ChannelExec) session.openChannel("exec"); // (re)define the "local.address" variable so that we can use for resolving the platform optionsByType.add(Variable.with("local.address", socketFactory.getLastLocalAddress().getHostAddress())); // ----- establish the remote environment variables ----- String environmentVariables = ""; // get the remote environment variables for the remote application Properties variables = launchable.getEnvironmentVariables(platform, optionsByType); // determine the format to use for setting variables String format; Shell shell = optionsByType.getOrSetDefault(Shell.class, Shell.isUnknown()); switch (shell.getType()) { case SH : case BASH : format = "export %s=%s ; "; break; case CSH : case TSCH : format = "setenv %s %s ; "; break; default : // when we don't know, assume something bash-like format = "export %s=%s ; "; break; } List arguments = launchable.getCommandLineArguments(platform, optionsByType); CommandInterceptor interceptor = optionsByType.get(CommandInterceptor.class); String executableName = launchable.getCommandToExecute(platform, optionsByType); File workingDirectory = optionsByType.get(WorkingDirectory.class).resolve(platform, optionsByType); String remoteCommand; if (interceptor == null) { for (String variableName : variables.stringPropertyNames()) { environmentVariables += String.format(format, variableName, StringHelper.doubleQuoteIfNecessary(variables.getProperty(variableName))); } // ----- establish the application command line to execute ----- // determine the command to execute remotely StringBuilder command = new StringBuilder(executableName); // add the arguments for (String arg : arguments) { command.append(" ").append(arg); } // the actual remote command must include changing to the remote directory remoteCommand = environmentVariables + String.format("cd %s ; %s", workingDirectory, command); } else { remoteCommand = interceptor.onExecute(executableName, arguments, variables, workingDirectory); } execChannel.setCommand(remoteCommand); // ----- establish the remote application process to represent the remote application ----- // establish a RemoteApplicationProcess representing the remote application RemoteApplicationProcess process = new JschRemoteApplicationProcess(session, execChannel); // ----- start the remote application ----- if (optionsByType.get(LaunchLogging.class).isEnabled()) { Table diagnosticsTable = optionsByType.get(Table.class); if (diagnosticsTable != null && LOGGER.isLoggable(Level.INFO)) { diagnosticsTable.addRow("Application Executable ", executableName); LOGGER.log(Level.INFO, "Oracle Bedrock " + Bedrock.getVersion() + ": Starting Application...\n" + "------------------------------------------------------------------------\n" + diagnosticsTable.toString() + "\n" + "------------------------------------------------------------------------\n"); } } // connect the channel execChannel.connect(session.getTimeout()); return process; } catch (JSchException e) { if (session != null) { session.disconnect(); } throw new RuntimeException("Failed to create remote application", e); } } @Override public void makeDirectories(String directoryName, OptionsByType optionsByType) { Session session = null; try { // acquire the remote platform RemotePlatform platform = getRemotePlatform(); // establish a specialized SocketFactory for JSch JSchSocketFactory socketFactory = new JSchSocketFactory(); // create the remote session session = sessionFactory.createSession(platform.getAddress().getHostName(), platform.getPort(), platform.getUserName(), platform.getAuthentication(), socketFactory, optionsByType); ChannelExec execChannel = (ChannelExec) session.openChannel("exec"); execChannel.setCommand("mkdir -p " + directoryName); RemoteApplicationProcess process = new JschRemoteApplicationProcess(session, execChannel); execChannel.connect(session.getTimeout()); process.waitFor(); } catch (Exception e) { throw new RuntimeException("Error creating remote directories " + directoryName, e); } finally { if (session != null) { session.disconnect(); } } } public void moveFile(String source, String destination, OptionsByType optionsByType) { Session session = null; try { // acquire the remote platform RemotePlatform platform = getRemotePlatform(); // establish a specialized SocketFactory for JSch JSchSocketFactory socketFactory = new JSchSocketFactory(); // create the remote session session = sessionFactory.createSession(platform.getAddress().getHostName(), platform.getPort(), platform.getUserName(), platform.getAuthentication(), socketFactory, optionsByType); ChannelExec execChannel = (ChannelExec) session.openChannel("exec"); String moveCommand = String.format("mv %s %s", source, destination); execChannel.setCommand(moveCommand); RemoteApplicationProcess process = new JschRemoteApplicationProcess(session, execChannel); execChannel.connect(session.getTimeout()); process.waitFor(); } catch (Exception e) { throw new RuntimeException("Error moving file from " + source + " to " + destination, e); } finally { if (session != null) { session.disconnect(); } } } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy