https://github.com/cerbero90/lazy-json
๐ผ Framework-agnostic package to load JSON of any dimension and from any source into Laravel lazy collections recursively.
https://github.com/cerbero90/lazy-json
json laravel lexer parser stream
Last synced: 2 months ago
JSON representation
๐ผ Framework-agnostic package to load JSON of any dimension and from any source into Laravel lazy collections recursively.
- Host: GitHub
- URL: https://github.com/cerbero90/lazy-json
- Owner: cerbero90
- License: mit
- Created: 2021-05-02T03:27:15.000Z (over 4 years ago)
- Default Branch: develop
- Last Pushed: 2023-11-29T23:53:18.000Z (almost 2 years ago)
- Last Synced: 2024-06-18T21:43:33.970Z (over 1 year ago)
- Topics: json, laravel, lexer, parser, stream
- Language: PHP
- Homepage:
- Size: 61.5 KB
- Stars: 233
- Watchers: 3
- Forks: 3
- Open Issues: 0
-
Metadata Files:
- Readme: README.md
- Changelog: CHANGELOG.md
- Contributing: CONTRIBUTING.md
- Funding: .github/FUNDING.yml
- License: LICENSE.md
- Code of conduct: CODE_OF_CONDUCT.md
Awesome Lists containing this project
README
# ๐ผ Lazy JSON
[![Author][ico-author]][link-author]
[![PHP Version][ico-php]][link-php]
[![Build Status][ico-actions]][link-actions]
[![Coverage Status][ico-scrutinizer]][link-scrutinizer]
[![Quality Score][ico-code-quality]][link-code-quality]
[![PHPStan Level][ico-phpstan]][link-phpstan]
[![Latest Version][ico-version]][link-packagist]
[![Software License][ico-license]](LICENSE.md)
[![PER][ico-per]][link-per]
[![Total Downloads][ico-downloads]][link-downloads]```php
LazyCollection::fromJson($source, 'data.*.users.*')
->map($this->mapToUser(...))
->filter($this->filterUser(...))
->values()
->chunk(1_000)
->each($this->storeUsersChunk(...));
```Framework-agnostic package to load JSON of any size and from any source into [Laravel lazy collections](https://laravel.com/docs/collections#lazy-collections).
Lazy JSON recursively turns any JSON array or object into a lazy collection, consuming only a few KB of memory while parsing JSON of any dimension.
It optionally allows to extract only some sub-trees, instead of the whole JSON, with an easy dot-notation syntax.
Under the hood, [๐งฉ JSON Parser](https://github.com/cerbero90/json-parser) is used to parse JSONs and extract sub-trees.
Need to lazy load items from paginated JSON APIs? Consider using [๐ผ Lazy JSON Pages](https://github.com/cerbero90/lazy-json-pages) instead.
## ๐ฆ Install
Via Composer:
``` bash
composer require cerbero/lazy-json
```## ๐ฎ Usage
* [๐ฃ Basics](#-basics)
* [๐ง Sources](#-sources)
* [๐ฏ Dots](#-dots)### ๐ฃ Basics
Depending on our coding style, we can call Lazy JSON in 3 different ways:
```php
use Cerbero\LazyJson\LazyJson;
use Illuminate\Support\LazyCollection;use function Cerbero\LazyJson\lazyJson;
// auto-registered lazy collection macro
$lazyCollection = LazyCollection::fromJson($source);// static method
$lazyCollection = LazyJson::from($source);// namespaced helper
$lazyCollection = lazyJson($source);
```The variable `$source` in our examples represents any [JSON source](#-sources). Once we define the source, we can chain any method of the [Laravel lazy collection](https://laravel.com/docs/collections#lazy-collections) to process the JSON in a memory-efficient way:
```php
LazyCollection::fromJson($source)
->values()
->map(/* ... */)
->where(/* ... */)
->each(/* ... */);
```### ๐ง Sources
A JSON source is any data point that provides a JSON. A wide range of sources are supported by default:
- **strings**, e.g. `{"foo":"bar"}`
- **iterables**, i.e. arrays or instances of `Traversable`
- **file paths**, e.g. `/path/to/large.json`
- **resources**, e.g. streams
- **API endpoint URLs**, e.g. `https://endpoint.json` or any instance of `Psr\Http\Message\UriInterface`
- **PSR-7 requests**, i.e. any instance of `Psr\Http\Message\RequestInterface`
- **PSR-7 messages**, i.e. any instance of `Psr\Http\Message\MessageInterface`
- **PSR-7 streams**, i.e. any instance of `Psr\Http\Message\StreamInterface`
- **Laravel HTTP client requests**, i.e. any instance of `Illuminate\Http\Client\Request`
- **Laravel HTTP client responses**, i.e. any instance of `Illuminate\Http\Client\Response`
- **user-defined sources**, i.e. any instance of `Cerbero\JsonParser\Sources\Source`For more information about JSON sources, please consult the [๐งฉ JSON Parser documentation](https://github.com/cerbero90/json-parser).
### ๐ฏ Dots
If we only need a sub-tree of a large JSON, we can use a simple dot-notation syntax to extract the desired path (or **dot**).
Consider [this JSON](https://randomuser.me/api/1.4?seed=json-parser&results=5) for example. To extract only the cities and avoid parsing the rest of the JSON, we can set the `results.*.location.city` dot:
```php
$source = 'https://randomuser.me/api/1.4?seed=json-parser&results=5';$dot = 'results.*.location.city';
LazyCollection::fromJson($source, $dot)->each(function (string $value, string $key) {
// 1st iteration: $key === 'city', $value === 'Sontra'
// 2nd iteration: $key === 'city', $value === 'San Rafael Tlanalapan'
// 3rd iteration: $key === 'city', $value === 'ฺฏุฑฺฏุงู'
// ...
});
```The dot-notation syntax is very simple and it can include any of the following 4 elements:
- a key of a JSON array, e.g. `0`
- a key of a JSON object, e.g. `results`
- a dot to indicate the nesting level within a JSON, e.g. `results.0`
- an asterisk to indicate all items within an array, e.g. `results.*`If we need to extract several sub-trees, Lazy JSON supports multiple dots:
```php
$dots = ['results.*.gender', 'results.*.email'];LazyCollection::fromJson($source, $dots)->each(function (string $value, string $key) {
// 1st iteration: $key === 'gender', $value === 'female'
// 2nd iteration: $key === 'email', $value === 'sara.meder@example.com'
// 3rd iteration: $key === 'gender', $value === 'female'
// 4th iteration: $key === 'email', $value === 'andrea.roque@example.com'
// ...
});
```## ๐ Change log
Please see [CHANGELOG](CHANGELOG.md) for more information on what has changed recently.
## ๐งช Testing
``` bash
composer test
```## ๐ Contributing
Please see [CONTRIBUTING](CONTRIBUTING.md) and [CODE_OF_CONDUCT](CODE_OF_CONDUCT.md) for details.
## ๐งฏ Security
If you discover any security related issues, please email andrea.marco.sartori@gmail.com instead of using the issue tracker.
## ๐ Credits
- [Andrea Marco Sartori][link-author]
- [All Contributors][link-contributors]## โ๏ธ License
The MIT License (MIT). Please see [License File](LICENSE.md) for more information.
[ico-author]: https://img.shields.io/static/v1?label=author&message=cerbero90&color=50ABF1&logo=twitter&style=flat-square
[ico-php]: https://img.shields.io/packagist/php-v/cerbero/lazy-json?color=%234F5B93&logo=php&style=flat-square
[ico-version]: https://img.shields.io/packagist/v/cerbero/lazy-json.svg?label=version&style=flat-square
[ico-actions]: https://img.shields.io/github/actions/workflow/status/cerbero90/json-parser/build.yml?branch=master&style=flat-square&logo=github
[ico-license]: https://img.shields.io/badge/license-MIT-brightgreen.svg?style=flat-square
[ico-per]: https://img.shields.io/static/v1?label=compliance&message=PER&color=blue&style=flat-square
[ico-scrutinizer]: https://img.shields.io/scrutinizer/coverage/g/cerbero90/lazy-json.svg?style=flat-square&logo=scrutinizer
[ico-code-quality]: https://img.shields.io/scrutinizer/g/cerbero90/lazy-json.svg?style=flat-square&logo=scrutinizer
[ico-phpstan]: https://img.shields.io/badge/level-max-success?style=flat-square&logo=
[ico-downloads]: https://img.shields.io/packagist/dt/cerbero/lazy-json.svg?style=flat-square[link-author]: https://twitter.com/cerbero90
[link-php]: https://www.php.net
[link-packagist]: https://packagist.org/packages/cerbero/lazy-json
[link-actions]: https://github.com/cerbero90/lazy-json/actions?query=workflow%3Abuild
[link-per]: https://www.php-fig.org/per/coding-style/
[link-scrutinizer]: https://scrutinizer-ci.com/g/cerbero90/lazy-json/code-structure
[link-code-quality]: https://scrutinizer-ci.com/g/cerbero90/lazy-json
[link-downloads]: https://packagist.org/packages/cerbero/lazy-json
[link-phpstan]: https://phpstan.org/
[link-contributors]: ../../contributors