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

com.github.sbaudoin.sonar.plugins.shellcheck.highlighting.ShellLocation Maven / Gradle / Ivy

/**
 * Copyright (c) 2018-2019, Sylvain Baudoin
 *
 * 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.github.sbaudoin.sonar.plugins.shellcheck.highlighting;

import com.github.sbaudoin.sonar.plugins.shellcheck.lexer.Token;

/**
 * Wrapper class used to locate tokens in Shell scripts and help highlight them
 */
public class ShellLocation {
    private final String content;
    private final int line;
    private final int column;
    private final int characterOffset;


    /**
     * Constructor
     *
     * @param content the Shell script to highlight
     * @param token a token to be pointed to (and certainly highlighted) in the Shell script
     */
    public ShellLocation(String content, Token token) {
        this(content, token.line + 1, token.column + 1, token.start);
    }

    /**
     * Constructor
     *
     * @param content the Shell script to highlight
     * @param line a line to point to (by convention, lines start at 1)
     * @param column a column to point to (by convention, columns start at 1)
     * @param characterOffset a character offset in the content equals to the number of characters read so far between
     *                        the content's start and the given column of the given line
     */
    public ShellLocation(String content, int line, int column, int characterOffset) {
        this.content = content;
        this.line = line;
        this.column = column;
        this.characterOffset = characterOffset;
    }

    /**
     * Returns the line pointer remembered by this class
     *
     * @return the line pointer remembered by this class
     */
    public int line() {
        return line;
    }

    /**
     * Returns the column pointer remembered by this class
     *
     * @return the column pointer remembered by this class
     */
    public int column() {
        return column;
    }

    /**
     * Returns a string representation of this class
     *
     * @return a string representation of this class
     */
    public String toString() {
        StringBuilder sb = new StringBuilder("{ ");
        sb.append("content: \"").append(content).append("\"; ");
        sb.append("line: ").append(line).append("; ");
        sb.append("column: ").append(column).append("; ");
        sb.append("characterOffset: ").append(characterOffset);
        sb.append(" }");

        return sb.toString();
    }

    /**
     * Returns a {@code ShellLocation} that corresponds to this instance + line, column and offset parameters moved
     * forward of the passed number of characters
     *
     * @param nbChar a number of character to move forward
     * @return a {@code ShellLocation} updated with the passed additional offset
     * @throws IllegalStateException if the passed number of character added to the offset will point to a character
     * beyond the content's end (i.e. if {@code offset + nbChar > content.length()})
     * @see #shift(char)
     */
    public ShellLocation shift(int nbChar) {
        if (characterOffset + nbChar > content.length()) {
            throw new IllegalStateException("Cannot shift by " + nbChar + " characters");
        }
        ShellLocation res = this;
        for (int i = 0; i < nbChar; i++) {
            res = res.shift(res.readChar());
        }
        return res;
    }

    /**
     * Returns the character of the Shell script pointed to by the offset
     *
     * @return a character
     */
    public char readChar() {
        return content.charAt(characterOffset);
    }

    /**
     * Tells if the passed {@code ShellLocation} has the same offset or not (this is a kind of shallow version of the
     * {@code equals} method)
     *
     * @param other another {@code ShellLocation}
     * @return {@code true} if the offset of the passed {@code ShellLocation} equals to the offset of this class,
     * {@code false} if not
     */
    public boolean isSameAs(ShellLocation other) {
        return this.characterOffset == other.characterOffset;
    }


    /**
     * Shifts the internal pointers (line, column and offset) of one character. The column and line pointers are
     * updated depending on the fact that the passed character is a new line character or not (the passed character
     * is supposed to be the character currently pointed to by the offset)
     *
     * @param c a character
     * @return a new {@code ShellLocation} moved forward of 1 character
     * @throws IllegalStateException if the passed number of character added to the offset will point to a character
     * beyond the content's end (i.e. if {@code offset + nbChar > content.length()})
     * @see #shift(int)
     */
    private ShellLocation shift(char c) {
        if (c == '\n') {
            return new ShellLocation(content, line + 1, 1, characterOffset + 1);
        }
        return new ShellLocation(content, line, column + 1, characterOffset + 1);
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy