1
0
Fork 0
composer/tests/Composer/Test/Json/JsonFileTest.php

415 lines
13 KiB
PHP
Raw Permalink Normal View History

2022-02-23 15:58:18 +00:00
<?php declare(strict_types=1);
2011-10-31 13:43:41 +00:00
/*
* This file is part of Composer.
*
* (c) Nils Adermann <naderman@naderman.de>
* Jordi Boggiano <j.boggiano@seld.be>
*
* For the full copyright and license information, please view the LICENSE
* file that was distributed with this source code.
*/
namespace Composer\Test\Json;
use Seld\JsonLint\ParsingException;
2011-10-31 13:43:41 +00:00
use Composer\Json\JsonFile;
2021-05-24 09:51:04 +00:00
use Composer\Json\JsonValidationException;
2020-02-07 03:18:45 +00:00
use Composer\Test\TestCase;
2011-10-31 13:43:41 +00:00
class JsonFileTest extends TestCase
2011-10-31 13:43:41 +00:00
{
public function testParseErrorDetectExtraComma(): void
2011-10-31 13:43:41 +00:00
{
$json = '{
"foo": "bar",
}';
$this->expectParseException('Parse error on line 2', $json);
2011-10-31 13:43:41 +00:00
}
public function testParseErrorDetectExtraCommaInArray(): void
{
$json = '{
"foo": [
"bar",
]
}';
$this->expectParseException('Parse error on line 3', $json);
}
public function testParseErrorDetectUnescapedBackslash(): void
{
$json = '{
"fo\o": "bar"
}';
$this->expectParseException('Parse error on line 1', $json);
}
public function testParseErrorSkipsEscapedBackslash(): void
2011-11-05 22:51:35 +00:00
{
$json = '{
"fo\\\\o": "bar"
"a": "b"
}';
$this->expectParseException('Parse error on line 2', $json);
2011-11-05 22:51:35 +00:00
}
public function testParseErrorDetectSingleQuotes(): void
2011-10-31 13:43:41 +00:00
{
2015-11-04 06:17:49 +00:00
if (defined('JSON_PARSER_NOTSTRICT') && version_compare(phpversion('json'), '1.3.9', '<')) {
2015-07-16 07:21:13 +00:00
$this->markTestSkipped('jsonc issue, see https://github.com/remicollet/pecl-json-c/issues/23');
}
2011-10-31 13:43:41 +00:00
$json = '{
\'foo\': "bar"
}';
$this->expectParseException('Parse error on line 1', $json);
2011-10-31 13:43:41 +00:00
}
public function testParseErrorDetectMissingQuotes(): void
2011-10-31 13:43:41 +00:00
{
$json = '{
foo: "bar"
}';
$this->expectParseException('Parse error on line 1', $json);
2011-10-31 13:43:41 +00:00
}
public function testParseErrorDetectArrayAsHash(): void
2011-10-31 13:43:41 +00:00
{
$json = '{
"foo": ["bar": "baz"]
}';
$this->expectParseException('Parse error on line 2', $json);
2011-10-31 13:43:41 +00:00
}
public function testParseErrorDetectMissingComma(): void
2011-10-31 13:43:41 +00:00
{
$json = '{
"foo": "bar"
"bar": "foo"
}';
$this->expectParseException('Parse error on line 2', $json);
}
public function testSchemaValidation(): void
{
self::expectNotToPerformAssertions();
2015-04-29 23:08:45 +00:00
$json = new JsonFile(__DIR__.'/Fixtures/composer.json');
$json->validateSchema();
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 09:51:04 +00:00
}
public function testSchemaValidationError(): void
2021-05-24 09:51:04 +00:00
{
2021-12-10 12:14:04 +00:00
$file = $this->createTempFile();
2021-05-24 09:51:04 +00:00
file_put_contents($file, '{ "name": null }');
$json = new JsonFile($file);
$expectedMessage = sprintf('"%s" does not match the expected JSON schema', $file);
$expectedError = 'name : NULL value found, but a string is required';
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
self::assertContains($expectedError, $e->getErrors());
2021-05-24 09:51:04 +00:00
}
try {
$json->validateSchema(JsonFile::LAX_SCHEMA);
$this->fail('Expected exception to be thrown (lax)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
self::assertContains($expectedError, $e->getErrors());
2021-05-24 09:51:04 +00:00
}
unlink($file);
}
public function testSchemaValidationLaxAdditionalProperties(): void
2021-05-24 09:51:04 +00:00
{
2021-12-10 12:14:04 +00:00
$file = $this->createTempFile();
2021-05-24 09:51:04 +00:00
file_put_contents($file, '{ "name": "vendor/package", "description": "generic description", "foo": "bar" }');
$json = new JsonFile($file);
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals(sprintf('"%s" does not match the expected JSON schema', $file), $e->getMessage());
self::assertEquals(['The property foo is not defined and the definition does not allow additional properties'], $e->getErrors());
2021-05-24 09:51:04 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 09:51:04 +00:00
unlink($file);
}
public function testSchemaValidationLaxRequired(): void
2021-05-24 09:51:04 +00:00
{
2021-12-10 12:14:04 +00:00
$file = $this->createTempFile();
2021-05-24 09:51:04 +00:00
$json = new JsonFile($file);
$expectedMessage = sprintf('"%s" does not match the expected JSON schema', $file);
file_put_contents($file, '{ }');
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
2021-05-24 09:51:04 +00:00
$errors = $e->getErrors();
self::assertContains('name : The property name is required', $errors);
self::assertContains('description : The property description is required', $errors);
2021-05-24 09:51:04 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 09:51:04 +00:00
file_put_contents($file, '{ "name": "vendor/package" }');
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
self::assertEquals(['description : The property description is required'], $e->getErrors());
2021-05-24 09:51:04 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 09:51:04 +00:00
file_put_contents($file, '{ "description": "generic description" }');
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
self::assertEquals(['name : The property name is required'], $e->getErrors());
2021-05-24 09:51:04 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 09:51:04 +00:00
2021-05-24 12:42:23 +00:00
file_put_contents($file, '{ "type": "library" }');
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
2021-05-24 12:42:23 +00:00
$errors = $e->getErrors();
self::assertContains('name : The property name is required', $errors);
self::assertContains('description : The property description is required', $errors);
2021-05-24 12:42:23 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 12:42:23 +00:00
file_put_contents($file, '{ "type": "project" }');
try {
$json->validateSchema();
$this->fail('Expected exception to be thrown (strict)');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
2021-05-24 12:42:23 +00:00
$errors = $e->getErrors();
self::assertContains('name : The property name is required', $errors);
self::assertContains('description : The property description is required', $errors);
2021-05-24 12:42:23 +00:00
}
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 12:42:23 +00:00
file_put_contents($file, '{ "name": "vendor/package", "description": "generic description" }');
$json->validateSchema();
$json->validateSchema(JsonFile::LAX_SCHEMA);
2021-05-24 12:42:23 +00:00
2021-05-24 09:51:04 +00:00
unlink($file);
2011-10-31 13:43:41 +00:00
}
public function testCustomSchemaValidationLax(): void
{
self::expectNotToPerformAssertions();
2021-12-10 12:14:04 +00:00
$file = $this->createTempFile();
file_put_contents($file, '{ "custom": "property", "another custom": "property" }');
2021-12-10 12:14:04 +00:00
$schema = $this->createTempFile();
file_put_contents($schema, '{ "properties": { "custom": { "type": "string" }}}');
$json = new JsonFile($file);
$json->validateSchema(JsonFile::LAX_SCHEMA, $schema);
unlink($file);
unlink($schema);
}
public function testCustomSchemaValidationStrict(): void
{
self::expectNotToPerformAssertions();
2021-12-10 12:14:04 +00:00
$file = $this->createTempFile();
file_put_contents($file, '{ "custom": "property" }');
2021-12-10 12:14:04 +00:00
$schema = $this->createTempFile();
file_put_contents($schema, '{ "properties": { "custom": { "type": "string" }}}');
$json = new JsonFile($file);
$json->validateSchema(JsonFile::STRICT_SCHEMA, $schema);
unlink($file);
unlink($schema);
}
public function testAuthSchemaValidationWithCustomDataSource(): void
{
$json = json_decode('{"github-oauth": "foo"}');
$expectedMessage = sprintf('"COMPOSER_AUTH" does not match the expected JSON schema');
$expectedError = 'github-oauth : String value found, but an object is required';
try {
JsonFile::validateJsonSchema('COMPOSER_AUTH', $json, JsonFile::AUTH_SCHEMA);
$this->fail('Expected exception to be thrown');
} catch (JsonValidationException $e) {
self::assertEquals($expectedMessage, $e->getMessage());
self::assertSame([$expectedError], $e->getErrors());
}
}
public function testParseErrorDetectMissingCommaMultiline(): void
{
$json = '{
"foo": "barbar"
"bar": "foo"
}';
$this->expectParseException('Parse error on line 2', $json);
}
public function testParseErrorDetectMissingColon(): void
{
$json = '{
"foo": "bar",
"bar" "foo"
}';
$this->expectParseException('Parse error on line 3', $json);
}
public function testSimpleJsonString(): void
{
2022-08-17 12:20:07 +00:00
$data = ['name' => 'composer/composer'];
$json = '{
"name": "composer/composer"
}';
self::assertJsonFormat($json, $data);
}
public function testTrailingBackslash(): void
{
2022-08-17 12:20:07 +00:00
$data = ['Metadata\\' => 'src/'];
$json = '{
"Metadata\\\\": "src/"
}';
self::assertJsonFormat($json, $data);
}
public function testFormatEmptyArray(): void
{
2022-08-17 12:20:07 +00:00
$data = ['test' => [], 'test2' => new \stdClass];
$json = '{
"test": [],
"test2": {}
}';
self::assertJsonFormat($json, $data);
}
public function testEscape(): void
{
2022-08-17 12:20:07 +00:00
$data = ["Metadata\\\"" => 'src/'];
$json = '{
"Metadata\\\\\\"": "src/"
}';
self::assertJsonFormat($json, $data);
}
public function testUnicode(): void
{
2022-08-17 12:20:07 +00:00
$data = ["Žluťoučký \" kůň" => "úpěl ďábelské ódy za €"];
$json = '{
"Žluťoučký \" kůň": "úpěl ďábelské ódy za €"
}';
self::assertJsonFormat($json, $data);
}
public function testOnlyUnicode(): void
{
$data = "\\";
self::assertJsonFormat('"\\\\\\/ƌ"', $data, JSON_UNESCAPED_UNICODE);
}
public function testEscapedSlashes(): void
{
$data = "\\/foo";
self::assertJsonFormat('"\\\\\\/foo"', $data, 0);
}
public function testEscapedBackslashes(): void
{
$data = "a\\b";
self::assertJsonFormat('"a\\\\b"', $data, 0);
}
public function testEscapedUnicode(): void
{
$data = "ƌ";
self::assertJsonFormat('"\\u018c"', $data, 0);
}
public function testDoubleEscapedUnicode(): void
{
$jsonFile = new JsonFile('composer.json');
2022-08-17 12:20:07 +00:00
$data = ["Zdjęcia","hjkjhl\\u0119kkjk"];
$encodedData = $jsonFile->encode($data);
2022-08-17 12:20:07 +00:00
$doubleEncodedData = $jsonFile->encode(['t' => $encodedData]);
$decodedData = json_decode($doubleEncodedData, true);
$doubleData = json_decode($decodedData['t'], true);
self::assertEquals($data, $doubleData);
}
public function testPreserveIndentationAfterRead(): void
{
copy(__DIR__.'/Fixtures/tabs.json', __DIR__.'/Fixtures/tabs2.json');
$jsonFile = new JsonFile(__DIR__.'/Fixtures/tabs2.json');
$data = $jsonFile->read();
$jsonFile->write(['foo' => 'baz']);
self::assertSame("{\n\t\"foo\": \"baz\"\n}\n", file_get_contents(__DIR__.'/Fixtures/tabs2.json'));
unlink(__DIR__.'/Fixtures/tabs2.json');
}
public function testOverwritesIndentationByDefault(): void
{
copy(__DIR__.'/Fixtures/tabs.json', __DIR__.'/Fixtures/tabs2.json');
$jsonFile = new JsonFile(__DIR__.'/Fixtures/tabs2.json');
$jsonFile->write(['foo' => 'baz']);
self::assertSame("{\n \"foo\": \"baz\"\n}\n", file_get_contents(__DIR__.'/Fixtures/tabs2.json'));
unlink(__DIR__.'/Fixtures/tabs2.json');
}
2022-02-22 15:47:09 +00:00
private function expectParseException(string $text, string $json): void
2011-10-31 13:43:41 +00:00
{
try {
2015-05-05 16:13:16 +00:00
$result = JsonFile::parseJson($json);
$this->fail(sprintf("Parsing should have failed but didn't.\nExpected:\n\"%s\"\nFor:\n\"%s\"\nGot:\n\"%s\"", $text, $json, var_export($result, true)));
2012-07-03 12:19:19 +00:00
} catch (ParsingException $e) {
self::assertStringContainsString($text, $e->getMessage());
2011-10-31 13:43:41 +00:00
}
}
/**
* @param mixed $data
*/
2022-02-22 15:47:09 +00:00
private function assertJsonFormat(string $json, $data, ?int $options = null): void
{
$file = new JsonFile('composer.json');
$json = str_replace("\r", '', $json);
if (null === $options) {
self::assertEquals($json, $file->encode($data));
} else {
self::assertEquals($json, $file->encode($data, $options));
}
}
2011-10-31 13:43:41 +00:00
}