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

com.twosigma.webtau.cli.interactive.ParsedCommand.groovy Maven / Gradle / Ivy

/*
 * Copyright 2019 TWO SIGMA OPEN SOURCE, LLC
 *
 * Licensed 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 com.twosigma.webtau.cli.interactive

import com.twosigma.webtau.utils.NumberUtils
import groovy.transform.PackageScope

@PackageScope
class ParsedCommand {
    List indexes = []
    List commands = []
    List unrecognized = []
    String error
    int errorTextIdx

    ParsedCommand(String text) {
        def parser = new Parser(text)
        parser.parse()
    }

    private static InteractiveCommand findCommand(String text) {
        return InteractiveCommand.values().find { it.matches(text) }
    }

    private static Integer convertToNumber(String text) {
        return NumberUtils.convertStringToNumber(text)
    }

    private static enum ParserState {
        Seeking,
        SeekingPotentialRange,
        Command,
        Number,
        RangeSecondNumber,
        Error
    }

    private class Parser {
        private final String text

        private StringBuilder current = new StringBuilder()

        private ParserState state
        private int idx

        Parser(String text) {
            this.text = text
            this.state = ParserState.Seeking
        }

        void parse() {
            for (idx = 0; idx < text.length(); idx++) {
                def c = text.charAt(idx)

                handle(c)

                if (state == Error) {
                    break
                }
            }

            handleDelimiter()
        }

        void handle(char c) {
            if (isDelimiter(c)) {
                if (current.size() > 0) {
                    handleDelimiter()
                }
                return
            }

            switch (state) {
                case ParserState.Seeking:
                    handleFirstChar(c)
                    break
                case ParserState.Command:
                    handleCommandChar(c)
                    break
                case ParserState.Number:
                    handleNumberChar(c)
                    break
                case ParserState.SeekingPotentialRange:
                    handlePotentialRange(c)
                    break
                case ParserState.RangeSecondNumber:
                    handleSecondNumberChar(c)
                    break
            }
        }

        void handleDelimiter() {
            switch (state) {
                case ParserState.Command:
                    def commandAsText = flush()
                    def command = findCommand(commandAsText)
                    if (command) {
                        commands.add(command)
                        state = ParserState.Seeking
                    } else {
                        unrecognized.add(commandAsText)
                        markError('unrecognized command: ' + commandAsText)
                    }
                    break

                case ParserState.Number:
                    indexes.add(convertToNumber(flush()))
                    state = ParserState.SeekingPotentialRange
                    break

                case ParserState.RangeSecondNumber:
                    indexes.addAll(createRange(indexes.last() + 1, convertToNumber(flush())))
                    state = ParserState.Seeking
                    break
            }
        }

        void handleFirstChar(char c) {
            if (c.isDigit()) {
                state = ParserState.Number
            } else {
                state = ParserState.Command
            }

            current.append(c)
        }

        void handleCommandChar(char c) {
            current.append(c)
        }

        void handleNumberChar(char c) {
            if (isRangeSymbol(c)) {
                handleDelimiter()
                state = ParserState.RangeSecondNumber
                return
            }

            if (!c.isDigit()) {
                markError('incorrect number')
            }

            current.append(c)

            if (current.length() > 3) {
                markError('number is too big')
            }
        }

        void handleSecondNumberChar(char c) {
            if (!c.isDigit()) {
                markError('incorrect number')
            }

            current.append(c)
        }

        void handlePotentialRange(char c) {
            if (isRangeSymbol(c)) {
                state = ParserState.RangeSecondNumber
            } else {
                state = ParserState.Seeking
                handle(c)
            }
        }

        List createRange(Integer a, Integer b) {
            return a..b
        }

        String flush() {
            String result = current.toString()
            current = new StringBuilder()
            return result
        }

        void markError(message) {
            error = message
            errorTextIdx = idx
            state = ParserState.Error
        }

        private static boolean isRangeSymbol(char c) {
            return c == '-'
        }

        private static boolean isDelimiter(char c) {
            return c == ' ' || c == ','
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy