This repository has been archived on 2025-09-14. You can view files and clone it, but cannot push or open issues or pull requests.
Files
stellar-stellar/decoders/http/http_decoder_string.c

289 lines
6.6 KiB
C

#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <assert.h>
#include "http_decoder_private.h"
static const char *string_state_to_desc(enum string_state state)
{
switch (state)
{
case STRING_STATE_INIT:
return "init";
break;
case STRING_STATE_REFER:
return "refer";
break;
case STRING_STATE_CACHE:
return "cache";
break;
case STRING_STATE_COMMIT:
return "commit";
break;
default:
return "unknown";
break;
}
}
void http_decoder_string_refer(struct http_decoder_string *rstr, const char *at, size_t length)
{
if (NULL == rstr)
{
return;
}
switch (rstr->state)
{
case STRING_STATE_INIT:
case STRING_STATE_CACHE:
rstr->refer.iov_base = (char *)at;
rstr->refer.iov_len = length;
break;
default:
abort();
break;
}
rstr->state = STRING_STATE_REFER;
}
static void string_refer2cache(struct http_decoder_string *rstr)
{
if (0 == rstr->refer.iov_len)
{
return;
}
if (rstr->cache.iov_len >= rstr->max_cache_size)
{
return;
}
size_t length = rstr->cache.iov_len + rstr->refer.iov_len;
if (length > rstr->max_cache_size)
{
length = rstr->max_cache_size;
}
if (NULL == rstr->cache.iov_base)
{
rstr->cache.iov_base = CALLOC(char, length + 1);
memcpy(rstr->cache.iov_base, rstr->refer.iov_base, length);
}
else
{
rstr->cache.iov_base = REALLOC(char, rstr->cache.iov_base, length + 1);
memcpy((char *)rstr->cache.iov_base + rstr->cache.iov_len, rstr->refer.iov_base,
(length - rstr->cache.iov_len));
}
rstr->cache.iov_len = length;
rstr->refer.iov_base = NULL;
rstr->refer.iov_len = 0;
}
static void string_commit2cache(struct http_decoder_string *rstr)
{
if (rstr->cache.iov_len == rstr->commit.iov_len &&
rstr->cache.iov_base == rstr->commit.iov_base)
{
rstr->commit.iov_base = NULL;
rstr->commit.iov_len = 0;
return;
}
// Only http header key need to backward to cache
size_t length = 0;
if (rstr->commit.iov_len > rstr->max_cache_size)
{
length = rstr->max_cache_size;
}
else
{
length = rstr->commit.iov_len;
}
if (length > 0)
{
if (NULL == rstr->cache.iov_base)
{
rstr->cache.iov_base = CALLOC(char, length + 1);
}
else
{
abort();
}
memcpy(rstr->cache.iov_base, rstr->commit.iov_base, length);
rstr->cache.iov_len = length;
rstr->commit.iov_base = NULL;
rstr->commit.iov_len = 0;
}
}
void http_decoder_string_cache(struct http_decoder_string *rstr)
{
if (NULL == rstr)
{
return;
}
switch (rstr->state)
{
case STRING_STATE_REFER:
string_refer2cache(rstr);
break;
case STRING_STATE_CACHE:
break;
case STRING_STATE_COMMIT:
// commit backward to cache
string_commit2cache(rstr);
break;
default:
abort();
break;
}
rstr->state = STRING_STATE_CACHE;
}
void http_decoder_string_commit(struct http_decoder_string *rstr)
{
if (NULL == rstr)
{
return;
}
switch (rstr->state)
{
case STRING_STATE_REFER:
if (rstr->cache.iov_len)
{
http_decoder_string_cache(rstr);
rstr->commit.iov_base = rstr->cache.iov_base;
rstr->commit.iov_len = rstr->cache.iov_len;
// not overwrite rstr->cache.iov_base
}
else
{
rstr->commit.iov_base = rstr->refer.iov_base;
rstr->commit.iov_len = rstr->refer.iov_len;
rstr->refer.iov_base = NULL;
rstr->refer.iov_len = 0;
}
break;
case STRING_STATE_CACHE:
rstr->commit.iov_base = rstr->cache.iov_base;
rstr->commit.iov_len = rstr->cache.iov_len;
// not overwrite rstr->cache.iov_base
break;
default:
// abort();
break;
}
rstr->state = STRING_STATE_COMMIT;
}
void http_decoder_string_reset(struct http_decoder_string *rstr)
{
assert(rstr);
switch (rstr->state)
{
case STRING_STATE_INIT:
case STRING_STATE_REFER:
case STRING_STATE_CACHE:
case STRING_STATE_COMMIT:
FREE(rstr->cache.iov_base);
memset(rstr, 0, sizeof(struct http_decoder_string));
break;
default:
abort();
break;
}
rstr->state = STRING_STATE_INIT;
}
void http_decoder_string_init(struct http_decoder_string *rstr, size_t max_cache_size)
{
rstr->max_cache_size = max_cache_size;
}
void http_decoder_string_reinit(struct http_decoder_string *rstr)
{
if (rstr->state == STRING_STATE_CACHE)
{
return;
}
if (rstr->state == STRING_STATE_COMMIT &&
rstr->cache.iov_base == rstr->commit.iov_base &&
rstr->cache.iov_len == rstr->commit.iov_len)
{
return;
}
if (rstr->cache.iov_base != NULL)
{
FREE(rstr->cache.iov_base);
rstr->cache.iov_len = 0;
}
#if 0
rstr->refer.iov_base = NULL;
rstr->refer.iov_len = 0;
rstr->commit.iov_base = NULL;
rstr->commit.iov_len = 0;
rstr->state = STRING_STATE_INIT;
#endif
}
enum string_state http_decoder_string_state(const struct http_decoder_string *rstr)
{
return rstr->state;
}
int http_decoder_string_get(const struct http_decoder_string *rstr, char **name, size_t *name_len)
{
if (NULL == rstr || NULL == name || 0 == name_len)
{
return -1;
}
if (http_decoder_string_state(rstr) == STRING_STATE_COMMIT)
{
*name = rstr->commit.iov_base;
*name_len = rstr->commit.iov_len;
}
else
{
*name = NULL;
*name_len = 0;
}
return 0;
}
void http_decoder_string_dump(struct http_decoder_string *rstr, const char *desc)
{
if (NULL == rstr)
{
return;
}
char *refer_str = http_safe_dup((char *)rstr->refer.iov_base, rstr->refer.iov_len);
char *cache_str = http_safe_dup((char *)rstr->cache.iov_base, rstr->cache.iov_len);
char *commit_str = http_safe_dup((char *)rstr->commit.iov_base, rstr->commit.iov_len);
printf("%s: state: %s, refer: {len: %02zu, iov_base: %s}, cache: {len: %02zu, iov_base: %s}, commit: {len: %02zu, iov_base: %s}\n",
desc, string_state_to_desc(rstr->state),
rstr->refer.iov_len, refer_str,
rstr->cache.iov_len, cache_str,
rstr->commit.iov_len, commit_str);
FREE(refer_str);
FREE(cache_str);
FREE(commit_str);
}