JSON5 Parser
A lossless JSON5 tokenizer and parser for Node.js that maintains indentation, spacing, and comments.
📦 Releases
·
🐞 Report Bug
·
✨ Request Feature
This library provides an API for working with JSON and JSON5 documents while preserving their original structure and formatting. Unlike traditional JSON parsers that use an Abstract Syntax Tree (AST) and discard formatting, this library leverages a Concrete Syntax Tree (CST) to retain every detail—comments, indentation, and whitespace.
Ideal for editing configuration files (e.g., package.json, tsconfig.json) or any user-generated JSON5 content, this library ensures that formatting remains intact throughout modifications.
- Preserve formatting – Read, modify, and write JSON5 files without losing comments, indentation, or spacing.
- Style learning – Automatically applies the document's existing formatting style to new entries.
- Reformatting – Customize output formatting with flexible options.
- Type Safety – Fully typed API for working with JSON5 documents.
Install via NPM:
npm install @croct/json5-parserThe library provides a simple API for parsing, manipulating, and serializing JSON5 documents.
Usually, you don't need to interact with the lexer directly. However, you can use it to tokenize a JSON5 document:
import {JsonLexer} from '@croct/json5-parser';
const tokens = JsonLexer.tokenize(
`{
// Comment
"name": "John Doe",
"age": 42,
}`
);To parse a JSON5 document:
import {JsonParser} from '@croct/json5-parser';
const node = JsonParser.parse(
`{
// Comment
"name": "John Doe",
"age": 42,
}`
);Optionally, specify the expected root node type to narrow down the result:
import {JsonParser, JsonObjectNode} from '@croct/json5-parser';
const node = JsonParser.parse(
`{
// Comment
"name": "John Doe",
"age": 42,
}`,
JsonObjectNode
);Modify values while preserving formatting:
// Get the value of a property
const name = node.get('name').toJSON();
// Update a property
node.set('age', 43);
console.log(node.toString());New entries adopt the document's existing style:
node.set('country', 'USA');
console.log(node.toString());Output:
{
// Comment
"name": "John Doe",
"age": 43,
"country": "USA",
}Formatting is applied at a block level, handling different styles within the same document:
{
"name": "My Project",
"version": "1.0.0",
"keywords": ["json5", "parser"],
}Adding an array entry keeps the existing format:
node.set('stack', ['react', 'typescript']);Output:
{
"name": "My Project",
"version": "1.0.0",
"keywords": ["json5", "parser"],
"stack": ["react", "typescript"],
}To reset formatting and apply a new style:
node.reset();
console.log(node.toString({/* formatting options */}));Output:
{
"name": "My Project",
"version": "1.0.0",
"keywords": [
"json5",
"parser"
],
"stack": [
"react",
"typescript"
]
}Refer to the Formatting section for details on available options.
To update the document while preserving formatting, use the update method:
node.update({
...node.toJSON(),
"version": "2.0.0",
});The update method reconciles the new content with the existing document, preserving comments, indentation, and spacing.
For single updates, prefer the set method for better performance:
node.set('version', '2.0.0');To merge two documents while preserving comments and formatting from both, use the merge method:
const destinationCode = `{
// Destination pre-foo comment
"foo": "value",
// Destination post-foo comment
"baz": [1, 2, 3]
}
`;
const sourceCode = `{
/* Source pre-bar comment */
"bar": 123, /* Inline comment */
/* Source post-bar comment */
"baz": true /* Another inline comment */
}
`;
const source = JsonParser.parse(sourceCode, JsonObjectNode);
const destination = JsonParser.parse(destinationCode, JsonObjectNode);
destination.merge(source)
console.log(destination.toString());Output:
{
// Destination pre-foo comment
"foo": "value",
/* Source pre-bar comment */
"bar": 123, /* Inline comment */
/* Source post-bar comment */
"baz": true /* Another inline comment */
}Here’s a drop-in README section that documents those options clearly, with concise tables and runnable examples.
When you call toString(options) you can control how the output is formatted.
If an option is omitted, the formatter learns from the current document and keeps its style. If no style can be inferred,
the formatter falls back to a compact style with no extra spaces or indentation.
| Option | Type/Values | Description |
|---|---|---|
indentationLevel |
number |
Base indentation level for the document. |
indentationCharacter |
'space'|'tab' |
Character used for indentation. |
string.quote |
'single'|'double' |
Quotation style for string values. |
property.quote |
'single'|'double' |
Quotation style for property keys. |
property.unquoted |
boolean |
Allow unquoted property keys when valid. |
array.indentationSize |
number |
Indentation size for array entries. |
array.trailingIndentation |
boolean |
Indent closing bracket on a new line. |
array.leadingIndentation |
boolean |
Indent opening bracket on a new line. |
array.entryIndentation |
boolean |
Indent each array entry. |
array.trailingComma |
boolean |
Append a trailing comma after the last entry. |
array.commaSpacing |
boolean |
Add space after commas in arrays. |
array.colonSpacing |
boolean |
Add space after colons in arrays (for objects inline). |
object.indentationSize |
number |
Indentation size for object properties. |
object.trailingIndentation |
boolean |
Indent closing brace on a new line. |
object.leadingIndentation |
boolean |
Indent opening brace on a new line. |
object.entryIndentation |
boolean |
Indent each object property. |
object.trailingComma |
boolean |
Append a trailing comma after the last property. |
object.commaSpacing |
boolean |
Add space after commas in objects. |
object.colonSpacing |
boolean |
Add space after colons in objects. |
Here are some common formatting styles you can achieve by combining different options.
node.toString({
array: {
leadingIndentation: false, // keep arrays inline
trailingIndentation: false,
trailingComma: false,
commaSpacing: true,
},
object: {
entryIndentation: true, // multi-line objects
leadingIndentation: true,
trailingIndentation: true,
indentationSize: 2,
trailingComma: true,
commaSpacing: true,
colonSpacing: true,
},
});Output style:
{
"name": "My Project",
"keywords": ["json5", "parser"], // array kept inline
"deps": {
"typescript": "^5.6.0", // multi-line object
"vite": "^5.4.0",
}
}node.toString({
indentationCharacter: 'tab',
string: {
quote: 'single'
},
property: {
quote: 'single',
unquoted: true
},
object: {
leadingIndentation: true,
entryIndentation: true,
trailingIndentation: true,
trailingComma: false,
commaSpacing: true,
colonSpacing: true,
},
});Output style:
{
foo: 'bar',
items: [
'a',
'b'
]
}node.toString();Output style:
{"a":1,"b":[1,2,3]}Contributions are welcome!
- Report issues on the issue tracker.
- For major changes, open an issue first to discuss.
- Ensure test coverage is updated accordingly.
Install dependencies:
npm installRun tests:
npm run testLint code to check for style issues:
npm run lint