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

org.glassfish.osgi.cli.interactive.LocalOSGiShellCommand Maven / Gradle / Ivy

There is a newer version: 8.0.0-JDK17-M7
Show newest version
/*
 * Copyright (c) 2022, 2024 Contributors to the Eclipse Foundation.
 * Copyright (c) 2012, 2018 Oracle and/or its affiliates. All rights reserved.
 *
 * This program and the accompanying materials are made available under the
 * terms of the Eclipse Public License v. 2.0, which is available at
 * http://www.eclipse.org/legal/epl-2.0.
 *
 * This Source Code may also be made available under the following Secondary
 * Licenses when the conditions for such availability set forth in the
 * Eclipse Public License v. 2.0 are satisfied: GNU General Public License,
 * version 2 with the GNU Classpath Exception, which is available at
 * https://www.gnu.org/software/classpath/license.html.
 *
 * SPDX-License-Identifier: EPL-2.0 OR GPL-2.0 WITH Classpath-exception-2.0
 */

package org.glassfish.osgi.cli.interactive;

import com.sun.enterprise.admin.cli.ArgumentTokenizer;
import com.sun.enterprise.admin.cli.ArgumentTokenizer.ArgumentException;
import com.sun.enterprise.admin.cli.CLICommand;
import com.sun.enterprise.admin.cli.CLIUtil;
import com.sun.enterprise.admin.cli.Environment;
import com.sun.enterprise.admin.cli.MultimodeCommand;
import com.sun.enterprise.admin.cli.ProgramOptions;
import com.sun.enterprise.admin.cli.remote.RemoteCLICommand;
import com.sun.enterprise.admin.util.CommandModelData.ParamModelData;
import com.sun.enterprise.universal.i18n.LocalStringsImpl;

import jakarta.inject.Inject;

import java.io.File;
import java.io.FileInputStream;
import java.io.IOException;
import java.nio.charset.Charset;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.LinkedHashSet;
import java.util.List;
import java.util.Set;
import java.util.logging.Level;

import org.glassfish.api.I18n;
import org.glassfish.api.Param;
import org.glassfish.api.admin.CommandException;
import org.glassfish.api.admin.CommandModel.ParamModel;
import org.glassfish.api.admin.CommandValidationException;
import org.glassfish.common.util.io.EmptyOutputStream;
import org.glassfish.hk2.api.ActiveDescriptor;
import org.glassfish.hk2.api.DynamicConfiguration;
import org.glassfish.hk2.api.DynamicConfigurationService;
import org.glassfish.hk2.api.MultiException;
import org.glassfish.hk2.api.PerLookup;
import org.glassfish.hk2.api.ServiceLocator;
import org.jline.reader.Completer;
import org.jline.reader.EndOfFileException;
import org.jline.reader.LineReader;
import org.jline.reader.LineReaderBuilder;
import org.jline.reader.impl.completer.NullCompleter;
import org.jline.reader.impl.completer.StringsCompleter;
import org.jline.terminal.Terminal;
import org.jline.terminal.TerminalBuilder;
import org.jline.terminal.impl.ExternalTerminal;
import org.jvnet.hk2.annotations.Service;

import static org.glassfish.hk2.utilities.BuilderHelper.createConstantDescriptor;
import static org.glassfish.hk2.utilities.BuilderHelper.createContractFilter;

