summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java212
1 files changed, 121 insertions, 91 deletions
diff --git a/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java b/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java
index 3fd9538fbc..5291870390 100644
--- a/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java
+++ b/src/com/vaadin/terminal/gwt/server/AbstractCommunicationManager.java
@@ -417,8 +417,6 @@ public abstract class AbstractCommunicationManager implements
contentLength -= (boundary.length() + 2); // 2 == CRLF
- final char[] charArray = boundary.toCharArray();
-
/*
* Reads bytes from the underlying stream. Compares the read bytes to
* the boundary string and returns -1 if met.
@@ -432,95 +430,8 @@ public abstract class AbstractCommunicationManager implements
* Note, if this is someday needed elsewhere, don't shoot yourself to
* foot and split to a top level helper class.
*/
- InputStream simpleMultiPartReader = new InputStream() {
-
- /**
- * Counter of how many characters have been matched to boundary
- * string from the stream
- */
- int matchedCount = 0;
-
- /**
- * Used as pointer when returning bytes after partly matched
- * boundary string.
- */
- int curBoundaryIndex = 0;
- /**
- * The byte found after a "promising start for boundary"
- */
- private int bufferedByte = -1;
- private boolean atTheEnd = false;
-
- @Override
- public int read() throws IOException {
- if (atTheEnd) {
- return -1;
- } else if (bufferedByte >= 0) {
- /* "buffered mode", purge partially matched boundary */
- return getBuffered();
- } else {
- int fromActualStream = inputStream.read();
- if (fromActualStream == -1) {
- // unexpected end of stream
- throw new IOException(
- "The multipart stream ended unexpectedly");
- }
- if (charArray[matchedCount] == fromActualStream) {
- /*
- * Going to "buffered mode". Read until full boundary
- * match or a different character.
- */
- while (true) {
- matchedCount++;
- if (matchedCount == charArray.length) {
- // reached the end of file
- atTheEnd = true;
- return -1;
- }
- fromActualStream = inputStream.read();
- if (fromActualStream != charArray[matchedCount]) {
- // Did not find full boundary, cache the last
- // byte
- bufferedByte = fromActualStream;
- return getBuffered();
- }
- }
- }
- return fromActualStream;
- }
- }
-
- private int getBuffered() throws IOException {
- int b;
- if (matchedCount == 0) {
- b = bufferedByte;
- bufferedByte = -1;
- } else {
- b = charArray[curBoundaryIndex++];
- if (curBoundaryIndex == matchedCount) {
- curBoundaryIndex = 0;
- if (bufferedByte != charArray[0]) {
- // next call for getBuffered will return the
- // bufferedByte, not from the char array.
- matchedCount = 0;
- } else {
- /*
- * Special case where buffered byte again matches
- * the boundaryString. Step over one byte matching
- * the boundary.
- */
- matchedCount = 1;
- bufferedByte = -1;
- }
- }
- }
- if (b == -1) {
- throw new IOException(
- "The multipart stream ended unexpectedly");
- }
- return b;
- }
- };
+ InputStream simpleMultiPartReader = new SimpleMultiPartInputStream(
+ inputStream, boundary);
/*
* Should report only the filename even if the browser sends the path
@@ -2210,4 +2121,123 @@ public abstract class AbstractCommunicationManager implements
abstract protected void cleanStreamVariable(VariableOwner owner, String name);
+ private static class SimpleMultiPartInputStream extends InputStream {
+
+ /**
+ * Counter of how many characters have been matched to boundary string
+ * from the stream
+ */
+ int matchedCount = 0;
+
+ /**
+ * Used as pointer when returning bytes after partly matched boundary
+ * string.
+ */
+ int curBoundaryIndex = 0;
+ /**
+ * The byte found after a "promising start for boundary"
+ */
+ private int bufferedByte = -1;
+ private boolean atTheEnd = false;
+
+ private final char[] boundary;
+
+ private final InputStream realInputStream;
+
+ public SimpleMultiPartInputStream(InputStream realInputStream,
+ String boundaryString) {
+ boundary = boundaryString.toCharArray();
+ this.realInputStream = realInputStream;
+ }
+
+ @Override
+ public int read() throws IOException {
+ if (atTheEnd) {
+ // End boundary reached, nothing more to read
+ return -1;
+ } else if (bufferedByte >= 0) {
+ /* Purge partially matched boundary if there was such */
+ return getBuffered();
+ } else {
+ int fromActualStream = realInputStream.read();
+ if (fromActualStream == -1) {
+ // unexpected end of stream
+ throw new IOException(
+ "The multipart stream ended unexpectedly");
+ }
+ if (boundary[matchedCount] == fromActualStream) {
+ /*
+ * Going to "buffered mode". Read until full boundary match
+ * or a different character.
+ */
+ while (true) {
+ matchedCount++;
+ if (matchedCount == boundary.length) {
+ /*
+ * The whole boundary matched so we have reached the
+ * end of file
+ */
+ atTheEnd = true;
+ return -1;
+ }
+ fromActualStream = realInputStream.read();
+ if (fromActualStream != boundary[matchedCount]) {
+ /*
+ * Did not find full boundary, cache the mismatching
+ * byte and start returning the partially matched
+ * boundary.
+ */
+ bufferedByte = fromActualStream;
+ return getBuffered();
+ }
+ }
+ }
+ return fromActualStream;
+ }
+ }
+
+ /**
+ * Returns the partly matched boundary string and the byte following
+ * that.
+ *
+ * @return
+ * @throws IOException
+ */
+ private int getBuffered() throws IOException {
+ int b;
+ if (matchedCount == 0) {
+ // The boundary has been returned, return the buffered byte.
+ b = bufferedByte;
+ bufferedByte = -1;
+ } else {
+ b = boundary[curBoundaryIndex++];
+ if (curBoundaryIndex == matchedCount) {
+ // The full boundary has been returned, remaining is the
+ // char that did not match the boundary.
+
+ curBoundaryIndex = 0;
+ if (bufferedByte != boundary[0]) {
+ /*
+ * next call for getBuffered will return the
+ * bufferedByte that came after the partial boundary
+ * match
+ */
+ matchedCount = 0;
+ } else {
+ /*
+ * Special case where buffered byte again matches the
+ * boundaryString. This could be the start of the real
+ * end boundary.
+ */
+ matchedCount = 1;
+ bufferedByte = -1;
+ }
+ }
+ }
+ if (b == -1) {
+ throw new IOException("The multipart stream ended unexpectedly");
+ }
+ return b;
+ }
+ }
}