com.couchbase.client.core.endpoint.util.ClosingPositionBufProcessor Maven / Gradle / Ivy
Go to download
Show more of this group Show more artifacts with this name
Show all versions of core-io Show documentation
Show all versions of core-io Show documentation
The official Couchbase JVM Core IO Library
/**
* Copyright (c) 2015 Couchbase, Inc.
*
* Permission is hereby granted, free of charge, to any person obtaining a copy
* of this software and associated documentation files (the "Software"), to deal
* in the Software without restriction, including without limitation the rights
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
* copies of the Software, and to permit persons to whom the Software is
* furnished to do so, subject to the following conditions:
*
* The above copyright notice and this permission notice shall be included in
* all copies or substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALING
* IN THE SOFTWARE.
*/
package com.couchbase.client.core.endpoint.util;
import com.couchbase.client.deps.io.netty.buffer.ByteBuf;
import com.couchbase.client.deps.io.netty.buffer.ByteBufProcessor;
/**
* A {@link ByteBufProcessor} to find a closing character position. Applying this to a
* buffer will output the position of the closing of the section, relative to that buffer's
* readerIndex, or -1 if the end of the section couldn't be found.
*
* Note that this processor will only work correctly if the number of opening and closing
* characters match up. This is typically the case when searching for open and closing {}
* in a streaming JSON response (in which case the constructor variant that detects JSON
* strings should be used).
*
* It is invoked on a {@link ByteBuf} by calling {@link ByteBuf#forEachByte(ByteBufProcessor)} methods.
*
* @author Simon Baslé
* @since 1.1.0
*/
public class ClosingPositionBufProcessor implements ByteBufProcessor {
/**
* The number of open characters found so far.
*/
private int openCount = 0;
/**
* The open character to search for.
*/
private final char openingChar;
/**
* The close character to search for.
*/
private final char closingChar;
/**
* Should we detect opening and closing of JSON strings and ignore characters in there?
*/
private final boolean detectJsonString;
/**
* @param openingChar the opening section character (used to detect a sub-section).
* @param closingChar the closing section character to search for.
*/
public ClosingPositionBufProcessor(char openingChar, char closingChar) {
this(openingChar, closingChar, false);
}
/**
* @param openingChar the opening section character (used to detect a sub-section)
* @param closingChar the closing section character to search for.
* @param detectJsonString set to true to not inspect bytes detected as being part of a String.
*/
public ClosingPositionBufProcessor(char openingChar, char closingChar, boolean detectJsonString) {
if (openingChar == closingChar) {
throw new IllegalArgumentException("only asymmetric section enclosing characters are supported");
}
this.openingChar = openingChar;
this.closingChar = closingChar;
this.detectJsonString = detectJsonString;
}
@Override
public boolean process(final byte current) throws Exception {
//don't look into escaped bytes for opening/closing section
//note that we wait for the opening of the section to do string checking
if (detectJsonString && openCount > 0 && isEscaped(current)) {
return true;
}
//now lookout for sub-sections, try to find the closing of the root section
if (current == openingChar) {
openCount++;
} else if (current == closingChar && openCount > 0) {
openCount--;
if (openCount == 0) {
return false;
}
}
return true;
}
/** previous byte inspected by string detection (useful to detect escaped quotes) */
private byte lastByte = 0;
/** flag to indicate that we are currently reading a JSON string */
private boolean inString = false;
/**
* Detects opening and closing of JSON strings and keep track of it in order
* to mark characters in the string (delimiter quotes included) as escaped.
*
* Quotes escaped by a \ are correctly detected and do not mark a closing of
* a JSON string.
*
* @param nextByte the next byte to inspect.
* @return true if the byte should be ignored as part of a JSON string, false otherwise.
*/
private boolean isEscaped(byte nextByte) {
boolean result = false;
if (inString) {
if (nextByte == '\"'
&& lastByte != '\\') {
inString = false;
}
result = true;
} else {
if (nextByte == '\"') {
inString = true;
result = true;
}
}
lastByte = nextByte;
return result;
}
}