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

com.squarespace.less.parse.Stream Maven / Gradle / Ivy

/**
 * Copyright (c) 2014 SQUARESPACE, Inc.
 *
 * 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.squarespace.less.parse;

import org.apache.commons.lang3.StringEscapeUtils;

import com.squarespace.less.core.CharClass;
import com.squarespace.less.core.CharPattern;
import com.squarespace.less.core.Chars;


/**
 * Wraps a String and provides an interface for interacting with the sequence of characters.
 */
public class Stream {

  protected static final boolean DEBUG = false;

  protected final String raw;

  protected final int length;

  protected int index;

  protected int furthest;

  /**
   * Offset of the current line. NOTE: zero-based.
   */
  protected int lineOffset;

  /**
   * Offset of the character position of the current line. NOTE: zero-based.
   */
  protected int charOffset;

  public Stream(String raw) {
    this.raw = raw;
    this.length = raw.length();
  }

  public int getLineOffset() {
    return lineOffset;
  }

  public int getCharOffset() {
    return charOffset;
  }

  protected void dump() {
    char ch = (index >= length) ? Chars.EOF : raw.charAt(index);
    String esc = StringEscapeUtils.escapeJava(ch + "");
    System.out.printf("Stream: index=%d len=%d line=%d char=%d ch=\"%s\"\n",
          index, length, lineOffset, charOffset, esc);
  }

  public String raw() {
    return raw;
  }

  public int position() {
    return index;
  }

  public Mark mark() {
    Mark pos = new Mark();
    mark(pos);
    return pos;
  }

  /**
   * Mark current position in the stream so that we can restore it later.
   */
  public void mark(Mark mark) {
    mark.index = index;
    mark.lineOffset = lineOffset;
    mark.charOffset = charOffset;
  }

  /**
   * Restore the stream to the marked position.
   */
  public int restore(Mark mark) {
    index = mark.index;
    lineOffset = mark.lineOffset;
    charOffset = mark.charOffset;
    return index;
  }

  /**
   * Return the character under the stream pointer. Does not increment the
   * stream pointer.
   */
  public char peek() {
    return (index >= length) ? Chars.EOF : raw.charAt(index);
  }

  /**
   * Return the character at position 'index + offset'. Does not increment the
   * stream pointer.
   */
  public char peek(int offset) {
    int pos = index + offset;
    return (pos < 0 || pos >= length) ? Chars.EOF : raw.charAt(pos);
  }

  /**
   * Seek ahead in the stream 'offset' characters. Increments the stream pointer and
   * increments the line/offset counters.
   */
  public char seek(int offset) {
    int limit = Math.min(length, index + offset);
    while (index < limit) {
      consume(raw.charAt(index));
      index++;
    }
    furthest = Math.max(index, furthest);
    return peek();
  }

  public char seek1() {
    if (index < length) {
      consume(raw.charAt(index));
      index++;
    }
    furthest = Math.max(index, furthest);
    return peek();
  }

  /**
   * Seek ahead until 'ch' is found. Increments the stream pointer, leaving it pointing at
   * the character just after 'ch', or the end of the string if not matched.
   */
  public void seekTo(char ch) {
    while (index < length) {
      char curr = raw.charAt(index);
      consume(curr);
      index++;
      if (curr == ch) {
        break;
      }
    }
    furthest = Math.max(index, furthest);
  }

  /**
   * If the character under the cursor equals 'ch', consume it and return
   * true; else return false;
   */
  public boolean seekIf(char ch) {
    if (peek() == ch) {
      seek1();
      return true;
    }
    return false;
  }

  /**
   * Consume all whitespace characters, positioning the pointer over the next non-whitespace character.
   * Returns the number of characters skipped.
   */
  public int skipWs() {
    int start = index;
    while (index < length) {
      char curr = raw.charAt(index);
      if (!CharClass.whitespace(curr)) {
        break;
      }
      consume(curr);
      index++;
    }
    // Important not to update 'furthest' pointer when skipping whitespace
    return index - start;
  }

  public int skipEmpty() {
    int start = index;
    while (index < length) {
      char curr = raw.charAt(index);
      if (!CharClass.skippable(curr)) {
        break;
      }
      consume(curr);
      index++;
    }
    // Important not to update 'furthest' pointer when skipping 'empty' chars.
    return index - start;
  }

  /**
   * Searches the stream to find the character pattern, consuming every character
   * it sees along the way.  Searching the string "ABCDEFGH" for the pattern "DE",
   * this would position the cursor over the 'F'.
   *
   * Searching for a pattern P in a string S of length N, this runs O(N).
   */
  public boolean seek(CharPattern table) {
    char[] pattern = table.pattern();
    int[] next = table.next();
    int patternLen = pattern.length;
    int j;
    for (j = 0; index < length && j < patternLen; index++) {
      char ch = raw.charAt(index);
      consume(ch);
      while (j >= 0 && ch != pattern[j]) {
        j = next[j];
      }
      j++;
    }
    furthest = Math.max(index, furthest);
    return (j == patternLen);
  }


  @Override
  public String toString() {
    return "Stream(\"" + StringEscapeUtils.escapeJava(raw.substring(index)) + "\")";
  }

  public String furthest() {
    return raw.substring(Math.min(furthest, length - 1));
  }

  private void consume(char ch) {
    if (ch == Chars.LINE_FEED) {
      lineOffset++;
      charOffset = 0;
    } else {
      charOffset++;
    }
  }

  /**
   * Useful for debugging the parser.
   */
  @SuppressWarnings("unused")
  private void stack() {
    StackTraceElement[] elems = Thread.currentThread().getStackTrace();
    for (int i = 1; i < 5; i++) {
      System.out.println(elems[i].getLineNumber() + " " + elems[i].getFileName() + " " + elems[i].getMethodName());
    }
    System.out.println();
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy