") !== false) return false; if ($strict) { if (class_exists($f)) return false; if (!function_exists($f)) return false; } return true; } private static function is_invalid(?string $m): bool { return $m === null || $m === "" || $m === "::" || $m === "->"; } private static function is_nfunction(?string $m): bool { return strpos($m, "\\") !== false; } private static function parse_static(?string &$m): bool { $pos = strpos($m, "::"); if ($pos === false) return false; $m = substr($m, $pos + 2); return true; } private static function parse_method(?string &$m): bool { $pos = strpos($m, "->"); if ($pos === false) return false; $m = substr($m, $pos + 2); return true; } private static function parse_class_s(?string $cs, ?string &$c, ?string &$s): bool { if (self::is_invalid($cs) || self::parse_method($cs)) return false; $pos = strpos($cs, "::"); if ($pos === false) return false; if ($pos === 0) return false; $tmpc = substr($cs, 0, $pos); $cs = substr($cs, $pos + 2); if (self::is_nfunction($cs)) return false; [$c, $s] = [$tmpc, cv::vn($cs)]; return true; } private static function parse_c_static(?string $cs, ?string &$c, ?string &$s, ?bool &$bound): bool { if (self::is_invalid($cs) || self::parse_method($cs)) return false; $pos = strpos($cs, "::"); if ($pos === false) return false; if ($pos == strlen($cs) - 2) return false; if ($pos > 0) { $tmpc = substr($cs, 0, $pos); $bound = true; } else { $tmpc = null; $bound = false; } $cs = substr($cs, $pos + 2); if (self::is_nfunction($cs)) return false; [$c, $s] = [$tmpc, cv::vn($cs)]; return true; } /** * vérifier que $func est une méthode statique, et la normaliser le cas * échéant. retourner true si c'est une méthode statique, false sinon * * les formes suivantes sont supportées (XXX étant null ou n'importe quelle * valeur scalaire de n'importe quel type sauf false) * - "XXX::function" * - ["XXX::function", ...$args] * - [XXX, "::function", ...$args] * - [XXX, "function", ...$args] c'est la forme normalisée * * Si XXX est une classe, la méthode statique est liée. sinon, elle doit être * liée à une classe avant d'être utilisée * * @param bool $strict vérifier l'existence de la classe et de la méthode si * la méthode est liée (ne pas uniquement faire une vérification syntaxique) */ static function verifix_static(&$func, bool $strict=true, ?bool &$bound=null): bool { if (is_string($func)) { if (!self::parse_c_static($func, $c, $f, $bound)) return false; $func = [$c, $f]; } elseif (is_array($func)) { if (!array_key_exists(0, $func)) return false; $c = $func[0]; if ($c === false) return false; if (self::parse_class_s($c, $c, $f)) { $func[0] = $c; if ($f !== null) { # ["class::method"] --> ["class", "method"] array_splice($func, 1, 0, [$f]); } $bound = true; } elseif (self::parse_c_static($c, $c, $f, $bound)) { # ["::method"] --> [null, "method"] array_splice($func, 0, 0, [null]); $func[1] = $f; } else { if (self::is_invalid($c)) $c = null; $func[0] = $c; $bound = is_string($c); } # if (!array_key_exists(1, $func)) return false; $f = $func[1]; if (!is_string($f)) return false; if (self::parse_c_static($f, $rc, $f, $rbound)) { if ($rc !== null && $c === null) { $c = $rc; $bound = $rbound; } } elseif (self::is_invalid($f)) { return false; } elseif (self::is_nfunction($f)) { return false; } elseif (self::parse_method($f)) { return false; } elseif (self::parse_static($f)) { } $func[1] = $f; } else { return false; } if ($strict && $bound && !method_exists($c, $f)) return false; return true; } private static function parse_class_m(?string $cm, ?string &$c, ?string &$m): bool { if (self::is_invalid($cm) || self::parse_static($cm)) return false; $pos = strpos($cm, "->"); if ($pos === false) return false; if ($pos === 0) return false; $tmpc = substr($cm, 0, $pos); $cm = substr($cm, $pos + 2); if (self::is_nfunction($cm)) return false; [$c, $m] = [$tmpc, cv::vn($cm)]; return true; } private static function parse_c_method(?string $cm, ?string &$c, ?string &$m, ?bool &$bound): bool { if (self::is_invalid($cm) || self::parse_static($cm)) return false; $pos = strpos($cm, "->"); if ($pos === false) return false; if ($pos == strlen($cm) - 2) return false; if ($pos > 0) { $tmpc = substr($cm, 0, $pos); $bound = true; } else { $tmpc = null; $bound = false; } $cm = substr($cm, $pos + 2); if (self::is_nfunction($cm)) return false; [$c, $m] = [$tmpc, cv::vn($cm)]; return true; } /** * vérifier que $func est une méthode non statique, et la normaliser le cas * échéant. retourner true si c'est une méthode non statique, false sinon * * les formes suivantes sont supportées (XXX étant null ou n'importe quelle * valeur scalaire de n'importe quel type sauf false) * - "XXX->function" * - ["XXX->function", ...$args] * - [XXX, "->function", ...$args] * - [XXX, "function", ...$args] c'est la forme normalisée * * Si XXX est une classe, la méthode est liée. sinon, elle doit être liée à un * objet avant d'être utilisée * * @param bool $strict vérifier l'existence de la classe et de la méthode si * la méthode est liée (ne pas uniquement faire une vérification syntaxique) */ static function verifix_method(&$func, bool $strict=true, ?bool &$bound=null): bool { if (is_string($func)) { if (!self::parse_c_method($func, $c, $f, $bound)) return false; $func = [$c, $f]; } elseif (is_array($func)) { if (!array_key_exists(0, $func)) return false; $c = $func[0]; if ($c === false) return false; if (self::parse_class_m($c, $c, $f)) { $func[0] = $c; if ($f !== null) { # ["class->method"] --> ["class", "method"] array_splice($func, 1, 0, [$f]); } $bound = true; } elseif (self::parse_c_method($c, $c, $f, $bound)) { # ["->method"] --> [null, "method"] array_splice($func, 0, 0, [null]); $func[1] = $f; } else { if (self::is_invalid($c)) $c = null; $func[0] = $c; $bound = is_string($c); } # if (!array_key_exists(1, $func)) return false; $f = $func[1]; if (!is_string($f)) return false; if (self::parse_c_method($f, $rc, $f, $rbound)) { if ($rc !== null && $c === null) { $c = $rc; $bound = $rbound; } } elseif (self::is_invalid($f)) { return false; } elseif (self::is_nfunction($f)) { return false; } elseif (self::parse_static($f)) { return false; } elseif (self::parse_method($f)) { } $func[1] = $f; } else { return false; } if ($strict && $bound && !method_exists($c, $f)) return false; return true; } static function verifix_cons(&$func): bool { if (is_string($func)) { return true; } elseif (is_array($func)) { return true; } else { return false; } } }