202 lines
5.0 KiB
C++
202 lines
5.0 KiB
C++
/*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*
|
|
* Copyright 2017 Danny Robson <danny@nerdcruft.net>
|
|
*/
|
|
|
|
#include "./jsonish.hpp"
|
|
|
|
#include "./base.hpp"
|
|
#include "../event.hpp"
|
|
#include "../except.hpp"
|
|
#include "../../debug.hpp"
|
|
|
|
using util::json2::personality::jsonish;
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
const char*
|
|
jsonish::consume_whitespace (const char *first, const char *last) noexcept
|
|
{
|
|
auto cursor = base<jsonish>::consume_whitespace (first, last);
|
|
|
|
// consume a comment
|
|
if (cursor != last && *cursor == '#') {
|
|
while (cursor != last && *cursor != '\n')
|
|
++cursor;
|
|
|
|
return consume_whitespace (cursor, last);
|
|
}
|
|
|
|
return cursor;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
// format is:
|
|
// int: '0x' hex+ | '0' oct+ | '0b' bit+
|
|
//
|
|
// float: significand exp?
|
|
// significand: digit+ ('.' digit*)?
|
|
// exp: [eE] sign? digit+
|
|
//
|
|
// number: [+-] (int | float)
|
|
const char*
|
|
jsonish::parse_number (const std::function<callback_t> &cb,
|
|
const char *first,
|
|
const char *last)
|
|
{
|
|
auto cursor = first;
|
|
|
|
if (cursor != last && (*cursor == '+' || *cursor == '-'))
|
|
++cursor;
|
|
|
|
if (cursor != last && *cursor == '0') {
|
|
++cursor;
|
|
|
|
if (cursor == last)
|
|
throw parse_error {cursor};
|
|
|
|
char max = '9';
|
|
switch (*cursor) {
|
|
case 'x': {
|
|
// parse the hex integer here because we can simplify the
|
|
// remaining cases somewhat if we don't need to care about the
|
|
// multiple ranges of valid digits.
|
|
++cursor;
|
|
|
|
auto digit_start = cursor;
|
|
while (cursor != last && ((('0' <= *cursor) && (*cursor <= '9')) ||
|
|
(('a' <= *cursor) && (*cursor <= 'f')) ||
|
|
(('A' <= *cursor) && (*cursor <= 'F'))))
|
|
++cursor;
|
|
if (digit_start == cursor)
|
|
throw parse_error {cursor};
|
|
|
|
cb ({first, cursor});
|
|
return cursor;
|
|
};
|
|
|
|
case 'b': max = '1'; break;
|
|
case '0'...'7': max = '7'; break;
|
|
|
|
case '.':
|
|
goto frac;
|
|
}
|
|
|
|
auto digit_start = ++cursor;
|
|
while (cursor != last && '0' <= *cursor && *cursor <= max)
|
|
++cursor;
|
|
if (digit_start == cursor)
|
|
throw parse_error {cursor};
|
|
|
|
cb ({first, cursor});
|
|
return cursor;
|
|
}
|
|
|
|
while (cursor != last && '0' <= *cursor && *cursor <= '9')
|
|
++cursor;
|
|
if (cursor == last)
|
|
goto done;
|
|
|
|
if (*cursor != '.')
|
|
goto exp;
|
|
|
|
frac:
|
|
++cursor;
|
|
while (cursor != last && *cursor >= '0' && *cursor <= '9')
|
|
++cursor;
|
|
if (cursor == last)
|
|
goto done;
|
|
|
|
exp:
|
|
if (cursor != last && (*cursor == 'e' || *cursor == 'E')) {
|
|
++cursor;
|
|
|
|
if (cursor != last && (*cursor == '+' || *cursor == '-'))
|
|
++cursor;
|
|
|
|
auto digit_start = cursor;
|
|
while (cursor != last && '0' <= *cursor && *cursor <= '9')
|
|
++cursor;
|
|
if (digit_start == cursor)
|
|
throw parse_error {cursor};
|
|
}
|
|
|
|
if (first == cursor)
|
|
throw parse_error {cursor};
|
|
|
|
done:
|
|
cb ({first, cursor});
|
|
return cursor;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
const char*
|
|
jsonish::parse_key (const std::function<callback_t> &cb,
|
|
const char *first,
|
|
const char *last)
|
|
{
|
|
auto cursor = first;
|
|
if (cursor == last)
|
|
throw parse_error {cursor};
|
|
|
|
// must start with alpha or underscore
|
|
switch (*cursor) {
|
|
case 'a'...'z':
|
|
case 'A'...'Z':
|
|
case '_':
|
|
++cursor;
|
|
break;
|
|
|
|
default:
|
|
throw parse_error {cursor};
|
|
}
|
|
|
|
|
|
while (cursor != last) {
|
|
switch (*cursor) {
|
|
case 'a'...'z':
|
|
case 'A'...'Z':
|
|
case '_':
|
|
case '0'...'9':
|
|
++cursor;
|
|
break;
|
|
|
|
default:
|
|
cb ({first, cursor});
|
|
return cursor;
|
|
}
|
|
}
|
|
|
|
cb ({first, cursor});
|
|
return cursor;
|
|
}
|
|
|
|
|
|
///////////////////////////////////////////////////////////////////////////////
|
|
const char*
|
|
jsonish::parse_string (const std::function<callback_t> &cb,
|
|
const char *first,
|
|
const char *last)
|
|
{
|
|
if (first == last)
|
|
throw parse_error {first};
|
|
|
|
if (*first == '"')
|
|
return base<jsonish>::parse_string (cb, first, last);
|
|
else
|
|
return parse_key (cb, first, last);
|
|
}
|