diff --git a/.devcontainer/devcontainer.json b/.devcontainer/devcontainer.json new file mode 100644 index 00000000..a103e397 --- /dev/null +++ b/.devcontainer/devcontainer.json @@ -0,0 +1,9 @@ +{ + "image": "mcr.microsoft.com/devcontainers/universal:2", + "features": { + "ghcr.io/devcontainers/features/php:1": { + "version": "8.3", + "installComposer": true + } + } + } \ No newline at end of file diff --git a/exercises/practice/list-ops/ListOpsTest.php b/exercises/practice/list-ops/ListOpsTest.php index 4d0f35ea..e2a2cb8f 100644 --- a/exercises/practice/list-ops/ListOpsTest.php +++ b/exercises/practice/list-ops/ListOpsTest.php @@ -33,231 +33,311 @@ public static function setUpBeforeClass(): void require_once 'ListOps.php'; } + /** * @testdox append entries to a list and return the new list -> empty lists */ - public function testAppendEmptyLists() + public function testAppendEntriesToAListAndReturnTheNewListWithEmptyLists() { $listOps = new ListOps(); - $this->assertEquals([], $listOps->append([], [])); + $list1 = []; + $list2 = []; + + $result = $listOps->append($list1, $list2); + + $this->assertEquals([], $result); } /** * @testdox append entries to a list and return the new list -> list to empty list */ - public function testAppendNonEmptyListToEmptyList() + public function testAppendEntriesToAListAndReturnTheNewListWithListToEmptyList() { $listOps = new ListOps(); - $this->assertEquals([1, 2, 3, 4], $listOps->append([1, 2, 3, 4], [])); + $list1 = []; + $list2 = [1, 2, 3, 4]; + + $result = $listOps->append($list1, $list2); + + $this->assertEquals([1, 2, 3, 4], $result); } /** * @testdox append entries to a list and return the new list -> empty list to list */ - public function testAppendEmptyListToNonEmptyList() + public function testAppendEntriesToAListAndReturnTheNewListWithEmptyListToList() { $listOps = new ListOps(); - $this->assertEquals([1, 2, 3, 4], $listOps->append([], [1, 2, 3, 4])); + $list1 = [1, 2, 3, 4]; + $list2 = []; + + $result = $listOps->append($list1, $list2); + + $this->assertEquals([1, 2, 3, 4], $result); } /** * @testdox append entries to a list and return the new list -> non-empty lists */ - public function testAppendNonEmptyLists() + public function testAppendEntriesToAListAndReturnTheNewListWithNonEmptyLists() { $listOps = new ListOps(); - $this->assertEquals([1, 2, 2, 3, 4, 5], $listOps->append([1, 2], [2, 3, 4, 5])); + $list1 = [1, 2]; + $list2 = [2, 3, 4, 5]; + + $result = $listOps->append($list1, $list2); + + $this->assertEquals([1, 2, 2, 3, 4, 5], $result); } /** * @testdox concatenate a list of lists -> empty list */ - public function testConcatEmptyLists() + public function testConcatenateAListOfListsWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals([], $listOps->concat([], [])); + $lists = []; + + $result = $listOps->concat($lists); + + $this->assertEquals([], $result); } /** * @testdox concatenate a list of lists -> list of lists */ - public function testConcatLists() + public function testConcatenateAListOfListsWithListOfLists() { $listOps = new ListOps(); - $this->assertEquals([1, 2, 3, 4, 5, 6], $listOps->concat([1, 2], [3], [], [4, 5, 6])); + $lists = [[1, 2], [3], [], [4, 5, 6]]; + + $result = $listOps->concat($lists); + + $this->assertEquals([1, 2, 3, 4, 5, 6], $result); } /** * @testdox concatenate a list of lists -> list of nested lists */ - public function testConcatNestedLists() + public function testConcatenateAListOfListsWithListOfNestedLists() { $listOps = new ListOps(); - $this->assertEquals([[1], [2], [3], [], [4, 5, 6]], $listOps->concat([[1], [2]], [[3]], [[]], [[4, 5, 6]])); + $lists = [[[1], [2]], [[3]], [[]], [[4, 5, 6]]]; + + $result = $listOps->concat($lists); + + $this->assertEquals([[1], [2], [3], [], [4, 5, 6]], $result); } /** * @testdox filter list returning only values that satisfy the filter function -> empty list */ - public function testFilterEmptyList() + public function testFilterListReturningOnlyValuesThatSatisfyTheFilterFunctionWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - [], - $listOps->filter(static fn ($el) => $el % 2 === 1, []) - ); + $list = []; + $function = static fn ($el) => $el % 2 === 1; + + $result = $listOps->filter($list, $function); + + $this->assertEquals([], $result); } /** - * @testdox filter list returning only values that satisfy the filter function -> non empty list + * @testdox filter list returning only values that satisfy the filter function -> non-empty list */ - public function testFilterNonEmptyList() + public function testFilterListReturningOnlyValuesThatSatisfyTheFilterFunctionWithNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - [1, 3, 5], - $listOps->filter(static fn ($el) => $el % 2 === 1, [1, 2, 3, 5]) - ); + $list = [1, 2, 3, 5]; + $function = static fn ($el) => $el % 2 === 1; + + $result = $listOps->filter($list, $function); + + $this->assertEquals([1, 3, 5], $result); } /** * @testdox returns the length of a list -> empty list */ - public function testLengthEmptyList() + public function testReturnsTheLengthOfAListWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals(0, $listOps->length([])); + $list = []; + + $result = $listOps->length($list); + + $this->assertEquals(0, $result); } /** * @testdox returns the length of a list -> non-empty list */ - public function testLengthNonEmptyList() + public function testReturnsTheLengthOfAListWithNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals(4, $listOps->length([1, 2, 3, 4])); + $list = [1, 2, 3, 4]; + + $result = $listOps->length($list); + + $this->assertEquals(4, $result); } /** - * @testdox returns a list of elements whose values equal the list value transformed by the mapping function -> empty list + * @testdox return a list of elements whose values equal the list value transformed by the mapping function -> empty list */ - public function testMapEmptyList() + public function testReturnAListOfElementsWhoseValuesEqualTheListValueTransformedByTheMappingFunctionWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - [], - $listOps->map(static fn ($el) => $el + 1, []) - ); + $list = []; + $function = static fn ($el) => $el + 1; + + $result = $listOps->map($list, $function); + + $this->assertEquals([], $result); } /** - * @testdox returns a list of elements whose values equal the list value transformed by the mapping function -> non-empty list + * @testdox return a list of elements whose values equal the list value transformed by the mapping function -> non-empty list */ - public function testMapNonEmptyList() + public function testReturnAListOfElementsWhoseValuesEqualTheListValueTransformedByTheMappingFunctionWithNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - [2, 4, 6, 8], - $listOps->map(static fn ($el) => $el + 1, [1, 3, 5, 7]) - ); + $list = [1, 3, 5, 7]; + $function = static fn ($el) => $el + 1; + + $result = $listOps->map($list, $function); + + $this->assertEquals([2, 4, 6, 8], $result); } /** * @testdox folds (reduces) the given list from the left with a function -> empty list */ - public function testFoldlEmptyList() + public function testFoldsReducesTheGivenListFromTheLeftWithAFunctionWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 2, - $listOps->foldl(static fn ($acc, $el) => $el * $acc, [], 2) - ); + $list = []; + $initial = 2; + $function = static fn ($acc, $el) => $el * $acc; + + $result = $listOps->foldl($list, $initial, $function); + + $this->assertEquals(2, $result); } /** * @testdox folds (reduces) the given list from the left with a function -> direction independent function applied to non-empty list */ - public function testFoldlDirectionIndependentNonEmptyList() + public function testFoldsReducesTheGivenListFromTheLeftWithAFunctionWithDirectionIndependentFunctionAppliedToNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 15, - $listOps->foldl(static fn ($acc, $el) => $acc + $el, [1, 2, 3, 4], 5) - ); + $list = [1, 2, 3, 4]; + $initial = 5; + $function = static fn ($acc, $el) => $el + $acc; + + $result = $listOps->foldl($list, $initial, $function); + + $this->assertEquals(15, $result); } /** * @testdox folds (reduces) the given list from the left with a function -> direction dependent function applied to non-empty list */ - public function testFoldlDirectionDependentNonEmptyList() + public function testFoldsReducesTheGivenListFromTheLeftWithAFunctionWithDirectionDependentFunctionAppliedToNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 64, - $listOps->foldl(static fn ($acc, $el) => $el / $acc, [1, 2, 3, 4], 24) - ); + $list = [1, 2, 3, 4]; + $initial = 24; + $function = static fn ($acc, $el) => $el / $acc; + + $result = $listOps->foldl($list, $initial, $function); + + $this->assertEquals(64, $result); } /** * @testdox folds (reduces) the given list from the right with a function -> empty list */ - public function testFoldrEmptyList() + public function testFoldsReducesTheGivenListFromTheRightWithAFunctionWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 2, - $listOps->foldr(static fn ($acc, $el) => $el * $acc, [], 2) - ); + $list = []; + $initial = 2; + $function = static fn ($acc, $el) => $el * $acc; + + $result = $listOps->foldr($list, $initial, $function); + + $this->assertEquals(2, $result); } /** * @testdox folds (reduces) the given list from the right with a function -> direction independent function applied to non-empty list */ - public function testFoldrDirectionIndependentNonEmptyList() + public function testFoldsReducesTheGivenListFromTheRightWithAFunctionWithDirectionIndependentFunctionAppliedToNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 15, - $listOps->foldr(static fn ($acc, $el) => $acc + $el, [1, 2, 3, 4], 5) - ); + $list = [1, 2, 3, 4]; + $initial = 5; + $function = static fn ($acc, $el) => $el + $acc; + + $result = $listOps->foldr($list, $initial, $function); + + $this->assertEquals(15, $result); } /** * @testdox folds (reduces) the given list from the right with a function -> direction dependent function applied to non-empty list */ - public function testFoldrDirectionDependentNonEmptyList() + public function testFoldsReducesTheGivenListFromTheRightWithAFunctionWithDirectionDependentFunctionAppliedToNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals( - 9, - $listOps->foldr(static fn ($acc, $el) => $el / $acc, [1, 2, 3, 4], 24) - ); + $list = [1, 2, 3, 4]; + $initial = 24; + $function = static fn ($acc, $el) => $el / $acc; + + $result = $listOps->foldr($list, $initial, $function); + + $this->assertEquals(9, $result); } /** - * @testdox reverse the elements of a list -> empty list + * @testdox reverse the elements of the list -> empty list */ - public function testReverseEmptyList() + public function testReverseTheElementsOfTheListWithEmptyList() { $listOps = new ListOps(); - $this->assertEquals([], $listOps->reverse([])); + $list = []; + + $result = $listOps->reverse($list); + + $this->assertEquals([], $result); } /** - * @testdox reverse the elements of a list -> non-empty list + * @testdox reverse the elements of the list -> non-empty list */ - public function testReverseNonEmptyList() + public function testReverseTheElementsOfTheListWithNonEmptyList() { $listOps = new ListOps(); - $this->assertEquals([7, 5, 3, 1], $listOps->reverse([1, 3, 5, 7])); + $list = [1, 3, 5, 7]; + + $result = $listOps->reverse($list); + + $this->assertEquals([7, 5, 3, 1], $result); } /** - * @testdox reverse the elements of a list -> list of lists is not flattened + * @testdox reverse the elements of the list -> list of lists is not flattened */ - public function testReverseNonEmptyListIsNotFlattened() + public function testReverseTheElementsOfTheListWithListOfListsIsNotFlattened() { $listOps = new ListOps(); - $this->assertEquals([[4, 5, 6], [], [3], [1, 2]], $listOps->reverse([[1, 2], [3], [], [4, 5, 6]])); + $list = [[1, 2], [3], [], [4, 5, 6]]; + + $result = $listOps->reverse($list); + + $this->assertEquals([[4, 5, 6], [], [3], [1, 2]], $result); + } + } -} diff --git a/exercises/practice/list-ops/ListOpsTest.php.twig b/exercises/practice/list-ops/ListOpsTest.php.twig new file mode 100644 index 00000000..056bf27d --- /dev/null +++ b/exercises/practice/list-ops/ListOpsTest.php.twig @@ -0,0 +1,65 @@ + x modulo 2 == 1': 'static fn ($el) => $el % 2 === 1', + '(x) -> x + 1': 'static fn ($el) => $el + 1', + '(acc, el) -> el * acc': 'static fn ($acc, $el) => $el * $acc', + '(acc, el) -> el + acc': 'static fn ($acc, $el) => $el + $acc', + '(acc, el) -> el / acc': 'static fn ($acc, $el) => $el / $acc', +} +-%} + +/* + * By adding type hints and enabling strict type checking, code can become + * easier to read, self-documenting and reduce the number of potential bugs. + * By default, type declarations are non-strict, which means they will attempt + * to change the original type to match the type specified by the + * type-declaration. + * + * In other words, if you pass a string to a function requiring a float, + * it will attempt to convert the string value to a float. + * + * To enable strict mode, a single declare directive must be placed at the top + * of the file. + * This means that the strictness of typing is configured on a per-file basis. + * This directive not only affects the type declarations of parameters, but also + * a function's return type. + * + * For more info review the Concept on strict type checking in the PHP track + * . + * + * To disable strict typing, comment out the directive below. + */ + +declare(strict_types=1); + +use PHPUnit\Framework\ExpectationFailedException; + +class ListOpsTest extends PHPUnit\Framework\TestCase +{ + public static function setUpBeforeClass(): void + { + require_once 'ListOps.php'; + } + + + {% for case0 in cases -%} + {% for case in case0.cases -%} + /** + * @testdox {{ case0.description }} -> {{ case.description }} + */ + public function {{ testfn(case0.description ~ ' with ' ~ case.description) }}() + { + $listOps = new ListOps(); + {% for property, value in case.input -%} + ${{ property }} = {{ property == 'function' ? callbacks[value] : export(value) }}; + {% endfor %} + + $result = $listOps->{{ case.property }}({{ case.input | keys | map(p => '$' ~ p) | join(', ')}}); + + $this->assertEquals({{ export(case.expected) }}, $result); + } + + {% endfor -%} + {% endfor -%} +} diff --git a/test-generator/.gitignore b/test-generator/.gitignore new file mode 100644 index 00000000..1940dfd2 --- /dev/null +++ b/test-generator/.gitignore @@ -0,0 +1,3 @@ +.phpunit.cache/ +.phpcs-cache +vendor/ diff --git a/test-generator/README.md b/test-generator/README.md new file mode 100644 index 00000000..dd1b0a35 --- /dev/null +++ b/test-generator/README.md @@ -0,0 +1,26 @@ +TODO: +- [ ] Readme + - [ ] Requirements (php 8.3) + - [ ] Usage `php test-generator/main.php exercises/practice/list-ops/ /home/codespace/.cache/exercism/configlet/problem-specifications/exercises/list-ops/canonical-data.json -vv` + - [ ] https://twig.symfony.com/ + - [ ] custom functions `export` / `testf` +- [ ] CI (generator) + - [ ] `phpstan` + - [ ] `phpcs` + - [ ] `phpunit` +- [ ] CI (exercises): iterate over each exercise and run the generator in check mode +- [ ] Write tests +- [ ] Path to convert existing exercises to the test-generator +- [ ] `@TODO` +- [ ] Upgrade https://github.com/brick/varexporter +- [ ] TOML Library for php (does not seem to exist any maitained library) +- [ ] Default templates: + - [ ] Test function header (automatic docblock, automatic name) +- [ ] Going further + - [ ] Skip re-implements + - [x] Read .meta/tests.toml to skip `include=false` cases by uuid + - [ ] Ensure correctness between toml and effectively generated files + - [ ] Default templates to include (strict_types header, require_once based on config, testfn header [testdox, uuid, task_id]) + - [ ] devcontainer for easy contribution in github codespace directly + - [ ] Automatically fetch configlet and exercise informations + - [x] Disable twig automatic isset diff --git a/test-generator/composer.json b/test-generator/composer.json new file mode 100644 index 00000000..b6a4ecb5 --- /dev/null +++ b/test-generator/composer.json @@ -0,0 +1,41 @@ +{ + "name": "exercism/test-generator", + "type": "project", + "require": { + "brick/varexporter": "^0.4.0", + "league/flysystem": "^3.26", + "league/flysystem-memory": "^3.25", + "psr/log": "^3.0", + "symfony/console": "^6.0", + "twig/twig": "^3.8" + }, + "require-dev": { + "doctrine/coding-standard": "^12.0", + "phpstan/phpstan": "^1.10", + "phpunit/phpunit": "^10.0", + "squizlabs/php_codesniffer": "^3.9" + }, + "license": "MIT", + "autoload": { + "psr-4": { + "App\\": "src/" + } + }, + "autoload-dev": { + "psr-4": { + "App\\Tests\\": "tests" + } + }, + "scripts": { + "phpstan": "phpstan analyse src tests --configuration phpstan.neon --memory-limit=2G", + "test": "phpunit", + "lint": "phpcs", + "lint:fix": "phpcbf" + }, + "config": { + "allow-plugins": { + "dealerdirect/phpcodesniffer-composer-installer": true + }, + "sort-packages": true + } +} diff --git a/test-generator/main.php b/test-generator/main.php new file mode 100644 index 00000000..a9a519e6 --- /dev/null +++ b/test-generator/main.php @@ -0,0 +1,8 @@ +run(); diff --git a/test-generator/phpcs.xml.dist b/test-generator/phpcs.xml.dist new file mode 100644 index 00000000..8a22e1cb --- /dev/null +++ b/test-generator/phpcs.xml.dist @@ -0,0 +1,37 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + src/ + tests/ + \ No newline at end of file diff --git a/test-generator/phpstan.neon b/test-generator/phpstan.neon new file mode 100644 index 00000000..22254bcd --- /dev/null +++ b/test-generator/phpstan.neon @@ -0,0 +1,2 @@ +parameters: + level: max diff --git a/test-generator/phpunit.xml.dist b/test-generator/phpunit.xml.dist new file mode 100644 index 00000000..1444de8b --- /dev/null +++ b/test-generator/phpunit.xml.dist @@ -0,0 +1,24 @@ + + + + + ./tests + + + + + src/ + + + + + + + + + + \ No newline at end of file diff --git a/test-generator/src/Application.php b/test-generator/src/Application.php new file mode 100644 index 00000000..0fc9049e --- /dev/null +++ b/test-generator/src/Application.php @@ -0,0 +1,150 @@ +setVersion('1.0.0'); + // @TODO + $this->addArgument('exercise-path', InputArgument::REQUIRED, 'Path of the exercise.'); + $this->addArgument('canonical-data', InputArgument::REQUIRED, 'Path of the canonical data for the exercise. (Use `bin/configlet -verbosity info --offline`)'); + $this->addOption('check', null, InputOption::VALUE_NONE, 'Checks whether the existing files are the same as generated one.'); + } + + protected function execute(InputInterface $input, OutputInterface $output): int + { + $exercisePath = $input->getArgument('exercise-path'); + $canonicalPath = $input->getArgument('canonical-data'); + $exerciseCheck = $input->getOption('check'); + assert(is_string($exercisePath), 'exercise-path must be a string'); + assert(is_string($canonicalPath), 'canonical-data must be a string'); + assert(is_bool($exerciseCheck), 'check must be a bool'); + + $logger = new ConsoleLogger($output); + $logger->info('Exercise path: ' . $exercisePath); + $logger->info('canonical-data path: ' . $canonicalPath); + + $canonicalDataJson = file_get_contents($canonicalPath); + if ($canonicalDataJson === false) { + throw new RuntimeException('Faield to fetch canonical-data.json, check you `canonical-data` argument.'); + } + + $canonicalData = json_decode($canonicalDataJson, true, flags: JSON_THROW_ON_ERROR); + assert(is_array($canonicalData), 'json_decode(..., true) should return an array'); + $exerciseAdapter = new LocalFilesystemAdapter($exercisePath); + $exerciseFilesystem = new Filesystem($exerciseAdapter); + + $success = $this->generate($exerciseFilesystem, $exerciseCheck, $canonicalData, $logger); + + return $success ? self::SUCCESS : self::FAILURE; + } + + /** @param array $canonicalData */ + public function generate(Filesystem $exerciseDir, bool $check, array $canonicalData, LoggerInterface $logger): bool + { + // 1. Read config.json + $configJson = $exerciseDir->read('/.meta/config.json'); + $config = json_decode($configJson, true, flags: JSON_THROW_ON_ERROR); + assert(is_array($config), 'json_decode(..., true) should return an array'); + + if (! isset($config['files']['test']) || ! is_array($config['files']['test'])) { + throw new RuntimeException('.meta/config.json: missing or invalid `files.test` key'); + } + + $testsPaths = $config['files']['test']; + $logger->info('.meta/config.json: tests files: ' . implode(', ', $testsPaths)); + + if (empty($testsPaths)) { + $logger->warning('.meta/config.json: `files.test` key is empty'); + } + + // 2. Read test.toml + $testsToml = $exerciseDir->read('/.meta/tests.toml'); + $tests = TomlParser::parse($testsToml); + + // 3. Remove `include = false` tests + $excludedTests = array_filter($tests, static fn (array $props) => isset($props['include']) && $props['include'] === false); + $this->removeExcludedTests($excludedTests, $canonicalData['cases']); + + // 4. foreach tests files, check if there is a twig file + $twigLoader = new ArrayLoader(); + $twigEnvironment = new Environment($twigLoader, ['strict_variables' => true, 'autoescape' => false]); + $twigEnvironment->addFunction(new TwigFunction('export', static fn (mixed $value) => VarExporter::export($value, VarExporter::INLINE_ARRAY))); + $twigEnvironment->addFunction(new TwigFunction('testfn', static fn (string $label) => 'test' . str_replace(' ', '', ucwords(preg_replace('/[^a-zA-Z0-9]/', ' ', $label))))); + foreach ($testsPaths as $testPath) { + // 5. generate the file + $twigFilename = $testPath . '.twig'; + // @TODO warning or error if it does not exist + $testTemplate = $exerciseDir->read($twigFilename); + $rendered = $twigEnvironment->createTemplate($testTemplate, $twigFilename)->render($canonicalData); + + if ($check) { + // 6. Compare it if check mode + if ($exerciseDir->read($testPath) !== $rendered) { + // return false; + throw new Exception('Differences between generated and existing file'); + } + } else { + $exerciseDir->write($testPath, $rendered); + } + } + + return true; + } + + private function removeExcludedTests(array $tests, array &$cases): void + { + foreach ($cases as $key => &$case) { + if (array_key_exists('cases', $case)) { + $this->removeExcludedTests($tests, $case['cases']); + } else { + assert(array_key_exists('uuid', $case)); + if (array_key_exists($case['uuid'], $tests)) { + unset($cases[$key]); + } + } + } + } +} diff --git a/test-generator/src/TomlParser.php b/test-generator/src/TomlParser.php new file mode 100644 index 00000000..1d88fc55 --- /dev/null +++ b/test-generator/src/TomlParser.php @@ -0,0 +1,79 @@ +write('.meta/config.json', '{"files":{"test":["test.php"]}}'); + $exerciseFs->write('.meta/tests.toml', ''); + $exerciseFs->write('test.php.twig', ' [1, 2], 'l' => 'this-Is_a test fn', 'cases' => []]; + + $application = new Application(); + $success = $application->generate($exerciseFs, false, $canonicalData, new NullLogger()); + + $this->assertTrue($success); + $this->assertSame('read('/test.php')); + } +}