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

org.apache.jackrabbit.oak.console.GroovyConsole.groovy Maven / Gradle / Ivy

/*
 * 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.jackrabbit.oak.console

import groovy.transform.CompileStatic
import groovy.transform.TypeCheckingMode
import jline.Terminal
import jline.TerminalFactory
import jline.console.history.FileHistory
import org.apache.jackrabbit.oak.run.commons.Utils
import org.apache.jackrabbit.oak.console.commands.*
import org.apache.jackrabbit.oak.plugins.document.DocumentNodeStore
import org.codehaus.groovy.runtime.StackTraceUtils
import org.codehaus.groovy.tools.shell.*
import org.codehaus.groovy.tools.shell.Command as ShellCommand
import org.codehaus.groovy.tools.shell.commands.*
import org.codehaus.groovy.tools.shell.util.Logger
import org.codehaus.groovy.tools.shell.util.Preferences
import org.fusesource.jansi.Ansi
import org.fusesource.jansi.AnsiConsole
import org.fusesource.jansi.AnsiRenderer

/**
 * Some part of logic is based on usage in org.codehaus.groovy.tools.shell.Main
 */
@CompileStatic
class GroovyConsole {
    static {
        try {
            // Install the system adapters
            AnsiConsole.systemInstall()

            // Register jline ansi detector
            Ansi.setDetector(new AnsiDetector())
        } catch (UnsatisfiedLinkError e){
            Logger.create(GroovyConsole.class).warn("Error loading console support. Some console features might not work properly. See " +
                    "https://issues.apache.org/jira/browse/OAK-5961 for details", e)
        }
    }

    private final ConsoleSession session
    private final Groovysh shell
    private final IO io;

    GroovyConsole(ConsoleSession session, IO io, Closeable closeable) {
        this.session = session
        this.io = io
        this.shell = prepareShell()
        addShutdownHook {
            if (shell.history) {
                shell.history.flush()
            }
            closeable.close()
        }
    }

    int run(){
        shell.run(null, null)
    }

    int execute(List args){
        try {
            shell.execute(args.join(' '))
            return 0;
        }catch(Throwable t){
            t.printStackTrace();
            return 1;
        }
    }

    private Groovysh prepareShell() {
        Binding binding = new Binding()
        binding['session'] = session
        Groovysh sh = new OakSh(getClass().getClassLoader(),
                binding, io, this.®isterCommands)
        sh.imports << 'org.apache.jackrabbit.oak.plugins.document.*'
        sh.imports << 'org.apache.jackrabbit.oak.plugins.segment.*'
        return sh
    }

    private void registerCommands(Groovysh shell){
        List commands = []
        commands.addAll([
                new ExitCommand(shell),
                new ImportCommand(shell),
                new DisplayCommand(shell),
                new ClearCommand(shell),
                new ShowCommand(shell),
                new InspectCommand(shell),
                new PurgeCommand(shell),
                new EditCommand(shell),
                new LoadCommand(shell),
                new SaveCommand(shell),
                new RecordCommand(shell),
                new HistoryCommand(shell),
                new AliasCommand(shell),
                new SetCommand(shell),
                // does not do anything
                //new ShadowCommand(shell),
                new RegisterCommand(shell),
                new DocCommand(shell),
        ]);

        commands.addAll([
                //Oak Commands
                new OakHelpCommand(shell),
                new CdCommand(shell),
                new CheckpointCommand(shell),
                new LsCommand(shell),
                new PnCommand(shell),
                new RefreshCommand(shell),
                new RetrieveCommand(shell),
                new LuceneCommand(shell),
                new ExportRelevantDocumentsCommand(shell),
                new ExportCommand(shell)
        ])

        if(session.store instanceof DocumentNodeStore){
            commands.addAll([
                    //Oak Commands
                    new PrintDocumentCommand(shell),
                    new LsdDocumentCommand(shell),
            ])
        }

        commands.each {ShellCommand command ->
            shell.register(command)
        }
    }

    private class OakSh extends Groovysh {
        private boolean colored = false

        OakSh(ClassLoader classLoader, Binding binding, IO io, Closure registrar) {
            super(classLoader, binding, io, registrar)
        }

