<?php
abstract class Filter
{
abstract public static function check($value, array $options = array());
}
final class IntFilter
{
public static function check($value, array $options = array())
{
return $value;
}
}
final class StringFilter
{
public static function check($value, array $options = array())
{
if (false === is_string($value)) {
return false;
}
if (false !== ($options['trim'] ?? true)) {
$value = trim($value);
}
$length = mb_strlen($value, '8bit');
if ((false === ($options['empty'] ?? true)) && (0 === $length)) {
return false;
}
foreach (['min', 'max'] as $name) {
if (isset($options[$name])) {
$optionLength = IntFilter::check($options[$name], ['min' => 1]);
if (('min' === $name && $length < $optionLength) || ('max' === $name && $length > $optionLength)) {
return false;
}
}
}
foreach (['startsWith', 'endsWith'] as $name) {
if (isset($options[$name])) {
$needle = StringFilter::check($options[$name]);
if (false === self::$name($value, $needle)) {
return false;
}
}
}
return $value;
}
private static function startsWith($haystack, $needle)
{
return $haystack[0] === $needle[0] ? strncmp($haystack, $needle, mb_strlen($needle, '8bit')) === 0 : false;
}
private static function endsWith($haystack, $needle)
{
return self::startsWith(mb_substr($haystack, -mb_strlen($needle, '8bit')), $needle);
}
}
var_dump(StringFilter::check(' r a ', ['min' => 4]));
var_dump(StringFilter::check(' r a ', ['max' => 4]));
var_dump(StringFilter::check(' r a ', ['min' => 4, 'max' => 4]));
var_dump(StringFilter::check(' r a ', ['min' => 5]));
var_dump(StringFilter::check(' r a ', ['max' => 3]));
var_dump(StringFilter::check(' r a ', ['min' => 5, 'max' => 4]));
var_dump(StringFilter::check(' r a ', ['min' => 4, 'max' => 3]));
var_dump(StringFilter::check(' r a ', ['min' => 5, 'max' => 3]));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'r']));
var_dump(StringFilter::check(' r a ', ['endsWith' => 'a']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'r', 'endsWith' => 'a']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'r at']));
var_dump(StringFilter::check(' r a ', ['endsWith' => 'r at']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'r at', 'endsWith' => 'r at']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'z']));
var_dump(StringFilter::check(' r a ', ['endsWith' => 'z']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'r', 'endsWith' => 'z']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'z', 'endsWith' => 'a']));
var_dump(StringFilter::check(' r a ', ['startsWith' => 'z', 'endsWith' => 'y']));
- Output for git.master, git.master_jit, rfc.property-hooks
- string(4) "r a"
string(4) "r a"
string(4) "r a"
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
string(4) "r a"
string(4) "r a"
string(4) "r a"
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
bool(false)
This tab shows result from various feature-branches currently under review by the php developers. Contact me to have additional branches featured.
Active branches
Archived branches
Once feature-branches are merged or declined, they are no longer available. Their functionality (when merged) can be viewed from the main output page
preferences:
55.18 ms | 401 KiB | 8 Q