mirror of
git://git.psyc.eu/libpsyc
synced 2024-08-15 03:19:02 +00:00
parser: fixes routing-only mode, set valueParsed & valueLength for data as well; render: added support for rendering raw content
This commit is contained in:
parent
fea30e3688
commit
fd5d886053
6 changed files with 116 additions and 46 deletions
|
@ -39,12 +39,12 @@ typedef enum
|
||||||
typedef enum
|
typedef enum
|
||||||
{
|
{
|
||||||
PSYC_PART_RESET = -1,
|
PSYC_PART_RESET = -1,
|
||||||
PSYC_PART_ROUTING,
|
PSYC_PART_ROUTING = 0,
|
||||||
PSYC_PART_LENGTH,
|
PSYC_PART_LENGTH = 1,
|
||||||
PSYC_PART_CONTENT,
|
PSYC_PART_CONTENT = 2,
|
||||||
PSYC_PART_METHOD,
|
PSYC_PART_METHOD = 3,
|
||||||
PSYC_PART_DATA,
|
PSYC_PART_DATA = 4,
|
||||||
PSYC_PART_END,
|
PSYC_PART_END = 5,
|
||||||
} psycPart;
|
} psycPart;
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
|
|
@ -56,6 +56,7 @@ typedef struct
|
||||||
psycHeader entity; ///< Entity header.
|
psycHeader entity; ///< Entity header.
|
||||||
psycString method;
|
psycString method;
|
||||||
psycString data;
|
psycString data;
|
||||||
|
psycString content;
|
||||||
size_t routingLength; ///< Length of routing part.
|
size_t routingLength; ///< Length of routing part.
|
||||||
size_t contentLength; ///< Length of content part.
|
size_t contentLength; ///< Length of content part.
|
||||||
size_t length; ///< Total length of packet.
|
size_t length; ///< Total length of packet.
|
||||||
|
@ -135,4 +136,11 @@ psycPacket psyc_newPacket2 (psycModifier *routing, size_t routinglen,
|
||||||
const char *data, size_t datalen,
|
const char *data, size_t datalen,
|
||||||
psycPacketFlag flag);
|
psycPacketFlag flag);
|
||||||
|
|
||||||
|
psycPacket psyc_newPacketContent (psycHeader *routing, psycString *content,
|
||||||
|
psycPacketFlag flag);
|
||||||
|
|
||||||
|
psycPacket psyc_newPacketContent2 (psycModifier *routing, size_t routinglen,
|
||||||
|
const char *content, size_t contentlen,
|
||||||
|
psycPacketFlag flag);
|
||||||
|
|
||||||
#endif // PSYC_PACKET_H
|
#endif // PSYC_PACKET_H
|
||||||
|
|
|
@ -4,7 +4,7 @@
|
||||||
*
|
*
|
||||||
* All parsing functions and the definitions they use are
|
* All parsing functions and the definitions they use are
|
||||||
* defined in this file.
|
* defined in this file.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* @defgroup parser Parsing Functions
|
* @defgroup parser Parsing Functions
|
||||||
|
|
31
src/packet.c
31
src/packet.c
|
@ -114,6 +114,10 @@ size_t psyc_setPacketLength(psycPacket *p)
|
||||||
for (i = 0; i < p->routing.lines; i++)
|
for (i = 0; i < p->routing.lines; i++)
|
||||||
p->routingLength += psyc_getModifierLength(&(p->routing.modifiers[i]));
|
p->routingLength += psyc_getModifierLength(&(p->routing.modifiers[i]));
|
||||||
|
|
||||||
|
if (p->content.length)
|
||||||
|
p->contentLength = p->content.length;
|
||||||
|
else
|
||||||
|
{
|
||||||
// add entity header length
|
// add entity header length
|
||||||
for (i = 0; i < p->entity.lines; i++)
|
for (i = 0; i < p->entity.lines; i++)
|
||||||
p->contentLength += psyc_getModifierLength(&(p->entity.modifiers[i]));
|
p->contentLength += psyc_getModifierLength(&(p->entity.modifiers[i]));
|
||||||
|
@ -123,6 +127,7 @@ size_t psyc_setPacketLength(psycPacket *p)
|
||||||
p->contentLength += p->method.length + 1; // method\n
|
p->contentLength += p->method.length + 1; // method\n
|
||||||
if (p->data.length)
|
if (p->data.length)
|
||||||
p->contentLength += p->data.length + 1; // data\n
|
p->contentLength += p->data.length + 1; // data\n
|
||||||
|
}
|
||||||
|
|
||||||
// set total length: routing-header content |\n
|
// set total length: routing-header content |\n
|
||||||
p->length = p->routingLength + p->contentLength + 2;
|
p->length = p->routingLength + p->contentLength + 2;
|
||||||
|
@ -141,7 +146,7 @@ psycPacket psyc_newPacket (psycHeader *routing, psycHeader *entity,
|
||||||
psycString *method, psycString *data,
|
psycString *method, psycString *data,
|
||||||
psycPacketFlag flag)
|
psycPacketFlag flag)
|
||||||
{
|
{
|
||||||
psycPacket p = {*routing, *entity, *method, *data, 0, 0, flag};
|
psycPacket p = {*routing, *entity, *method, *data, {0,0}, 0, 0, flag};
|
||||||
|
|
||||||
if (flag == PSYC_PACKET_CHECK_LENGTH) // find out if it needs a length
|
if (flag == PSYC_PACKET_CHECK_LENGTH) // find out if it needs a length
|
||||||
p.flag = psyc_checkPacketLength(&p);
|
p.flag = psyc_checkPacketLength(&p);
|
||||||
|
@ -164,3 +169,27 @@ psycPacket psyc_newPacket2 (psycModifier *routing, size_t routinglen,
|
||||||
|
|
||||||
return psyc_newPacket(&r, &e, &m, &d, flag);
|
return psyc_newPacket(&r, &e, &m, &d, flag);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
inline
|
||||||
|
psycPacket psyc_newPacketContent (psycHeader *routing, psycString *content,
|
||||||
|
psycPacketFlag flag)
|
||||||
|
{
|
||||||
|
psycPacket p = {*routing, {0,0}, {0,0}, {0,0}, *content, 0, 0, flag};
|
||||||
|
|
||||||
|
if (flag == PSYC_PACKET_CHECK_LENGTH) // find out if it needs a length
|
||||||
|
p.flag = psyc_checkPacketLength(&p);
|
||||||
|
|
||||||
|
psyc_setPacketLength(&p);
|
||||||
|
return p;
|
||||||
|
}
|
||||||
|
|
||||||
|
inline
|
||||||
|
psycPacket psyc_newPacketContent2 (psycModifier *routing, size_t routinglen,
|
||||||
|
const char *content, size_t contentlen,
|
||||||
|
psycPacketFlag flag)
|
||||||
|
{
|
||||||
|
psycHeader r = {routinglen, routing};
|
||||||
|
psycString c = {contentlen, content};
|
||||||
|
|
||||||
|
return psyc_newPacketContent(&r, &c, flag);
|
||||||
|
}
|
||||||
|
|
49
src/parser.c
49
src/parser.c
|
@ -222,6 +222,7 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
case PSYC_PART_RESET: // New packet starts here, reset state.
|
case PSYC_PART_RESET: // New packet starts here, reset state.
|
||||||
state->valueParsed = 0;
|
state->valueParsed = 0;
|
||||||
state->valueLength = 0;
|
state->valueLength = 0;
|
||||||
|
state->valueLengthFound = 0;
|
||||||
state->routingLength = 0;
|
state->routingLength = 0;
|
||||||
state->contentParsed = 0;
|
state->contentParsed = 0;
|
||||||
state->contentLength = 0;
|
state->contentLength = 0;
|
||||||
|
@ -354,6 +355,10 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
if (state->buffer.ptr[state->cursor] != '\n')
|
if (state->buffer.ptr[state->cursor] != '\n')
|
||||||
return PSYC_PARSE_ERROR_METHOD;
|
return PSYC_PARSE_ERROR_METHOD;
|
||||||
|
|
||||||
|
state->valueLengthFound = 0;
|
||||||
|
state->valueParsed = 0;
|
||||||
|
state->valueLength = 0;
|
||||||
|
|
||||||
if (state->contentLengthFound)
|
if (state->contentLengthFound)
|
||||||
{ // if length was found set start position to the beginning of data
|
{ // if length was found set start position to the beginning of data
|
||||||
state->cursor++;
|
state->cursor++;
|
||||||
|
@ -362,7 +367,9 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
state->part = PSYC_PART_DATA;
|
state->part = PSYC_PART_DATA;
|
||||||
}
|
}
|
||||||
else // otherwise keep it at the beginning of method
|
else // otherwise keep it at the beginning of method
|
||||||
|
{
|
||||||
ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT);
|
ADVANCE_CURSOR_OR_RETURN(PSYC_PARSE_INSUFFICIENT);
|
||||||
|
}
|
||||||
|
|
||||||
// fall thru
|
// fall thru
|
||||||
}
|
}
|
||||||
|
@ -380,19 +387,21 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
|
|
||||||
if (state->contentLengthFound) // We know the length of the packet.
|
if (state->contentLengthFound) // We know the length of the packet.
|
||||||
{
|
{
|
||||||
if (state->contentParsed < state->contentLength &&
|
if (!state->valueLengthFound) // start of data
|
||||||
psyc_parseBinaryValue(state, value, &(state->contentLength),
|
{
|
||||||
&(state->contentParsed)) == PSYC_PARSE_INCOMPLETE)
|
state->valueLengthFound = 1;
|
||||||
|
state->valueLength = state->contentLength - state->contentParsed; // length of data
|
||||||
|
if (state->valueLength && !(state->flags & PSYC_PARSE_ROUTING_ONLY))
|
||||||
|
state->valueLength--; // \n at the end is not part of data
|
||||||
|
}
|
||||||
|
if (state->valueParsed < state->valueLength)
|
||||||
|
{
|
||||||
|
ret = psyc_parseBinaryValue(state, value, &(state->valueLength), &(state->valueParsed));
|
||||||
|
state->contentParsed += value->length;
|
||||||
|
|
||||||
|
if (ret == PSYC_PARSE_INCOMPLETE)
|
||||||
return PSYC_PARSE_BODY_INCOMPLETE;
|
return PSYC_PARSE_BODY_INCOMPLETE;
|
||||||
|
}
|
||||||
if (value->length)
|
|
||||||
value->length--; // \n at the end is not part of the body
|
|
||||||
|
|
||||||
if (state->cursor >= state->buffer.length)
|
|
||||||
return PSYC_PARSE_BODY;
|
|
||||||
|
|
||||||
if (state->buffer.ptr[state->cursor] != '|')
|
|
||||||
return PSYC_PARSE_ERROR_BODY;
|
|
||||||
|
|
||||||
state->part = PSYC_PART_END;
|
state->part = PSYC_PART_END;
|
||||||
return PSYC_PARSE_BODY;
|
return PSYC_PARSE_BODY;
|
||||||
|
@ -415,6 +424,9 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
if (state->buffer.ptr[state->cursor+nl] == '|' &&
|
if (state->buffer.ptr[state->cursor+nl] == '|' &&
|
||||||
state->buffer.ptr[state->cursor+1+nl] == '\n') // packet ends here
|
state->buffer.ptr[state->cursor+1+nl] == '\n') // packet ends here
|
||||||
{
|
{
|
||||||
|
if (state->flags & PSYC_PARSE_ROUTING_ONLY)
|
||||||
|
value->length++;
|
||||||
|
|
||||||
state->contentParsed += state->cursor - pos;
|
state->contentParsed += state->cursor - pos;
|
||||||
state->cursor += nl;
|
state->cursor += nl;
|
||||||
state->part = PSYC_PART_END;
|
state->part = PSYC_PART_END;
|
||||||
|
@ -428,6 +440,19 @@ psycParseRC psyc_parse (psycParseState *state, char *oper,
|
||||||
|
|
||||||
case PSYC_PART_END:
|
case PSYC_PART_END:
|
||||||
PSYC_PART_END:
|
PSYC_PART_END:
|
||||||
|
if (state->contentLengthFound && state->valueLengthFound && state->valueLength &&
|
||||||
|
!(state->flags & PSYC_PARSE_ROUTING_ONLY))
|
||||||
|
{ // if data was not empty next is the \n at the end of data
|
||||||
|
state->valueLength = 0;
|
||||||
|
state->valueLengthFound = 0;
|
||||||
|
|
||||||
|
if (state->buffer.ptr[state->cursor] != '\n')
|
||||||
|
return PSYC_PARSE_ERROR_END;
|
||||||
|
|
||||||
|
state->contentParsed++;
|
||||||
|
state->cursor++;
|
||||||
|
}
|
||||||
|
|
||||||
// End of packet, at this point we have already passed a \n
|
// End of packet, at this point we have already passed a \n
|
||||||
// and the cursor should point to |
|
// and the cursor should point to |
|
||||||
if (state->cursor+1 >= state->buffer.length) // incremented cursor inside length?
|
if (state->cursor+1 >= state->buffer.length) // incremented cursor inside length?
|
||||||
|
|
10
src/render.c
10
src/render.c
|
@ -82,10 +82,17 @@ psycRenderRC psyc_render (psycPacket *packet, char *buffer, size_t buflen)
|
||||||
if (packet->flag == PSYC_PACKET_NEED_LENGTH)
|
if (packet->flag == PSYC_PACKET_NEED_LENGTH)
|
||||||
cur += itoa(packet->contentLength, buffer + cur, 10);
|
cur += itoa(packet->contentLength, buffer + cur, 10);
|
||||||
|
|
||||||
if (packet->flag == PSYC_PACKET_NEED_LENGTH ||
|
if (packet->flag == PSYC_PACKET_NEED_LENGTH || packet->content.length ||
|
||||||
packet->entity.lines || packet->method.length || packet->data.length)
|
packet->entity.lines || packet->method.length || packet->data.length)
|
||||||
buffer[cur++] = '\n'; // start of content part if there's content or length
|
buffer[cur++] = '\n'; // start of content part if there's content or length
|
||||||
|
|
||||||
|
if (packet->content.length) // render raw content if present
|
||||||
|
{
|
||||||
|
memcpy(buffer + cur, packet->content.ptr, packet->content.length);
|
||||||
|
cur += packet->content.length;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
// render entity modifiers
|
// render entity modifiers
|
||||||
for (i = 0; i < packet->entity.lines; i++)
|
for (i = 0; i < packet->entity.lines; i++)
|
||||||
cur += psyc_renderModifier(&packet->entity.modifiers[i], buffer + cur);
|
cur += psyc_renderModifier(&packet->entity.modifiers[i], buffer + cur);
|
||||||
|
@ -105,6 +112,7 @@ psycRenderRC psyc_render (psycPacket *packet, char *buffer, size_t buflen)
|
||||||
}
|
}
|
||||||
else if (packet->data.length) // error, we have data but no modifier
|
else if (packet->data.length) // error, we have data but no modifier
|
||||||
return PSYC_RENDER_ERROR_METHOD_MISSING;
|
return PSYC_RENDER_ERROR_METHOD_MISSING;
|
||||||
|
}
|
||||||
|
|
||||||
// add packet delimiter
|
// add packet delimiter
|
||||||
buffer[cur++] = C_GLYPH_PACKET_DELIMITER;
|
buffer[cur++] = C_GLYPH_PACKET_DELIMITER;
|
||||||
|
|
Loading…
Reference in a new issue