From ff2e790e8936772ba952471e951e0c504b220983 Mon Sep 17 00:00:00 2001 From: Chris Vest Date: Tue, 3 Nov 2020 16:12:54 +0100 Subject: [PATCH] Revert "ByteProcessor shouldn't throw checked exception" This reverts commit b70d0fa6e3aeadc7863d65aff62a96d1aa301425. --- .../java/io/netty/buffer/AbstractByteBuf.java | 33 ++++++++-- .../io/netty/buffer/CompositeByteBuf.java | 4 +- .../io/netty/buffer/AbstractByteBufTest.java | 10 +-- .../codec/http/DefaultHttpHeaders.java | 11 +++- .../handler/codec/http/HttpObjectDecoder.java | 4 +- .../HttpPostStandardRequestDecoder.java | 2 +- .../codec/http/websocketx/Utf8Validator.java | 2 +- .../codec/http2/DefaultHttp2Headers.java | 15 ++++- .../codec/http2/HpackHuffmanEncoder.java | 15 +++-- .../codec/http2/HttpConversionUtil.java | 62 +++++++++++-------- .../handler/codec/redis/RedisDecoder.java | 2 +- .../codec/stomp/StompSubframeDecoder.java | 4 +- .../io/netty/handler/codec/base64/Base64.java | 2 +- .../main/java/io/netty/util/AsciiString.java | 12 ++-- .../java/io/netty/util/ByteProcessor.java | 2 +- .../io/netty/util/AsciiStringMemoryTest.java | 20 +++--- 16 files changed, 128 insertions(+), 72 deletions(-) diff --git a/buffer/src/main/java/io/netty/buffer/AbstractByteBuf.java b/buffer/src/main/java/io/netty/buffer/AbstractByteBuf.java index 7851ebaf2d..e0116e5db1 100644 --- a/buffer/src/main/java/io/netty/buffer/AbstractByteBuf.java +++ b/buffer/src/main/java/io/netty/buffer/AbstractByteBuf.java @@ -1262,37 +1262,58 @@ public abstract class AbstractByteBuf extends ByteBuf { @Override public int forEachByte(ByteProcessor processor) { ensureAccessible(); - return forEachByteAsc0(readerIndex, writerIndex, processor); + try { + return forEachByteAsc0(readerIndex, writerIndex, processor); + } catch (Exception e) { + PlatformDependent.throwException(e); + return -1; + } } @Override public int forEachByte(int index, int length, ByteProcessor processor) { checkIndex(index, length); - return forEachByteAsc0(index, index + length, processor); + try { + return forEachByteAsc0(index, index + length, processor); + } catch (Exception e) { + PlatformDependent.throwException(e); + return -1; + } } - int forEachByteAsc0(int start, int end, ByteProcessor processor) { + int forEachByteAsc0(int start, int end, ByteProcessor processor) throws Exception { for (; start < end; ++start) { if (!processor.process(_getByte(start))) { return start; } } + return -1; } @Override public int forEachByteDesc(ByteProcessor processor) { ensureAccessible(); - return forEachByteDesc0(writerIndex - 1, readerIndex, processor); + try { + return forEachByteDesc0(writerIndex - 1, readerIndex, processor); + } catch (Exception e) { + PlatformDependent.throwException(e); + return -1; + } } @Override public int forEachByteDesc(int index, int length, ByteProcessor processor) { checkIndex(index, length); - return forEachByteDesc0(index + length - 1, index, processor); + try { + return forEachByteDesc0(index + length - 1, index, processor); + } catch (Exception e) { + PlatformDependent.throwException(e); + return -1; + } } - int forEachByteDesc0(int rStart, final int rEnd, ByteProcessor processor) { + int forEachByteDesc0(int rStart, final int rEnd, ByteProcessor processor) throws Exception { for (; rStart >= rEnd; --rStart) { if (!processor.process(_getByte(rStart))) { return rStart; diff --git a/buffer/src/main/java/io/netty/buffer/CompositeByteBuf.java b/buffer/src/main/java/io/netty/buffer/CompositeByteBuf.java index 0b8106b2b1..f9ebb47da2 100644 --- a/buffer/src/main/java/io/netty/buffer/CompositeByteBuf.java +++ b/buffer/src/main/java/io/netty/buffer/CompositeByteBuf.java @@ -660,7 +660,7 @@ public class CompositeByteBuf extends AbstractReferenceCountedByteBuf implements } @Override - protected int forEachByteAsc0(int start, int end, ByteProcessor processor) { + protected int forEachByteAsc0(int start, int end, ByteProcessor processor) throws Exception { if (end <= start) { return -1; } @@ -686,7 +686,7 @@ public class CompositeByteBuf extends AbstractReferenceCountedByteBuf implements } @Override - protected int forEachByteDesc0(int rStart, int rEnd, ByteProcessor processor) { + protected int forEachByteDesc0(int rStart, int rEnd, ByteProcessor processor) throws Exception { if (rEnd > rStart) { // rStart *and* rEnd are inclusive return -1; } diff --git a/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java b/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java index 1467a7898e..c7e0f18446 100644 --- a/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java +++ b/buffer/src/test/java/io/netty/buffer/AbstractByteBufTest.java @@ -2284,7 +2284,7 @@ public abstract class AbstractByteBufTest { int i = CAPACITY / 4; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertThat(value, is((byte) (i + 1))); lastIndex.set(i); i ++; @@ -2307,7 +2307,7 @@ public abstract class AbstractByteBufTest { int i = CAPACITY / 3; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertThat(value, is((byte) (i + 1))); if (i == stop) { return false; @@ -4608,7 +4608,7 @@ public abstract class AbstractByteBufTest { private int index = bytes.length - 1; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { bytes[index--] = value; return true; } @@ -4631,7 +4631,7 @@ public abstract class AbstractByteBufTest { private int index; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { bytes[index++] = value; return true; } @@ -4804,7 +4804,7 @@ public abstract class AbstractByteBufTest { private static final class TestByteProcessor implements ByteProcessor { @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { return true; } } diff --git a/codec-http/src/main/java/io/netty/handler/codec/http/DefaultHttpHeaders.java b/codec-http/src/main/java/io/netty/handler/codec/http/DefaultHttpHeaders.java index 9611336ba3..322e0f7c2c 100644 --- a/codec-http/src/main/java/io/netty/handler/codec/http/DefaultHttpHeaders.java +++ b/codec-http/src/main/java/io/netty/handler/codec/http/DefaultHttpHeaders.java @@ -24,6 +24,7 @@ import io.netty.handler.codec.HeadersUtils; import io.netty.handler.codec.ValueConverter; import io.netty.util.AsciiString; import io.netty.util.ByteProcessor; +import io.netty.util.internal.PlatformDependent; import java.util.ArrayList; import java.util.Calendar; @@ -49,10 +50,14 @@ public class DefaultHttpHeaders extends HttpHeaders { }; static final NameValidator HttpNameValidator = name -> { if (name == null || name.length() == 0) { - throw new IllegalArgumentException("empty headers are not allowed [" + name + ']'); + throw new IllegalArgumentException("empty headers are not allowed [" + name + "]"); } if (name instanceof AsciiString) { - ((AsciiString) name).forEachByte(HEADER_NAME_VALIDATOR); + try { + ((AsciiString) name).forEachByte(HEADER_NAME_VALIDATOR); + } catch (Exception e) { + PlatformDependent.throwException(e); + } } else { // Go through each character in the name for (int index = 0; index < name.length(); ++index) { @@ -68,7 +73,7 @@ public class DefaultHttpHeaders extends HttpHeaders { } /** - * Warning! Setting {@code validate} to {@code false} will mean that Netty won't + * Warning! Setting validate to false will mean that Netty won't * validate & protect against user-supplied header values that are malicious. * This can leave your server implementation vulnerable to * diff --git a/codec-http/src/main/java/io/netty/handler/codec/http/HttpObjectDecoder.java b/codec-http/src/main/java/io/netty/handler/codec/http/HttpObjectDecoder.java index f2cc0084b1..559ad12e08 100644 --- a/codec-http/src/main/java/io/netty/handler/codec/http/HttpObjectDecoder.java +++ b/codec-http/src/main/java/io/netty/handler/codec/http/HttpObjectDecoder.java @@ -936,7 +936,7 @@ public abstract class HttpObjectDecoder extends ByteToMessageDecoder { } @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { char nextByte = (char) (value & 0xFF); if (nextByte == HttpConstants.LF) { int len = seq.length(); @@ -983,7 +983,7 @@ public abstract class HttpObjectDecoder extends ByteToMessageDecoder { } @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { if (currentState == State.SKIP_CONTROL_CHARS) { char c = (char) (value & 0xFF); if (Character.isISOControl(c) || Character.isWhitespace(c)) { diff --git a/codec-http/src/main/java/io/netty/handler/codec/http/multipart/HttpPostStandardRequestDecoder.java b/codec-http/src/main/java/io/netty/handler/codec/http/multipart/HttpPostStandardRequestDecoder.java index bec61ae8b4..d4d73c0275 100644 --- a/codec-http/src/main/java/io/netty/handler/codec/http/multipart/HttpPostStandardRequestDecoder.java +++ b/codec-http/src/main/java/io/netty/handler/codec/http/multipart/HttpPostStandardRequestDecoder.java @@ -706,7 +706,7 @@ public class HttpPostStandardRequestDecoder implements InterfaceHttpPostRequestD private static final class UrlEncodedDetector implements ByteProcessor { @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { return value != '%' && value != '+'; } } diff --git a/codec-http/src/main/java/io/netty/handler/codec/http/websocketx/Utf8Validator.java b/codec-http/src/main/java/io/netty/handler/codec/http/websocketx/Utf8Validator.java index 0928698aff..ccd8768376 100644 --- a/codec-http/src/main/java/io/netty/handler/codec/http/websocketx/Utf8Validator.java +++ b/codec-http/src/main/java/io/netty/handler/codec/http/websocketx/Utf8Validator.java @@ -84,7 +84,7 @@ final class Utf8Validator implements ByteProcessor { } @Override - public boolean process(byte b) { + public boolean process(byte b) throws Exception { byte type = TYPES[b & 0xFF]; codep = state != UTF8_ACCEPT ? b & 0x3f | codep << 6 : 0xff >> type & b; diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Headers.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Headers.java index d35a2052a9..8968258335 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Headers.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/DefaultHttp2Headers.java @@ -38,7 +38,18 @@ public class DefaultHttp2Headers "empty headers are not allowed [%s]", name)); } if (name instanceof AsciiString) { - int index = ((AsciiString) name).forEachByte(HTTP2_NAME_VALIDATOR_PROCESSOR); + final int index; + try { + index = ((AsciiString) name).forEachByte(HTTP2_NAME_VALIDATOR_PROCESSOR); + } catch (Http2Exception e) { + PlatformDependent.throwException(e); + return; + } catch (Throwable t) { + PlatformDependent.throwException(connectionError(PROTOCOL_ERROR, t, + "unexpected error. invalid header name [%s]", name)); + return; + } + if (index != -1) { PlatformDependent.throwException(connectionError(PROTOCOL_ERROR, "invalid header name [%s]", name)); @@ -98,7 +109,7 @@ public class DefaultHttp2Headers @Override public Http2Headers clear() { - firstNonPseudo = head; + this.firstNonPseudo = head; return super.clear(); } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/HpackHuffmanEncoder.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/HpackHuffmanEncoder.java index f7adfe8af6..ddc2b9386b 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/HpackHuffmanEncoder.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/HpackHuffmanEncoder.java @@ -70,9 +70,11 @@ final class HpackHuffmanEncoder { requireNonNull(out, "out"); if (data instanceof AsciiString) { AsciiString string = (AsciiString) data; - encodeProcessor.out = out; try { + encodeProcessor.out = out; string.forEachByte(encodeProcessor); + } catch (Exception e) { + PlatformDependent.throwException(e); } finally { encodeProcessor.end(); } @@ -116,9 +118,14 @@ final class HpackHuffmanEncoder { int getEncodedLength(CharSequence data) { if (data instanceof AsciiString) { AsciiString string = (AsciiString) data; - encodedLengthProcessor.reset(); - string.forEachByte(encodedLengthProcessor); - return encodedLengthProcessor.length(); + try { + encodedLengthProcessor.reset(); + string.forEachByte(encodedLengthProcessor); + return encodedLengthProcessor.length(); + } catch (Exception e) { + PlatformDependent.throwException(e); + return -1; + } } else { return getEncodedLengthSlowPath(data); } diff --git a/codec-http2/src/main/java/io/netty/handler/codec/http2/HttpConversionUtil.java b/codec-http2/src/main/java/io/netty/handler/codec/http2/HttpConversionUtil.java index b8649e923c..7f7d23ff3d 100644 --- a/codec-http2/src/main/java/io/netty/handler/codec/http2/HttpConversionUtil.java +++ b/codec-http2/src/main/java/io/netty/handler/codec/http2/HttpConversionUtil.java @@ -427,17 +427,23 @@ public final class HttpConversionUtil { while (valuesIter.hasNext()) { AsciiString lowerCased = AsciiString.of(valuesIter.next()).toLowerCase(); - int index = lowerCased.forEachByte(FIND_COMMA); - if (index != -1) { - int start = 0; - do { - result.add(lowerCased.subSequence(start, index, false).trim(), EMPTY_STRING); - start = index + 1; - } while (start < lowerCased.length() && - (index = lowerCased.forEachByte(start, lowerCased.length() - start, FIND_COMMA)) != -1); - result.add(lowerCased.subSequence(start, lowerCased.length(), false).trim(), EMPTY_STRING); - } else { - result.add(lowerCased.trim(), EMPTY_STRING); + try { + int index = lowerCased.forEachByte(FIND_COMMA); + if (index != -1) { + int start = 0; + do { + result.add(lowerCased.subSequence(start, index, false).trim(), EMPTY_STRING); + start = index + 1; + } while (start < lowerCased.length() && + (index = lowerCased.forEachByte(start, lowerCased.length() - start, FIND_COMMA)) != -1); + result.add(lowerCased.subSequence(start, lowerCased.length(), false).trim(), EMPTY_STRING); + } else { + result.add(lowerCased.trim(), EMPTY_STRING); + } + } catch (Exception e) { + // This is not expect to happen because FIND_COMMA never throws but must be caught + // because of the ByteProcessor interface. + throw new IllegalStateException(e); } } return result; @@ -483,21 +489,27 @@ public final class HttpConversionUtil { AsciiString value = AsciiString.of(entry.getValue()); // split up cookies to allow for better compression // https://tools.ietf.org/html/rfc7540#section-8.1.2.5 - int index = value.forEachByte(FIND_SEMI_COLON); - if (index != -1) { - int start = 0; - do { - out.add(COOKIE, value.subSequence(start, index, false)); - // skip 2 characters "; " (see https://tools.ietf.org/html/rfc6265#section-4.2.1) - start = index + 2; - } while (start < value.length() && - (index = value.forEachByte(start, value.length() - start, FIND_SEMI_COLON)) != -1); - if (start >= value.length()) { - throw new IllegalArgumentException("cookie value is of unexpected format: " + value); + try { + int index = value.forEachByte(FIND_SEMI_COLON); + if (index != -1) { + int start = 0; + do { + out.add(COOKIE, value.subSequence(start, index, false)); + // skip 2 characters "; " (see https://tools.ietf.org/html/rfc6265#section-4.2.1) + start = index + 2; + } while (start < value.length() && + (index = value.forEachByte(start, value.length() - start, FIND_SEMI_COLON)) != -1); + if (start >= value.length()) { + throw new IllegalArgumentException("cookie value is of unexpected format: " + value); + } + out.add(COOKIE, value.subSequence(start, value.length(), false)); + } else { + out.add(COOKIE, value); } - out.add(COOKIE, value.subSequence(start, value.length(), false)); - } else { - out.add(COOKIE, value); + } catch (Exception e) { + // This is not expect to happen because FIND_SEMI_COLON never throws but must be caught + // because of the ByteProcessor interface. + throw new IllegalStateException(e); } } else { out.add(aName, entry.getValue()); diff --git a/codec-redis/src/main/java/io/netty/handler/codec/redis/RedisDecoder.java b/codec-redis/src/main/java/io/netty/handler/codec/redis/RedisDecoder.java index 231a349888..4eda472dca 100644 --- a/codec-redis/src/main/java/io/netty/handler/codec/redis/RedisDecoder.java +++ b/codec-redis/src/main/java/io/netty/handler/codec/redis/RedisDecoder.java @@ -311,7 +311,7 @@ public final class RedisDecoder extends ByteToMessageDecoder { private long result; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { if (value < '0' || value > '9') { throw new RedisCodecException("bad byte in number: " + value); } diff --git a/codec-stomp/src/main/java/io/netty/handler/codec/stomp/StompSubframeDecoder.java b/codec-stomp/src/main/java/io/netty/handler/codec/stomp/StompSubframeDecoder.java index 107b8589c2..4ecb09b32e 100644 --- a/codec-stomp/src/main/java/io/netty/handler/codec/stomp/StompSubframeDecoder.java +++ b/codec-stomp/src/main/java/io/netty/handler/codec/stomp/StompSubframeDecoder.java @@ -274,7 +274,7 @@ public class StompSubframeDecoder extends ReplayingDecoder { } @Override - public boolean process(byte nextByte) { + public boolean process(byte nextByte) throws Exception { if (nextByte == StompConstants.CR) { ++lineLength; return true; @@ -353,7 +353,7 @@ public class StompSubframeDecoder extends ReplayingDecoder { } @Override - public boolean process(byte nextByte) { + public boolean process(byte nextByte) throws Exception { if (nextByte == StompConstants.COLON) { if (name == null) { AppendableCharSequence charSeq = charSequence(); diff --git a/codec/src/main/java/io/netty/handler/codec/base64/Base64.java b/codec/src/main/java/io/netty/handler/codec/base64/Base64.java index e5d1b1f5ce..e9e6672c65 100644 --- a/codec/src/main/java/io/netty/handler/codec/base64/Base64.java +++ b/codec/src/main/java/io/netty/handler/codec/base64/Base64.java @@ -331,7 +331,7 @@ public final class Base64 { } @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { if (value > 0) { byte sbiDecode = decodabet[value]; if (sbiDecode >= WHITE_SPACE_ENC) { // White space, Equals sign or better diff --git a/common/src/main/java/io/netty/util/AsciiString.java b/common/src/main/java/io/netty/util/AsciiString.java index b423502b0f..a571cf807a 100644 --- a/common/src/main/java/io/netty/util/AsciiString.java +++ b/common/src/main/java/io/netty/util/AsciiString.java @@ -255,7 +255,7 @@ public final class AsciiString implements CharSequence, Comparable * @return {@code -1} if the processor iterated to or beyond the end of the readable bytes. * The last-visited index If the {@link ByteProcessor#process(byte)} returned {@code false}. */ - public int forEachByte(ByteProcessor visitor) { + public int forEachByte(ByteProcessor visitor) throws Exception { return forEachByte0(0, length(), visitor); } @@ -266,7 +266,7 @@ public final class AsciiString implements CharSequence, Comparable * @return {@code -1} if the processor iterated to or beyond the end of the specified area. * The last-visited index If the {@link ByteProcessor#process(byte)} returned {@code false}. */ - public int forEachByte(int index, int length, ByteProcessor visitor) { + public int forEachByte(int index, int length, ByteProcessor visitor) throws Exception { if (isOutOfBounds(index, length, length())) { throw new IndexOutOfBoundsException("expected: " + "0 <= index(" + index + ") <= start + length(" + length + ") <= " + "length(" + length() + ')'); @@ -274,7 +274,7 @@ public final class AsciiString implements CharSequence, Comparable return forEachByte0(index, length, visitor); } - private int forEachByte0(int index, int length, ByteProcessor visitor) { + private int forEachByte0(int index, int length, ByteProcessor visitor) throws Exception { final int len = offset + index + length; for (int i = offset + index; i < len; ++i) { if (!visitor.process(value[i])) { @@ -290,7 +290,7 @@ public final class AsciiString implements CharSequence, Comparable * @return {@code -1} if the processor iterated to or beyond the beginning of the readable bytes. * The last-visited index If the {@link ByteProcessor#process(byte)} returned {@code false}. */ - public int forEachByteDesc(ByteProcessor visitor) { + public int forEachByteDesc(ByteProcessor visitor) throws Exception { return forEachByteDesc0(0, length(), visitor); } @@ -301,7 +301,7 @@ public final class AsciiString implements CharSequence, Comparable * @return {@code -1} if the processor iterated to or beyond the beginning of the specified area. * The last-visited index If the {@link ByteProcessor#process(byte)} returned {@code false}. */ - public int forEachByteDesc(int index, int length, ByteProcessor visitor) { + public int forEachByteDesc(int index, int length, ByteProcessor visitor) throws Exception { if (isOutOfBounds(index, length, length())) { throw new IndexOutOfBoundsException("expected: " + "0 <= index(" + index + ") <= start + length(" + length + ") <= " + "length(" + length() + ')'); @@ -309,7 +309,7 @@ public final class AsciiString implements CharSequence, Comparable return forEachByteDesc0(index, length, visitor); } - private int forEachByteDesc0(int index, int length, ByteProcessor visitor) { + private int forEachByteDesc0(int index, int length, ByteProcessor visitor) throws Exception { final int end = offset + index; for (int i = offset + index + length - 1; i >= end; --i) { if (!visitor.process(value[i])) { diff --git a/common/src/main/java/io/netty/util/ByteProcessor.java b/common/src/main/java/io/netty/util/ByteProcessor.java index 2bc5b6f492..e971983740 100644 --- a/common/src/main/java/io/netty/util/ByteProcessor.java +++ b/common/src/main/java/io/netty/util/ByteProcessor.java @@ -124,5 +124,5 @@ public interface ByteProcessor { * @return {@code true} if the processor wants to continue the loop and handle the next byte in the buffer. * {@code false} if the processor wants to stop handling bytes and abort the loop. */ - boolean process(byte value); + boolean process(byte value) throws Exception; } diff --git a/common/src/test/java/io/netty/util/AsciiStringMemoryTest.java b/common/src/test/java/io/netty/util/AsciiStringMemoryTest.java index abe7e326b7..ca8e10ec0f 100644 --- a/common/src/test/java/io/netty/util/AsciiStringMemoryTest.java +++ b/common/src/test/java/io/netty/util/AsciiStringMemoryTest.java @@ -78,13 +78,13 @@ public class AsciiStringMemoryTest { } @Test - public void forEachTest() { + public void forEachTest() throws Exception { final AtomicReference aCount = new AtomicReference<>(0); final AtomicReference bCount = new AtomicReference<>(0); aAsciiString.forEachByte(new ByteProcessor() { int i; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertEquals("failed at index: " + i, value, bAsciiString.byteAt(i++)); aCount.set(aCount.get() + 1); return true; @@ -93,7 +93,7 @@ public class AsciiStringMemoryTest { bAsciiString.forEachByte(new ByteProcessor() { int i; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertEquals("failed at index: " + i, value, aAsciiString.byteAt(i++)); bCount.set(bCount.get() + 1); return true; @@ -104,25 +104,25 @@ public class AsciiStringMemoryTest { } @Test - public void forEachWithIndexEndTest() { + public void forEachWithIndexEndTest() throws Exception { assertNotEquals(-1, aAsciiString.forEachByte(aAsciiString.length() - 1, 1, new IndexOfProcessor(aAsciiString.byteAt(aAsciiString.length() - 1)))); } @Test - public void forEachWithIndexBeginTest() { + public void forEachWithIndexBeginTest() throws Exception { assertNotEquals(-1, aAsciiString.forEachByte(0, 1, new IndexOfProcessor(aAsciiString.byteAt(0)))); } @Test - public void forEachDescTest() { + public void forEachDescTest() throws Exception { final AtomicReference aCount = new AtomicReference<>(0); final AtomicReference bCount = new AtomicReference<>(0); aAsciiString.forEachByteDesc(new ByteProcessor() { int i = 1; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertEquals("failed at index: " + i, value, bAsciiString.byteAt(bAsciiString.length() - (i++))); aCount.set(aCount.get() + 1); return true; @@ -131,7 +131,7 @@ public class AsciiStringMemoryTest { bAsciiString.forEachByteDesc(new ByteProcessor() { int i = 1; @Override - public boolean process(byte value) { + public boolean process(byte value) throws Exception { assertEquals("failed at index: " + i, value, aAsciiString.byteAt(aAsciiString.length() - (i++))); bCount.set(bCount.get() + 1); return true; @@ -142,13 +142,13 @@ public class AsciiStringMemoryTest { } @Test - public void forEachDescWithIndexEndTest() { + public void forEachDescWithIndexEndTest() throws Exception { assertNotEquals(-1, bAsciiString.forEachByteDesc(bAsciiString.length() - 1, 1, new IndexOfProcessor(bAsciiString.byteAt(bAsciiString.length() - 1)))); } @Test - public void forEachDescWithIndexBeginTest() { + public void forEachDescWithIndexBeginTest() throws Exception { assertNotEquals(-1, bAsciiString.forEachByteDesc(0, 1, new IndexOfProcessor(bAsciiString.byteAt(0)))); }