langtools/src/jdk.jshell/share/classes/jdk/jshell/MaskCommentsAndModifiers.java
author rfield
Wed, 08 Jun 2016 00:32:31 -0700
changeset 38908 f0c186d76c8a
parent 38835 37280d52d723
child 41940 048d559e9da7
permissions -rw-r--r--
8139829: JShell API: No use of fields to return information from public types Reviewed-by: vromero

/*
 * Copyright (c) 2015, Oracle and/or its affiliates. All rights reserved.
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
 *
 * This code is free software; you can redistribute it and/or modify it
 * under the terms of the GNU General Public License version 2 only, as
 * published by the Free Software Foundation.  Oracle designates this
 * particular file as subject to the "Classpath" exception as provided
 * by Oracle in the LICENSE file that accompanied this code.
 *
 * This code is distributed in the hope that it will be useful, but WITHOUT
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
 * version 2 for more details (a copy is included in the LICENSE file that
 * accompanied this code).
 *
 * You should have received a copy of the GNU General Public License version
 * 2 along with this work; if not, write to the Free Software Foundation,
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
 *
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
 * or visit www.oracle.com if you need additional information or have any
 * questions.
 */

package jdk.jshell;

import java.util.Set;
import java.util.stream.Collectors;
import java.util.stream.Stream;

/**
 * Within a String, mask code comments and ignored modifiers (within context).
 *
 * @author Robert Field
 */
class MaskCommentsAndModifiers {

    private final static Set<String> IGNORED_MODIFERS =
            Stream.of( "public", "protected", "private", "static", "final" )
                    .collect( Collectors.toSet() );

    // Builder to accumulate non-masked characters
    private final StringBuilder sbCleared = new StringBuilder();

    // Builder to accumulate masked characters
    private final StringBuilder sbMask = new StringBuilder();

    // The input string
    private final String str;

    // Entire input string length
    private final int length;

    // Should leading modifiers be masked away
    private final boolean maskModifiers;

    // The next character
    private int next = 0;

    // We have past any point where a top-level modifier could be
    private boolean inside = false;

    // Does the string end with an unclosed '/*' style comment?
    private boolean openComment = false;

    @SuppressWarnings("empty-statement")
    MaskCommentsAndModifiers(String s, boolean maskModifiers) {
        this.str = s;
        this.length = s.length();
        this.maskModifiers = maskModifiers;
        do { } while (next());
    }

    String cleared() {
        return sbCleared.toString();
    }

    String mask() {
        return sbMask.toString();
    }

    boolean endsWithOpenComment() {
        return openComment;
    }

    /****** private implementation methods ******/

    /**
     * Read the next character
     */
    private int read() {
        if (next >= length) {
            return -1;
        }
        return str.charAt(next++);
    }

    private void write(StringBuilder sb, int ch) {
        sb.append((char)ch);
    }

    private void write(int ch) {
        write(sbCleared, ch);
        write(sbMask, Character.isWhitespace(ch) ? ch : ' ');
    }

    private void writeMask(int ch) {
        write(sbMask, ch);
        write(sbCleared, Character.isWhitespace(ch) ? ch : ' ');
    }

    private void write(CharSequence s) {
        for (int cp : s.chars().toArray()) {
            write(cp);
        }
    }

    private void writeMask(CharSequence s) {
        for (int cp : s.chars().toArray()) {
            writeMask(cp);
        }
    }

    private boolean next() {
        return next(read());
    }

    private boolean next(int c) {
        if (c < 0) {
            return false;
        }

        if (c == '\'' || c == '"') {
            inside = true;
            write(c);
            int match = c;
            c = read();
            while (c != match) {
                if (c < 0) {
                    return false;
                }
                if (c == '\n' || c == '\r') {
                    write(c);
                    return true;
                }
                if (c == '\\') {
                    write(c);
                    c = read();
                }
                write(c);
                c = read();
            }
            write(c);
            return true;
        }

        if (c == '/') {
            c = read();
            if (c == '*') {
                writeMask('/');
                writeMask(c);
                int prevc = 0;
                while ((c = read()) != '/' || prevc != '*') {
                    if (c < 0) {
                        openComment = true;
                        return false;
                    }
                    writeMask(c);
                    prevc = c;
                }
                writeMask(c);
                return true;
            } else if (c == '/') {
                writeMask('/');
                writeMask(c);
                while ((c = read()) != '\n' && c != '\r') {
                    if (c < 0) {
                        return false;
                    }
                    writeMask(c);
                }
                writeMask(c);
                return true;
            } else {
                inside = true;
                write('/');
                // read character falls through
            }
        }

        if (Character.isJavaIdentifierStart(c)) {
            if (maskModifiers && !inside) {
                StringBuilder sb = new StringBuilder();
                do {
                    write(sb, c);
                    c = read();
                } while (Character.isJavaIdentifierPart(c));
                String id = sb.toString();
                if (IGNORED_MODIFERS.contains(id)) {
                    writeMask(sb);
                } else {
                    write(sb);
                    if (id.equals("import")) {
                        inside = true;
                    }
                }
                return next(c); // recurse to handle left-over character
            }
        } else if (!Character.isWhitespace(c)) {
            inside = true;
        }

        if (c < 0) {
            return false;
        }
        write(c);
        return true;
    }
}