the upcoming 2.0 version
https://redmine.lighttpd.net/projects/lighttpd2
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
106 lines
2.8 KiB
106 lines
2.8 KiB
|
|
#include <lighttpd/base.h> |
|
|
|
void li_chunk_parser_init(liChunkParserCtx *ctx, liChunkQueue *cq) { |
|
ctx->cq = cq; |
|
li_chunk_parser_reset(ctx); |
|
} |
|
|
|
void li_chunk_parser_reset(liChunkParserCtx *ctx) { |
|
ctx->bytes_in = 0; |
|
ctx->curi.element = NULL; |
|
ctx->start = 0; |
|
ctx->length = 0; |
|
ctx->buf = NULL; |
|
} |
|
|
|
liHandlerResult li_chunk_parser_prepare(liChunkParserCtx *ctx) { |
|
if (NULL == ctx->curi.element) { |
|
ctx->curi = li_chunkqueue_iter(ctx->cq); |
|
if (NULL == ctx->curi.element) return LI_HANDLER_WAIT_FOR_EVENT; |
|
} |
|
return LI_HANDLER_GO_ON; |
|
} |
|
|
|
liHandlerResult li_chunk_parser_next(liVRequest *vr, liChunkParserCtx *ctx, char **p, char **pe) { |
|
off_t l; |
|
liHandlerResult res; |
|
|
|
if (NULL == ctx->curi.element) return LI_HANDLER_WAIT_FOR_EVENT; |
|
|
|
while (ctx->start >= (l = li_chunkiter_length(ctx->curi))) { |
|
liChunkIter i = ctx->curi; |
|
/* Wait at the end of the last chunk if it gets extended */ |
|
if (!li_chunkiter_next(&i)) return LI_HANDLER_WAIT_FOR_EVENT; |
|
ctx->curi = i; |
|
ctx->start -= l; |
|
} |
|
|
|
if (NULL == ctx->curi.element) return LI_HANDLER_WAIT_FOR_EVENT; |
|
|
|
if (LI_HANDLER_GO_ON != (res = li_chunkiter_read(vr, ctx->curi, ctx->start, l - ctx->start, &ctx->buf, &ctx->length))) { |
|
return res; |
|
} |
|
|
|
*p = ctx->buf; |
|
*pe = ctx->buf + ctx->length; |
|
return LI_HANDLER_GO_ON; |
|
} |
|
|
|
void li_chunk_parser_done(liChunkParserCtx *ctx, goffset len) { |
|
ctx->bytes_in += len; |
|
ctx->start += len; |
|
} |
|
|
|
gboolean li_chunk_extract_to(liVRequest *vr, liChunkParserMark from, liChunkParserMark to, GString *dest) { |
|
liChunkParserMark i; |
|
|
|
g_string_set_size(dest, to.abs_pos - from.abs_pos); |
|
li_g_string_clear(dest); |
|
|
|
for ( i = from; i.ci.element != to.ci.element; li_chunkiter_next(&i.ci) ) { |
|
goffset len = li_chunkiter_length(i.ci); |
|
while (i.pos < len) { |
|
char *buf; |
|
off_t we_have; |
|
if (LI_HANDLER_GO_ON != li_chunkiter_read(vr, i.ci, i.pos, len - i.pos, &buf, &we_have)) goto error; |
|
if (dest->len + we_have < dest->allocated_len) { |
|
/* "fast" append */ |
|
memcpy(dest->str + dest->len, buf, we_have); |
|
dest->len += we_have; |
|
dest->str[dest->len] = '\0'; |
|
} else { |
|
g_string_append_len(dest, buf, we_have); |
|
} |
|
i.pos += we_have; |
|
} |
|
i.pos = 0; |
|
} |
|
while (i.pos < to.pos) { |
|
char *buf; |
|
off_t we_have; |
|
if (LI_HANDLER_GO_ON != li_chunkiter_read(vr, i.ci, i.pos, to.pos - i.pos, &buf, &we_have)) goto error; |
|
if (dest->len + we_have < dest->allocated_len) { |
|
/* "fast" append */ |
|
memcpy(dest->str + dest->len, buf, we_have); |
|
dest->len += we_have; |
|
dest->str[dest->len] = '\0'; |
|
} else { |
|
g_string_append_len(dest, buf, we_have); |
|
} |
|
i.pos += we_have; |
|
} |
|
|
|
return TRUE; |
|
|
|
error: |
|
li_g_string_clear(dest); |
|
return FALSE; |
|
} |
|
|
|
GString* li_chunk_extract(liVRequest *vr, liChunkParserMark from, liChunkParserMark to) { |
|
GString *str = g_string_sized_new(0); |
|
if (li_chunk_extract_to(vr, from, to, str)) return str; |
|
g_string_free(str, TRUE); |
|
return NULL; |
|
}
|
|
|