Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Draft: Controller namespace issue for modules in urlPrefixes #38

Draft
wants to merge 27 commits into
base: master
Choose a base branch
from
Draft
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
27 commits
Select commit Hold shift + click to select a range
73aa9b3
Add failing test
SOHELAHMED7 Sep 2, 2024
1657e95
Merge branches 'master' and 'controller-namespace-issue-for-modules-i…
SOHELAHMED7 Mar 18, 2025
0abba24
Stub
SOHELAHMED7 Mar 21, 2025
9701230
Fix issue https://github.com/php-openapi/yii2-openapi/issues/39
SOHELAHMED7 Mar 21, 2025
0b106c5
Fix test
SOHELAHMED7 Mar 21, 2025
697af0e
Fix error in test
SOHELAHMED7 Mar 21, 2025
8240cfb
Complete test - follow up of 0b106c556aea76a2964c5ed5e0ae75b063258d54
SOHELAHMED7 Mar 21, 2025
7660080
Fix error in test
SOHELAHMED7 Mar 21, 2025
f60960c
Fix failing test
SOHELAHMED7 Mar 21, 2025
e824afb
Fix another test
SOHELAHMED7 Mar 21, 2025
7649d0f
Fix another test 2
SOHELAHMED7 Mar 21, 2025
232627a
Implement nested modules in `x-route` - WIP
SOHELAHMED7 Mar 24, 2025
5d2d63c
Fix bug in module path
SOHELAHMED7 Mar 24, 2025
17a40fa
Add Module.php file generator - WIP
SOHELAHMED7 Mar 25, 2025
1bc56b8
Fix failing tests
SOHELAHMED7 Mar 25, 2025
d0f86ce
Support for nested modules
SOHELAHMED7 Mar 26, 2025
e571d2c
Refactor + fix failing test
SOHELAHMED7 Mar 26, 2025
424a6c1
Cleanup
SOHELAHMED7 Mar 26, 2025
3720768
Refactor + complete the test
SOHELAHMED7 Mar 26, 2025
c567892
Fix failing test
SOHELAHMED7 Mar 27, 2025
452aa57
Fix failing test
SOHELAHMED7 Mar 27, 2025
4df4cea
Fix failing test 2
SOHELAHMED7 Mar 27, 2025
ef2536a
Generate Module.php file for `urlPrefixes.module` config - WIP
SOHELAHMED7 Mar 27, 2025
417bfc4
Implement Module.php file generation for `urlPrefixes.module` config …
SOHELAHMED7 Mar 28, 2025
963c7e2
Implement Module.php file generation for `urlPrefixes.module` config …
SOHELAHMED7 Mar 28, 2025
db9a018
Implement for `FractalAction`
SOHELAHMED7 Mar 29, 2025
fe97ab4
Implement for `FractalAction` - WIP
SOHELAHMED7 Mar 29, 2025
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 13 additions & 3 deletions README.md
Original file line number Diff line number Diff line change
Expand Up @@ -610,8 +610,8 @@ then the value for `comments` can be

### `x-route`

