HomePort
All Data Structures Files Functions Variables Typedefs Enumerations Enumerator Macros Pages
httpd_header_parser.c
Go to the documentation of this file.
1 /*
2  * Copyright 2011 Aalborg University. All rights reserved.
3  *
4  * Redistribution and use in source and binary forms, with or without modification, are
5  * permitted provided that the following conditions are met:
6  *
7  * 1. Redistributions of source code must retain the above copyright notice, this list of
8  * conditions and the following disclaimer.
9  *
10  * 2. Redistributions in binary form must reproduce the above copyright notice, this list
11  * of conditions and the following disclaimer in the documentation and/or other materials
12  * provided with the distribution.
13  *
14  * THIS SOFTWARE IS PROVIDED BY Aalborg University ''AS IS'' AND ANY EXPRESS OR IMPLIED
15  * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND
16  * FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL Aalborg University OR
17  * CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR
18  * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
19  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
20  * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING
21  * NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF
22  * ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
23  *
24  * The views and conclusions contained in the software and documentation are those of the
25  * authors and should not be interpreted as representing official policies, either expressed
26  */
27 
28 #include <string.h>
29 #include <stdlib.h>
30 #include <stdio.h>
31 
32 #include "httpd_header_parser.h"
33 #include "hpd/hpd_shared_api.h"
34 
35 enum hp_state {
36  S_ERROR = -1,
40 };
41 
42 struct hp {
45 
47 
48  char* field_buffer;
50 
51  char* value_buffer;
53 };
54 
55 static void hp_reset_buffers(struct hp *instance)
56 {
57  if(instance->field_buffer) free(instance->field_buffer);
58  if(instance->value_buffer) free(instance->value_buffer);
59 
60  instance -> field_buffer_size = 0;
61  instance -> field_buffer = NULL;
62 
63  instance -> value_buffer_size = 0;
64  instance -> value_buffer = NULL;
65 }
66 
67 hpd_error_t hp_create(struct hp **instance, struct hp_settings *settings, const hpd_module_t *context)
68 {
69  if (!context) return HPD_E_NULL;
70  if (!instance || !settings) HPD_LOG_RETURN_E_NULL(context);
71 
72  (*instance) = malloc(sizeof(struct hp));
73  if (!(*instance)) HPD_LOG_RETURN_E_ALLOC(context);
74 
75  (*instance)->context = context;
76 
77  memcpy(&(*instance)->settings, settings, sizeof(struct hp_settings));
78 
79  (*instance)->state = S_FIELD;
80 
81  (*instance)->field_buffer_size = 0;
82  (*instance)->field_buffer = NULL;
83 
84  (*instance)->value_buffer_size = 0;
85  (*instance)->value_buffer = NULL;
86 
87  return HPD_E_SUCCESS;
88 }
89 
90 hpd_error_t hp_on_header_field(struct hp *instance, const char *field_chunk, size_t length)
91 {
92  if (!instance)
93  return HPD_E_NULL;
94  if (!field_chunk)
95  HPD_LOG_RETURN_E_NULL(instance->context);
96 
97  hpd_error_t rc;
98  size_t old_buffer_size = instance->field_buffer_size;
99 
100  switch (instance->state) {
101  case S_VALUE: {
103  if (on_field_value_pair && (rc = on_field_value_pair(instance->settings.data,
104  instance->field_buffer, instance->field_buffer_size,
105  instance->value_buffer, instance->value_buffer_size))) {
106  instance->state = S_ERROR;
107  return rc;
108  }
109  hp_reset_buffers(instance);
110  old_buffer_size = 0;
111  instance->state = S_FIELD;
112  }
113  case S_FIELD: {
114  size_t new_len = instance->field_buffer_size + length;
115  char *new_buf = realloc(instance->field_buffer, new_len * (sizeof(char)));
116  if (!new_buf) HPD_LOG_RETURN_E_ALLOC(instance->context);
117  instance->field_buffer = new_buf;
118  instance->field_buffer_size = new_len;
119  memcpy(instance->field_buffer+old_buffer_size, field_chunk, length);
120  return HPD_E_SUCCESS;
121  }
122  case S_COMPLETED:
123  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Received additional data after hp_on_header_complete().");
124  case S_ERROR:
125  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Cannot receive data: In an error state.");
126  default:
127  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Unexpected state.");
128  }
129 }
130 
131 hpd_error_t hp_on_header_value(struct hp *instance, const char *value_chunk, size_t length)
132 {
133  if (!instance || !value_chunk) return HPD_E_NULL;
134 
135  size_t old_buffer_size = instance->value_buffer_size;
136 
137  switch (instance->state) {
138  case S_FIELD:
139  instance->state = S_VALUE;
140  case S_VALUE: {
141  size_t new_len = instance->value_buffer_size + length;
142  char *new_buf = realloc(instance->value_buffer, new_len * (sizeof(char)));
143  if (!new_buf) HPD_LOG_RETURN_E_ALLOC(instance->context);
144  instance->value_buffer = new_buf;
145  instance->value_buffer_size = new_len;
146  memcpy(instance->value_buffer+old_buffer_size, value_chunk, length);
147  return HPD_E_SUCCESS;
148  }
149  case S_COMPLETED:
150  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Received additional data after hp_on_header_complete().");
151  case S_ERROR:
152  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Cannot receive data: In an error state.");
153  default:
154  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Unexpected state.");
155  }
156 }
157 
159 {
160  if (!instance) return HPD_E_NULL;
161 
162  hpd_error_t rc;
163 
164  switch (instance->state) {
165  case S_FIELD:
166  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Cannot complete headers: Missing value to last key.");
167  case S_VALUE:
168  if ((rc = instance->settings.on_field_value_pair(instance->settings.data, instance->field_buffer, instance->field_buffer_size, instance->value_buffer, instance->value_buffer_size))) {
169  instance->state = S_ERROR;
170  return rc;
171  }
172  instance->state = S_COMPLETED;
173  return HPD_E_SUCCESS;
174  case S_COMPLETED:
175  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Headers are already completed.");
176  case S_ERROR:
177  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Cannot complete headers: In an error state.");
178  default:
179  HPD_LOG_RETURN(instance->context, HPD_E_STATE, "Unexpected state.");
180  }
181 }
182 
184 {
185  if (!instance) return HPD_E_NULL;
186 
187  hp_reset_buffers(instance);
188  free(instance);
189 
190  return HPD_E_SUCCESS;
191 }
hpd_error_t hp_on_header_field(struct hp *instance, const char *field_chunk, size_t length)
size_t value_buffer_size
#define HPD_LOG_RETURN(CONTEXT, E, FMT,...)
hpd_error_t hp_destroy(struct hp *instance)
struct up * instance
free(data.url)
enum hp_state state
struct hp_settings settings
hp_string_cb on_field_value_pair
char * value_buffer
enum hpd_error hpd_error_t
Definition: hpd_types.h:167
#define HPD_LOG_RETURN_E_ALLOC(CONTEXT)
#define HPD_LOG_RETURN_E_NULL(CONTEXT)
hpd_error_t hp_create(struct hp **instance, struct hp_settings *settings, const hpd_module_t *context)
settings on_field_value_pair
char * field_buffer
size_t field_buffer_size
struct hp_settings settings
const hpd_module_t * context
hpd_error_t(* hp_string_cb)(void *data, const char *field, size_t field_length, const char *value, size_t value_length)
hpd_error_t hp_on_header_complete(struct hp *instance)
static void hp_reset_buffers(struct hp *instance)
hpd_error_t hp_on_header_value(struct hp *instance, const char *value_chunk, size_t length)