        public String renderPrompt() {
            return AnsiRenderer.render( buildPrompt() )
        }

        //Following methods are copied because they are private in parent however
        //they are referred via method handle which somehow looks for method in
        //derived class. And we need to customize the welcome banner in run method
        private String buildPrompt() {
            def prefix = session.workingPath
            def groovyShellProperty = System.getProperty("groovysh.prompt")
            def groovyShellEnv = System.getenv("GROOVYSH_PROMPT")
            if (groovyShellProperty) {
                prefix = groovyShellProperty
            } else if (groovyShellEnv) {
                prefix = groovyShellEnv
            }
            return colored ? "@|bold,blue ${prefix}>|@ " : "${prefix}>"
        }

        private void displayError(final Throwable cause) {
            if (errorHook == null) {
                throw new IllegalStateException("Error hook is not set")
            }
            if (cause instanceof MissingPropertyException) {
                if (cause.type && cause.type.canonicalName == Interpreter.SCRIPT_FILENAME) {
                    io.err.println("@|bold,red Unknown property|@: " + cause.property)
                    return
                }
            }

            errorHook.call(cause)
        }

        @CompileStatic(TypeCheckingMode.SKIP)
        protected void maybeRecordError(Throwable cause) {
            RecordCommand record = registry[RecordCommand.COMMAND_NAME]

            if (record != null) {
                boolean sanitize = Preferences.sanitizeStackTrace

                if (sanitize) {
                    cause = StackTraceUtils.deepSanitize(cause);
                }

                record.recordError(cause)
            }
        }

        @CompileStatic(TypeCheckingMode.SKIP)
        int run(final String commandLine) {
            Terminal term = TerminalFactory.create()
            colored = term.ansiSupported
            if (log.debug) {
                log.debug("Terminal ($term)")
                log.debug("    Supported:  $term.supported")
                log.debug("    ECHO:       (enabled: $term.echoEnabled)")
                log.debug("    H x W:      ${term.getHeight()} x ${term.getWidth()}")
                log.debug("    ANSI:       ${term.isAnsiSupported()}")

                if (term instanceof jline.WindowsTerminal) {
                    jline.WindowsTerminal winterm = (jline.WindowsTerminal) term
                    log.debug("    Direct:     ${winterm.directConsole}")
                }
            }

            def code

            try {
                loadUserScript('groovysh.profile')

                // if args were passed in, just execute as a command
                // (but cygwin gives an empty string, so ignore that)
                if (commandLine != null && commandLine.trim().size() > 0) {
                    // Run the given commands
                    execute(commandLine)
                } else {
                    loadUserScript('groovysh.rc')

                    // Setup the interactive runner
                    runner = new InteractiveShellRunner(
                            this,
                            this.&renderPrompt as Closure)

                    // Setup the history
                    File histFile = new File(userStateDirectory, 'groovysh.history')
                    history = new FileHistory(histFile)
                    runner.setHistory(history)

                    // Setup the error handler
                    runner.errorHandler = this.&displayError

                    //
                    // TODO: See if we want to add any more language specific completions, like for println for example?
                    //

                    // Display the welcome banner
                    if (!io.quiet) {
                        int width = term.getWidth()

                        // If we can't tell, or have something bogus then use a reasonable default
                        if (width < 1) {
                            width = 80
                        }

                        io.out.println("@|green Jackrabbit Oak Shell|@ (${getProductInfo()}, " +
                                "JVM: ${System.properties['java.version']})")
                        io.out.println("Type '@|bold :help|@' or '@|bold :h|@' for help.")
                        io.out.println('-' * (width - 1))
                    }

                    // And let 'er rip... :-)
                    runner.run()
                }

                code = 0
            }
            catch (ExitNotification n) {
                log.debug("Exiting w/code: ${n.code}")

                code = n.code
            }
            catch (Throwable t) {
                io.err.println("FATAL : " + t)
                t.printStackTrace(io.err)

                code = 1
            }

            assert code != null // This should never happen

            return code
        }

    }

    private static String getProductInfo(){
        return Utils.getProductInfo(
                GroovyConsole.class.getResourceAsStream("/META-INF/maven/org.apache.jackrabbit/oak-run/pom.properties"));
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy