163 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
			
		
		
	
	
			163 lines
		
	
	
		
			5.6 KiB
		
	
	
	
		
			PHP
		
	
	
<?php
 | 
						|
 | 
						|
/*
 | 
						|
 * This file is part of the Symfony package.
 | 
						|
 *
 | 
						|
 * (c) Fabien Potencier <fabien@symfony.com>
 | 
						|
 *
 | 
						|
 * For the full copyright and license information, please view the LICENSE
 | 
						|
 * file that was distributed with this source code.
 | 
						|
 */
 | 
						|
 | 
						|
namespace Symfony\Component\HttpFoundation;
 | 
						|
 | 
						|
/**
 | 
						|
 * StreamedJsonResponse represents a streamed HTTP response for JSON.
 | 
						|
 *
 | 
						|
 * A StreamedJsonResponse uses a structure and generics to create an
 | 
						|
 * efficient resource-saving JSON response.
 | 
						|
 *
 | 
						|
 * It is recommended to use flush() function after a specific number of items to directly stream the data.
 | 
						|
 *
 | 
						|
 * @see flush()
 | 
						|
 *
 | 
						|
 * @author Alexander Schranz <alexander@sulu.io>
 | 
						|
 *
 | 
						|
 * Example usage:
 | 
						|
 *
 | 
						|
 *     function loadArticles(): \Generator
 | 
						|
 *         // some streamed loading
 | 
						|
 *         yield ['title' => 'Article 1'];
 | 
						|
 *         yield ['title' => 'Article 2'];
 | 
						|
 *         yield ['title' => 'Article 3'];
 | 
						|
 *         // recommended to use flush() after every specific number of items
 | 
						|
 *     }),
 | 
						|
 *
 | 
						|
 *     $response = new StreamedJsonResponse(
 | 
						|
 *         // json structure with generators in which will be streamed
 | 
						|
 *         [
 | 
						|
 *             '_embedded' => [
 | 
						|
 *                 'articles' => loadArticles(), // any generator which you want to stream as list of data
 | 
						|
 *             ],
 | 
						|
 *         ],
 | 
						|
 *     );
 | 
						|
 */
 | 
						|
class StreamedJsonResponse extends StreamedResponse
 | 
						|
{
 | 
						|
    private const PLACEHOLDER = '__symfony_json__';
 | 
						|
 | 
						|
    /**
 | 
						|
     * @param mixed[]                        $data            JSON Data containing PHP generators which will be streamed as list of data or a Generator
 | 
						|
     * @param int                            $status          The HTTP status code (200 "OK" by default)
 | 
						|
     * @param array<string, string|string[]> $headers         An array of HTTP headers
 | 
						|
     * @param int                            $encodingOptions Flags for the json_encode() function
 | 
						|
     */
 | 
						|
    public function __construct(
 | 
						|
        private readonly iterable $data,
 | 
						|
        int $status = 200,
 | 
						|
        array $headers = [],
 | 
						|
        private int $encodingOptions = JsonResponse::DEFAULT_ENCODING_OPTIONS,
 | 
						|
    ) {
 | 
						|
        parent::__construct($this->stream(...), $status, $headers);
 | 
						|
 | 
						|
        if (!$this->headers->get('Content-Type')) {
 | 
						|
            $this->headers->set('Content-Type', 'application/json');
 | 
						|
        }
 | 
						|
    }
 | 
						|
 | 
						|
    private function stream(): void
 | 
						|
    {
 | 
						|
        $jsonEncodingOptions = \JSON_THROW_ON_ERROR | $this->encodingOptions;
 | 
						|
        $keyEncodingOptions = $jsonEncodingOptions & ~\JSON_NUMERIC_CHECK;
 | 
						|
 | 
						|
        $this->streamData($this->data, $jsonEncodingOptions, $keyEncodingOptions);
 | 
						|
    }
 | 
						|
 | 
						|
    private function streamData(mixed $data, int $jsonEncodingOptions, int $keyEncodingOptions): void
 | 
						|
    {
 | 
						|
        if (\is_array($data)) {
 | 
						|
            $this->streamArray($data, $jsonEncodingOptions, $keyEncodingOptions);
 | 
						|
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        if (is_iterable($data) && !$data instanceof \JsonSerializable) {
 | 
						|
            $this->streamIterable($data, $jsonEncodingOptions, $keyEncodingOptions);
 | 
						|
 | 
						|
            return;
 | 
						|
        }
 | 
						|
 | 
						|
        echo json_encode($data, $jsonEncodingOptions);
 | 
						|
    }
 | 
						|
 | 
						|
    private function streamArray(array $data, int $jsonEncodingOptions, int $keyEncodingOptions): void
 | 
						|
    {
 | 
						|
        $generators = [];
 | 
						|
 | 
						|
        array_walk_recursive($data, function (&$item, $key) use (&$generators) {
 | 
						|
            if (self::PLACEHOLDER === $key) {
 | 
						|
                // if the placeholder is already in the structure it should be replaced with a new one that explode
 | 
						|
                // works like expected for the structure
 | 
						|
                $generators[] = $key;
 | 
						|
            }
 | 
						|
 | 
						|
            // generators should be used but for better DX all kind of Traversable and objects are supported
 | 
						|
            if (\is_object($item)) {
 | 
						|
                $generators[] = $item;
 | 
						|
                $item = self::PLACEHOLDER;
 | 
						|
            } elseif (self::PLACEHOLDER === $item) {
 | 
						|
                // if the placeholder is already in the structure it should be replaced with a new one that explode
 | 
						|
                // works like expected for the structure
 | 
						|
                $generators[] = $item;
 | 
						|
            }
 | 
						|
        });
 | 
						|
 | 
						|
        $jsonParts = explode('"'.self::PLACEHOLDER.'"', json_encode($data, $jsonEncodingOptions));
 | 
						|
 | 
						|
        foreach ($generators as $index => $generator) {
 | 
						|
            // send first and between parts of the structure
 | 
						|
            echo $jsonParts[$index];
 | 
						|
 | 
						|
            $this->streamData($generator, $jsonEncodingOptions, $keyEncodingOptions);
 | 
						|
        }
 | 
						|
 | 
						|
        // send last part of the structure
 | 
						|
        echo $jsonParts[array_key_last($jsonParts)];
 | 
						|
    }
 | 
						|
 | 
						|
    private function streamIterable(iterable $iterable, int $jsonEncodingOptions, int $keyEncodingOptions): void
 | 
						|
    {
 | 
						|
        $isFirstItem = true;
 | 
						|
        $startTag = '[';
 | 
						|
 | 
						|
        foreach ($iterable as $key => $item) {
 | 
						|
            if ($isFirstItem) {
 | 
						|
                $isFirstItem = false;
 | 
						|
                // depending on the first elements key the generator is detected as a list or map
 | 
						|
                // we can not check for a whole list or map because that would hurt the performance
 | 
						|
                // of the streamed response which is the main goal of this response class
 | 
						|
                if (0 !== $key) {
 | 
						|
                    $startTag = '{';
 | 
						|
                }
 | 
						|
 | 
						|
                echo $startTag;
 | 
						|
            } else {
 | 
						|
                // if not first element of the generic, a separator is required between the elements
 | 
						|
                echo ',';
 | 
						|
            }
 | 
						|
 | 
						|
            if ('{' === $startTag) {
 | 
						|
                echo json_encode((string) $key, $keyEncodingOptions).':';
 | 
						|
            }
 | 
						|
 | 
						|
            $this->streamData($item, $jsonEncodingOptions, $keyEncodingOptions);
 | 
						|
        }
 | 
						|
 | 
						|
        if ($isFirstItem) { // indicates that the generator was empty
 | 
						|
            echo '[';
 | 
						|
        }
 | 
						|
 | 
						|
        echo '[' === $startTag ? ']' : '}';
 | 
						|
    }
 | 
						|
}
 |