Skip to content

Commit 4abf17b

Browse files
authored
feat(symfony): CSS Color Schema Restriction for Property Validation (api-platform#7215)
1 parent fb96f33 commit 4abf17b

File tree

3 files changed

+153
-0
lines changed

3 files changed

+153
-0
lines changed

src/Symfony/Bundle/Resources/config/metadata/validator.xml

Lines changed: 4 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -22,6 +22,10 @@
2222
<tag name="api_platform.metadata.property_schema_restriction"/>
2323
</service>
2424

25+
<service id="api_platform.metadata.property_schema.css_color_restriction" class="ApiPlatform\Symfony\Validator\Metadata\Property\Restriction\PropertySchemaCssColorRestriction" public="false">
26+
<tag name="api_platform.metadata.property_schema_restriction"/>
27+
</service>
28+
2529
<service id="api_platform.metadata.property_schema.greater_than_or_equal_restriction" class="ApiPlatform\Symfony\Validator\Metadata\Property\Restriction\PropertySchemaGreaterThanOrEqualRestriction" public="false">
2630
<tag name="api_platform.metadata.property_schema_restriction"/>
2731
</service>
Lines changed: 78 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,78 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the API Platform project.
5+
*
6+
* (c) Kévin Dunglas <[email protected]>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
declare(strict_types=1);
13+
14+
namespace ApiPlatform\Symfony\Validator\Metadata\Property\Restriction;
15+
16+
use ApiPlatform\Metadata\ApiProperty;
17+
use Symfony\Component\Validator\Constraint;
18+
use Symfony\Component\Validator\Constraints\CssColor;
19+
20+
/**
21+
* Class PropertySchemaRegexRestriction.
22+
*/
23+
class PropertySchemaCssColorRestriction implements PropertySchemaRestrictionMetadataInterface
24+
{
25+
private const PATTERN_HEX_LONG = '/^#[0-9a-f]{6}$/iD';
26+
private const PATTERN_HEX_LONG_WITH_ALPHA = '/^#[0-9a-f]{8}$/iD';
27+
private const PATTERN_HEX_SHORT = '/^#[0-9a-f]{3}$/iD';
28+
private const PATTERN_HEX_SHORT_WITH_ALPHA = '/^#[0-9a-f]{4}$/iD';
29+
// List comes from https://www.w3.org/wiki/CSS/Properties/color/keywords#Basic_Colors
30+
private const PATTERN_BASIC_NAMED_COLORS = '/^(black|silver|gray|white|maroon|red|purple|fuchsia|green|lime|olive|yellow|navy|blue|teal|aqua)$/iD';
31+
// List comes from https://www.w3.org/wiki/CSS/Properties/color/keywords#Extended_colors
32+
private const PATTERN_EXTENDED_NAMED_COLORS = '/^(aliceblue|antiquewhite|aqua|aquamarine|azure|beige|bisque|black|blanchedalmond|blue|blueviolet|brown|burlywood|cadetblue|chartreuse|chocolate|coral|cornflowerblue|cornsilk|crimson|cyan|darkblue|darkcyan|darkgoldenrod|darkgray|darkgreen|darkgrey|darkkhaki|darkmagenta|darkolivegreen|darkorange|darkorchid|darkred|darksalmon|darkseagreen|darkslateblue|darkslategray|darkslategrey|darkturquoise|darkviolet|deeppink|deepskyblue|dimgray|dimgrey|dodgerblue|firebrick|floralwhite|forestgreen|fuchsia|gainsboro|ghostwhite|gold|goldenrod|gray|green|greenyellow|grey|honeydew|hotpink|indianred|indigo|ivory|khaki|lavender|lavenderblush|lawngreen|lemonchiffon|lightblue|lightcoral|lightcyan|lightgoldenrodyellow|lightgray|lightgreen|lightgrey|lightpink|lightsalmon|lightseagreen|lightskyblue|lightslategray|lightslategrey|lightsteelblue|lightyellow|lime|limegreen|linen|magenta|maroon|mediumaquamarine|mediumblue|mediumorchid|mediumpurple|mediumseagreen|mediumslateblue|mediumspringgreen|mediumturquoise|mediumvioletred|midnightblue|mintcream|mistyrose|moccasin|navajowhite|navy|oldlace|olive|olivedrab|orange|orangered|orchid|palegoldenrod|palegreen|paleturquoise|palevioletred|papayawhip|peachpuff|peru|pink|plum|powderblue|purple|red|rosybrown|royalblue|saddlebrown|salmon|sandybrown|seagreen|seashell|sienna|silver|skyblue|slateblue|slategray|slategrey|snow|springgreen|steelblue|tan|teal|thistle|tomato|turquoise|violet|wheat|white|whitesmoke|yellow|yellowgreen)$/iD';
33+
// List comes from https://drafts.csswg.org/css-color/#css-system-colors
34+
private const PATTERN_SYSTEM_COLORS = '/^(Canvas|CanvasText|LinkText|VisitedText|ActiveText|ButtonFace|ButtonText|ButtonBorder|Field|FieldText|Highlight|HighlightText|SelectedItem|SelectedItemText|Mark|MarkText|GrayText)$/iD';
35+
private const PATTERN_KEYWORDS = '/^(transparent|currentColor)$/iD';
36+
private const PATTERN_RGB = '/^rgb\(\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d)\s*\)$/iD';
37+
private const PATTERN_RGBA = '/^rgba\(\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|255|25[0-4]|2[0-4]\d|1\d\d|0?\d?\d),\s*(0|0?\.\d+|1(\.0)?)\s*\)$/iD';
38+
private const PATTERN_HSL = '/^hsl\(\s*(0|360|35\d|3[0-4]\d|[12]\d\d|0?\d?\d),\s*(0|100|\d{1,2})%,\s*(0|100|\d{1,2})%\s*\)$/iD';
39+
private const PATTERN_HSLA = '/^hsla\(\s*(0|360|35\d|3[0-4]\d|[12]\d\d|0?\d?\d),\s*(0|100|\d{1,2})%,\s*(0|100|\d{1,2})%,\s*(0|0?\.\d+|1(\.0)?)\s*\)$/iD';
40+
41+
private const COLOR_PATTERNS = [
42+
CssColor::HEX_LONG => self::PATTERN_HEX_LONG,
43+
CssColor::HEX_LONG_WITH_ALPHA => self::PATTERN_HEX_LONG_WITH_ALPHA,
44+
CssColor::HEX_SHORT => self::PATTERN_HEX_SHORT,
45+
CssColor::HEX_SHORT_WITH_ALPHA => self::PATTERN_HEX_SHORT_WITH_ALPHA,
46+
CssColor::BASIC_NAMED_COLORS => self::PATTERN_BASIC_NAMED_COLORS,
47+
CssColor::EXTENDED_NAMED_COLORS => self::PATTERN_EXTENDED_NAMED_COLORS,
48+
CssColor::SYSTEM_COLORS => self::PATTERN_SYSTEM_COLORS,
49+
CssColor::KEYWORDS => self::PATTERN_KEYWORDS,
50+
CssColor::RGB => self::PATTERN_RGB,
51+
CssColor::RGBA => self::PATTERN_RGBA,
52+
CssColor::HSL => self::PATTERN_HSL,
53+
CssColor::HSLA => self::PATTERN_HSLA,
54+
];
55+
56+
/**
57+
* {@inheritdoc}
58+
*
59+
* @param CssColor $constraint
60+
*/
61+
public function create(Constraint $constraint, ApiProperty $propertyMetadata): array
62+
{
63+
return [
64+
'pattern' => '^('.implode('|', array_map(
65+
fn ($format) => trim(self::COLOR_PATTERNS[$format], '/iD^$'),
66+
(array) $constraint->formats
67+
)).')$',
68+
];
69+
}
70+
71+
/**
72+
* {@inheritdoc}
73+
*/
74+
public function supports(Constraint $constraint, ApiProperty $propertyMetadata): bool
75+
{
76+
return $constraint instanceof CssColor;
77+
}
78+
}
Lines changed: 71 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,71 @@
1+
<?php
2+
3+
/*
4+
* This file is part of the API Platform project.
5+
*
6+
* (c) Kévin Dunglas <[email protected]>
7+
*
8+
* For the full copyright and license information, please view the LICENSE
9+
* file that was distributed with this source code.
10+
*/
11+
12+
declare(strict_types=1);
13+
14+
namespace ApiPlatform\Tests\Symfony\Validator\Metadata\Property\Restriction;
15+
16+
use ApiPlatform\Metadata\ApiProperty;
17+
use ApiPlatform\Symfony\Validator\Metadata\Property\Restriction\PropertySchemaCssColorRestriction;
18+
use PHPUnit\Framework\Attributes\DataProvider;
19+
use PHPUnit\Framework\TestCase;
20+
use Symfony\Component\Validator\Constraints\CssColor;
21+
use Symfony\Component\Validator\Constraints\PositiveOrZero;
22+
23+
final class PropertySchemaCssColorRestrictionTest extends TestCase
24+
{
25+
private PropertySchemaCssColorRestriction $restriction;
26+
27+
protected function setUp(): void
28+
{
29+
$this->restriction = new PropertySchemaCssColorRestriction();
30+
}
31+
32+
#[DataProvider('supportsProvider')]
33+
public function testSupports(mixed $constraint, ApiProperty $propertyMetadata, bool $expected): void
34+
{
35+
self::assertSame($expected, $this->restriction->supports($constraint, $propertyMetadata));
36+
}
37+
38+
public static function supportsProvider(): \Generator
39+
{
40+
yield 'supported CssColor' => [new CssColor(), new ApiProperty(), true];
41+
yield 'unsupported other constraint' => [new PositiveOrZero(), new ApiProperty(), false];
42+
}
43+
44+
#[DataProvider('createProvider')]
45+
public function testCreate(CssColor $constraint, string $expectedPattern): void
46+
{
47+
$property = new ApiProperty();
48+
$result = $this->restriction->create($constraint, $property);
49+
50+
self::assertArrayHasKey('pattern', $result);
51+
self::assertSame($expectedPattern, $result['pattern']);
52+
}
53+
54+
public static function createProvider(): \Generator
55+
{
56+
yield 'HEX_LONG only' => [
57+
new CssColor(formats: [CssColor::HEX_LONG]),
58+
'^(#[0-9a-f]{6})$',
59+
];
60+
61+
yield 'HEX_SHORT and KEYWORDS' => [
62+
new CssColor(formats: [CssColor::HEX_SHORT, CssColor::KEYWORDS]),
63+
'^(#[0-9a-f]{3}|(transparent|currentColor))$',
64+
];
65+
66+
yield 'RGB + NAMED COLORS' => [
67+
new CssColor(formats: [CssColor::RGB, CssColor::BASIC_NAMED_COLORS]),
68+
'^(rgb\\(\\s*(0|255|25[0-4]|2[0-4]\\d|1\\d\\d|0?\\d?\\d),\\s*(0|255|25[0-4]|2[0-4]\\d|1\\d\\d|0?\\d?\\d),\\s*(0|255|25[0-4]|2[0-4]\\d|1\\d\\d|0?\\d?\\d)\\s*\\)|(black|silver|gray|white|maroon|red|purple|fuchsia|green|lime|olive|yellow|navy|blue|teal|aqua))$',
69+
];
70+
}
71+
}

0 commit comments

Comments
 (0)