diff --git a/src/Api/Groups.php b/src/Api/Groups.php index 7104d345..74d63f13 100644 --- a/src/Api/Groups.php +++ b/src/Api/Groups.php @@ -854,4 +854,18 @@ public function search(int|string $id, array $parameters = []): mixed return $this->get('groups/'.self::encodePath($id).'/search', $resolver->resolve($parameters)); } + + /** + * Get a list of registry repositories in a group. + * + * @see https://docs.gitlab.com/ee/api/container_registry.html#within-a-group + * + * @param $id: The ID of a group + * + * @return mixed + */ + public function registryRepositories(int $id) + { + return $this->get('groups/'.self::encodePath($id).'/registry/repositories'); + } } diff --git a/src/Api/Projects.php b/src/Api/Projects.php index 32aa168a..97c29ecb 100644 --- a/src/Api/Projects.php +++ b/src/Api/Projects.php @@ -1280,4 +1280,33 @@ public function search(int|string $id, array $parameters = []): mixed return $this->get('projects/'.self::encodePath($id).'/search', $resolver->resolve($parameters)); } + + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#within-a-project + * + * @param int|string $project_id + * @param array $parameters { + * + * @var bool $tags if the parameter is included as true, each repository includes an array of "tags" in the response + * @var bool $tags_count If the parameter is included as true, each repository includes "tags_count" in the response. + * } + * + * @return mixed + */ + public function registryRepositories($project_id, array $parameters = []) + { + $resolver = $this->createOptionsResolver(); + $booleanNormalizer = function (Options $resolver, $value): string { + return $value ? 'true' : 'false'; + }; + + $resolver->setDefined('tags') + ->setAllowedTypes('tags', 'bool') + ->setNormalizer('tags', $booleanNormalizer); + $resolver->setDefined('tags_count') + ->setAllowedTypes('tags_count', 'bool') + ->setNormalizer('tags_count', $booleanNormalizer); + + return $this->get($this->getProjectPath($project_id, 'registry/repositories'), $resolver->resolve($parameters)); + } } diff --git a/src/Api/Registry.php b/src/Api/Registry.php new file mode 100644 index 00000000..5fdc0103 --- /dev/null +++ b/src/Api/Registry.php @@ -0,0 +1,136 @@ +<?php + +declare(strict_types=1); + +/* + * This file is part of the Gitlab API library. + * + * (c) Matt Humphrey <matth@windsor-telecom.co.uk> + * (c) Graham Campbell <hello@gjcampbell.co.uk> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace Gitlab\Api; + +use Symfony\Component\OptionsResolver\Options; + +class Registry extends AbstractApi +{ + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#get-details-of-a-single-repository + * + * @param int|string $repository_id The ID of the registry repository accessible by the authenticated user + * @param array $parameters { + * + * @var bool $tags + * @var bool $tags_count + * @var bool $size + * } + * + * @return mixed + */ + public function repositories($repository_id, array $parameters = []) + { + $resolver = $this->createOptionsResolver(); + $booleanNormalizer = function (Options $resolver, $value): string { + return $value ? 'true' : 'false'; + }; + + $resolver->setDefined('tags') + ->setAllowedTypes('tags', 'bool') + ->setNormalizer('tags', $booleanNormalizer); + $resolver->setDefined('tags_count') + ->setAllowedTypes('tags_count', 'bool') + ->setNormalizer('tags_count', $booleanNormalizer); + $resolver->setDefined('size') + ->setAllowedTypes('size', 'bool') + ->setNormalizer('size', $booleanNormalizer); + + return $this->get('registry/repositories/'.self::encodePath($repository_id), $resolver->resolve($parameters)); + } + + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#list-registry-repository-tags + * + * @param int|string $project_id + * + * @return mixed + */ + public function repositoryTags($project_id, int $repository_id) + { + return $this->get( + $this->getProjectPath($project_id, 'registry/repositories/'.self::encodePath($repository_id).'/tags') + ); + } + + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#get-details-of-a-registry-repository-tag + * + * @param int|string $project_id + * + * @return mixed + */ + public function repositoryTag($project_id, int $repository_id, string $tag_name) + { + return $this->get( + $this->getProjectPath( + $project_id, + 'registry/repositories/'.self::encodePath($repository_id).'/tags/'.self::encodePath($tag_name) + ) + ); + } + + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#delete-a-registry-repository-tag + * + * @param int|string $project_id + * + * @return mixed + */ + public function removeRepositoryTag($project_id, int $repository_id, string $tag_name) + { + return $this->delete( + $this->getProjectPath( + $project_id, + 'registry/repositories/'.self::encodePath($repository_id).'/tags/'.self::encodePath($tag_name) + ) + ); + } + + /** + * @see https://docs.gitlab.com/ee/api/container_registry.html#delete-registry-repository-tags-in-bulk + * + * @param int|string $project_id + * @param array $parameters { + * + * @var string $name_regex_delete + * @var string $name_regex_keep + * @var int $keep_n + * @var string $older_than + * } + * + * @return mixed + */ + public function removeRepositoryTags($project_id, int $repository_id, array $parameters = []) + { + $resolver = $this->createOptionsResolver(); + $resolver->setRequired('name_regex_delete') + ->setAllowedTypes('name_regex_delete', 'string'); + $resolver->setDefined('name_regex_keep') + ->setAllowedTypes('name_regex_keep', 'string'); + $resolver->setDefined('keep_n') + ->setAllowedTypes('keep_n', 'int'); + $resolver->setDefined('older_than') + ->setAllowedTypes('older_than', 'string'); + + return $this->delete( + $this->getProjectPath( + $project_id, + 'registry/repositories/'.self::encodePath($repository_id).'/tags' + ), + $resolver->resolve($parameters) + ); + } +} diff --git a/src/Client.php b/src/Client.php index ee4ff8a7..c74ec9da 100644 --- a/src/Client.php +++ b/src/Client.php @@ -32,6 +32,7 @@ use Gitlab\Api\Milestones; use Gitlab\Api\ProjectNamespaces; use Gitlab\Api\Projects; +use Gitlab\Api\Registry; use Gitlab\Api\Repositories; use Gitlab\Api\RepositoryFiles; use Gitlab\Api\ResourceIterationEvents; @@ -245,6 +246,11 @@ public function projects(): Projects return new Projects($this); } + public function registry(): Registry + { + return new Registry($this); + } + public function repositories(): Repositories { return new Repositories($this); diff --git a/tests/Api/GroupsTest.php b/tests/Api/GroupsTest.php index 3b90e6f2..dfe02289 100644 --- a/tests/Api/GroupsTest.php +++ b/tests/Api/GroupsTest.php @@ -882,4 +882,24 @@ public function shouldSearchGroups(): void 'sort' => 'desc', ])); } + + /** + * @test + */ + public function shouldGetGroupRegistryRepositories(): void + { + $expectedArray = [ + ['id' => 1, 'name' => 'A registry'], + ['id' => 2, 'name' => 'Another registry'], + ]; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('groups/1/registry/repositories') + ->will($this->returnValue($expectedArray)) + ; + + $this->assertEquals($expectedArray, $api->registryRepositories(1)); + } } diff --git a/tests/Api/ProjectsTest.php b/tests/Api/ProjectsTest.php index 1ef556ea..9c004ed8 100644 --- a/tests/Api/ProjectsTest.php +++ b/tests/Api/ProjectsTest.php @@ -2765,4 +2765,45 @@ public function shouldSearchGroups(): void 'sort' => 'desc', ])); } + + /** + * @test + */ + public function shouldGetProjectRegistryRepositories(): void + { + $expectedArray = [ + ['id' => 1, 'name' => 'A registry'], + ['id' => 2, 'name' => 'Another registry'], + ]; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('projects/123/registry/repositories') + ->will($this->returnValue($expectedArray)); + + $this->assertEquals($expectedArray, $api->registryRepositories(123)); + } + + /** + * @test + */ + public function shouldGetProjectRegistryRepositoriesTags(): void + { + $expectedArray = [ + ['id' => 1, 'name' => 'A registry', 'tags' => ['1.0', '1.1'], 'tags_count' => 2], + ['id' => 2, 'name' => 'Another registry', 'tags' => ['2.0', '2.1'], 'tags_count' => 2], + ]; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('projects/123/registry/repositories', [ + 'tags' => true, + 'tags_count' => true, + ]) + ->will($this->returnValue($expectedArray)); + + $this->assertEquals($expectedArray, $api->registryRepositories(123, ['tags' => true, 'tags_count' => true])); + } } diff --git a/tests/Api/RegistryTest.php b/tests/Api/RegistryTest.php new file mode 100644 index 00000000..7c3ceeb6 --- /dev/null +++ b/tests/Api/RegistryTest.php @@ -0,0 +1,120 @@ +<?php + +declare(strict_types=1); + +/* + * This file is part of the Gitlab API library. + * + * (c) Matt Humphrey <matth@windsor-telecom.co.uk> + * (c) Graham Campbell <hello@gjcampbell.co.uk> + * + * For the full copyright and license information, please view the LICENSE + * file that was distributed with this source code. + */ + +namespace Gitlab\Tests\Api; + +use Gitlab\Api\Registry; + +class RegistryTest extends TestCase +{ + protected function getApiClass(): string + { + return Registry::class; + } + + /** + * @test + */ + public function shouldGetSingleRepository(): void + { + $expectedArray = ['id' => 1, 'name' => 'John Doe']; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('registry/repositories/1') + ->willReturn($expectedArray); + + $this->assertEquals($expectedArray, $api->repositories(1)); + } + + /** + * @test + */ + public function shouldGetSingleRepositoryWithParams(): void + { + $expectedArray = ['id' => 1, 'name' => 'John Doe', 'tags' => ['tag1', 'tag2'], 'tags_count' => 2]; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('registry/repositories/1', ['tags' => 'true', 'tags_count' => 'true']) + ->willReturn($expectedArray); + + $this->assertEquals($expectedArray, $api->repositories(1, ['tags' => true, 'tags_count' => true])); + } + + /** + * @test + */ + public function shouldGetRepositoryTags(): void + { + $expectedArray = [['name' => 'A', 'path' => 'group/project:A'], ['name' => 'B', 'path' => 'group/project:B']]; + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('projects/1/registry/repositories/1/tags') + ->willReturn($expectedArray); + + $this->assertEquals($expectedArray, $api->repositoryTags(1, 1)); + } + + /** + * @test + */ + public function shouldGetRepositoryTag(): void + { + $expectedArray = ['name' => 'A', 'path' => 'group/project:A']; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('get') + ->with('projects/1/registry/repositories/1/tags/A') + ->willReturn($expectedArray); + + $this->assertEquals($expectedArray, $api->repositoryTag(1, 1, 'A')); + } + + /** + * @test + */ + public function shouldRemoveRepositoryTag(): void + { + $expectedBool = true; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('delete') + ->with('projects/1/registry/repositories/1/tags/A') + ->will($this->returnValue($expectedBool)); + + $this->assertEquals($expectedBool, $api->removeRepositoryTag(1, 1, 'A')); + } + + /** + * @test + */ + public function shouldRemoveRepositoryTags(): void + { + $expectedBool = true; + + $api = $this->getApiMock(); + $api->expects($this->once()) + ->method('delete') + ->with('projects/1/registry/repositories/1/tags', ['name_regex_delete' => '.*', 'keep_n' => 12]) + ->will($this->returnValue($expectedBool)); + + $this->assertEquals($expectedBool, $api->removeRepositoryTags(1, 1, ['name_regex_delete' => '.*', 'keep_n' => 12])); + } +}