Compare commits

...

8 Commits

Author SHA1 Message Date
Pavel Shevaev d56e6dceea Improving line error reporter
Publish PHP Package / docker (push) Successful in 7s Details
2025-02-28 01:15:44 +03:00
Pavel Shevaev 1a24065c72 Fixing minor bugs in parser error line reporter
Publish PHP Package / docker (push) Failing after 3s Details
2025-02-26 09:52:24 +03:00
Pavel Shevaev 4d080bacd8 Better parsing error reporting
Publish PHP Package / docker (push) Successful in 6s Details
2025-02-25 20:03:53 +03:00
Maxim Kuleshov dd1c8ad1ba v1.0.8
Publish PHP Package / docker (push) Successful in 7s Details
2024-03-21 16:15:11 +03:00
Maxim Kuleshov 8221b4c151 Merge pull request 'Parse number as floating-point if it doesn't fit in integer; try hard to use integer when possible though' (#1) from feature/ints_overflow into master
Reviewed-on: #1
2024-03-21 16:04:16 +03:00
Maxim Kuleshov ce8fe5c99b Parse number as floating-point if it doesn't fit in integer; try hard to use integer when possible though 2024-03-21 15:54:50 +03:00
Pavel Shevaev dde7df8320 Добавить CHANGELOG.md 2024-03-14 17:39:02 +03:00
Pavel Shevaev 1f6c89684f Обновить README.md 2024-03-14 17:38:02 +03:00
3 changed files with 116 additions and 22 deletions

8
CHANGELOG.md Normal file
View File

@ -0,0 +1,8 @@
## v1.0.8
- Parse number as floating-point if it doesn't fit in integer; trying hard to
use integer when possible though to not lose precision
## v1.0.7
- Removing slashes which escape quotes inside strings

View File

@ -0,0 +1,63 @@
# PHP JZON #
This is a small PHP library for parsing JZON documents.
## What is JZON? ##
JZON is a *superset* of JSON which is designed to be actively edited by *humans*.
PHP JZON features:
* no need to worry about trailing commas
* one line comments starting with #
* simple array keys don't need to be quoted with ""
* parsing errors are shown in a friendly manner
### Quick Example ###
Say, you have the following JZON file example.json:
```
["foo",
"bar",
{meaning_of_life: 42,
#NOTE: no comma ahead
# and yes comments are supported!
hey: "bar"
"thatscool": 1
}
]
```
Now you can parse it this way:
```
<?php
$str = file_get_contents('example.json');
var_dump(jzon_parse($str));
```
And it should output something as follows:
```
array(3) {
[0]=>
string(3) "foo"
[1]=>
string(3) "bar"
[2]=>
array(3) {
["thatscool"]=>
int(1)
["meaning_of_life"]=>
int(42)
["hey"]=>
string(3) "bar"
}
}
```

View File

@ -6,29 +6,44 @@ if(!defined('JZON_VERSION'))
define('JZON_EXT_VERSION', phpversion('jzon'));
}
function jzon_show_position($p, $in, $context_chars)
function jzon_show_position(int $p, string $in, int $context_lines = 5) : string
{
$pre = str_replace("\n", '', jzon_get_past_input($p, $in, $context_chars));
$c = str_repeat('-', max(0, strlen($pre) - 1));
$out = array();
$lines = explode("\n", $in);
foreach($lines as $line_idx => $line)
{
//normalizing tabs by converting them to spaces
$out[] = str_replace("\t", " ", $line);
$left = $p - (strlen($line) + 1/*taking into account \n*/);
if($left <= 0)
{
$arrow = '';
//let's take into account tabs
for($i=0;$i<$p;++$i)
$arrow .= ($line[$i] === "\t" ? '----' : '-');
$arrow .= '^';
return $pre . str_replace("\n", '', jzon_get_upcoming_input($p, $in, $context_chars)) . "\n" . $c . "^";
}
$out[] = $arrow;
break;
}
$p = $left;
}
if(count($out) > $context_lines)
$out = array_slice($out, -$context_lines);
function jzon_get_past_input($c, $in, $context_chars)
{
$past = substr($in, 0, $c+1);
return (strlen($past) > $context_chars ? '...' : '') . substr($past, -$context_chars);
}
//let's add line numbers
end($out);
//let's find out the maximum leading zeros for line numbers
$fmt_num = (int)round(log10(key($out)))+1;
foreach($out as $idx => $line)
$out[$idx] = sprintf('%0'.$fmt_num.'d', ($idx+1)) . ' ' . $line;
function jzon_get_upcoming_input($c, $in, $context_chars)
{
$next = substr($in, $c+1);
return substr($next, 0, $context_chars) . (strlen($next) > $context_chars ? '...' : '');
return implode("\n", $out);
}
class jzonParser
{
const ERR_CONTEXT_CHARS = 200;
const ERR_CONTEXT_LINES = 5;
private $in;
private $len;
@ -86,9 +101,9 @@ class jzonParser
private function _error($error)
{
if($this->c < $this->len)
throw new Exception("Parse error: $error\n" . jzon_show_position($this->c, $this->in, self::ERR_CONTEXT_CHARS));
throw new Exception("Parse error: $error\n" . jzon_show_position($this->c, $this->in, self::ERR_CONTEXT_LINES));
else
throw new Exception("Parse error: $error\n" . jzon_show_position($this->len-1, $this->in, self::ERR_CONTEXT_CHARS));
throw new Exception("Parse error: $error\n" . jzon_show_position($this->len-1, $this->in, self::ERR_CONTEXT_LINES));
}
private function skip_whitespace()
@ -293,10 +308,16 @@ class jzonParser
}
$str_num = substr($this->in, $start, $this->c - $start);
$fval = floatval($str_num);
$ival = intval($str_num);
if($is_float)
$out = floatval($str_num);
$out = $fval;
else if(($ival < PHP_INT_MAX && $ival > PHP_INT_MIN) || strval($ival) === $str_num)
$out = $ival;
else
$out = intval($str_num);
$out = $fval;
}
private function parse_true(&$out)
@ -331,13 +352,13 @@ class jzonParser
else
$this->_error("'null' expected");
}
}
function jzon_parse($str)
function jzon_parse($str, ?int &$parser = null) : array
{
//NOTE: using super fast built-in implementation and making a gracefull fallback to a slower and
// more relaxed implementation
$parser = 1;
$res = json_decode($str, true);
if(is_array($res))
return $res;
@ -345,13 +366,15 @@ function jzon_parse($str)
//NOTE: only allowing extension implementation if versions match
if(defined('JZON_EXT_VERSION') && JZON_EXT_VERSION === JZON_VERSION)
{
$parser = 2;
list($ok, $err, $err_pos, $res) = jzon_parse_c($str);
if(!$ok)
throw new Exception($err . "\n" . jzon_show_position($err_pos, $str, jzonParser::ERR_CONTEXT_CHARS));
throw new Exception($err . "\n" . jzon_show_position($err_pos, $str, jzonParser::ERR_CONTEXT_LINES));
return $res;
}
else
{
$parser = 3;
$p = new jzonParser($str);
$res = $p->parse();
return $res;