diff --git a/framework/core/src/Extend/ApiResource.php b/framework/core/src/Extend/ApiResource.php index bdd682de9..2c6634e11 100644 --- a/framework/core/src/Extend/ApiResource.php +++ b/framework/core/src/Extend/ApiResource.php @@ -25,6 +25,8 @@ class ApiResource implements ExtenderInterface private array $removeEndpoints = []; private array $endpoint = []; private array $fields = []; + private array $fieldsBefore = []; + private array $fieldsAfter = []; private array $removeFields = []; private array $field = []; private array $sorts = []; @@ -93,6 +95,32 @@ class ApiResource implements ExtenderInterface return $this; } + /** + * Add fields to the resource before a certain field. + * + * @param string $before the name of the field to add the new fields before. + * @param callable|class-string $fields must be a callable that returns an array of objects that implement \Tobyz\JsonApiServer\Schema\Field. + */ + public function fieldsBefore(string $before, callable|string $fields): self + { + $this->fieldsBefore[] = [$before, $fields]; + + return $this; + } + + /** + * Add fields to the resource after a certain field. + * + * @param string $after the name of the field to add the new fields after. + * @param callable|class-string $fields must be a callable that returns an array of objects that implement \Tobyz\JsonApiServer\Schema\Field. + */ + public function fieldsAfter(string $after, callable|string $fields): self + { + $this->fieldsAfter[] = [$after, $fields]; + + return $this; + } + /** * Remove fields from the resource. * @@ -221,6 +249,26 @@ class ApiResource implements ExtenderInterface $fields = array_merge($fields, $newFieldsCallback()); } + foreach ($this->fieldsBefore as [$before, $newFieldsCallback]) { + $newFieldsCallback = ContainerUtil::wrapCallback($newFieldsCallback, $container); + $newFields = $newFieldsCallback(); + $beforeIndex = array_search($before, array_column($fields, 'name')); + + if ($beforeIndex !== false) { + array_splice($fields, $beforeIndex, 0, $newFields); + } + } + + foreach ($this->fieldsAfter as [$after, $newFieldsCallback]) { + $newFieldsCallback = ContainerUtil::wrapCallback($newFieldsCallback, $container); + $newFields = $newFieldsCallback(); + $afterIndex = array_search($after, array_column($fields, 'name')); + + if ($afterIndex !== false) { + array_splice($fields, $afterIndex + 1, 0, $newFields); + } + } + foreach ($this->removeFields as $field) { [$fieldsToRemove, $condition] = $field; diff --git a/framework/core/tests/integration/extenders/ApiControllerTest.php b/framework/core/tests/integration/extenders/ApiResourceTest.php similarity index 64% rename from framework/core/tests/integration/extenders/ApiControllerTest.php rename to framework/core/tests/integration/extenders/ApiResourceTest.php index 340ecde5f..088bc51e5 100644 --- a/framework/core/tests/integration/extenders/ApiControllerTest.php +++ b/framework/core/tests/integration/extenders/ApiResourceTest.php @@ -15,8 +15,9 @@ use Flarum\Api\Endpoint\Index; use Flarum\Api\Endpoint\Show; use Flarum\Api\Resource\AbstractDatabaseResource; use Flarum\Api\Resource\DiscussionResource; +use Flarum\Api\Resource\ForumResource; use Flarum\Api\Resource\UserResource; -use Flarum\Api\Schema\Relationship\ToMany; +use Flarum\Api\Schema; use Flarum\Api\Sort\SortColumn; use Flarum\Discussion\Discussion; use Flarum\Extend; @@ -25,11 +26,12 @@ use Flarum\Post\Post; use Flarum\Testing\integration\RetrievesAuthorizedUsers; use Flarum\Testing\integration\TestCase; use Flarum\User\User; +use Illuminate\Database\PostgresConnection; use Illuminate\Support\Arr; use PHPUnit\Framework\Attributes\Test; use Tobyz\JsonApiServer\Schema\Field\Field; -class ApiControllerTest extends TestCase +class ApiResourceTest extends TestCase { use RetrievesAuthorizedUsers; @@ -45,15 +47,21 @@ class ApiControllerTest extends TestCase $this->normalUser() ], Discussion::class => [ - ['id' => 1, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], - ['id' => 2, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 3, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], - ['id' => 3, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 1, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + ['id' => 1, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(1)->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + ['id' => 2, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(2)->toDateTimeString(), 'user_id' => 3, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + ['id' => 3, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(3)->toDateTimeString(), 'user_id' => 1, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + + ['id' => 4, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(4)->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + ['id' => 5, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(5)->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], + ['id' => 6, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->addMinutes(6)->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], ], Post::class => [ ['id' => 1, 'discussion_id' => 3, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], ['id' => 2, 'discussion_id' => 2, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], ['id' => 3, 'discussion_id' => 1, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], ['id' => 3, 'discussion_id' => 1, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 1, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], + + ['id' => 5, 'discussion_id' => 6, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], ], ]); } @@ -243,7 +251,7 @@ class ApiControllerTest extends TestCase ->hasMany('customApiControllerRelation', Discussion::class, 'user_id'), (new Extend\ApiResource(UserResource::class)) ->fields(fn () => [ - ToMany::make('customApiControllerRelation') + Schema\Relationship\ToMany::make('customApiControllerRelation') ->type('discussions') ->includable(), ]) @@ -271,7 +279,7 @@ class ApiControllerTest extends TestCase ->hasMany('customApiControllerRelation2', Discussion::class, 'user_id'), (new Extend\ApiResource(UserResource::class)) ->fields(fn () => [ - ToMany::make('customApiControllerRelation2') + Schema\Relationship\ToMany::make('customApiControllerRelation2') ->type('discussions') ->includable(), ]) @@ -333,7 +341,7 @@ class ApiControllerTest extends TestCase ->hasMany('customApiControllerRelation2', Discussion::class, 'user_id'), (new Extend\ApiResource(UserResource::class)) ->fields(fn () => [ - ToMany::make('customApiControllerRelation2') + Schema\Relationship\ToMany::make('customApiControllerRelation2') ->type('discussions') ->includable(), ]) @@ -362,7 +370,7 @@ class ApiControllerTest extends TestCase $payload = json_decode($response->getBody()->getContents(), true); - $this->assertCount(3, $payload['data']); + $this->assertCount(6, $payload['data']); } #[Test] @@ -444,7 +452,7 @@ class ApiControllerTest extends TestCase $payload = json_decode($body = $response->getBody()->getContents(), true); $this->assertEquals(200, $response->getStatusCode(), $body); - $this->assertEquals([3, 1, 2], Arr::pluck($payload['data'], 'id')); + $this->assertEquals([3, 1, 4, 5, 6, 2], Arr::pluck($payload['data'], 'id')); } #[Test] @@ -502,7 +510,12 @@ class ApiControllerTest extends TestCase $payload = json_decode($response->getBody()->getContents(), true); $this->assertEquals(200, $response->getStatusCode()); - $this->assertEquals([2, 1, 3], Arr::pluck($payload['data'], 'id')); + + if ($this->database() instanceof PostgresConnection) { + $this->assertEquals([2, 1, 4, 5, 6, 3], Arr::pluck($payload['data'], 'id')); + } else { + $this->assertEquals([2, 6, 5, 4, 1, 3], Arr::pluck($payload['data'], 'id')); + } } #[Test] @@ -709,6 +722,326 @@ class ApiControllerTest extends TestCase $this->assertFalse($users->pluck('firstLevelRelation')->filter->relationLoaded('secondLevelRelation')->isEmpty()); } + + #[Test] + public function custom_attributes_dont_exist_by_default() + { + $this->app(); + + $response = $this->send( + $this->request('GET', '/api', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayNotHasKey('customAttribute', $payload['data']['attributes']); + } + + #[Test] + public function custom_attributes_exist_if_added() + { + $this->extend( + (new Extend\ApiResource(ForumResource::class)) + ->fields(fn () => [ + Schema\Boolean::make('customAttribute') + ->get(fn () => true), + ]) + ); + + $this->app(); + + $response = $this->send( + $this->request('GET', '/api', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); + } + + #[Test] + public function custom_attributes_exist_if_added_before_field() + { + $this->extend( + (new Extend\ApiResource(DiscussionResource::class)) + ->fieldsBefore('title', fn () => [ + Schema\Boolean::make('customAttribute') + ->writable() + ->set(fn (Discussion $discussion) => $this->assertNull($discussion->title)) + ]) + ); + + $response = $this->send( + $this->request('POST', '/api/discussions', [ + 'authenticatedAs' => 1, + 'json' => [ + 'data' => [ + 'type' => 'discussions', + 'attributes' => [ + 'title' => 'Custom Discussion Title', + 'customAttribute' => true, + 'content' => 'Custom Discussion Content', + ], + ], + ], + ]) + ); + + $this->assertEquals(201, $response->getStatusCode(), $response->getBody()->getContents()); + } + + #[Test] + public function custom_attributes_exist_if_added_after_field() + { + $this->extend( + (new Extend\ApiResource(DiscussionResource::class)) + ->fieldsAfter('title', fn () => [ + Schema\Boolean::make('customAttribute') + ->writable() + ->set(fn (Discussion $discussion) => $this->assertNotNull($discussion->title)) + ]) + ); + + $response = $this->send( + $this->request('POST', '/api/discussions', [ + 'authenticatedAs' => 1, + 'json' => [ + 'data' => [ + 'type' => 'discussions', + 'attributes' => [ + 'title' => 'Custom Discussion Title', + 'customAttribute' => true, + 'content' => 'Custom Discussion Content', + ], + ], + ], + ]) + ); + + $this->assertEquals(201, $response->getStatusCode(), $response->getBody()->getContents()); + } + + #[Test] + public function custom_attributes_with_invokable_exist_if_added() + { + $this->extend( + (new Extend\ApiResource(ForumResource::class)) + ->fields(CustomAttributesInvokableClass::class) + ); + + $this->app(); + + $response = $this->send( + $this->request('GET', '/api', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayHasKey('customAttributeFromInvokable', $payload['data']['attributes']); + } + + #[Test] + public function custom_attributes_exist_if_added_to_parent_class() + { + $this->extend( + (new Extend\ApiResource(AbstractDatabaseResource::class)) + ->fields(fn () => [ + Schema\Boolean::make('customAttribute') + ->get(fn () => true), + ]) + ); + + $this->app(); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); + } + + #[Test] + public function custom_attributes_prioritize_child_classes() + { + $this->extend( + (new Extend\ApiResource(AbstractDatabaseResource::class)) + ->fields(fn () => [ + Schema\Str::make('customAttribute') + ->get(fn () => 'initialValue') + ]), + (new Extend\ApiResource(UserResource::class)) + ->fields(fn () => [ + Schema\Str::make('customAttribute') + ->get(fn () => 'newValue') + ]), + ); + + $this->app(); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); + $this->assertEquals('newValue', $payload['data']['attributes']['customAttribute']); + } + + #[Test] + public function custom_attributes_can_be_overridden() + { + $this->extend( + (new Extend\ApiResource(UserResource::class)) + ->fields(fn () => [ + Schema\Str::make('someCustomAttribute') + ->get(fn () => 'newValue'), + ]) + ->fields(fn () => [ + Schema\Str::make('someCustomAttribute') + ->get(fn () => 'secondValue'), + Schema\Str::make('someOtherCustomAttribute') + ->get(fn () => 'secondValue'), + ]) + ->fields(fn () => [ + Schema\Str::make('someOtherCustomAttribute') + ->get(fn () => 'newValue'), + ]) + ); + + $this->app(); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $payload = json_decode($response->getBody()->getContents(), true); + + $this->assertArrayHasKey('someCustomAttribute', $payload['data']['attributes']); + $this->assertEquals('secondValue', $payload['data']['attributes']['someCustomAttribute']); + $this->assertArrayHasKey('someOtherCustomAttribute', $payload['data']['attributes']); + $this->assertEquals('newValue', $payload['data']['attributes']['someOtherCustomAttribute']); + } + + #[Test] + public function custom_relations_dont_exist_by_default() + { + $this->extend( + (new Extend\ApiResource(UserResource::class)) + ->endpoint(Show::class, function (Show $endpoint): Show { + return $endpoint->addDefaultInclude(['customSerializerRelation', 'postCustomRelation', 'anotherCustomRelation']); + }) + ); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $this->assertEquals(400, $response->getStatusCode()); + } + + #[Test] + public function custom_hasMany_relationship_exists_if_added() + { + $this->extend( + (new Extend\Model(User::class)) + ->hasMany('customSerializerRelation', Discussion::class, 'user_id'), + (new Extend\ApiResource(UserResource::class)) + ->fields(fn () => [ + Schema\Relationship\ToMany::make('customSerializerRelation') + ->type('discussions') + ->includable() + ]) + ->endpoint(Show::class, function (Show $endpoint) { + return $endpoint->addDefaultInclude(['customSerializerRelation']); + }) + ); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $responseJson = json_decode($response->getBody(), true); + + $this->assertArrayHasKey('customSerializerRelation', $responseJson['data']['relationships']); + $this->assertCount(4, $responseJson['data']['relationships']['customSerializerRelation']['data']); + } + + #[Test] + public function custom_hasOne_relationship_exists_if_added() + { + $this->extend( + (new Extend\Model(User::class)) + ->hasOne('customSerializerRelation', Discussion::class, 'user_id'), + (new Extend\ApiResource(UserResource::class)) + ->fields(fn () => [ + Schema\Relationship\ToOne::make('customSerializerRelation') + ->type('discussions') + ->includable() + ]) + ->endpoint(Show::class, function (Show $endpoint) { + return $endpoint->addDefaultInclude(['customSerializerRelation']); + }) + ); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $responseJson = json_decode($response->getBody(), true); + + $this->assertArrayHasKey('customSerializerRelation', $responseJson['data']['relationships']); + $this->assertEquals('discussions', $responseJson['data']['relationships']['customSerializerRelation']['data']['type']); + } + + #[Test] + public function custom_relationship_is_inherited_to_child_classes() + { + $this->extend( + (new Extend\Model(User::class)) + ->hasMany('anotherCustomRelation', Discussion::class, 'user_id'), + (new Extend\ApiResource(AbstractDatabaseResource::class)) + ->fields(fn () => [ + Schema\Relationship\ToMany::make('anotherCustomRelation') + ->type('discussions') + ->includable() + ]) + ->endpoint(Show::class, function (Show $endpoint) { + return $endpoint->addDefaultInclude(['anotherCustomRelation']); + }) + ); + + $response = $this->send( + $this->request('GET', '/api/users/2', [ + 'authenticatedAs' => 1, + ]) + ); + + $responseJson = json_decode($response->getBody(), true); + + $this->assertArrayHasKey('anotherCustomRelation', $responseJson['data']['relationships']); + $this->assertCount(4, $responseJson['data']['relationships']['anotherCustomRelation']['data']); + } } class CustomAfterEndpointInvokableClass @@ -720,3 +1053,14 @@ class CustomAfterEndpointInvokableClass return $discussion; } } + +class CustomAttributesInvokableClass +{ + public function __invoke(): array + { + return [ + Schema\Boolean::make('customAttributeFromInvokable') + ->get(fn () => true), + ]; + } +} diff --git a/framework/core/tests/integration/extenders/ApiSerializerTest.php b/framework/core/tests/integration/extenders/ApiSerializerTest.php deleted file mode 100644 index 1ec6d367c..000000000 --- a/framework/core/tests/integration/extenders/ApiSerializerTest.php +++ /dev/null @@ -1,320 +0,0 @@ -prepareDatabase([ - User::class => [ - $this->normalUser() - ], - Discussion::class => [ - ['id' => 1, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], - ['id' => 2, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], - ['id' => 3, 'title' => 'Custom Discussion Title', 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'first_post_id' => 0, 'comment_count' => 1, 'is_private' => 0], - ], - Post::class => [ - ['id' => 1, 'discussion_id' => 3, 'created_at' => Carbon::now()->toDateTimeString(), 'user_id' => 2, 'type' => 'discussionRenamed', 'content' => '

can i haz relationz?

'], - ], - ]); - } - - #[Test] - public function custom_attributes_dont_exist_by_default() - { - $this->app(); - - $response = $this->send( - $this->request('GET', '/api', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayNotHasKey('customAttribute', $payload['data']['attributes']); - } - - #[Test] - public function custom_attributes_exist_if_added() - { - $this->extend( - (new Extend\ApiResource(ForumResource::class)) - ->fields(fn () => [ - Schema\Boolean::make('customAttribute') - ->get(fn () => true), - ]) - ); - - $this->app(); - - $response = $this->send( - $this->request('GET', '/api', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); - } - - #[Test] - public function custom_attributes_with_invokable_exist_if_added() - { - $this->extend( - (new Extend\ApiResource(ForumResource::class)) - ->fields(CustomAttributesInvokableClass::class) - ); - - $this->app(); - - $response = $this->send( - $this->request('GET', '/api', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayHasKey('customAttributeFromInvokable', $payload['data']['attributes']); - } - - #[Test] - public function custom_attributes_exist_if_added_to_parent_class() - { - $this->extend( - (new Extend\ApiResource(AbstractDatabaseResource::class)) - ->fields(fn () => [ - Schema\Boolean::make('customAttribute') - ->get(fn () => true), - ]) - ); - - $this->app(); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); - } - - #[Test] - public function custom_attributes_prioritize_child_classes() - { - $this->extend( - (new Extend\ApiResource(AbstractDatabaseResource::class)) - ->fields(fn () => [ - Schema\Str::make('customAttribute') - ->get(fn () => 'initialValue') - ]), - (new Extend\ApiResource(UserResource::class)) - ->fields(fn () => [ - Schema\Str::make('customAttribute') - ->get(fn () => 'newValue') - ]), - ); - - $this->app(); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayHasKey('customAttribute', $payload['data']['attributes']); - $this->assertEquals('newValue', $payload['data']['attributes']['customAttribute']); - } - - #[Test] - public function custom_attributes_can_be_overridden() - { - $this->extend( - (new Extend\ApiResource(UserResource::class)) - ->fields(fn () => [ - Schema\Str::make('someCustomAttribute') - ->get(fn () => 'newValue'), - ]) - ->fields(fn () => [ - Schema\Str::make('someCustomAttribute') - ->get(fn () => 'secondValue'), - Schema\Str::make('someOtherCustomAttribute') - ->get(fn () => 'secondValue'), - ]) - ->fields(fn () => [ - Schema\Str::make('someOtherCustomAttribute') - ->get(fn () => 'newValue'), - ]) - ); - - $this->app(); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $payload = json_decode($response->getBody()->getContents(), true); - - $this->assertArrayHasKey('someCustomAttribute', $payload['data']['attributes']); - $this->assertEquals('secondValue', $payload['data']['attributes']['someCustomAttribute']); - $this->assertArrayHasKey('someOtherCustomAttribute', $payload['data']['attributes']); - $this->assertEquals('newValue', $payload['data']['attributes']['someOtherCustomAttribute']); - } - - #[Test] - public function custom_relations_dont_exist_by_default() - { - $this->extend( - (new Extend\ApiResource(UserResource::class)) - ->endpoint(Show::class, function (Show $endpoint): Show { - return $endpoint->addDefaultInclude(['customSerializerRelation', 'postCustomRelation', 'anotherCustomRelation']); - }) - ); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $this->assertEquals(400, $response->getStatusCode()); - } - - #[Test] - public function custom_hasMany_relationship_exists_if_added() - { - $this->extend( - (new Extend\Model(User::class)) - ->hasMany('customSerializerRelation', Discussion::class, 'user_id'), - (new Extend\ApiResource(UserResource::class)) - ->fields(fn () => [ - Schema\Relationship\ToMany::make('customSerializerRelation') - ->type('discussions') - ->includable() - ]) - ->endpoint(Show::class, function (Show $endpoint) { - return $endpoint->addDefaultInclude(['customSerializerRelation']); - }) - ); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $responseJson = json_decode($response->getBody(), true); - - $this->assertArrayHasKey('customSerializerRelation', $responseJson['data']['relationships']); - $this->assertCount(3, $responseJson['data']['relationships']['customSerializerRelation']['data']); - } - - #[Test] - public function custom_hasOne_relationship_exists_if_added() - { - $this->extend( - (new Extend\Model(User::class)) - ->hasOne('customSerializerRelation', Discussion::class, 'user_id'), - (new Extend\ApiResource(UserResource::class)) - ->fields(fn () => [ - Schema\Relationship\ToOne::make('customSerializerRelation') - ->type('discussions') - ->includable() - ]) - ->endpoint(Show::class, function (Show $endpoint) { - return $endpoint->addDefaultInclude(['customSerializerRelation']); - }) - ); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $responseJson = json_decode($response->getBody(), true); - - $this->assertArrayHasKey('customSerializerRelation', $responseJson['data']['relationships']); - $this->assertEquals('discussions', $responseJson['data']['relationships']['customSerializerRelation']['data']['type']); - } - - #[Test] - public function custom_relationship_is_inherited_to_child_classes() - { - $this->extend( - (new Extend\Model(User::class)) - ->hasMany('anotherCustomRelation', Discussion::class, 'user_id'), - (new Extend\ApiResource(AbstractDatabaseResource::class)) - ->fields(fn () => [ - Schema\Relationship\ToMany::make('anotherCustomRelation') - ->type('discussions') - ->includable() - ]) - ->endpoint(Show::class, function (Show $endpoint) { - return $endpoint->addDefaultInclude(['anotherCustomRelation']); - }) - ); - - $response = $this->send( - $this->request('GET', '/api/users/2', [ - 'authenticatedAs' => 1, - ]) - ); - - $responseJson = json_decode($response->getBody(), true); - - $this->assertArrayHasKey('anotherCustomRelation', $responseJson['data']['relationships']); - $this->assertCount(3, $responseJson['data']['relationships']['anotherCustomRelation']['data']); - } -} - -class CustomAttributesInvokableClass -{ - public function __invoke(): array - { - return [ - Schema\Boolean::make('customAttributeFromInvokable') - ->get(fn () => true), - ]; - } -}