2016-12-29 14:09:49 -06:00
|
|
|
<?php
|
2016-08-15 23:46:26 -05:00
|
|
|
|
2014-11-11 11:11:58 -05:00
|
|
|
namespace Amp;
|
|
|
|
|
2015-01-28 22:30:02 -05:00
|
|
|
/**
|
2017-04-24 15:39:08 +02:00
|
|
|
* A "safe" struct trait for public property aggregators.
|
2015-02-05 18:17:05 +01:00
|
|
|
*
|
2015-04-03 11:56:16 -04:00
|
|
|
* This trait is intended to make using public properties a little safer by throwing when
|
|
|
|
* nonexistent property names are read or written.
|
2015-01-28 22:30:02 -05:00
|
|
|
*/
|
2018-06-18 20:00:01 +02:00
|
|
|
trait Struct
|
|
|
|
{
|
2015-05-13 10:05:23 -04:00
|
|
|
/**
|
|
|
|
* The minimum percentage [0-100] at which to recommend a similar property
|
|
|
|
* name when generating error messages.
|
|
|
|
*/
|
2020-09-24 22:17:13 -05:00
|
|
|
private int $__propertySuggestThreshold = 70;
|
2015-05-13 10:05:23 -04:00
|
|
|
|
2020-03-28 22:20:44 +01:00
|
|
|
/**
|
|
|
|
* @param string $property
|
|
|
|
*
|
|
|
|
* @psalm-return no-return
|
|
|
|
*/
|
2018-06-18 20:00:01 +02:00
|
|
|
public function __get(string $property)
|
|
|
|
{
|
2016-08-12 16:58:53 -05:00
|
|
|
throw new \Error(
|
2015-04-03 11:56:16 -04:00
|
|
|
$this->generateStructPropertyError($property)
|
2014-11-11 11:11:58 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-03-28 22:20:44 +01:00
|
|
|
/**
|
|
|
|
* @param string $property
|
|
|
|
* @param mixed $value
|
|
|
|
*
|
|
|
|
* @psalm-return no-return
|
|
|
|
*/
|
2018-06-18 20:00:01 +02:00
|
|
|
public function __set(string $property, $value)
|
|
|
|
{
|
2016-08-12 16:58:53 -05:00
|
|
|
throw new \Error(
|
2015-04-03 11:56:16 -04:00
|
|
|
$this->generateStructPropertyError($property)
|
2014-11-11 11:11:58 -05:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-06-18 20:00:01 +02:00
|
|
|
private function generateStructPropertyError(string $property): string
|
|
|
|
{
|
2015-05-13 10:05:23 -04:00
|
|
|
$suggestion = $this->suggestPropertyName($property);
|
|
|
|
$suggestStr = ($suggestion == "") ? "" : " ... did you mean \"{$suggestion}?\"";
|
2017-01-07 22:55:34 +01:00
|
|
|
|
2016-08-17 22:25:54 -05:00
|
|
|
return \sprintf(
|
2015-05-13 10:05:23 -04:00
|
|
|
"%s property \"%s\" does not exist%s",
|
2016-08-17 22:25:54 -05:00
|
|
|
\str_replace("\0", "@", \get_class($this)), // Handle anonymous class names.
|
2015-05-13 10:05:23 -04:00
|
|
|
$property,
|
|
|
|
$suggestStr
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2018-06-18 20:00:01 +02:00
|
|
|
private function suggestPropertyName(string $badProperty): string
|
|
|
|
{
|
2016-08-17 22:25:54 -05:00
|
|
|
$badProperty = \strtolower($badProperty);
|
2015-05-13 10:05:23 -04:00
|
|
|
$bestMatch = "";
|
2017-01-07 22:55:34 +01:00
|
|
|
$bestMatchPercentage = 0;
|
|
|
|
|
2020-03-28 22:20:44 +01:00
|
|
|
/** @psalm-suppress RawObjectIteration */
|
2015-05-13 10:05:23 -04:00
|
|
|
foreach ($this as $property => $value) {
|
|
|
|
// Never suggest properties that begin with an underscore
|
|
|
|
if ($property[0] === "_") {
|
|
|
|
continue;
|
|
|
|
}
|
2016-08-17 22:25:54 -05:00
|
|
|
\similar_text($badProperty, \strtolower($property), $byRefPercentage);
|
2015-05-13 10:05:23 -04:00
|
|
|
if ($byRefPercentage > $bestMatchPercentage) {
|
|
|
|
$bestMatchPercentage = $byRefPercentage;
|
|
|
|
$bestMatch = $property;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return ($bestMatchPercentage >= $this->__propertySuggestThreshold) ? $bestMatch : "";
|
2014-11-11 11:11:58 -05:00
|
|
|
}
|
|
|
|
}
|