ILIAS  trunk Revision v11.0_alpha-1702-gfd3ecb7f852
All Data Structures Namespaces Files Functions Variables Enumerations Enumerator Modules Pages
AbstractFactoryTestCase.php
Go to the documentation of this file.
1 <?php
2 
19 declare(strict_types=1);
20 
21 require_once("vendor/composer/vendor/autoload.php");
22 
26 
36 abstract class AbstractFactoryTestCase extends TestCase
37 {
38  public const COMPONENT = 1;
39  public const FACTORY = 2;
40 
41  /* This allows to omit checking of certain factory methods, use prudently...
42  */
43  private static array $omit_factory_methods = [
44  "helpTopics"
45  ];
46 
47  /* kitchensink info test configuration:
48  * true = should be there, check
49  * false = may be there, don't check
50  * Notice, some properties (MUST/MUST NOT) will always be checked.
51  */
52  private static array $kitchensink_info_settings_default = [
53  'description' => true,
54  'background' => false,
55  'context' => true,
56  'featurewiki' => false,
57  'javascript' => false,
58  'rules' => true
59  ];
60 
61  /* You can overwrite these settings per factory method when using this test
62  * by writing $kitchensink_info_settings. See GlyphFactoryTest for an example.
63  */
64 
65 
66  // Definitions and Helpers:
67 
68  private static array $description_categories = ['purpose', 'composition', 'effect', 'rival'];
69 
70  private static array $rules_categories = [
71  'usage',
72  'interaction',
73  'wording',
74  'style',
75  'ordering',
76  'responsiveness',
77  'composition',
78  'accessibility'
79  ];
80 
81  private Crawler\EntriesYamlParser $yaml_parser;
82  private static ReflectionClass $reflection;
83  public static string $factory_title = '';
84 
85  final protected function returnsFactory(array $docstring_data): bool
86  {
87  return $this->isFactoryName($docstring_data["namespace"]);
88  }
89 
90  final protected function returnsComponent(array $docstring_data): bool
91  {
92  $reflection = new ReflectionClass($docstring_data["namespace"]);
93  return in_array("ILIAS\\UI\\Component\\Component", $reflection->getInterfaceNames());
94  }
95 
96  final protected function isFactoryName(string $name): bool
97  {
98  return preg_match("#^(\\\\)?ILIAS\\\\UI\\\\Component\\\\([a-zA-Z]+\\\\)*Factory$#", $name) === 1;
99  }
100 
101  final public static function buildFactoryReflection(): ReflectionClass
102  {
103  return new ReflectionClass(static::$factory_title);
104  }
105 
106  final public static function getMethodsProvider(): array
107  {
108 
109  $reflection = self::buildFactoryReflection();
110  return array_filter(
111  array_map(function ($element) {
112  if (!in_array($element->getName(), self::$omit_factory_methods)) {
113  return array($element, $element->getName());
114  }
115  return false;
116  }, $reflection->getMethods())
117  );
118  }
119 
120  // Setup
121 
122  public function setUp(): void
123  {
124  $this->yaml_parser = new Crawler\EntriesYamlParser();
125  self::$reflection = $this->buildFactoryReflection();
126  }
127 
128  public function testProperNamespace(): void
129  {
130  $message = "TODO: Put your factory into the proper namespace.";
131  $this->assertMatchesRegularExpression(
132  "#^ILIAS\\\\UI\\\\Component.#",
133  self::$reflection->getNamespaceName(),
134  $message
135  );
136  }
137 
138  public function testProperName(): void
139  {
140  $name = self::$reflection->getName();
141  $message = "TODO: Give your factory a proper name.";
142  $this->assertTrue($this->isFactoryName($name), $message);
143  }
144 
150  final public function testCheckYamlExtraction(ReflectionMethod $method_reflection, string $name): array
151  {
152  try {
153  //Todo (TA) this is not pretty. We should think about using only reflection in the parser as well.
154  $function_name_string = "\n public function " . $method_reflection->getName() . "()";
155  $docstring_data = $this->yaml_parser->parseArrayFromString(
156  $method_reflection->getDocComment() . $function_name_string
157  );
158  $this->assertTrue(true);
160  $message = "TODO ($name): fix parse error in kitchen sink yaml: " . $e->getMessage();
161  $this->assertTrue(false, $message);
162  }
163  $this->assertCount(1, $docstring_data);
164  return $docstring_data[0];
165  }
166 
172  final public function testReturnType(ReflectionMethod $method_reflection, string $name): void
173  {
174  $message = "TODO ($name): fix return type, it must be a factory or a component.";
175  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
176  if ($this->returnsFactory($docstring_data)) {
177  $this->assertTrue(true);
178  } elseif ($this->returnsComponent($docstring_data)) {
179  $this->assertTrue(true);
180  } else {
181  $this->assertTrue(false, $message);
182  }
183  }
184 
191  ReflectionMethod $method_reflection,
192  string $name
193  ): void {
194  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
195  $this->testReturnType($method_reflection, $name);
196 
197  $return_doc = $docstring_data["namespace"];
198  $name_uppercase = ucwords($name);
199  $regex_factory_namespace = $this->getRegexFactoryNamespace();
200  $regex_head = "#^(\\\\?)$regex_factory_namespace";
201 
202  $message = "TODO ($name): fix @return, it does not match the method name.";
203  if ($this->returnsFactory($docstring_data)) {
204  $this->assertMatchesRegularExpression(
205  "$regex_head\\\\$name_uppercase\\\\Factory$#",
206  $return_doc,
207  $message
208  );
209  } else { // returnsComponent
210  // Every component MUST be described by a single interface, where the name of
211  // the interface corresponds to the name of the component.
212  $standard_pattern = "$regex_head\\\\$name_uppercase#";
213  $standard_case = preg_match($standard_pattern, $return_doc);
214 
215  // unless they only differ in a type and share a common prefix to their pathes.
216  $namespace_parts = explode("\\", self::$reflection->getNamespaceName());
217  $typediff_only_pattern = "$regex_head\\\\" . array_pop($namespace_parts) . "#";
218  $typediff_only_case = preg_match($typediff_only_pattern, $return_doc);
219 
220  $this->assertTrue($standard_case || $typediff_only_case, $message);
221  }
222  }
223 
224  protected function getRegexFactoryNamespace(): string
225  {
226  return str_replace("\\", "\\\\", self::$reflection->getNamespaceName());
227  }
228 
234  final public function testMethodParams(ReflectionMethod $method_reflection, string $name): void
235  {
236  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
237  if ($this->returnsFactory($docstring_data)) {
238  $message = "TODO ($name): remove params from method that returns Factory.";
239  $this->assertEquals(0, $method_reflection->getNumberOfParameters(), $message);
240  }
241  }
242 
243  // Common rules for all factory methods, regardless whether they return other
244  // factories or components.
245 
249  final public function testKitchensinkInfoDescription(ReflectionMethod $method_reflection, string $name): void
250  {
251  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
252  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
253 
254  if ($kitchensink_info_settings['description']) {
255  $message = "TODO ($name): add a description.";
256  $this->assertArrayHasKey('description', $docstring_data, $message);
257 
258  $desc_fields = implode(", ", static::$description_categories);
259  $message = "TODO ($name): the description field should at least contain one of these: $desc_fields.";
260  $existing_keys = array_keys($docstring_data["description"]);
261  $existing_expected_keys = array_intersect(static::$description_categories, $existing_keys);
262  $this->assertGreaterThanOrEqual(
263  1,
264  $existing_expected_keys,
265  $message
266  );
267  }
268  }
269 
273  final public function testKitchensinkInfoRivals(ReflectionMethod $method_reflection, string $name): void
274  {
275  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
276  if (isset($docstring_data["description"]) && isset($docstring_data["description"]["rivals"])) {
277  $rules = $docstring_data["description"]["rivals"];
278  $message = "TODO ($name): The Rivals field has a non-string index. Format like 'rival_name': 'description'";
279  $this->assertTrue(array_unique(array_map("is_string", array_keys($rules))) === array(true), $message);
280  }
281  $this->assertTrue(true);
282  }
283 
287  final public function testKitchensinkInfoBackground(ReflectionMethod $method_reflection, string $name): void
288  {
289  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
290  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
291 
292  if ($kitchensink_info_settings['background']) {
293  $message = "TODO ($name): add a background field.";
294  $this->assertArrayHasKey('background', $docstring_data, $message);
295  }
296  }
297 
301  final public function testKitchensinkInfoFeatureWiki(ReflectionMethod $method_reflection, string $name): void
302  {
303  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
304  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
305 
306  if ($kitchensink_info_settings['featurewiki']) {
307  $message = "TODO ($name): add a featurewiki field.";
308  $this->assertArrayHasKey('featurewiki', $docstring_data, $message);
309  }
310  }
311 
315  final public function testKitchensinkInfoJavaScript(ReflectionMethod $method_reflection, string $name): void
316  {
317  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
318  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
319 
320  if ($kitchensink_info_settings['javascript']) {
321  $message = "TODO ($name): add a javascript field.";
322  $this->assertArrayHasKey('javascript', $docstring_data, $message);
323  }
324  }
325 
329  final public function testKitchensinkInfoRules(ReflectionMethod $method_reflection, string $name): void
330  {
331  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
332  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
333 
334  if ($kitchensink_info_settings['rules']) {
335  $message = "TODO ($name): add a rules field.";
336  $this->assertArrayHasKey('rules', $docstring_data, $message);
337 
338  $rules_fields = implode(", ", static::$rules_categories);
339  $message = "TODO ($name): the rules field should at least contain one of these: $rules_fields.";
340  $existing_keys = array_keys($docstring_data["rules"]);
341  $existing_expected_keys = array_intersect(static::$rules_categories, $existing_keys);
342  $this->assertGreaterThanOrEqual(
343  1,
344  $existing_expected_keys,
345  $message
346  );
347  }
348  }
349 
353  final public function testKitchensinkInfoContext(ReflectionMethod $method_reflection, string $name): void
354  {
355  $docstring_data = $this->testCheckYamlExtraction($method_reflection, $name);
356  $kitchensink_info_settings = $this->kitchensinkInfoSettingsMergedWithDefaults($name);
357  if (!$this->returnsFactory($docstring_data) && $kitchensink_info_settings["context"]) {
358  $message = "TODO ($name): factory method returning component should have context field. Add it.";
359  $this->assertArrayHasKey("context", $docstring_data, $message);
360  }
361  }
362 
363  final public function kitchensinkInfoSettingsMergedWithDefaults(string $name): array
364  {
365  if (array_key_exists($name, static::$kitchensink_info_settings)) {
366  return array_merge(
367  static::$kitchensink_info_settings_default,
368  static::$kitchensink_info_settings[$name]
369  );
370  } else {
371  return static::$kitchensink_info_settings_default;
372  }
373  }
374 }
returnsFactory(array $docstring_data)
testKitchensinkInfoDescription(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
testReturnType(ReflectionMethod $method_reflection, string $name)
Tests whether the method either returns a factory or a component.
testKitchensinkInfoFeatureWiki(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
testKitchensinkInfoBackground(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
testKitchensinkInfoContext(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
returnsComponent(array $docstring_data)
static array static array $description_categories
testKitchensinkInfoRules(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
testFactoryMethodNameCompatibleDocstring(ReflectionMethod $method_reflection, string $name)
Tests whether the method name matches the return doctring?
testMethodParams(ReflectionMethod $method_reflection, string $name)
Tests whether methods returning factories have no parameters.
static ReflectionClass $reflection
testKitchensinkInfoRivals(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
$message
Definition: xapiexit.php:31
static array $kitchensink_info_settings_default
Crawler EntriesYamlParser $yaml_parser
kitchensinkInfoSettingsMergedWithDefaults(string $name)
testKitchensinkInfoJavaScript(ReflectionMethod $method_reflection, string $name)
getMethodsProvider
Defines tests every SHOULD pass UI-factory.
testCheckYamlExtraction(ReflectionMethod $method_reflection, string $name)
Tests whether the YAML Kitchen Sink info can be parsed.