All Downloads are FREE. Search and download functionalities are using the official Maven repository.
Please wait. This can take some minutes ...
Many resources are needed to download a project. Please understand that we have to compensate our server costs. Thank you in advance.
Project price only 1 $
You can buy this project and download/modify it how often you want.
swim.http.HttpChunkedDecoder Maven / Gradle / Ivy
// Copyright 2015-2019 SWIM.AI 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 swim.http;
import swim.codec.Decoder;
import swim.codec.DecoderException;
import swim.codec.InputBuffer;
import swim.codec.Parser;
import swim.codec.Utf8;
import swim.http.header.ContentType;
final class HttpChunkedDecoder extends Decoder> {
final HttpParser http;
final HttpMessage> message;
final Decoder content;
final HttpChunkHeader header;
final Parser> part;
final int offset;
final int step;
HttpChunkedDecoder(HttpParser http, HttpMessage> message, Decoder content,
HttpChunkHeader header, Parser> part, int offset, int step) {
this.http = http;
this.message = message;
this.content = content;
this.header = header;
this.part = part;
this.offset = offset;
this.step = step;
}
HttpChunkedDecoder(HttpParser http, HttpMessage> message, Decoder content) {
this(http, message, content, null, null, 0, 1);
}
@Override
public Decoder> feed(InputBuffer input) {
return decode(input, this.http, this.message, this.content, this.header,
this.part, this.offset, this.step);
}
static Decoder> decode(InputBuffer input, HttpParser http,
HttpMessage> message, Decoder content,
HttpChunkHeader header, Parser> part,
int offset, int step) {
do {
if (step == 1) { // chunk header
if (part == null) {
part = Utf8.parseDecoded(http.chunkHeaderParser(), input);
} else {
part = part.feed(input);
}
if (part.isDone()) {
header = (HttpChunkHeader) part.bind();
part = null;
step = 2;
} else if (part.isError()) {
return error(part.trap());
}
}
if (step == 2) { // chunk data
final int inputStart = input.index();
final int inputLimit = input.limit();
final int inputRemaining = inputLimit - inputStart;
final long chunkSize = header.size();
final long chunkRemaining = chunkSize - offset;
final boolean inputPart = input.isPart();
if (chunkRemaining < inputRemaining) {
input = input.limit(inputStart + (int) chunkRemaining).isPart(chunkSize != 0);
content = content.feed(input);
input = input.limit(inputLimit);
} else {
input = input.isPart(chunkSize != 0);
content = content.feed(input);
}
input = input.isPart(inputPart);
offset += input.index() - inputStart;
if (offset >= chunkSize) {
offset = 0;
if (chunkSize > 0) {
step = 3;
} else {
step = 5;
break;
}
} else if (content.isError()) {
return content.asError();
}
}
if (step == 3) {
if (input.isCont()) {
if (input.head() == '\r') {
input = input.step();
step = 4;
} else {
return error(new DecoderException("carriage return"));
}
}
}
if (step == 4) {
if (input.isCont()) {
if (input.head() == '\n') {
input = input.step();
step = 1;
continue;
} else {
return error(new DecoderException("line feed"));
}
}
}
break;
} while (true);
if (step == 5) { // chunk trailer
if (part == null) {
part = Utf8.parseDecoded(http.chunkTrailerParser(), input);
} else {
part = part.feed(input);
}
if (part.isDone()) {
final HttpChunkTrailer trailer = (HttpChunkTrailer) part.bind();
message = message.appendedHeaders(trailer.headers());
final MediaType mediaType;
final ContentType contentType = message.getHeader(ContentType.class);
if (contentType != null) {
mediaType = contentType.mediaType();
} else {
mediaType = null;
}
final HttpValue entity = HttpValue.from(content.bind(), mediaType);
return done(message.entity(entity));
} else if (part.isError()) {
return error(part.trap());
}
}
return new HttpChunkedDecoder(http, message, content, header, part, offset, step);
}
static Decoder> decode(InputBuffer input, HttpParser http,
HttpMessage> message, Decoder content) {
return decode(input, http, message, content, null, null, 0, 1);
}
}