/**
 * A simple local asadmin sub-command to establish an interactive osgi shell.
 *
 * 

This class is forked from com.sun.enterprise.admin.cli.MultimodeCommand. * *

Original code authors: * केदार([email protected]) * Bill Shannon * @author ancoron */ @Service(name = "osgi-shell") @I18n("osgi-shell") @PerLookup public class LocalOSGiShellCommand extends CLICommand { private static final String REMOTE_COMMAND = "osgi"; private static final String SESSIONID_OPTION = "--session-id"; private static final String SESSION_OPTION = "--session"; private static final String SESSION_OPTION_EXECUTE = "execute"; private static final String SESSION_OPTION_START = "new"; private static final String SESSION_OPTION_STOP = "stop"; private static final LocalStringsImpl STRINGS = new LocalStringsImpl(MultimodeCommand.class); @Inject private ServiceLocator locator; @Param(name = "instance", optional = true) private String instance; @Param(optional = true, shortName = "f") private File file; @Param(name = "printprompt", optional = true) private Boolean printPromptOpt; @Param(optional = true) private String encoding; private boolean echo; private RemoteCLICommand cmd; private String shellType; @Override public void postConstruct() { super.postConstruct(); try { cmd = new RemoteCLICommand(REMOTE_COMMAND, locator.getService(ProgramOptions.class), locator.getService(Environment.class)); } catch (MultiException | CommandException e) { logger.log(Level.SEVERE, "postConstruct failed!", e); } } /** * The validate method validates that the type and quantity of * parameters and operands matches the requirements for this * command. The validate method supplies missing options from * the environment. */ @Override protected void validate() { // Save value of --echo because CLICommand will reset it // before calling our executeCommand method but we want it // to also apply to all commands in multimode. echo = programOpts.isEcho(); } /** * In the usage message modify the --printprompt option to have a * default based on the --interactive option. */ @Override protected Collection usageOptions() { Collection opts = commandModel.getParameters(); Set uopts = new LinkedHashSet<>(); String interactive = Boolean.toString(programOpts.isInteractive()); ParamModel p = new ParamModelData("printprompt", boolean.class, true, interactive); for (ParamModel pm : opts) { if (pm.getName().equals("printprompt")) { uopts.add(p); } else { uopts.add(pm); } } return uopts; } @Override protected int executeCommand() throws CommandException { if (cmd == null) { throw new CommandException("Remote command 'osgi' is not available."); } programOpts.setEcho(echo); // restore echo flag, saved in validate if (encoding != null) { // see Configuration.getEncoding()... System.setProperty("input.encoding", encoding); } final String[] args = enhanceForTarget(new String[] {REMOTE_COMMAND, "asadmin-osgi-shell"}); logger.log(Level.FINEST, "executeCommand: args {0}", Arrays.toString(args)); shellType = cmd.executeAndReturnOutput(args).trim(); try (Terminal terminal = createTerminal()) { LineReaderBuilder builder = LineReaderBuilder.builder().appName(REMOTE_COMMAND).terminal(terminal); if (isInteractive()) { builder.completer(getCommandCompleter()); builder.option(LineReader.Option.INSERT_TAB, false); } return executeCommands(builder.build()); } catch (IOException e) { throw new CommandException(e); } } private String[] enhanceForTarget(String[] args) { if (instance == null) { return args; } String[] targetArgs = new String[args.length + 2]; targetArgs[1] = "--instance"; targetArgs[2] = instance; System.arraycopy(args, 0, targetArgs, 0, 1); System.arraycopy(args, 1, targetArgs, 3, args.length - 1); return targetArgs; } private Terminal createTerminal() throws IOException, CommandException { if (!isInteractive()) { if (!file.canRead()) { throw new CommandException("File: " + file + " can not be read"); } Charset charset = encoding == null ? Charset.defaultCharset() : Charset.forName(encoding); return new ExternalTerminal(REMOTE_COMMAND, "dumb", new FileInputStream(file), new EmptyOutputStream(), charset); } System.out.println(STRINGS.get("multimodeIntro")); return TerminalBuilder.builder().system(true).build(); } /** * Get the command completion. * * @return The command completer */ private Completer getCommandCompleter() { if ("gogo".equals(shellType)) { return new StringsCompleter( "bundlelevel", "cd", "frameworklevel", "headers", "help", "inspect", "install", "lb", "log", "ls", "refresh", "resolve", "start", "stop", "uninstall", "update", "which", "cat", "each", "echo", "format", "getopt", "gosh", "grep", "not", "set", "sh", "source", "tac", "telnetd", "type", "until", "deploy", "info", "javadoc", "list", "repos", "source" ); } else if ("felix".equals(shellType)) { return new StringsCompleter( "exit", "quit", "help", "bundlelevel", "cd", "find", "headers", "inspect", "install", "log", "ps", "refresh", "resolve", "scr", "shutdown", "start", "startlevel", "stop", "sysprop", "uninstall", "update", "version" ); } return new NullCompleter(); } /** * Read commands from the specified {@link java.io.BufferedReader} * and execute them. If printPrompt is set, prompt first. * * @return the exit code of the last command executed */ private int executeCommands(LineReader reader) throws CommandException { String line; int rc = 0; /* * Any program options we start with are copied to the environment * to serve as defaults for commands we run, and then we give each * command an empty program options. */ programOpts.toEnvironment(env); String sessionId = startSession(); try { while (true) { try { if (isInteractive()) { line = reader.readLine(shellType + "$ "); } else { line = reader.readLine(); } } catch (EndOfFileException e) { break; } if (line == null || line.isBlank()) { // ignore blank lines continue; } if (line.trim().startsWith("#")) { // ignore comment lines continue; } final String[] args; try { args = getArgs(line); } catch (ArgumentException ex) { logger.severe(ex.getMessage()); continue; } final String command = args[0]; // handle built-in exit and quit commands if ("exit".equals(command) || "quit".equals(command)) { break; } final String[] arguments = enhanceForTarget(prepareArguments(sessionId, args)); try { /* * Every command gets its own copy of program options * so that any program options specified in its * command line options don't effect other commands. * But all commands share the same environment. */ final ProgramOptions programOptions = new ProgramOptions(env); // copy over AsadminMain info programOptions.setClassPath(programOpts.getClassPath()); programOptions.setClassName(programOpts.getClassName()); // remove the old one and replace it atomicReplace(locator, programOptions); String output = cmd.executeAndReturnOutput(arguments).trim(); if (output != null && !output.isEmpty()) { logger.info(output); } } catch (CommandValidationException cve) { logger.severe(cve.getMessage()); logger.severe(cmd.getUsage()); rc = ERROR; } catch (CommandException ce) { logger.severe(ce.getMessage()); rc = ERROR; } finally { // restore the original program options // XXX - is this necessary? atomicReplace(locator, programOpts); } CLIUtil.writeCommandToDebugLog(name, env, arguments, rc); } } finally { // what if something breaks on the wire? rc = stopSession(sessionId); } return rc; } private String[] prepareArguments(String sessionId, String[] args) { if (sessionId == null) { String[] osgiArgs = args == null ? new String[1] : new String[args.length + 1]; osgiArgs[0] = REMOTE_COMMAND; if (args != null && args.length > 0) { System.arraycopy(args, 0, osgiArgs, 1, args.length); } return osgiArgs; } // attach command to remote session... String[] sessionArgs = args == null ? new String[5] : new String[args.length + 5]; sessionArgs[0] = REMOTE_COMMAND; sessionArgs[1] = SESSION_OPTION; sessionArgs[2] = SESSION_OPTION_EXECUTE; sessionArgs[3] = SESSIONID_OPTION; sessionArgs[4] = sessionId; if (args != null && args.length > 0) { System.arraycopy(args, 0, sessionArgs, 5, args.length); } return sessionArgs; } private String startSession() throws CommandException { if (!"gogo".equals(shellType)) { return null; } String[] args = {REMOTE_COMMAND, SESSION_OPTION, SESSION_OPTION_START}; return cmd.executeAndReturnOutput(enhanceForTarget(args)).trim(); } private int stopSession(String sessionId) throws CommandException { if (sessionId == null) { return 0; } String[] args = {REMOTE_COMMAND, SESSION_OPTION, SESSION_OPTION_STOP, SESSIONID_OPTION, sessionId}; return cmd.execute(enhanceForTarget(args)); } private boolean isInteractive() { return file == null; } private static void atomicReplace(ServiceLocator locator, ProgramOptions options) { DynamicConfigurationService dcs = locator.getService(DynamicConfigurationService.class); DynamicConfiguration config = dcs.createDynamicConfiguration(); config.addUnbindFilter(createContractFilter(ProgramOptions.class.getName())); ActiveDescriptor desc = createConstantDescriptor(options, null, ProgramOptions.class); config.addActiveDescriptor(desc); config.commit(); } private static String[] getArgs(String line) throws ArgumentException { List args = new ArrayList<>(); ArgumentTokenizer t = new ArgumentTokenizer(line); while (t.hasMoreTokens()) { args.add(t.nextToken()); } return args.toArray(String[]::new); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy