blob: 99502d97cbbdea23f51295d43b118f4e79b7b44c [file] [log] [blame]
/*
* Licensed to the Apache Software Foundation (ASF) under one
* or more contributor license agreements. See the NOTICE file
* distributed with this work for additional information
* regarding copyright ownership. The ASF licenses this file
* to you under the Apache License, Version 2.0 (the
* "License"); you may not use this file except in compliance
* with the License. You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing,
* software distributed under the License is distributed on an
* "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
* KIND, either express or implied. See the License for the
* specific language governing permissions and limitations
* under the License.
*/
package org.apache.karaf.shell.ssh;
import java.io.ByteArrayInputStream;
import java.io.IOException;
import java.util.List;
import org.apache.felix.gogo.commands.Argument;
import org.apache.felix.gogo.commands.Command;
import org.apache.felix.gogo.commands.Option;
import org.apache.karaf.shell.console.BlueprintContainerAware;
import org.apache.karaf.shell.console.OsgiCommandSupport;
import org.apache.karaf.shell.console.jline.Console;
import org.apache.sshd.ClientChannel;
import org.apache.sshd.ClientSession;
import org.apache.sshd.SshClient;
import org.apache.sshd.client.channel.ChannelShell;
import org.apache.sshd.client.future.ConnectFuture;
import org.apache.sshd.common.util.NoCloseInputStream;
import org.apache.sshd.common.util.NoCloseOutputStream;
import org.osgi.service.blueprint.container.BlueprintContainer;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* Connect to a SSH server.
*
* @version $Rev: 721244 $ $Date: 2008-11-27 18:19:56 +0100 (Thu, 27 Nov 2008) $
*/
@Command(scope = "ssh", name = "ssh", description = "Connects to a remote SSH server")
public class SshAction
extends OsgiCommandSupport implements BlueprintContainerAware
{
private final Logger log = LoggerFactory.getLogger(getClass());
@Option(name="-l", aliases={"--username"}, description = "The user name for remote login", required = false, multiValued = false)
private String username;
@Option(name="-P", aliases={"--password"}, description = "The password for remote login", required = false, multiValued = false)
private String password;
@Option(name="-p", aliases={"--port"}, description = "The port to use for SSH connection", required = false, multiValued = false)
private int port = 22;
@Argument(index = 0, name = "hostname", description = "The host name to connect to via SSH", required = true, multiValued = false)
private String hostname;
@Argument(index = 1, name = "command", description = "Optional command to execute", required = false, multiValued = true)
private List<String> command;
private BlueprintContainer container;
private ClientSession sshSession;
private String sshClientId;
public void setBlueprintContainer(final BlueprintContainer container) {
assert container != null;
this.container = container;
}
public void setSshClientId(String sshClientId) {
this.sshClientId = sshClientId;
}
@Override
protected Object doExecute() throws Exception {
//
// TODO: Parse hostname for <username>@<hostname>
//
System.out.println("Connecting to host " + hostname + " on port " + port);
// If the username/password was not configured via cli, then prompt the user for the values
if (username == null || password == null) {
log.debug("Prompting user for credentials");
if (username == null) {
username = readLine("Login: ");
}
if (password == null) {
password = readLine("Password: ");
}
}
// Create the client from prototype
SshClient client = (SshClient) container.getComponentInstance(sshClientId);
log.debug("Created client: {}", client);
client.start();
try {
ConnectFuture future = client.connect(hostname, port);
future.await();
sshSession = future.getSession();
Object oldIgnoreInterrupts = this.session.get(Console.IGNORE_INTERRUPTS);
this.session.put( Console.IGNORE_INTERRUPTS, Boolean.TRUE );
try {
System.out.println("Connected");
sshSession.authPassword(username, password);
int ret = sshSession.waitFor(ClientSession.WAIT_AUTH | ClientSession.CLOSED | ClientSession.AUTHED, 0);
if ((ret & ClientSession.AUTHED) == 0) {
System.err.println("Authentication failed");
return null;
}
StringBuilder sb = new StringBuilder();
if (command != null) {
for (String cmd : command) {
if (sb.length() > 0) {
sb.append(' ');
}
sb.append(cmd);
}
}
ClientChannel channel;
if (sb.length() > 0) {
channel = sshSession.createChannel("exec", sb.append("\n").toString());
channel.setIn(new ByteArrayInputStream(new byte[0]));
} else {
channel = sshSession.createChannel("shell");
channel.setIn(new NoCloseInputStream(System.in));
((ChannelShell) channel).setupSensibleDefaultPty();
}
channel.setOut(new NoCloseOutputStream(System.out));
channel.setErr(new NoCloseOutputStream(System.err));
channel.open();
channel.waitFor(ClientChannel.CLOSED, 0);
} finally {
session.put( Console.IGNORE_INTERRUPTS, oldIgnoreInterrupts );
sshSession.close(false);
}
} finally {
client.stop();
}
return null;
}
public String readLine(String msg) throws IOException {
StringBuffer sb = new StringBuffer();
System.err.print(msg);
System.err.flush();
for (;;) {
int c = super.session.getKeyboard().read();
if (c < 0) {
return null;
}
System.err.print((char) c);
if (c == '\r' || c == '\n') {
break;
}
sb.append((char) c);
}
return sb.toString();
}
}