You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
256 lines
7.7 KiB
256 lines
7.7 KiB
<?php
|
|
|
|
namespace Luthfi\CrudGenerator;
|
|
|
|
use Illuminate\Console\Command;
|
|
use Illuminate\Filesystem\Filesystem;
|
|
use Luthfi\CrudGenerator\Generators\ControllerGenerator;
|
|
use Luthfi\CrudGenerator\Generators\FormViewGenerator;
|
|
use Luthfi\CrudGenerator\Generators\IndexViewGenerator;
|
|
use Luthfi\CrudGenerator\Generators\LangFileGenerator;
|
|
use Luthfi\CrudGenerator\Generators\MigrationGenerator;
|
|
use Luthfi\CrudGenerator\Generators\ModelFactoryGenerator;
|
|
use Luthfi\CrudGenerator\Generators\ModelGenerator;
|
|
use Luthfi\CrudGenerator\Generators\WebRouteGenerator;
|
|
|
|
class CrudMake extends Command
|
|
{
|
|
/**
|
|
* The injected Filesystem class
|
|
*
|
|
* @var Filesystem
|
|
*/
|
|
private $files;
|
|
|
|
/**
|
|
* Array of defined model names
|
|
*
|
|
* @var array
|
|
*/
|
|
public $modelNames = [];
|
|
|
|
/**
|
|
* Array of stub's model names
|
|
*
|
|
* @var array
|
|
*/
|
|
public $stubModelNames;
|
|
|
|
/**
|
|
* Construct CrudMake class
|
|
* @param Filesystem $files Put generated file content to application file system
|
|
*/
|
|
public function __construct(Filesystem $files)
|
|
{
|
|
parent::__construct();
|
|
|
|
$this->files = $files;
|
|
|
|
$this->stubModelNames = [
|
|
'model_namespace' => 'mstrNmspc',
|
|
'full_model_name' => 'fullMstr',
|
|
'plural_model_name' => 'Masters',
|
|
'model_name' => 'Master',
|
|
'table_name' => 'masters',
|
|
'lang_name' => 'master',
|
|
'collection_model_var_name' => 'mstrCollections',
|
|
'single_model_var_name' => 'singleMstr',
|
|
];
|
|
}
|
|
|
|
/**
|
|
* The name and signature of the console command.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $signature = 'make:crud {name} {--parent=}';
|
|
|
|
/**
|
|
* The console command description.
|
|
*
|
|
* @var string
|
|
*/
|
|
protected $description = 'Create simple Laravel complate CRUD files of given model name.';
|
|
|
|
/**
|
|
* Execute the console command.
|
|
*
|
|
* @return void
|
|
*/
|
|
public function handle()
|
|
{
|
|
$this->getModelName();
|
|
|
|
if (! $this->modelExists()) {
|
|
app(WebRouteGenerator::class, ['command' => $this])->generate();
|
|
app(ModelGenerator::class, ['command' => $this])->generate();
|
|
app(MigrationGenerator::class, ['command' => $this])->generate();
|
|
app(ControllerGenerator::class, ['command' => $this])->generate();
|
|
app(IndexViewGenerator::class, ['command' => $this])->generate();
|
|
app(FormViewGenerator::class, ['command' => $this])->generate();
|
|
app(LangFileGenerator::class, ['command' => $this])->generate();
|
|
app(ModelFactoryGenerator::class, ['command' => $this])->generate();
|
|
$this->generateTests();
|
|
|
|
$this->info('CRUD files generated successfully!');
|
|
} else {
|
|
$this->error("{$this->modelNames['model_name']} model already exists.");
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Generate class properties for model names in different usage
|
|
*
|
|
* @return void
|
|
*/
|
|
public function getModelName($modelName = null)
|
|
{
|
|
$modelName = is_null($modelName) ? $this->argument('name') : $modelName;
|
|
$model_name = ucfirst(class_basename($modelName));
|
|
$plural_model_name = str_plural($model_name);
|
|
$modelPath = $this->getModelPath($modelName);
|
|
$modelNamespace = $this->getModelNamespace($modelPath);
|
|
|
|
return $this->modelNames = [
|
|
'model_namespace' => $modelNamespace,
|
|
'full_model_name' => $modelNamespace.'\\'.$model_name,
|
|
'plural_model_name' => $plural_model_name,
|
|
'model_name' => $model_name,
|
|
'table_name' => snake_case($plural_model_name),
|
|
'lang_name' => snake_case($model_name),
|
|
'collection_model_var_name' => camel_case($plural_model_name),
|
|
'single_model_var_name' => camel_case($model_name),
|
|
'model_path' => $modelPath,
|
|
];
|
|
}
|
|
|
|
protected function getModelPath($modelName)
|
|
{
|
|
$inputName = explode('/', ucfirst($modelName));
|
|
array_pop($inputName);
|
|
|
|
return implode('/', $inputName);
|
|
}
|
|
|
|
protected function getModelNamespace($modelPath)
|
|
{
|
|
$modelNamespace = str_replace('/', '\\', 'App/'.ucfirst($modelPath));
|
|
return $modelNamespace == 'App\\' ? 'App' : $modelNamespace;
|
|
}
|
|
|
|
/**
|
|
* Check for Model file existance
|
|
*
|
|
* @return void
|
|
*/
|
|
public function modelExists()
|
|
{
|
|
return $this->files->exists(app_path($this->modelNames['model_name'].'.php'));
|
|
}
|
|
|
|
/**
|
|
* Generate Feature for CRUD Operation and and Unit Testing for Model behaviour
|
|
* @return void
|
|
*/
|
|
public function generateTests()
|
|
{
|
|
$this->createBrowserKitBaseTestClass();
|
|
|
|
$featureTestPath = $this->makeDirectory(base_path('tests/Feature'));
|
|
$this->generateFile("{$featureTestPath}/Manage{$this->modelNames['plural_model_name']}Test.php", $this->getFeatureTestContent());
|
|
$this->info('Manage'.$this->modelNames['plural_model_name'].'Test generated.');
|
|
|
|
$unitTestPath = $this->makeDirectory(base_path('tests/Unit/Models'));
|
|
$this->generateFile("{$unitTestPath}/{$this->modelNames['model_name']}Test.php", $this->getUnitTestContent());
|
|
$this->info($this->modelNames['model_name'].'Test (model) generated.');
|
|
}
|
|
|
|
/**
|
|
* Generate BrowserKitTest class for BaseTestCase
|
|
*
|
|
* @return void
|
|
*/
|
|
public function createBrowserKitBaseTestClass()
|
|
{
|
|
$testsPath = base_path('tests');
|
|
if (! $this->files->isDirectory($testsPath)) {
|
|
$this->files->makeDirectory($testsPath, 0777, true, true);
|
|
}
|
|
|
|
if (! $this->files->exists($testsPath.'/BrowserKitTest.php')) {
|
|
$this->generateFile($testsPath.'/BrowserKitTest.php', $this->getBrowserKitBaseTestContent());
|
|
|
|
$this->info('BrowserKitTest generated.');
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Get BrowserKitBaseTest class file content
|
|
*
|
|
* @return string
|
|
*/
|
|
public function getBrowserKitBaseTestContent()
|
|
{
|
|
return $this->files->get(__DIR__.'/stubs/test-browserkit-base-class.stub');
|
|
}
|
|
|
|
/**
|
|
* Get feature test file content from feature test stub
|
|
*
|
|
* @return string Replaced proper model names in feature test file content
|
|
*/
|
|
public function getFeatureTestContent()
|
|
{
|
|
$stub = $this->files->get(__DIR__.'/stubs/test-feature.stub');
|
|
return $this->replaceStubString($stub);
|
|
}
|
|
|
|
/**
|
|
* Get unit test file content from unit test stub
|
|
*
|
|
* @return string Replaced proper model names in unit test file content
|
|
*/
|
|
public function getUnitTestContent()
|
|
{
|
|
$stub = $this->files->get(__DIR__.'/stubs/test-unit.stub');
|
|
return $this->replaceStubString($stub);
|
|
}
|
|
|
|
/**
|
|
* Make directory if the path is not exists
|
|
* @param string $path Absolute path of targetted directory
|
|
* @return string Absolute path
|
|
*/
|
|
protected function makeDirectory($path)
|
|
{
|
|
if (! $this->files->isDirectory($path)) {
|
|
$this->files->makeDirectory($path, 0777, true, true);
|
|
}
|
|
|
|
return $path;
|
|
}
|
|
|
|
/**
|
|
* Replace all string of model names
|
|
*
|
|
* @param string $stub String of file or class stub with default content
|
|
* @return string Replaced content
|
|
*/
|
|
protected function replaceStubString($stub)
|
|
{
|
|
return str_replace($this->stubModelNames, $this->modelNames, $stub);
|
|
}
|
|
|
|
/**
|
|
* Generate file on filesystem
|
|
* @param string $path Absoute path of file
|
|
* @param string $content Generated file content
|
|
* @return string Absolute path of file
|
|
*/
|
|
protected function generateFile($path, $content)
|
|
{
|
|
$this->files->put($path, $content);
|
|
|
|
return $path;
|
|
}
|
|
}
|