To customize route (controller ID/action ID) for a path, use custom key `x-route` with value `<controller ID>/<action ID>`. It can be used for non-crud paths. It must be used under HTTP method key but not
directly under the `paths` key of OpenAPI spec. Example:
To customize [route](https://www.yiiframework.com/doc/guide/2.0/en/runtime-routing) (controller ID/action ID) for a path, use custom key `x-route` with value `[<module ID>/<child module ID>/...]<controller ID>/<action ID>`. It can be used for non-crud paths. It must be used under HTTP method key but not
directly under the `paths` key of OpenAPI spec. Providing `<controller ID>` and `<action ID>` is required. Example:

```yaml
paths:
Expand Down Expand Up @@ -681,7 +681,17 @@ Generated URL rules config for above is (in `urls.rest.php` or pertinent file):
'POST a1/b1' => 'abc/xyz',
'a1/b1' => 'abc/options',
```
`x-route` does not support [Yii Modules](https://www.yiiframework.com/doc/guide/2.0/en/structure-modules).

`x-route` must not start with slash `/`. For example `x-route: /user/posts` is incorrect. It must start with [module ID](https://www.yiiframework.com/doc/guide/2.0/en/structure-modules) or [controller ID](https://www.yiiframework.com/doc/guide/2.0/en/structure-controllers#controller-ids)

#### Route, path and namespace TODO

Route, path and namespace for controller/action will be resolved in following manner (from highest priority to lowest):

- [`x-route`](#x-route)
- [`urlPrefixes`](https://github.com/php-openapi/yii2-openapi/blob/649743cf0a78743f550edcbd4e93fdffc55c76fd/src/lib/Config.php#L51)
- [`controllerNamespace`](https://github.com/php-openapi/yii2-openapi/blob/649743cf0a78743f550edcbd4e93fdffc55c76fd/src/lib/Config.php#L77) of this lib
- [`controllerNamespace`](https://github.com/yiisoft/yii2/blob/16f50626e1aa81200f109c1a455a5c9b18acfdda/framework/base/Application.php#L93) of Yii app

### `x-description-is-comment`

Expand Down
7 changes: 7 additions & 0 deletions TODO.taskpaper
Original file line number Diff line number Diff line change
@@ -0,0 +1,7 @@
TODO.taskpaper

✔ fix failing tests @done (25-03-21 16:20)
✔ Implement code generation for nested module in `x-route` and other pertinent place @done (25-03-26 11:38)
☐ resolve all sub tasks of issue - Support for Yii Modules in x-route and pertinent places #14
☐ resolve all TODOs
☐ delete this file
2 changes: 1 addition & 1 deletion src/lib/Config.php
Original file line number Diff line number Diff line change
Expand Up @@ -183,7 +183,7 @@ public function getOpenApi():OpenApi
return $this->openApi;
}

public function getPathFromNamespace(string $namespace):string
public static function getPathFromNamespace(string $namespace): string
{
return Yii::getAlias('@' . str_replace('\\', '/', ltrim($namespace, '\\')));
}
Expand Down
67 changes: 60 additions & 7 deletions src/lib/generators/ControllersGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,6 @@
use Laminas\Code\Generator\ValueGenerator;
use Yii;
use yii\gii\CodeFile;
use yii\helpers\ArrayHelper;
use yii\helpers\Inflector;

class ControllersGenerator
Expand All @@ -38,7 +37,13 @@ class ControllersGenerator
public function __construct(Config $config, array $actions = [])
{
$this->config = $config;
$this->controllers = ArrayHelper::index($actions, null, 'controllerId');
foreach ($actions as $action) {
$r = $action->getRoute();
$r = explode('/', $r);
array_pop($r);
array_pop($r);
$this->controllers[implode('/', $r) . '/' . $action->controllerId][] = $action;
}
$this->files = new CodeFiles([]);
}

Expand All @@ -51,22 +56,25 @@ public function generate():CodeFiles
return new CodeFiles([]);
}
$namespace = $this->config->controllerNamespace ?? Yii::$app->controllerNamespace;
$path = $this->config->getPathFromNamespace($namespace);
$path = Config::getPathFromNamespace($namespace);
$templateName = $this->config->useJsonApi ? 'controller_jsonapi.php' : 'controller.php';

foreach ($this->controllers as $controller => $actions) {
foreach ($this->controllers as $controllerWithPrefix => $actions) {
$controllerNamespace = $namespace;
$controllerPath = $path;
/**
* @var RestAction|FractalAction $action
**/
$action = $actions[0];
if ($action->prefix && !empty($action->prefixSettings)) {
if (!empty($action->prefixSettings)) {
$controllerNamespace = trim($action->prefixSettings['namespace'], '\\');
$controllerPath = $action->prefixSettings['path']
?? $this->config->getPathFromNamespace($controllerNamespace);
?? Config::getPathFromNamespace($controllerNamespace);
}
$className = Inflector::id2camel($controller) . 'Controller';

$routeParts = explode('/', $controllerWithPrefix);

$className = Inflector::id2camel(end($routeParts)) . 'Controller';
$this->files->add(new CodeFile(
Yii::getAlias($controllerPath . "/base/$className.php"),
$this->config->render(
Expand All @@ -86,6 +94,18 @@ public function generate():CodeFiles
$classFileGenerator->generate()
));
}

// generate Module.php file for modules
foreach ($action->modulesList as $moduleId => $moduleDetail) {
// only generate Module.php file if they do not exist, do not override
if (!file_exists(Yii::getAlias($moduleDetail['path'] . "/Module.php"))) {
$moduleFileGenerator = $this->makeModuleFile('Module', $moduleDetail['namespace'], $moduleId, $action);
$this->files->add(new CodeFile(
Yii::getAlias($moduleDetail['path'] . "/Module.php"),
$moduleFileGenerator->generate()
));
}
}
}
return $this->files;
}
Expand Down Expand Up @@ -156,4 +176,37 @@ protected function makeCustomController(
$classFileGenerator->setClasses([$reflection]);
return $classFileGenerator;
}

/**
* @param RestAction|FractalAction $action
*/
public function makeModuleFile(string $class, string $namespace, $moduleId, $action): FileGenerator
{
$file = new FileGenerator;
$reflection = new ClassGenerator(
$class,
$namespace,
null,
'yii\base\Module'
);

$moduleIds = array_keys($action->modulesList);
$position = array_search($moduleId, $moduleIds);
$childModuleId = $childModuleCode = null;
if (array_key_exists($position + 1, $moduleIds)) { # if `true`, child module exists
$childModuleId = $moduleIds[$position + 1];
$childModuleNamespace = $action->modulesList[$childModuleId]['namespace'];
$childModuleCode = <<<PHP
\$this->modules = [
'{$childModuleId}' => [
'class' => \\{$childModuleNamespace}\Module::class,
],
];
PHP;
}

$reflection->addMethod('init', [], AbstractMemberGenerator::FLAG_PUBLIC, 'parent::init();' . PHP_EOL . $childModuleCode);
$file->setClasses([$reflection]);
return $file;
}
}
14 changes: 12 additions & 2 deletions src/lib/generators/JsonActionGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -74,12 +74,20 @@ protected function prepareAction(
}
}

$actionId = $routeData->isNonCrudAction() ? trim("{$actionType}-{$routeData->action}", '-')
: "$actionType{$routeData->action}";
if (!empty($customRoute)) {
// $actionType = '';
$parts = explode('/', $customRoute);
$controllerId = $parts[count($parts) - 2];
$actionId = $parts[count($parts) - 1];
}

return Yii::createObject(FractalAction::class, [
[
'singularResourceKey' => $this->config->singularResourceKeys,
'type' => $routeData->type,
'id' => $routeData->isNonCrudAction() ? trim("{$actionType}-{$routeData->action}", '-')
: "$actionType{$routeData->action}",
'id' => $actionId,
'controllerId' => $controllerId,
'urlPath' => $routeData->path,
'requestMethod' => strtoupper($method),
Expand All @@ -96,6 +104,8 @@ protected function prepareAction(
'expectedRelations' => $expectedRelations,
'prefix' => $routeData->getPrefix(),
'prefixSettings' => $routeData->getPrefixSettings(),
'xRoute' => $customRoute,
'modulesList' => $routeData->listModules()
],
]);
}
Expand Down
4 changes: 2 additions & 2 deletions src/lib/generators/ModelsGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -46,8 +46,8 @@ public function generate():CodeFiles
if (!$this->config->generateModels) {
return $this->files;
}
$modelPath = $this->config->getPathFromNamespace($this->config->modelNamespace);
$fakerPath = $this->config->getPathFromNamespace($this->config->fakerNamespace);
$modelPath = Config::getPathFromNamespace($this->config->modelNamespace);
$fakerPath = Config::getPathFromNamespace($this->config->fakerNamespace);
if ($this->config->generateModelFaker) {
$this->files->add(new CodeFile(
Yii::getAlias("$fakerPath/BaseModelFaker.php"),
Expand Down
12 changes: 8 additions & 4 deletions src/lib/generators/RestActionGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -71,8 +71,8 @@ protected function resolvePath(string $path, PathItem $pathItem):array
{
$actions = [];

$routeData = Yii::createObject(RouteData::class, [$pathItem, $path, $this->config->urlPrefixes]);
foreach ($pathItem->getOperations() as $method => $operation) {
$routeData = Yii::createObject(RouteData::class, [$path, $pathItem, $method, $operation, $this->config->urlPrefixes]);
$customRoute = null;
if (isset($operation->{CustomSpecAttr::ROUTE})) { # https://github.com/cebe/yii2-openapi/issues/144
$customRoute = $operation->{CustomSpecAttr::ROUTE};
Expand Down Expand Up @@ -130,14 +130,16 @@ protected function prepareAction(
? Inflector::camel2id($this->config->controllerModelMap[$modelClass])
: Inflector::camel2id($modelClass);
} elseif (!empty($customRoute)) {
$controllerId = explode('/', $customRoute)[0];
$parts = explode('/', $customRoute);
$controllerId = $parts[count($parts) - 2];
} else {
$controllerId = $routeData->controller;
}
$action = Inflector::camel2id($routeData->action);
if (!empty($customRoute)) {
$actionType = '';
$action = explode('/', $customRoute)[1];
$parts = explode('/', $customRoute);
$action = $parts[count($parts) - 1];
}
return Yii::createObject(RestAction::class, [
[
Expand All @@ -154,7 +156,9 @@ protected function prepareAction(
: null,
'responseWrapper' => $responseWrapper,
'prefix' => $routeData->getPrefix(),
'prefixSettings' => $routeData->getPrefixSettings()
'prefixSettings' => $routeData->getPrefixSettings(),
'xRoute' => $customRoute,
'modulesList' => $routeData->listModules()
],
]);
}
Expand Down
2 changes: 1 addition & 1 deletion src/lib/generators/TransformersGenerator.php
Original file line number Diff line number Diff line change
Expand Up @@ -48,7 +48,7 @@ public function generate():CodeFiles
if (!$this->config->generateControllers || !$this->config->useJsonApi) {
return $this->files;
}
$transformerPath = $this->config->getPathFromNamespace($this->config->transformerNamespace);
$transformerPath = Config::getPathFromNamespace($this->config->transformerNamespace);
foreach ($this->models as $model) {
$transformer = Yii::createObject(Transformer::class, [
$model,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -7,11 +7,33 @@

namespace cebe\yii2openapi\lib\items;

trait OptionsRoutesTrait
trait ActionHelperTrait
{
public ?string $xRoute = null;

# list of module this action is part of. 'key' is module ID and 'value' is path where Module.php file must be generated
public array $modulesList = [];

/**
* @see $isDuplicate
* https://github.com/cebe/yii2-openapi/issues/84
* see `x-route` in README.md
* Used for generating only one action for paths like: `GET /calendar/domains` and `GET /calendar/domains/{id}` given that they have same `x-route`.
* If duplicates routes have same params then `false`, else action is generated with no (0) params `true`
*/
public bool $zeroParams = false;

/**
* https://github.com/cebe/yii2-openapi/issues/84
* Generate only one action for paths like: `GET /calendar/domains` and `GET /calendar/domains/{id}` given that they have same `x-route`.
* @see $zeroParams
* see `x-route` in README.md
*/
public bool $isDuplicate = false;

public function getOptionsRoute():string
{
if ($this->prefix && !empty($this->prefixSettings)) {
if (!empty($this->prefixSettings)) {
if (isset($this->prefixSettings['module'])) {
$prefix = $this->prefixSettings['module'];
return static::finalOptionsRoute($prefix, $this->controllerId);
Expand Down
8 changes: 6 additions & 2 deletions src/lib/items/FractalAction.php
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@
*/
final class FractalAction extends BaseObject
{
use OptionsRoutesTrait;
use ActionHelperTrait;

/**@var string* */
public $id;
Expand Down Expand Up @@ -97,7 +97,11 @@ private function templateFactory():FractalActionTemplates

public function getRoute():string
{
if ($this->prefix && !empty($this->prefixSettings)) {
if ($this->xRoute) {
return $this->xRoute;
}

if (!empty($this->prefixSettings)) {
$prefix = $this->prefixSettings['module'] ?? $this->prefix;
return trim($prefix, '/').'/'.$this->controllerId.'/'.$this->id;
}
Expand Down
28 changes: 7 additions & 21 deletions src/lib/items/RestAction.php
Original file line number Diff line number Diff line change
Expand Up @@ -31,7 +31,7 @@
*/
final class RestAction extends BaseObject
{
use OptionsRoutesTrait;
use ActionHelperTrait;

/**@var string* */
public $id;
Expand Down Expand Up @@ -70,31 +70,17 @@ final class RestAction extends BaseObject
*/
public $responseWrapper;

/**
* @var bool
* @see $isDuplicate
* https://github.com/cebe/yii2-openapi/issues/84
* see `x-route` in README.md
* Used for generating only one action for paths like: `GET /calendar/domains` and `GET /calendar/domains/{id}` given that they have same `x-route`.
* If duplicates routes have same params then `false`, else action is generated with no (0) params `true`
*/
public $zeroParams = false;

/**
* @var bool
* https://github.com/cebe/yii2-openapi/issues/84
* Generate only one action for paths like: `GET /calendar/domains` and `GET /calendar/domains/{id}` given that they have same `x-route`.
* @see $zeroParams
* see `x-route` in README.md
*/
public $isDuplicate = false;

public function getRoute():string
{
if ($this->prefix && !empty($this->prefixSettings)) {
if ($this->xRoute) {
return $this->xRoute;
}

if (!empty($this->prefixSettings)) {
$prefix = $this->prefixSettings['module'] ?? $this->prefix;
return trim($prefix, '/') . '/' . $this->controllerId . '/' . $this->id;
}

return $this->controllerId . '/' . $this->id;
}

Expand Down
Loading
Loading