blob: 6f81da86f873e2b730049951f956466b067331d6 [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.Closeable;
import java.io.FilterOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintStream;
import java.nio.charset.Charset;
import java.security.PrivilegedAction;
import java.util.Map;
import javax.security.auth.Subject;
import jline.Terminal;
import org.apache.felix.service.command.CommandProcessor;
import org.apache.felix.service.command.CommandSession;
import org.apache.karaf.jaas.modules.JaasHelper;
import org.apache.karaf.shell.console.Console;
import org.apache.karaf.shell.console.ConsoleFactory;
import org.apache.felix.service.command.Function;
import org.apache.felix.service.threadio.ThreadIO;
import org.apache.karaf.util.StreamLoggerInterceptor;
import org.apache.sshd.common.Factory;
import org.apache.sshd.server.Command;
import org.apache.sshd.server.Environment;
import org.apache.sshd.server.ExitCallback;
import org.apache.sshd.server.SessionAware;
import org.apache.sshd.server.session.ServerSession;
import org.osgi.service.blueprint.container.ReifiedType;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* SSHD {@link org.apache.sshd.server.Command} factory which provides access to
* Shell.
*/
public class ShellFactoryImpl implements Factory<Command> {
private static final Logger LOGGER = LoggerFactory.getLogger(ShellFactoryImpl.class);
private CommandProcessor commandProcessor;
private ConsoleFactory consoleFactory;
private ThreadIO threadIO;
private boolean consoleLogger;
private String consoleLoggerName;
private String consoleLoggerOutLevel;
private String consoleLoggerErrLevel;
public ShellFactoryImpl(CommandProcessor commandProcessor, ConsoleFactory consoleFactory, ThreadIO threadIO, boolean consoleLogger, String consoleLoggerName, String consoleLoggerOutLevel, String consoleLoggerErrLevel) {
this.commandProcessor = commandProcessor;
this.consoleFactory = consoleFactory;
this.threadIO = threadIO;
this.consoleLogger = consoleLogger;
this.consoleLoggerName = consoleLoggerName;
this.consoleLoggerOutLevel = consoleLoggerOutLevel;
this.consoleLoggerErrLevel = consoleLoggerErrLevel;
}
public Command create() {
return new ShellImpl();
}
public class ShellImpl implements Command, SessionAware {
private InputStream in;
private OutputStream out;
private OutputStream err;
private ExitCallback callback;
private ServerSession session;
private boolean closed;
public void setInputStream(final InputStream in) {
this.in = in;
}
public void setOutputStream(final OutputStream out) {
if (consoleLogger) {
this.out = new StreamLoggerInterceptor(out, consoleLoggerName, consoleLoggerOutLevel);
} else {
this.out = out;
}
}
public void setErrorStream(final OutputStream err) {
if (consoleLogger) {
this.err = new StreamLoggerInterceptor(err, consoleLoggerName, consoleLoggerErrLevel);
} else {
this.err = err;
}
}
public void setExitCallback(ExitCallback callback) {
this.callback = callback;
}
public void setSession(ServerSession session) {
this.session = session;
}
public void start(final Environment env) throws IOException {
new Thread() {
@Override
public void run() {
final Subject subject = ShellImpl.this.session != null ? ShellImpl.this.session.getAttribute(KarafJaasAuthenticator.SUBJECT_ATTRIBUTE_KEY) : null;
if (subject != null) {
JaasHelper.doAs(subject, new PrivilegedAction<Object>() {
public Object run() {
String userName = JaasHelper.getUserName(subject);
Thread.currentThread().setName("Karaf Console ssh for user " + userName);
runConsole(userName);
return null;
}
});
} else {
Thread.currentThread().setName("Karaf Console ssh for anonymous user");
runConsole(null);
}
}
protected void runConsole(String userName) {
try {
final Terminal terminal = new SshTerminal(env);
Runnable destroyCallback = new Runnable() {
public void run() {
ShellImpl.this.destroy();
}
};
String encoding = getEncoding();
final Console console = consoleFactory.create(commandProcessor, threadIO, in,
lfToCrLfPrintStream(out), lfToCrLfPrintStream(err), terminal, encoding, destroyCallback);
final CommandSession session = console.getSession();
for (Map.Entry<String,String> e : env.getEnv().entrySet()) {
session.put(e.getKey(), e.getValue());
}
session.put("USER", userName);
console.run();
} catch (Exception e) {
LOGGER.warn("Unable to start shell", e);
//close the session to notify the ssh client if something wrong
//during starting shell
try {
out.write(("unable to start shell because " + e.getMessage() + "\n").getBytes());
out.flush();
} catch (IOException e1) {
LOGGER.warn("Unable to send back error message", e1);
}
ShellImpl.this.session.close(true);
}
}
}.start();
}
private PrintStream lfToCrLfPrintStream(OutputStream stream) {
return new PrintStream(new LfToCrLfFilterOutputStream(stream), true);
}
public void destroy() {
if (!closed) {
closed = true;
ShellFactoryImpl.flush(out, err);
ShellFactoryImpl.close(in, out, err);
callback.onExit(0);
}
}
}
/**
* Get the default encoding. Will first look at the LC_CTYPE environment variable, then the input.encoding
* system property, then the default charset according to the JVM.
*
* @return The default encoding to use when none is specified.
*/
public static String getEncoding() {
// LC_CTYPE is usually in the form en_US.UTF-8
String envEncoding = extractEncodingFromCtype(System.getenv("LC_CTYPE"));
if (envEncoding != null) {
return envEncoding;
}
return System.getProperty("input.encoding", Charset.defaultCharset().name());
}
/**
* Parses the LC_CTYPE value to extract the encoding according to the POSIX standard, which says that the LC_CTYPE
* environment variable may be of the format <code>[language[_territory][.codeset][@modifier]]</code>
*
* @param ctype The ctype to parse, may be null
* @return The encoding, if one was present, otherwise null
*/
static String extractEncodingFromCtype(String ctype) {
if (ctype != null && ctype.indexOf('.') > 0) {
String encodingAndModifier = ctype.substring(ctype.indexOf('.') + 1);
if (encodingAndModifier.indexOf('@') > 0) {
return encodingAndModifier.substring(0, encodingAndModifier.indexOf('@'));
} else {
return encodingAndModifier;
}
}
return null;
}
private static void flush(OutputStream... streams) {
for (OutputStream s : streams) {
try {
s.flush();
} catch (IOException e) {
// Ignore
}
}
}
private static void close(Closeable... closeables) {
for (Closeable c : closeables) {
try {
c.close();
} catch (IOException e) {
// Ignore
}
}
}
public static Converter getConverter() {
return new Converter();
}
public static class Converter implements org.osgi.service.blueprint.container.Converter {
public boolean canConvert(Object sourceObject, ReifiedType targetType) {
return ShellFactoryImpl.class.isAssignableFrom(sourceObject.getClass())
&& Factory.class.equals(targetType.getRawClass())
&& Command.class.equals(targetType.getActualTypeArgument(0).getRawClass());
}
public Object convert(Object sourceObject, ReifiedType targetType) throws Exception {
return sourceObject;
}
}
// TODO: remove this class when sshd use lf->crlf conversion by default
public class LfToCrLfFilterOutputStream extends FilterOutputStream {
private boolean lastWasCr;
public LfToCrLfFilterOutputStream(OutputStream out) {
super(out);
}
@Override
public void write(int b) throws IOException {
if (!lastWasCr && b == '\n') {
out.write('\r');
out.write('\n');
} else {
out.write(b);
}
lastWasCr = b == '\r';
}
}
}