blob: db23c1e1fe542e8d0e025106a37bf267dab9bc34 [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.console.jline;
import java.io.InputStream;
import java.io.OutputStream;
import java.io.PrintStream;
import java.lang.reflect.Method;
import java.security.PrivilegedExceptionAction;
import java.util.List;
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.felix.service.command.Function;
import org.apache.karaf.jaas.modules.UserPrincipal;
import org.fusesource.jansi.AnsiConsole;
import org.osgi.framework.BundleContext;
public class ConsoleFactory {
private BundleContext bundleContext;
private CommandProcessor commandProcessor;
private TerminalFactory terminalFactory;
private Console console;
private boolean start;
public void setBundleContext(BundleContext bundleContext) {
this.bundleContext = bundleContext;
}
public synchronized void registerCommandProcessor(CommandProcessor commandProcessor) throws Exception {
this.commandProcessor = commandProcessor;
start();
}
public synchronized void unregisterCommandProcessor(CommandProcessor commandProcessor) throws Exception {
this.commandProcessor = null;
stop();
}
public void setTerminalFactory(TerminalFactory terminalFactory) {
this.terminalFactory = terminalFactory;
}
public void setStart(boolean start) {
this.start = start;
}
protected void start() throws Exception {
if (start) {
Subject subject = new Subject();
final String user = "karaf";
subject.getPrincipals().add(new UserPrincipal(user));
Subject.doAs(subject, new PrivilegedExceptionAction<Object>() {
public Object run() throws Exception {
doStart(user);
return null;
}
});
}
}
public static Object invokePrivateMethod(Object o, String methodName, Object[] params) throws Exception {
final Method methods[] = o.getClass().getDeclaredMethods();
for (int i = 0; i < methods.length; ++i) {
if (methodName.equals(methods[i].getName())) {
methods[i].setAccessible(true);
return methods[i].invoke(o, params);
}
}
return null;
}
private static <T> T unwrapBIS(T stream) {
try {
return (T) invokePrivateMethod(stream, "getInIfOpen", null);
} catch (Throwable t) {
return stream;
}
}
protected void doStart(String user) throws Exception {
final Terminal terminal = terminalFactory.getTerminal();
// unwrap stream so it can be recognized by the terminal and wrapped to get
// special keys in windows
InputStream unwrappedIn = unwrapBIS(unwrap(System.in));
InputStream in = terminal.wrapInIfNeeded(unwrappedIn);
PrintStream out = unwrap(System.out);
PrintStream err = unwrap(System.err);
Runnable callback = new Runnable() {
public void run() {
try {
bundleContext.getBundle(0).stop();
} catch (Exception e) {
// Ignore
}
}
};
this.console = new Console(commandProcessor,
in,
wrap(out),
wrap(err),
terminal,
callback);
CommandSession session = console.getSession();
session.put("USER", user);
session.put("APPLICATION", System.getProperty("karaf.name", "root"));
session.put("#LINES", new Function() {
public Object execute(CommandSession session, List<Object> arguments) throws Exception {
return Integer.toString(terminal.getHeight());
}
});
session.put("#COLUMNS", new Function() {
public Object execute(CommandSession session, List<Object> arguments) throws Exception {
return Integer.toString(terminal.getWidth());
}
});
session.put(".jline.terminal", terminal);
new Thread(console, "Karaf Shell Console Thread").start();
}
protected void stop() throws Exception {
if (console != null) {
console.close();
}
}
private static PrintStream wrap(PrintStream stream) {
OutputStream o = AnsiConsole.wrapOutputStream(stream);
if (o instanceof PrintStream) {
return ((PrintStream) o);
} else {
return new PrintStream(o);
}
}
private static <T> T unwrap(T stream) {
try {
Method mth = stream.getClass().getMethod("getRoot");
return (T) mth.invoke(stream);
} catch (Throwable t) {
return stream;
}
}
}