mongrel_esi 0.5.2 → 0.5.4
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/ext/esi/esi_parser.c +1 -0
- data/ext/esi/extconf.rb +2 -0
- data/ext/esi/parser.c +2 -2
- data/ext/esi/parser.h +2 -0
- data/ext/esi/parser.rl +2 -2
- data/ext/esi/test/common.rl +46 -0
- data/ext/esi/test/parser.c +1901 -0
- data/ext/esi/test/parser.h +127 -0
- data/ext/esi/test/parser.rl +611 -0
- data/ext/esi/test/test.c +2 -2
- data/lib/esi/cache.rb +1 -1
- data/lib/esi/version.rb +1 -1
- data/test/benchmarks/perf-trunk-n1000-c1.csv +100 -100
- data/test/benchmarks/perf-trunk-n1000-c2.csv +100 -100
- data/test/benchmarks/perf-trunk-n1000-c3.csv +100 -100
- data/test/benchmarks/perf-trunk-n1000-c4.csv +100 -100
- data/test/benchmarks/perf-trunk-n1000-c5.csv +100 -100
- data/test/integration/docs/large-no-cache.html +688 -0
- data/test/integration/handler_test.rb +7 -2
- data/test/load_test_ab.rb +1 -1
- metadata +7 -222
- data/doc/rdoc/classes/ESI/CParser.html +0 -122
- data/doc/rdoc/classes/ESI/Cache.html +0 -178
- data/doc/rdoc/classes/ESI/Cache.src/M000087.html +0 -17
- data/doc/rdoc/classes/ESI/Cache.src/M000088.html +0 -20
- data/doc/rdoc/classes/ESI/Config.html +0 -314
- data/doc/rdoc/classes/ESI/Config.src/M000049.html +0 -18
- data/doc/rdoc/classes/ESI/Config.src/M000050.html +0 -18
- data/doc/rdoc/classes/ESI/Config.src/M000051.html +0 -35
- data/doc/rdoc/classes/ESI/Config.src/M000052.html +0 -40
- data/doc/rdoc/classes/ESI/Config.src/M000053.html +0 -23
- data/doc/rdoc/classes/ESI/Config.src/M000054.html +0 -18
- data/doc/rdoc/classes/ESI/Config.src/M000055.html +0 -20
- data/doc/rdoc/classes/ESI/Config.src/M000056.html +0 -24
- data/doc/rdoc/classes/ESI/Config/CacheConfig.html +0 -212
- data/doc/rdoc/classes/ESI/Config/CacheConfig.src/M000060.html +0 -19
- data/doc/rdoc/classes/ESI/Config/CacheConfig.src/M000061.html +0 -19
- data/doc/rdoc/classes/ESI/Config/CacheConfig.src/M000062.html +0 -18
- data/doc/rdoc/classes/ESI/Config/CacheConfig.src/M000063.html +0 -18
- data/doc/rdoc/classes/ESI/Config/CacheConfig.src/M000064.html +0 -18
- data/doc/rdoc/classes/ESI/Config/ConfigRouter.html +0 -187
- data/doc/rdoc/classes/ESI/Config/ConfigRouter.src/M000057.html +0 -19
- data/doc/rdoc/classes/ESI/Config/ConfigRouter.src/M000058.html +0 -21
- data/doc/rdoc/classes/ESI/Config/ConfigRouter.src/M000059.html +0 -21
- data/doc/rdoc/classes/ESI/Dispatcher.html +0 -174
- data/doc/rdoc/classes/ESI/Dispatcher.src/M000092.html +0 -19
- data/doc/rdoc/classes/ESI/Dispatcher.src/M000093.html +0 -27
- data/doc/rdoc/classes/ESI/Fragment.html +0 -218
- data/doc/rdoc/classes/ESI/Fragment.src/M000111.html +0 -20
- data/doc/rdoc/classes/ESI/Fragment.src/M000112.html +0 -18
- data/doc/rdoc/classes/ESI/Fragment.src/M000113.html +0 -18
- data/doc/rdoc/classes/ESI/Invalidator.html +0 -131
- data/doc/rdoc/classes/ESI/Invalidator.src/M000004.html +0 -41
- data/doc/rdoc/classes/ESI/Log.html +0 -221
- data/doc/rdoc/classes/ESI/Log.src/M000030.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000031.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000032.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000033.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000034.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000035.html +0 -18
- data/doc/rdoc/classes/ESI/Log.src/M000036.html +0 -18
- data/doc/rdoc/classes/ESI/MemcachedCache.html +0 -314
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000037.html +0 -24
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000038.html +0 -22
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000039.html +0 -19
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000040.html +0 -23
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000041.html +0 -18
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000042.html +0 -19
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000043.html +0 -18
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000044.html +0 -18
- data/doc/rdoc/classes/ESI/MemcachedCache.src/M000045.html +0 -17
- data/doc/rdoc/classes/ESI/OutputAdapter.html +0 -204
- data/doc/rdoc/classes/ESI/OutputAdapter.src/M000094.html +0 -37
- data/doc/rdoc/classes/ESI/OutputAdapter.src/M000095.html +0 -18
- data/doc/rdoc/classes/ESI/OutputAdapter.src/M000096.html +0 -18
- data/doc/rdoc/classes/ESI/OutputAdapter.src/M000097.html +0 -18
- data/doc/rdoc/classes/ESI/Parser.html +0 -184
- data/doc/rdoc/classes/ESI/Parser.src/M000065.html +0 -51
- data/doc/rdoc/classes/ESI/Parser.src/M000066.html +0 -39
- data/doc/rdoc/classes/ESI/Parser.src/M000067.html +0 -19
- data/doc/rdoc/classes/ESI/Processor.html +0 -194
- data/doc/rdoc/classes/ESI/Processor.src/M000046.html +0 -26
- data/doc/rdoc/classes/ESI/Processor.src/M000047.html +0 -44
- data/doc/rdoc/classes/ESI/Processor.src/M000048.html +0 -26
- data/doc/rdoc/classes/ESI/Proxy.html +0 -304
- data/doc/rdoc/classes/ESI/Proxy.src/M000071.html +0 -20
- data/doc/rdoc/classes/ESI/Proxy.src/M000072.html +0 -63
- data/doc/rdoc/classes/ESI/Proxy.src/M000073.html +0 -62
- data/doc/rdoc/classes/ESI/Proxy.src/M000074.html +0 -24
- data/doc/rdoc/classes/ESI/Proxy.src/M000075.html +0 -19
- data/doc/rdoc/classes/ESI/Proxy.src/M000076.html +0 -22
- data/doc/rdoc/classes/ESI/Proxy.src/M000077.html +0 -18
- data/doc/rdoc/classes/ESI/Proxy.src/M000078.html +0 -28
- data/doc/rdoc/classes/ESI/Proxy.src/M000079.html +0 -29
- data/doc/rdoc/classes/ESI/RackAdapter.html +0 -214
- data/doc/rdoc/classes/ESI/RackAdapter.src/M000068.html +0 -18
- data/doc/rdoc/classes/ESI/RackAdapter.src/M000069.html +0 -34
- data/doc/rdoc/classes/ESI/RackAdapter.src/M000070.html +0 -17
- data/doc/rdoc/classes/ESI/RackConfig.html +0 -173
- data/doc/rdoc/classes/ESI/RackConfig.src/M000089.html +0 -28
- data/doc/rdoc/classes/ESI/RackConfig.src/M000090.html +0 -18
- data/doc/rdoc/classes/ESI/RackConfig.src/M000091.html +0 -20
- data/doc/rdoc/classes/ESI/Request.html +0 -158
- data/doc/rdoc/classes/ESI/Request.src/M000098.html +0 -18
- data/doc/rdoc/classes/ESI/Request.src/M000099.html +0 -18
- data/doc/rdoc/classes/ESI/Response.html +0 -250
- data/doc/rdoc/classes/ESI/Response.src/M000080.html +0 -23
- data/doc/rdoc/classes/ESI/Response.src/M000081.html +0 -18
- data/doc/rdoc/classes/ESI/Response.src/M000082.html +0 -21
- data/doc/rdoc/classes/ESI/Response.src/M000083.html +0 -20
- data/doc/rdoc/classes/ESI/Response.src/M000084.html +0 -25
- data/doc/rdoc/classes/ESI/Response.src/M000085.html +0 -18
- data/doc/rdoc/classes/ESI/Response.src/M000086.html +0 -33
- data/doc/rdoc/classes/ESI/Router.html +0 -212
- data/doc/rdoc/classes/ESI/Router.src/M000109.html +0 -36
- data/doc/rdoc/classes/ESI/Router.src/M000110.html +0 -25
- data/doc/rdoc/classes/ESI/RubyCache.html +0 -278
- data/doc/rdoc/classes/ESI/RubyCache.src/M000100.html +0 -20
- data/doc/rdoc/classes/ESI/RubyCache.src/M000101.html +0 -20
- data/doc/rdoc/classes/ESI/RubyCache.src/M000102.html +0 -20
- data/doc/rdoc/classes/ESI/RubyCache.src/M000103.html +0 -22
- data/doc/rdoc/classes/ESI/RubyCache.src/M000104.html +0 -18
- data/doc/rdoc/classes/ESI/RubyCache.src/M000105.html +0 -22
- data/doc/rdoc/classes/ESI/RubyCache.src/M000106.html +0 -18
- data/doc/rdoc/classes/ESI/RubyCache.src/M000107.html +0 -18
- data/doc/rdoc/classes/ESI/RubyCache.src/M000108.html +0 -18
- data/doc/rdoc/classes/ESI/Tag.html +0 -141
- data/doc/rdoc/classes/ESI/Tag/Attempt.html +0 -113
- data/doc/rdoc/classes/ESI/Tag/Base.html +0 -237
- data/doc/rdoc/classes/ESI/Tag/Base.src/M000026.html +0 -23
- data/doc/rdoc/classes/ESI/Tag/Base.src/M000027.html +0 -22
- data/doc/rdoc/classes/ESI/Tag/Base.src/M000028.html +0 -22
- data/doc/rdoc/classes/ESI/Tag/Base.src/M000029.html +0 -17
- data/doc/rdoc/classes/ESI/Tag/Container.html +0 -206
- data/doc/rdoc/classes/ESI/Tag/Container.src/M000015.html +0 -20
- data/doc/rdoc/classes/ESI/Tag/Container.src/M000016.html +0 -22
- data/doc/rdoc/classes/ESI/Tag/Container.src/M000017.html +0 -25
- data/doc/rdoc/classes/ESI/Tag/Except.html +0 -184
- data/doc/rdoc/classes/ESI/Tag/Except.src/M000018.html +0 -21
- data/doc/rdoc/classes/ESI/Tag/Except.src/M000019.html +0 -20
- data/doc/rdoc/classes/ESI/Tag/Except.src/M000020.html +0 -18
- data/doc/rdoc/classes/ESI/Tag/Include.html +0 -257
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000005.html +0 -20
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000006.html +0 -18
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000007.html +0 -34
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000008.html +0 -22
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000009.html +0 -27
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000010.html +0 -30
- data/doc/rdoc/classes/ESI/Tag/Include.src/M000011.html +0 -51
- data/doc/rdoc/classes/ESI/Tag/Include/Request.html +0 -199
- data/doc/rdoc/classes/ESI/Tag/Include/Request.src/M000012.html +0 -18
- data/doc/rdoc/classes/ESI/Tag/Include/Request.src/M000013.html +0 -44
- data/doc/rdoc/classes/ESI/Tag/Include/Request/Error.html +0 -155
- data/doc/rdoc/classes/ESI/Tag/Include/Request/Error.src/M000014.html +0 -19
- data/doc/rdoc/classes/ESI/Tag/Invalidate.html +0 -171
- data/doc/rdoc/classes/ESI/Tag/Invalidate.src/M000023.html +0 -19
- data/doc/rdoc/classes/ESI/Tag/Invalidate.src/M000024.html +0 -51
- data/doc/rdoc/classes/ESI/Tag/Invalidate.src/M000025.html +0 -19
- data/doc/rdoc/classes/ESI/Tag/Try.html +0 -154
- data/doc/rdoc/classes/ESI/Tag/Try.src/M000021.html +0 -38
- data/doc/rdoc/classes/ESI/Tag/Try.src/M000022.html +0 -18
- data/doc/rdoc/classes/MultiDirHandler.html +0 -198
- data/doc/rdoc/classes/MultiDirHandler.src/M000001.html +0 -20
- data/doc/rdoc/classes/MultiDirHandler.src/M000002.html +0 -28
- data/doc/rdoc/classes/MultiDirHandler.src/M000003.html +0 -22
- data/doc/rdoc/created.rid +0 -1
- data/doc/rdoc/files/COPYING.html +0 -167
- data/doc/rdoc/files/LICENSE.html +0 -605
- data/doc/rdoc/files/README.html +0 -361
- data/doc/rdoc/files/ext/esi/common_rl.html +0 -160
- data/doc/rdoc/files/ext/esi/esi_parser_c.html +0 -101
- data/doc/rdoc/files/ext/esi/parser_c.html +0 -101
- data/doc/rdoc/files/ext/esi/parser_h.html +0 -101
- data/doc/rdoc/files/ext/esi/parser_rl.html +0 -867
- data/doc/rdoc/files/ext/esi/test/test_c.html +0 -101
- data/doc/rdoc/files/lib/esi/cache_rb.html +0 -120
- data/doc/rdoc/files/lib/esi/config_rb.html +0 -115
- data/doc/rdoc/files/lib/esi/dispatcher_rb.html +0 -110
- data/doc/rdoc/files/lib/esi/invalidator_rb.html +0 -114
- data/doc/rdoc/files/lib/esi/logger_rb.html +0 -108
- data/doc/rdoc/files/lib/esi/parser_rb.html +0 -120
- data/doc/rdoc/files/lib/esi/processor_rb.html +0 -115
- data/doc/rdoc/files/lib/esi/proxy_rb.html +0 -123
- data/doc/rdoc/files/lib/esi/rack_adapter_rb.html +0 -117
- data/doc/rdoc/files/lib/esi/response_rb.html +0 -116
- data/doc/rdoc/files/lib/esi/router_rb.html +0 -107
- data/doc/rdoc/files/lib/esi/tag/attempt_rb.html +0 -114
- data/doc/rdoc/files/lib/esi/tag/base_rb.html +0 -114
- data/doc/rdoc/files/lib/esi/tag/container_rb.html +0 -115
- data/doc/rdoc/files/lib/esi/tag/except_rb.html +0 -114
- data/doc/rdoc/files/lib/esi/tag/include_rb.html +0 -116
- data/doc/rdoc/files/lib/esi/tag/invalidate_rb.html +0 -115
- data/doc/rdoc/files/lib/esi/tag/try_rb.html +0 -115
- data/doc/rdoc/files/lib/esi/version_rb.html +0 -101
- data/doc/rdoc/files/lib/multi_dirhandler_rb.html +0 -109
- data/doc/rdoc/fr_class_index.html +0 -57
- data/doc/rdoc/fr_file_index.html +0 -55
- data/doc/rdoc/fr_method_index.html +0 -139
- data/doc/rdoc/index.html +0 -24
- data/doc/rdoc/rdoc-style.css +0 -208
|
@@ -0,0 +1,127 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2007 Todd A. Fisher
|
|
3
|
+
* You can redistribute it and/or modify it under the same terms as Mozilla Public License 1.1.
|
|
4
|
+
*/
|
|
5
|
+
#ifndef ESI_PARSER_H
|
|
6
|
+
#define ESI_PARSER_H
|
|
7
|
+
#include <sys/types.h>
|
|
8
|
+
|
|
9
|
+
/* how much output to hold in memory before sending out */
|
|
10
|
+
#define ESI_OUTPUT_BUFFER_SIZE 4096
|
|
11
|
+
#define ESI_ECHOBUFFER_SIZE 128
|
|
12
|
+
|
|
13
|
+
char *esi_strndup( const char *str, size_t len );
|
|
14
|
+
|
|
15
|
+
/*
|
|
16
|
+
* ESI Attribute is a single attribute with name and value
|
|
17
|
+
*
|
|
18
|
+
* e.g. for an esi include tag:
|
|
19
|
+
*
|
|
20
|
+
* <esi:include src='/foo/bar/' timeout='10'/>
|
|
21
|
+
*
|
|
22
|
+
* 2 attributes would be allocated
|
|
23
|
+
*
|
|
24
|
+
* attrs[0]->name => 'src'
|
|
25
|
+
* attrs[0]->value => '/foo/bar/'
|
|
26
|
+
*
|
|
27
|
+
* attrs[1]->name => 'timeout'
|
|
28
|
+
* attrs[1]->value => '10'
|
|
29
|
+
*
|
|
30
|
+
* */
|
|
31
|
+
typedef struct _ESIAttr {
|
|
32
|
+
const char *name;
|
|
33
|
+
const char *value;
|
|
34
|
+
size_t name_length;
|
|
35
|
+
size_t value_length;
|
|
36
|
+
struct _ESIAttr *next;
|
|
37
|
+
}ESIAttribute;
|
|
38
|
+
|
|
39
|
+
ESIAttribute *esi_attribute_new( const char *name, size_t name_length, const char *value, size_t value_length );
|
|
40
|
+
ESIAttribute *esi_attribute_copy( ESIAttribute *attribute );
|
|
41
|
+
void esi_attribute_free( ESIAttribute *attribute );
|
|
42
|
+
|
|
43
|
+
typedef void (*start_tag_cb)(const void *data,
|
|
44
|
+
const char *name_start,
|
|
45
|
+
size_t name_length,
|
|
46
|
+
ESIAttribute *attributes,
|
|
47
|
+
void *user_data);
|
|
48
|
+
|
|
49
|
+
typedef void (*end_tag_cb)(const void *data,
|
|
50
|
+
const char *name_start,
|
|
51
|
+
size_t name_length,
|
|
52
|
+
void *user_data);
|
|
53
|
+
|
|
54
|
+
typedef void (*output_cb)(const void *data,
|
|
55
|
+
size_t length,
|
|
56
|
+
void *user_data);
|
|
57
|
+
|
|
58
|
+
typedef struct _ESIParser {
|
|
59
|
+
int cs;
|
|
60
|
+
int prev_state;
|
|
61
|
+
|
|
62
|
+
void *user_data;
|
|
63
|
+
|
|
64
|
+
const char *mark;
|
|
65
|
+
size_t overflow_data_size; /* amount of the overflow buffer being used */
|
|
66
|
+
size_t overflow_data_allocated; /* amount of memory allocated to use for the overflow buffer */
|
|
67
|
+
char *overflow_data; /* overflow buffer if execute finishes and we are not in a final state store the parse data */
|
|
68
|
+
unsigned using_overflow:1; /* if this is 1, overflow data is in use. */
|
|
69
|
+
|
|
70
|
+
size_t echobuffer_allocated; /* amount of memory allocated for the echobuffer */
|
|
71
|
+
size_t echobuffer_index; /* current write position of the last echo'ed character */
|
|
72
|
+
char *echobuffer; /* echo buffer if the parse state is not 0 we store the characters here */
|
|
73
|
+
|
|
74
|
+
const char *tag_text; /* start pointer in data */
|
|
75
|
+
size_t tag_text_length; /* length from tag_text within data */
|
|
76
|
+
|
|
77
|
+
const char *attr_key; /* start pointer in data */
|
|
78
|
+
size_t attr_key_length;
|
|
79
|
+
|
|
80
|
+
const char *attr_value; /* start pointer in data */
|
|
81
|
+
size_t attr_value_length;
|
|
82
|
+
|
|
83
|
+
ESIAttribute *attributes, *last;
|
|
84
|
+
|
|
85
|
+
/* this memory will be pass to the output_cb when either it's full
|
|
86
|
+
* or eof is encountered */
|
|
87
|
+
char output_buffer[ESI_OUTPUT_BUFFER_SIZE+1];
|
|
88
|
+
size_t output_buffer_size;
|
|
89
|
+
|
|
90
|
+
start_tag_cb start_tag_handler;
|
|
91
|
+
end_tag_cb end_tag_handler;
|
|
92
|
+
output_cb output_handler;
|
|
93
|
+
|
|
94
|
+
} ESIParser;
|
|
95
|
+
|
|
96
|
+
/* create a new Edge Side Include Parser */
|
|
97
|
+
ESIParser *esi_parser_new();
|
|
98
|
+
void esi_parser_free( ESIParser *parser );
|
|
99
|
+
|
|
100
|
+
/* initialize the parser */
|
|
101
|
+
int esi_parser_init( ESIParser *parser );
|
|
102
|
+
|
|
103
|
+
/*
|
|
104
|
+
* send a chunk of data to the parser, the internal parser state is returned
|
|
105
|
+
*/
|
|
106
|
+
int esi_parser_execute( ESIParser *parser, const char *data, size_t length );
|
|
107
|
+
/*
|
|
108
|
+
* let the parser no that it has reached the end and it should flush any remaining data to the desired output device
|
|
109
|
+
*/
|
|
110
|
+
int esi_parser_finish( ESIParser *parser );
|
|
111
|
+
|
|
112
|
+
/*
|
|
113
|
+
* setup a callback to execute when a new esi: start tag is encountered
|
|
114
|
+
* this is will fire for all block tags e.g. <esi:try>, <esi:attempt> and also
|
|
115
|
+
* inline tags <esi:inline src='cache-key'/> <esi:include src='dest'/>
|
|
116
|
+
*/
|
|
117
|
+
void esi_parser_start_tag_handler( ESIParser *parser, start_tag_cb callback );
|
|
118
|
+
|
|
119
|
+
void esi_parser_end_tag_handler( ESIParser *parser, end_tag_cb callback );
|
|
120
|
+
|
|
121
|
+
/* setup a callback to recieve data ready for output */
|
|
122
|
+
void esi_parser_output_handler( ESIParser *parser, output_cb output_handler );
|
|
123
|
+
|
|
124
|
+
void esi_parser_flush_output( ESIParser *parser );
|
|
125
|
+
|
|
126
|
+
|
|
127
|
+
#endif
|
|
@@ -0,0 +1,611 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Copyright (c) 2008 Todd A. Fisher
|
|
3
|
+
* see LICENSE
|
|
4
|
+
*/
|
|
5
|
+
#include <stdio.h>
|
|
6
|
+
#include <stdlib.h>
|
|
7
|
+
#include <string.h>
|
|
8
|
+
#include <ctype.h>
|
|
9
|
+
#include "parser.h"
|
|
10
|
+
|
|
11
|
+
#ifdef DEBUG
|
|
12
|
+
static void debug_string( const char *msg, const char *str, size_t len )
|
|
13
|
+
{
|
|
14
|
+
char *pstr = esi_strndup( str, len );
|
|
15
|
+
printf( "%s :'%s'\n", msg, pstr );
|
|
16
|
+
free( pstr );
|
|
17
|
+
}
|
|
18
|
+
#else
|
|
19
|
+
#define debug_string(m,s,l)
|
|
20
|
+
#endif
|
|
21
|
+
|
|
22
|
+
/* define default callbacks */
|
|
23
|
+
static void
|
|
24
|
+
esi_parser_default_start_cb( const void *data,
|
|
25
|
+
const char *name_start,
|
|
26
|
+
size_t name_length,
|
|
27
|
+
ESIAttribute *attributes,
|
|
28
|
+
void *user_data )
|
|
29
|
+
{
|
|
30
|
+
}
|
|
31
|
+
static void
|
|
32
|
+
esi_parser_default_end_cb( const void *data,
|
|
33
|
+
const char *name_start,
|
|
34
|
+
size_t name_length,
|
|
35
|
+
void *user_data )
|
|
36
|
+
{
|
|
37
|
+
}
|
|
38
|
+
static void
|
|
39
|
+
esi_parser_default_output_cp(const void *data,
|
|
40
|
+
size_t length,
|
|
41
|
+
void *user_data)
|
|
42
|
+
{
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
/*
|
|
46
|
+
* flush output buffer
|
|
47
|
+
*/
|
|
48
|
+
void esi_parser_flush_output( ESIParser *parser )
|
|
49
|
+
{
|
|
50
|
+
if( parser->output_buffer_size > 0 ) {
|
|
51
|
+
//debug_string( "esi_parser_flush_output:", parser->output_buffer, parser->output_buffer_size );
|
|
52
|
+
parser->output_handler( (void*)parser->output_buffer, parser->output_buffer_size, parser->user_data );
|
|
53
|
+
parser->output_buffer_size = 0;
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
/* send the character to the output handler marking it
|
|
57
|
+
* as ready for consumption, e.g. not an esi tag
|
|
58
|
+
*/
|
|
59
|
+
static void esi_parser_echo_char( ESIParser *parser, char ch )
|
|
60
|
+
{
|
|
61
|
+
parser->output_buffer[parser->output_buffer_size++] = ch;
|
|
62
|
+
if( parser->output_buffer_size == ESI_OUTPUT_BUFFER_SIZE ) {
|
|
63
|
+
// flush the buffer to the consumer
|
|
64
|
+
esi_parser_flush_output( parser );
|
|
65
|
+
}
|
|
66
|
+
}
|
|
67
|
+
static void esi_parser_send_buffer( ESIParser *parser, const char *seq, size_t len )
|
|
68
|
+
{
|
|
69
|
+
const char *nseq = seq + len;
|
|
70
|
+
size_t nlen = parser->output_buffer_size + len;
|
|
71
|
+
|
|
72
|
+
if( nlen < ESI_OUTPUT_BUFFER_SIZE ) {
|
|
73
|
+
memcpy( parser->output_buffer + parser->output_buffer_size, seq, len );
|
|
74
|
+
parser->output_buffer_size += len;
|
|
75
|
+
}
|
|
76
|
+
else if( nlen > ESI_OUTPUT_BUFFER_SIZE ) {
|
|
77
|
+
while( seq != nseq ) {
|
|
78
|
+
esi_parser_echo_char( parser, *seq++ );
|
|
79
|
+
}
|
|
80
|
+
}
|
|
81
|
+
}
|
|
82
|
+
/* send any buffered characters to the output handler.
|
|
83
|
+
* This happens when we enter a case such as <em> where the
|
|
84
|
+
* first two characters < and e match the <esi: expression
|
|
85
|
+
*/
|
|
86
|
+
static void esi_parser_echo_buffer( ESIParser *parser )
|
|
87
|
+
{
|
|
88
|
+
size_t i = 0, len = parser->echobuffer_index + 1;;
|
|
89
|
+
//debug_string( "echobuffer", parser->echobuffer, parser->echobuffer_index+1 );
|
|
90
|
+
//parser->output_handler( parser->echobuffer, parser->echobuffer_index+1, parser->user_data );
|
|
91
|
+
for( ; i < len; ++i ) {
|
|
92
|
+
esi_parser_echo_char( parser, parser->echobuffer[i] );
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
/*
|
|
96
|
+
* clear the buffer, no buffered characters should be emitted .
|
|
97
|
+
* e.g. we matched an esi tag completely and all buffered characters can be tossed out
|
|
98
|
+
*/
|
|
99
|
+
static void esi_parser_echobuffer_clear( ESIParser *parser )
|
|
100
|
+
{
|
|
101
|
+
parser->echobuffer_index = -1;
|
|
102
|
+
}
|
|
103
|
+
|
|
104
|
+
/*
|
|
105
|
+
* add a character to the echobuffer.
|
|
106
|
+
* this happens when we can't determine if the character is allowed to be sent to the client device
|
|
107
|
+
* e.g. matching <e it's not yet determined if these characters are safe to send or not
|
|
108
|
+
*/
|
|
109
|
+
static void esi_parser_concat_to_echobuffer( ESIParser *parser, char ch )
|
|
110
|
+
{
|
|
111
|
+
parser->echobuffer_index++;
|
|
112
|
+
|
|
113
|
+
if( parser->echobuffer_allocated <= parser->echobuffer_index ) {
|
|
114
|
+
/* double the echobuffer size
|
|
115
|
+
* we're getting some crazy input if this case ever happens
|
|
116
|
+
*/
|
|
117
|
+
printf( "increase echobuffer: %lu, %s\n", (long)parser->echobuffer_allocated, parser->echobuffer );
|
|
118
|
+
parser->echobuffer_allocated *= 2;
|
|
119
|
+
parser->echobuffer = (char*)realloc( parser->echobuffer, parser->echobuffer_allocated );
|
|
120
|
+
}
|
|
121
|
+
parser->echobuffer[parser->echobuffer_index] = ch;
|
|
122
|
+
// debug_string( "echo buffer", parser->echobuffer, parser->echobuffer_index+1 );
|
|
123
|
+
}
|
|
124
|
+
/*
|
|
125
|
+
* the mark boundary is not always going to be exactly on the attribute or tag name boundary
|
|
126
|
+
* this trims characters from the left to right, advancing *ptr and reducing *len
|
|
127
|
+
*/
|
|
128
|
+
static void ltrim_pointer( const char **ptr, const char *bounds, size_t *len )
|
|
129
|
+
{
|
|
130
|
+
// remove any spaces or = at the before the value
|
|
131
|
+
while( (isspace( **ptr ) ||
|
|
132
|
+
**ptr == '=' ||
|
|
133
|
+
**ptr == '"' ||
|
|
134
|
+
**ptr == '<' ||
|
|
135
|
+
**ptr == '\'' ) && (*len > 0) && (*ptr != bounds) ) {
|
|
136
|
+
(*ptr)++;
|
|
137
|
+
(*len)--;
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
/*
|
|
141
|
+
* similar to ltrim_pointer, this walks from bounds to *ptr, reducing *len
|
|
142
|
+
*/
|
|
143
|
+
static void rtrim_pointer( const char **ptr, const char *bounds, size_t *len )
|
|
144
|
+
{
|
|
145
|
+
bounds = (*ptr+(*len-1));
|
|
146
|
+
// remove any spaces or = at the before the value
|
|
147
|
+
while( (isspace( *bounds ) ||
|
|
148
|
+
*bounds == '=' ||
|
|
149
|
+
*bounds == '"' ||
|
|
150
|
+
*bounds == '>' ||
|
|
151
|
+
*bounds == '\'') && (*len > 0) && (*ptr != bounds) ){
|
|
152
|
+
bounds--;
|
|
153
|
+
(*len)--;
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
%%{
|
|
158
|
+
machine esi;
|
|
159
|
+
|
|
160
|
+
action begin {
|
|
161
|
+
parser->mark = p;
|
|
162
|
+
//debug_string( "begin", p, 1 );
|
|
163
|
+
}
|
|
164
|
+
action finish {
|
|
165
|
+
// printf( "finish\n" );
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
# record the position of the start tag
|
|
169
|
+
action see_start_tag {
|
|
170
|
+
parser->tag_text = parser->mark+1;
|
|
171
|
+
parser->tag_text_length = p - (parser->mark+1);
|
|
172
|
+
parser->mark = p;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
# detected an inline tag end, sends the start tag and end tag callback
|
|
176
|
+
action see_end_tag {
|
|
177
|
+
/* trim the tag text */
|
|
178
|
+
ltrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
179
|
+
rtrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
180
|
+
|
|
181
|
+
/* send the start tag and end tag message */
|
|
182
|
+
esi_parser_flush_output( parser );
|
|
183
|
+
parser->start_tag_handler( data, parser->tag_text, parser->tag_text_length, parser->attributes, parser->user_data );
|
|
184
|
+
parser->end_tag_handler( data, parser->tag_text, parser->tag_text_length, parser->user_data );
|
|
185
|
+
//printf("\t[see inline tag]\n");
|
|
186
|
+
|
|
187
|
+
/* mark the position */
|
|
188
|
+
parser->tag_text = NULL;
|
|
189
|
+
parser->tag_text_length = 0;
|
|
190
|
+
parser->mark = p;
|
|
191
|
+
|
|
192
|
+
/* clear out the echo buffer */
|
|
193
|
+
esi_parser_echobuffer_clear( parser );
|
|
194
|
+
}
|
|
195
|
+
|
|
196
|
+
# block tag start, with attributes
|
|
197
|
+
action see_block_start_with_attributes {
|
|
198
|
+
/* trim tag text */
|
|
199
|
+
ltrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
200
|
+
rtrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
201
|
+
|
|
202
|
+
/* send the start and end tag message */
|
|
203
|
+
esi_parser_flush_output( parser );
|
|
204
|
+
parser->start_tag_handler( data, parser->tag_text, parser->tag_text_length, parser->attributes, parser->user_data );
|
|
205
|
+
//printf("\t[see start tag with attributes]\n");
|
|
206
|
+
|
|
207
|
+
parser->tag_text = NULL;
|
|
208
|
+
parser->tag_text_length = 0;
|
|
209
|
+
parser->mark = p;
|
|
210
|
+
|
|
211
|
+
/* clear out the echo buffer */
|
|
212
|
+
esi_parser_echobuffer_clear( parser );
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
# see an attribute key, /foo\s*=/
|
|
216
|
+
action see_attribute_key {
|
|
217
|
+
/* save the attribute key start */
|
|
218
|
+
parser->attr_key = parser->mark;
|
|
219
|
+
/* compute the length of the key */
|
|
220
|
+
parser->attr_key_length = p - parser->mark;
|
|
221
|
+
/* save the position following the key */
|
|
222
|
+
parser->mark = p;
|
|
223
|
+
|
|
224
|
+
/* trim the attribute key */
|
|
225
|
+
ltrim_pointer( &(parser->attr_key), p, &(parser->attr_key_length) );
|
|
226
|
+
rtrim_pointer( &(parser->attr_key), p, &(parser->attr_key_length) );
|
|
227
|
+
}
|
|
228
|
+
|
|
229
|
+
# see an attribute value, aprox ~= /['"].*['"]/
|
|
230
|
+
action see_attribute_value {
|
|
231
|
+
ESIAttribute *attr;
|
|
232
|
+
|
|
233
|
+
/* save the attribute value start */
|
|
234
|
+
parser->attr_value = parser->mark;
|
|
235
|
+
/* compute the length of the value */
|
|
236
|
+
parser->attr_value_length = p - parser->mark;
|
|
237
|
+
/* svae the position following the value */
|
|
238
|
+
parser->mark = p;
|
|
239
|
+
|
|
240
|
+
/* trim the attribute value */
|
|
241
|
+
ltrim_pointer( &(parser->attr_value), p, &(parser->attr_value_length) );
|
|
242
|
+
rtrim_pointer( &(parser->attr_value), p, &(parser->attr_value_length) );
|
|
243
|
+
|
|
244
|
+
/* using the attr_key and attr_value, allocate a new attribute object */
|
|
245
|
+
attr = esi_attribute_new( parser->attr_key, parser->attr_key_length,
|
|
246
|
+
parser->attr_value, parser->attr_value_length );
|
|
247
|
+
|
|
248
|
+
/* add the new attribute to the list of attributes */
|
|
249
|
+
if( parser->attributes ) {
|
|
250
|
+
parser->last->next = attr;
|
|
251
|
+
parser->last = attr;
|
|
252
|
+
}
|
|
253
|
+
else {
|
|
254
|
+
parser->last = parser->attributes = attr;
|
|
255
|
+
}
|
|
256
|
+
}
|
|
257
|
+
|
|
258
|
+
# simple block start tag detected, e.g. <esi:try> no attributes
|
|
259
|
+
action block_start_tag {
|
|
260
|
+
|
|
261
|
+
parser->tag_text = parser->mark;
|
|
262
|
+
parser->tag_text_length = p - parser->mark;
|
|
263
|
+
|
|
264
|
+
parser->mark = p;
|
|
265
|
+
|
|
266
|
+
ltrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
267
|
+
rtrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
268
|
+
|
|
269
|
+
esi_parser_flush_output( parser );
|
|
270
|
+
parser->start_tag_handler( data, parser->tag_text, parser->tag_text_length, NULL, parser->user_data );
|
|
271
|
+
//printf("\t[see start tag]\n");
|
|
272
|
+
|
|
273
|
+
esi_parser_echobuffer_clear( parser );
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
# block end tag detected, e.g. </esi:try>
|
|
277
|
+
action block_end_tag {
|
|
278
|
+
/* offset by 2 to account for the </ characters */
|
|
279
|
+
parser->tag_text = parser->mark+2;
|
|
280
|
+
parser->tag_text_length = p - (parser->mark+2);
|
|
281
|
+
|
|
282
|
+
parser->mark = p;
|
|
283
|
+
|
|
284
|
+
ltrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
285
|
+
rtrim_pointer( &(parser->tag_text), p, &(parser->tag_text_length) );
|
|
286
|
+
|
|
287
|
+
esi_parser_flush_output( parser );
|
|
288
|
+
parser->end_tag_handler( data, parser->tag_text, parser->tag_text_length, parser->user_data );
|
|
289
|
+
//printf("\t[see end tag]\n");
|
|
290
|
+
|
|
291
|
+
esi_parser_echobuffer_clear( parser );
|
|
292
|
+
}
|
|
293
|
+
|
|
294
|
+
# process each character in the input stream for output
|
|
295
|
+
action echo {
|
|
296
|
+
//printf( "[%c:%d],", *p, cs );
|
|
297
|
+
switch( cs ) {
|
|
298
|
+
case 0: /* non matching state */
|
|
299
|
+
if( parser->prev_state != 12 && parser->prev_state != 7 ){ /* states following a possible end state for a tag */
|
|
300
|
+
if( parser->echobuffer && parser->echobuffer_index != -1 ){
|
|
301
|
+
/* send the echo buffer */
|
|
302
|
+
esi_parser_echo_buffer( parser );
|
|
303
|
+
}
|
|
304
|
+
/* send the current character */
|
|
305
|
+
esi_parser_echo_char( parser, *p );
|
|
306
|
+
}
|
|
307
|
+
/* clear the echo buffer */
|
|
308
|
+
esi_parser_echobuffer_clear( parser );
|
|
309
|
+
break;
|
|
310
|
+
default:
|
|
311
|
+
/* append to the echo buffer */
|
|
312
|
+
esi_parser_concat_to_echobuffer( parser, *p );
|
|
313
|
+
}
|
|
314
|
+
/* save the previous state, necessary for end case detection such as /> and </esi:try> the trailing > character
|
|
315
|
+
is state 12 and 7
|
|
316
|
+
*/
|
|
317
|
+
parser->prev_state = cs;
|
|
318
|
+
}
|
|
319
|
+
|
|
320
|
+
include esi_common_parser "common.rl";
|
|
321
|
+
}%%
|
|
322
|
+
|
|
323
|
+
%%write data;
|
|
324
|
+
|
|
325
|
+
/* dup the string up to len */
|
|
326
|
+
char *esi_strndup( const char *str, size_t len )
|
|
327
|
+
{
|
|
328
|
+
char *s = (char*)malloc(sizeof(char)*(len+1));
|
|
329
|
+
memcpy( s, str, len );
|
|
330
|
+
s[len] = '\0';
|
|
331
|
+
return s;
|
|
332
|
+
}
|
|
333
|
+
|
|
334
|
+
ESIAttribute *esi_attribute_new( const char *name, size_t name_length, const char *value, size_t value_length )
|
|
335
|
+
{
|
|
336
|
+
ESIAttribute *attr = (ESIAttribute*)malloc(sizeof(ESIAttribute));
|
|
337
|
+
attr->name = name;//esi_strndup(name, name_length);
|
|
338
|
+
attr->value = value;//esi_strndup(value, value_length);
|
|
339
|
+
attr->name_length = name_length;
|
|
340
|
+
attr->value_length = value_length;
|
|
341
|
+
attr->next = NULL;
|
|
342
|
+
return attr;
|
|
343
|
+
}
|
|
344
|
+
|
|
345
|
+
ESIAttribute *esi_attribute_copy( ESIAttribute *attribute )
|
|
346
|
+
{
|
|
347
|
+
ESIAttribute *head, *nattr;
|
|
348
|
+
if( !attribute ){ return NULL; }
|
|
349
|
+
|
|
350
|
+
// copy the first attribute
|
|
351
|
+
nattr = esi_attribute_new( attribute->name, attribute->name_length,
|
|
352
|
+
attribute->value, attribute->value_length );
|
|
353
|
+
// save a pointer for return
|
|
354
|
+
head = nattr;
|
|
355
|
+
// copy next attributes
|
|
356
|
+
attribute = attribute->next;
|
|
357
|
+
while( attribute ) {
|
|
358
|
+
// set the next attribute
|
|
359
|
+
nattr->next = esi_attribute_new( attribute->name, attribute->name_length,
|
|
360
|
+
attribute->value, attribute->value_length );
|
|
361
|
+
// next attribute
|
|
362
|
+
nattr = nattr->next;
|
|
363
|
+
attribute = attribute->next;
|
|
364
|
+
}
|
|
365
|
+
return head;
|
|
366
|
+
}
|
|
367
|
+
|
|
368
|
+
void esi_attribute_free( ESIAttribute *attribute )
|
|
369
|
+
{
|
|
370
|
+
ESIAttribute *ptr;
|
|
371
|
+
while( attribute ){
|
|
372
|
+
// free( attribute->name );
|
|
373
|
+
// free( attribute->value );
|
|
374
|
+
ptr = attribute->next;
|
|
375
|
+
free( attribute );
|
|
376
|
+
attribute = ptr;
|
|
377
|
+
}
|
|
378
|
+
}
|
|
379
|
+
|
|
380
|
+
ESIParser *esi_parser_new()
|
|
381
|
+
{
|
|
382
|
+
ESIParser *parser = (ESIParser*)malloc(sizeof(ESIParser));
|
|
383
|
+
parser->cs = esi_start;
|
|
384
|
+
parser->mark = NULL;
|
|
385
|
+
parser->tag_text = NULL;
|
|
386
|
+
parser->attr_key = NULL;
|
|
387
|
+
parser->attr_value = NULL;
|
|
388
|
+
parser->overflow_data_size = 0;
|
|
389
|
+
parser->overflow_data = NULL;
|
|
390
|
+
parser->overflow_data_allocated = 0;
|
|
391
|
+
parser->using_overflow = 0;
|
|
392
|
+
|
|
393
|
+
/* allocate ESI_OUTPUT_BUFFER_SIZE bytes for the echobuffer */
|
|
394
|
+
parser->echobuffer_allocated = ESI_ECHOBUFFER_SIZE;
|
|
395
|
+
parser->echobuffer_index = -1;
|
|
396
|
+
parser->echobuffer = (char*)malloc(sizeof(char)*parser->echobuffer_allocated);
|
|
397
|
+
|
|
398
|
+
parser->attributes = NULL;
|
|
399
|
+
parser->last = NULL;
|
|
400
|
+
|
|
401
|
+
parser->start_tag_handler = esi_parser_default_start_cb;
|
|
402
|
+
parser->end_tag_handler = esi_parser_default_end_cb;
|
|
403
|
+
parser->output_handler = esi_parser_default_output_cp;
|
|
404
|
+
|
|
405
|
+
parser->output_buffer_size = 0;
|
|
406
|
+
memset( parser->output_buffer, 0, ESI_OUTPUT_BUFFER_SIZE );
|
|
407
|
+
|
|
408
|
+
return parser;
|
|
409
|
+
}
|
|
410
|
+
void esi_parser_free( ESIParser *parser )
|
|
411
|
+
{
|
|
412
|
+
if( parser->overflow_data ){ free( parser->overflow_data ); }
|
|
413
|
+
|
|
414
|
+
free( parser->echobuffer );
|
|
415
|
+
|
|
416
|
+
esi_attribute_free( parser->attributes );
|
|
417
|
+
|
|
418
|
+
free( parser );
|
|
419
|
+
}
|
|
420
|
+
|
|
421
|
+
void esi_parser_output_handler( ESIParser *parser, output_cb output_handler )
|
|
422
|
+
{
|
|
423
|
+
parser->output_handler = output_handler;
|
|
424
|
+
}
|
|
425
|
+
|
|
426
|
+
int esi_parser_init( ESIParser *parser )
|
|
427
|
+
{
|
|
428
|
+
int cs;
|
|
429
|
+
%% write init;
|
|
430
|
+
parser->prev_state = parser->cs = cs;
|
|
431
|
+
return 0;
|
|
432
|
+
}
|
|
433
|
+
|
|
434
|
+
static int compute_offset( const char *mark, const char *data )
|
|
435
|
+
{
|
|
436
|
+
if( mark ) {
|
|
437
|
+
return mark - data;
|
|
438
|
+
}
|
|
439
|
+
return -1;
|
|
440
|
+
}
|
|
441
|
+
|
|
442
|
+
/*
|
|
443
|
+
* scans the data buffer for a start sequence /<$/, /<e$/, /<es$/, /<esi$/, /<esi:$/
|
|
444
|
+
* returns index of if start sequence found else returns -1
|
|
445
|
+
*/
|
|
446
|
+
static int
|
|
447
|
+
esi_parser_scan_for_start( ESIParser *parser, const char *data, size_t length )
|
|
448
|
+
{
|
|
449
|
+
size_t i, f = -2, s = -2;
|
|
450
|
+
char ch;
|
|
451
|
+
|
|
452
|
+
for( i = 0; i < length; ++i ) {
|
|
453
|
+
ch = data[i];
|
|
454
|
+
switch( ch ) {
|
|
455
|
+
case '<':
|
|
456
|
+
f = s = i;
|
|
457
|
+
break;
|
|
458
|
+
case '/':
|
|
459
|
+
if( s == (i-1) && f != -2 ) { s = i; }
|
|
460
|
+
break;
|
|
461
|
+
case 'e':
|
|
462
|
+
if( s == (i-1) && f != -2 ) { s = i; }
|
|
463
|
+
break;
|
|
464
|
+
case 's':
|
|
465
|
+
if( s == (i-1) && f != -2 ) { s = i; }
|
|
466
|
+
break;
|
|
467
|
+
case 'i':
|
|
468
|
+
if( s == (i-1) && f != -2 ) { s = i; }
|
|
469
|
+
break;
|
|
470
|
+
case ':':
|
|
471
|
+
if( s == (i-1) && f != -2 ) { s = i; return f; }
|
|
472
|
+
break;
|
|
473
|
+
default:
|
|
474
|
+
f = s = -2;
|
|
475
|
+
break;
|
|
476
|
+
}
|
|
477
|
+
}
|
|
478
|
+
|
|
479
|
+
// if s and f are still valid at end of input return f
|
|
480
|
+
if( f != -2 && s != -2 ) {
|
|
481
|
+
return f;
|
|
482
|
+
}
|
|
483
|
+
else {
|
|
484
|
+
return -1;
|
|
485
|
+
}
|
|
486
|
+
}
|
|
487
|
+
|
|
488
|
+
static void
|
|
489
|
+
esi_parser_append_data_to_overflow_data( ESIParser *parser, const char *data, size_t length )
|
|
490
|
+
{
|
|
491
|
+
int mark_offset = 0;
|
|
492
|
+
int tag_text_offset = 0;
|
|
493
|
+
int attr_key_offset = 0;
|
|
494
|
+
int attr_value_offset = 0;
|
|
495
|
+
|
|
496
|
+
if( parser->using_overflow ) {
|
|
497
|
+
// recompute mark, tag_text, attr_key, and attr_value as they all exist within overflow_data
|
|
498
|
+
mark_offset = compute_offset( parser->mark, parser->overflow_data );
|
|
499
|
+
tag_text_offset = compute_offset( parser->tag_text, parser->overflow_data );
|
|
500
|
+
attr_key_offset = compute_offset( parser->attr_key, parser->overflow_data );
|
|
501
|
+
attr_value_offset = compute_offset( parser->attr_value, parser->overflow_data );
|
|
502
|
+
}
|
|
503
|
+
else {
|
|
504
|
+
// recompute mark, tag_text, attr_key, and attr_value as they all exist within data
|
|
505
|
+
mark_offset = compute_offset( parser->mark, data );
|
|
506
|
+
tag_text_offset = compute_offset( parser->tag_text, data );
|
|
507
|
+
attr_key_offset = compute_offset( parser->attr_key, data );
|
|
508
|
+
attr_value_offset = compute_offset( parser->attr_value, data );
|
|
509
|
+
}
|
|
510
|
+
|
|
511
|
+
if( parser->overflow_data ) {
|
|
512
|
+
// check if we need to resize or grow the buffer
|
|
513
|
+
if( (parser->overflow_data_size + length) > parser->overflow_data_allocated ) {
|
|
514
|
+
parser->overflow_data_allocated += length;
|
|
515
|
+
// printf( "realloc: %d bytes, using %d\n", (int)parser->overflow_data_allocated, (int)(parser->overflow_data_size+length) );
|
|
516
|
+
parser->overflow_data = (char*)realloc( parser->overflow_data, sizeof(char)*(parser->overflow_data_allocated) );
|
|
517
|
+
}
|
|
518
|
+
memcpy( parser->overflow_data+parser->overflow_data_size, data, length );
|
|
519
|
+
parser->overflow_data_size += length;
|
|
520
|
+
}
|
|
521
|
+
else {
|
|
522
|
+
parser->overflow_data_allocated = ESI_OUTPUT_BUFFER_SIZE > length ? ESI_OUTPUT_BUFFER_SIZE : length;
|
|
523
|
+
parser->overflow_data = (char*)malloc( sizeof( char ) * parser->overflow_data_allocated );
|
|
524
|
+
memcpy( parser->overflow_data, data, length );
|
|
525
|
+
parser->overflow_data_size = length;
|
|
526
|
+
// printf( "malloc: %d for %d bytes\n", (int)parser->overflow_data_allocated, (int)length );
|
|
527
|
+
}
|
|
528
|
+
|
|
529
|
+
// in our new memory space mark will now be
|
|
530
|
+
parser->mark = ( mark_offset >= 0 ) ? parser->overflow_data + mark_offset : NULL;
|
|
531
|
+
parser->tag_text = ( tag_text_offset >= 0 ) ? parser->overflow_data + tag_text_offset : NULL;
|
|
532
|
+
parser->attr_key = ( attr_key_offset >= 0 ) ? parser->overflow_data + attr_key_offset : NULL;
|
|
533
|
+
parser->attr_value = ( attr_value_offset >= 0 ) ? parser->overflow_data + attr_value_offset : NULL;
|
|
534
|
+
}
|
|
535
|
+
|
|
536
|
+
/* accept an arbitrary length string buffer
|
|
537
|
+
* when this methods exits it determines if an end state was reached
|
|
538
|
+
* if no end state was reached it saves the full input into an internal buffer
|
|
539
|
+
* when invoked next, it reuses that internable buffer copying all pointers into the
|
|
540
|
+
* newly allocated buffer. if it exits in a terminal state, e.g. 0 then it will dump these buffers
|
|
541
|
+
*/
|
|
542
|
+
int esi_parser_execute( ESIParser *parser, const char *data, size_t length )
|
|
543
|
+
{
|
|
544
|
+
int cs = parser->cs;
|
|
545
|
+
const char *p = data;
|
|
546
|
+
const char *eof = NULL; // ragel 6.x compat
|
|
547
|
+
const char *pe = data + length;
|
|
548
|
+
int pindex;
|
|
549
|
+
|
|
550
|
+
if( length == 0 ){ return cs; }
|
|
551
|
+
|
|
552
|
+
/* scan data for any '<esi:' start sequences, /<$/, /<e$/, /<es$/, /<esi$/, /<esi:$/ */
|
|
553
|
+
if( cs == esi_start || cs == 0 ) {
|
|
554
|
+
pindex = esi_parser_scan_for_start( parser, data, length );
|
|
555
|
+
if( pindex == -1 ) { /* this means there are no start sequences in the buffer we recieved */
|
|
556
|
+
esi_parser_send_buffer( parser, data, length );
|
|
557
|
+
return cs; /* no esi start sequences send it all out and break out early */
|
|
558
|
+
}
|
|
559
|
+
}
|
|
560
|
+
|
|
561
|
+
/* there's an existing overflow buffer and it's being used, append the new data */
|
|
562
|
+
//if( parser->overflow_data && parser->overflow_data_size > 0 ) {
|
|
563
|
+
if( parser->using_overflow ) {
|
|
564
|
+
int prev_overflow_size = parser->overflow_data_size; // save the current overflow size
|
|
565
|
+
esi_parser_append_data_to_overflow_data( parser, data, length );
|
|
566
|
+
|
|
567
|
+
/* set parser variables */
|
|
568
|
+
p = parser->overflow_data + prev_overflow_size;
|
|
569
|
+
data = parser->overflow_data;
|
|
570
|
+
length = parser->overflow_data_size;
|
|
571
|
+
pe = data + length;
|
|
572
|
+
}
|
|
573
|
+
|
|
574
|
+
if( !parser->mark ) {
|
|
575
|
+
parser->mark = p;
|
|
576
|
+
}
|
|
577
|
+
|
|
578
|
+
%% write exec;
|
|
579
|
+
|
|
580
|
+
parser->cs = cs;
|
|
581
|
+
|
|
582
|
+
if( cs != esi_start && cs != 0 ) {
|
|
583
|
+
|
|
584
|
+
/* reached the end and we're not at a termination point save the buffer as overflow */
|
|
585
|
+
if( !parser->using_overflow ) {
|
|
586
|
+
esi_parser_append_data_to_overflow_data( parser, data, length );
|
|
587
|
+
parser->using_overflow = 1;
|
|
588
|
+
}
|
|
589
|
+
|
|
590
|
+
} else if ( parser->using_overflow ) {
|
|
591
|
+
parser->using_overflow = 0;
|
|
592
|
+
parser->overflow_data_size = 0;
|
|
593
|
+
}
|
|
594
|
+
|
|
595
|
+
return cs;
|
|
596
|
+
}
|
|
597
|
+
int esi_parser_finish( ESIParser *parser )
|
|
598
|
+
{
|
|
599
|
+
esi_parser_flush_output( parser );
|
|
600
|
+
return 0;
|
|
601
|
+
}
|
|
602
|
+
|
|
603
|
+
void esi_parser_start_tag_handler( ESIParser *parser, start_tag_cb callback )
|
|
604
|
+
{
|
|
605
|
+
parser->start_tag_handler = callback;
|
|
606
|
+
}
|
|
607
|
+
|
|
608
|
+
void esi_parser_end_tag_handler( ESIParser *parser, end_tag_cb callback )
|
|
609
|
+
{
|
|
610
|
+
parser->end_tag_handler = callback;
|
|
611
|
+
}
|