Parse 'null'

This commit is contained in:
Benoit Blanchon
2014-10-15 23:27:38 +02:00
parent c61ee09d26
commit 3d92531ad3
5 changed files with 120 additions and 80 deletions

View File

@ -3,11 +3,6 @@
#include <stdlib.h> // for strtol, strtod #include <stdlib.h> // for strtol, strtod
#include <ctype.h> #include <ctype.h>
bool JsonParser::isEnd()
{
return *_ptr == 0;
}
bool JsonParser::isArrayStart() bool JsonParser::isArrayStart()
{ {
return *_ptr == '['; return *_ptr == '[';
@ -18,20 +13,14 @@ bool JsonParser::isArrayStop()
return *_ptr == ']'; return *_ptr == ']';
} }
bool JsonParser::isLong() bool JsonParser::isBoolean()
{ {
char* ptr = _ptr; return *_ptr == 't' || *_ptr == 'f';
}
// skip all digits bool JsonParser::isComma()
while (isdigit(*ptr)) {
ptr++; return *_ptr == ',';
// same position => 0 digits => not a number
if (ptr == _ptr)
return false;
// stopped on a decimal separator => not a long but a double
return *ptr != '.';
} }
bool JsonParser::isDouble() bool JsonParser::isDouble()
@ -50,21 +39,37 @@ bool JsonParser::isDouble()
return *ptr == '.'; return *ptr == '.';
} }
bool JsonParser::isEnd()
{
return *_ptr == 0;
}
bool JsonParser::isLong()
{
char* ptr = _ptr;
// skip all digits
while (isdigit(*ptr))
ptr++;
// same position => 0 digits => not a number
if (ptr == _ptr)
return false;
// stopped on a decimal separator => not a long but a double
return *ptr != '.';
}
bool JsonParser::isNull()
{
return *_ptr == 'n';
}
bool JsonParser::isSpace() bool JsonParser::isSpace()
{ {
return *_ptr == ' ' || *_ptr == '\t' || *_ptr == '\n' || *_ptr == '\r'; return *_ptr == ' ' || *_ptr == '\t' || *_ptr == '\n' || *_ptr == '\r';
} }
bool JsonParser::isComma()
{
return *_ptr == ',';
}
bool JsonParser::isBoolean()
{
return *_ptr == 't' || *_ptr == 'f';
}
void JsonParser::skipOneChar() void JsonParser::skipOneChar()
{ {
_ptr++; _ptr++;
@ -82,14 +87,17 @@ JsonNode* JsonParser::parseAnything()
if (isArrayStart()) if (isArrayStart())
return parseArray(); return parseArray();
if (isLong()) if (isBoolean())
return parseLong(); return parseBoolean();
if (isDouble()) if (isDouble())
return parseDouble(); return parseDouble();
if (isBoolean()) if (isLong())
return parseBoolean(); return parseLong();
if (isNull())
return parseNull();
return 0; return 0;
} }
@ -124,11 +132,15 @@ JsonNode* JsonParser::parseArray()
} }
} }
JsonNode* JsonParser::parseLong() JsonNode *JsonParser::parseBoolean()
{ {
long value = strtol(_ptr, &_ptr, 10); bool value = *_ptr == 't';
return _buffer->createLongNode(value); _ptr += value ? 4 : 5;
// 4 = strlen("true")
// 5 = strlen("false");
return _buffer->createBoolNode(value);
} }
JsonNode *JsonParser::parseDouble() JsonNode *JsonParser::parseDouble()
@ -142,13 +154,17 @@ JsonNode *JsonParser::parseDouble()
return _buffer->createDoubleNode(value, decimals); return _buffer->createDoubleNode(value, decimals);
} }
JsonNode *JsonParser::parseBoolean() JsonNode* JsonParser::parseLong()
{ {
bool value = *_ptr == 't'; long value = strtol(_ptr, &_ptr, 10);
_ptr += value ? 4 : 5; return _buffer->createLongNode(value);
// 4 = strlen("true") }
// 5 = strlen("false");
JsonNode* JsonParser::parseNull()
return _buffer->createBoolNode(value); {
_ptr += 4;
// 4 = strlen("null")
return _buffer->createStringNode(0);
} }

View File

@ -20,21 +20,23 @@ private:
JsonBuffer* _buffer; JsonBuffer* _buffer;
char* _ptr; char* _ptr;
inline bool isEnd();
inline bool isArrayStart(); inline bool isArrayStart();
inline bool isArrayStop(); inline bool isArrayStop();
inline bool isLong();
inline bool isDouble();
inline bool isSpace();
inline bool isComma();
inline bool isBoolean(); inline bool isBoolean();
inline bool isComma();
inline bool isDouble();
inline bool isEnd();
inline bool isLong();
inline bool isNull();
inline bool isSpace();
inline void skipOneChar(); inline void skipOneChar();
inline void skipSpaces(); inline void skipSpaces();
inline JsonNode* parseArray(); inline JsonNode* parseArray();
inline JsonNode* parseLong();
inline JsonNode* parseBoolean(); inline JsonNode* parseBoolean();
inline JsonNode* parseLong();
inline JsonNode* parseNull();
JsonNode *parseDouble(); JsonNode *parseDouble();
}; };

View File

@ -35,22 +35,12 @@ JsonNode *JsonBuffer::createArrayNode()
return node; return node;
} }
JsonNode *JsonBuffer::createObjectNode() JsonNode* JsonBuffer::createBoolNode(bool value)
{ {
JsonNode* node = createNode(); JsonNode* node = createNode();
if (node) if (node)
node->setAsObject(this); node->setAsBoolean(value);
return node;
}
JsonNode *JsonBuffer::createLongNode(long value)
{
JsonNode* node = createNode();
if (node)
node->setAsLong(value);
return node; return node;
} }
@ -65,12 +55,32 @@ JsonNode *JsonBuffer::createDoubleNode(double value, int decimals)
return node; return node;
} }
JsonNode *JsonBuffer::createBoolNode(bool value) JsonNode* JsonBuffer::createLongNode(long value)
{ {
JsonNode* node = createNode(); JsonNode* node = createNode();
if (node) if (node)
node->setAsBoolean(value); node->setAsLong(value);
return node;
}
JsonNode* JsonBuffer::createObjectNode()
{
JsonNode* node = createNode();
if (node)
node->setAsObject(this);
return node;
}
JsonNode* JsonBuffer::createStringNode(const char* value)
{
JsonNode* node = createNode();
if (node)
node->setAsString(value);
return node; return node;
} }

View File

@ -35,10 +35,10 @@ private:
JsonNode* createNode(); JsonNode* createNode();
JsonNode* createArrayNode(); JsonNode* createArrayNode();
JsonNode* createObjectNode();
JsonNode* createLongNode(long value);
JsonNode *createDoubleNode(double value, int decimals);
JsonNode* createBoolNode(bool value); JsonNode* createBoolNode(bool value);
JsonNode* createDoubleNode(double value, int decimals);
JsonNode* createLongNode(long value);
JsonNode* createObjectNode();
JsonNode* createStringNode(const char* value);
}; };

View File

@ -136,3 +136,15 @@ TEST_F(JsonArray_Parser_Tests, TwoBooleans)
firstElementMustBe(true); firstElementMustBe(true);
secondElementMustBe(false); secondElementMustBe(false);
} }
TEST_F(JsonArray_Parser_Tests, TwoNulls)
{
const char* const nullCharPtr = 0;
whenInputIs("[null,null]");
parseMustSucceed();
sizeMustBe(2);
firstElementMustBe(nullCharPtr);
secondElementMustBe(nullCharPtr);
}