123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510 |
- <?php
- /**
- * Laravel IDE Helper Generator
- *
- * @author Barry vd. Heuvel <barryvdh@gmail.com>
- * @copyright 2014 Barry vd. Heuvel / Fruitcake Studio (http://www.fruitcakestudio.nl)
- * @license http://www.opensource.org/licenses/mit-license.php MIT
- * @link https://github.com/barryvdh/laravel-ide-helper
- */
-
- namespace Barryvdh\LaravelIdeHelper\Console;
-
- use Illuminate\Console\Command;
- use Illuminate\Support\Str;
- use Symfony\Component\Console\Input\InputOption;
- use Symfony\Component\Console\Input\InputArgument;
- use Symfony\Component\ClassLoader\ClassMapGenerator;
- use phpDocumentor\Reflection\DocBlock;
- use phpDocumentor\Reflection\DocBlock\Context;
- use phpDocumentor\Reflection\DocBlock\Tag;
- use phpDocumentor\Reflection\DocBlock\Serializer as DocBlockSerializer;
-
- /**
- * A command to generate autocomplete information for your IDE
- *
- * @author Barry vd. Heuvel <barryvdh@gmail.com>
- */
- class ModelsCommand extends Command
- {
-
- /**
- * The console command name.
- *
- * @var string
- */
- protected $name = 'ide-helper:models';
- protected $filename = '_ide_helper_models.php';
-
- /**
- * The console command description.
- *
- * @var string
- */
- protected $description = 'Generate autocompletion for models';
-
- protected $properties = array();
- protected $methods = array();
- protected $write = false;
- protected $dirs = array();
- protected $reset;
-
-
- /**
- * Execute the console command.
- *
- * @return void
- */
- public function fire()
- {
- $filename = $this->option('filename');
- $this->write = $this->option('write');
- $this->dirs = array_merge(
- $this->laravel['config']->get('laravel-ide-helper::model_locations'),
- $this->option('dir')
- );
- $model = $this->argument('model');
- $ignore = $this->option('ignore');
- $this->reset = $this->option('reset');
-
- //If filename is default and Write is not specified, ask what to do
- if (!$this->write && $filename === $this->filename && !$this->option('nowrite')) {
- if ($this->confirm(
- "Do you want to overwrite the existing model files? Choose no to write to $filename instead? (Yes/No): "
- )
- ) {
- $this->write = true;
- }
- }
-
- $content = $this->generateDocs($model, $ignore);
-
- if (!$this->write) {
- $written = \File::put($filename, $content);
- if ($written !== false) {
- $this->info("Model information was written to $filename");
- } else {
- $this->error("Failed to write model information to $filename");
- }
- }
- }
-
-
- /**
- * Get the console command arguments.
- *
- * @return array
- */
- protected function getArguments()
- {
- return array(
- array('model', InputArgument::OPTIONAL | InputArgument::IS_ARRAY, 'Which models to include', array()),
- );
- }
-
- /**
- * Get the console command options.
- *
- * @return array
- */
- protected function getOptions()
- {
- return array(
- array('filename', 'F', InputOption::VALUE_OPTIONAL, 'The path to the helper file', $this->filename),
- array('dir', 'D', InputOption::VALUE_OPTIONAL | InputOption::VALUE_IS_ARRAY, 'The model dir', array()),
- array('write', 'W', InputOption::VALUE_NONE, 'Write to Model file'),
- array('nowrite', 'N', InputOption::VALUE_NONE, 'Don\'t write to Model file'),
- array('reset', 'R', InputOption::VALUE_NONE, 'Remove the original phpdocs instead of appending'),
- array('ignore', 'I', InputOption::VALUE_OPTIONAL, 'Which models to ignore', ''),
- );
- }
-
- protected function generateDocs($loadModels, $ignore = '')
- {
-
-
- $output = "<?php
- /**
- * An helper file for your Eloquent Models
- * Copy the phpDocs from this file to the correct Model,
- * And remove them from this file, to prevent double declarations.
- *
- * @author Barry vd. Heuvel <barryvdh@gmail.com>
- */
- \n\n";
-
- $hasDoctrine = interface_exists('Doctrine\DBAL\Driver');
-
- if (empty($loadModels)) {
- $models = $this->loadModels();
- } else {
- $models = array();
- foreach ($loadModels as $model) {
- $models = array_merge($models, explode(',', $model));
- }
- }
-
- $ignore = explode(',', $ignore);
-
- foreach ($models as $name) {
- if (in_array($name, $ignore)) {
- $this->comment("Ignoring model '$name'");
- continue;
- } else {
- $this->comment("Loading model '$name'");
- }
- $this->properties = array();
- $this->methods = array();
- if (class_exists($name)) {
- try {
- // handle abstract classes, interfaces, ...
- $reflectionClass = new \ReflectionClass($name);
- if (!$reflectionClass->IsInstantiable()) {
- throw new \Exception($name . ' is not instanciable.');
- } elseif (!$reflectionClass->isSubclassOf('Illuminate\Database\Eloquent\Model')) {
- $this->comment("Class '$name' is not a model");
- continue;
- }
-
- $model = new $name();
- if ($hasDoctrine) {
- $this->getPropertiesFromTable($model);
- }
- $this->getPropertiesFromMethods($model);
- $output .= $this->createPhpDocs($name);
- } catch (\Exception $e) {
- $this->error("Exception: " . $e->getMessage() . "\nCould not analyze class $name.");
- }
- } else {
- $this->error("Class $name does not exist");
- }
-
- }
-
- if (!$hasDoctrine) {
- $this->error(
- "Warning: 'doctrine/dbal: ~2.3' is required to load database information. Please require that in your composer.json and run 'composer update'."
- );
- }
-
- return $output;
-
- }
-
-
- protected function loadModels()
- {
- $models = array();
- foreach ($this->dirs as $dir) {
- $dir = base_path() . '/' . $dir;
- if (file_exists($dir)) {
- foreach (ClassMapGenerator::createMap($dir) as $model => $path) {
- $models[] = $model;
- }
- }
- }
- return $models;
- }
-
- /**
- * Load the properties from the database table.
- *
- * @param \Illuminate\Database\Eloquent\Model $model
- */
- protected function getPropertiesFromTable($model)
- {
- $table = $model->getConnection()->getTablePrefix() . $model->getTable();
- $schema = $model->getConnection()->getDoctrineSchemaManager($table);
- $schema->getDatabasePlatform()->registerDoctrineTypeMapping('enum', 'string');
-
- $columns = $schema->listTableColumns($table);
-
- if ($columns) {
- foreach ($columns as $column) {
- $name = $column->getName();
- if (in_array($name, $model->getDates())) {
- $type = '\Carbon\Carbon';
- } else {
- $type = $column->getType()->getName();
- switch ($type) {
- case 'string':
- case 'text':
- case 'date':
- case 'time':
- case 'guid':
- case 'datetimetz':
- case 'datetime':
- $type = 'string';
- break;
- case 'integer':
- case 'bigint':
- case 'smallint':
- $type = 'integer';
- break;
- case 'decimal':
- case 'float':
- $type = 'float';
- break;
- case 'boolean':
- $type = 'boolean';
- break;
- default:
- $type = 'mixed';
- break;
- }
- }
-
-
- $this->setProperty($name, $type, true, true);
- $this->setMethod(
- Str::camel("where_" . $name),
- '\Illuminate\Database\Query\Builder|\\' . get_class($model),
- array('$value')
- );
- }
- }
- }
-
- /**
- * @param \Illuminate\Database\Eloquent\Model $model
- */
- protected function getPropertiesFromMethods($model)
- {
- $methods = get_class_methods($model);
- if ($methods) {
- foreach ($methods as $method) {
- if (Str::startsWith($method, 'get') && Str::endsWith(
- $method,
- 'Attribute'
- ) && $method !== 'getAttribute'
- ) {
- //Magic get<name>Attribute
- $name = Str::snake(substr($method, 3, -9));
- if (!empty($name)) {
- $this->setProperty($name, null, true, null);
- }
- } elseif (Str::startsWith($method, 'set') && Str::endsWith(
- $method,
- 'Attribute'
- ) && $method !== 'setAttribute'
- ) {
- //Magic set<name>Attribute
- $name = Str::snake(substr($method, 3, -9));
- if (!empty($name)) {
- $this->setProperty($name, null, null, true);
- }
- } elseif (Str::startsWith($method, 'scope') && $method !== 'scopeQuery') {
- //Magic set<name>Attribute
- $name = Str::camel(substr($method, 5));
- if (!empty($name)) {
- $reflection = new \ReflectionMethod($model, $method);
- $args = $this->getParameters($reflection);
- //Remove the first ($query) argument
- array_shift($args);
- $this->setMethod($name, '\\' . $reflection->class, $args);
- }
- } elseif (!method_exists('Eloquent', $method) && !Str::startsWith($method, 'get')) {
-
- //Use reflection to inspect the code, based on Illuminate/Support/SerializableClosure.php
- $reflection = new \ReflectionMethod($model, $method);
-
- $file = new \SplFileObject($reflection->getFileName());
- $file->seek($reflection->getStartLine() - 1);
-
- $code = '';
- while ($file->key() < $reflection->getEndLine()) {
- $code .= $file->current();
- $file->next();
- }
- $begin = strpos($code, 'function(');
- $code = substr($code, $begin, strrpos($code, '}') - $begin + 1);
-
- foreach (array(
- 'hasMany',
- 'belongsToMany',
- 'hasOne',
- 'belongsTo',
- 'morphTo',
- 'morphMany',
- 'morphToMany'
- ) as $relation) {
- $search = '$this->' . $relation . '(';
- if ($pos = stripos($code, $search)) {
- $code = substr($code, $pos + strlen($search));
- $arguments = explode(',', substr($code, 0, stripos($code, ')')));
- //Remove quotes, ensure 1 \ in front of the model
- $returnModel = "\\" . ltrim(trim($arguments[0], " \"'"), "\\");
- if ($relation === "belongsToMany" or $relation === 'hasMany' or $relation === 'morphMany' or $relation === 'morphToMany') {
- //Collection or array of models (because Collection is Arrayable)
- $this->setProperty(
- $method,
- '\Illuminate\Database\Eloquent\Collection|' . $returnModel . '[]',
- true,
- null
- );
- } else {
- //Single model is returned
- $this->setProperty($method, $returnModel, true, null);
- }
- }
- }
-
- }
- }
- }
- }
-
- /**
- * @param string $name
- * @param string|null $type
- * @param bool|null $read
- * @param bool|null $write
- */
- protected function setProperty($name, $type = null, $read = null, $write = null)
- {
- if (!isset($this->properties[$name])) {
- $this->properties[$name] = array();
- $this->properties[$name]['type'] = 'mixed';
- $this->properties[$name]['read'] = false;
- $this->properties[$name]['write'] = false;
- }
- if ($type !== null) {
- $this->properties[$name]['type'] = $type;
- }
- if ($read !== null) {
- $this->properties[$name]['read'] = $read;
- }
- if ($write !== null) {
- $this->properties[$name]['write'] = $write;
- }
- }
-
- protected function setMethod($name, $type = '', $arguments = array())
- {
- if (!isset($this->methods[$name])) {
- $this->methods[$name] = array();
- $this->methods[$name]['type'] = $type;
- $this->methods[$name]['arguments'] = $arguments;
- }
- }
-
- /**
- * @param string $class
- * @return string
- */
- protected function createPhpDocs($class)
- {
-
- $reflection = new \ReflectionClass($class);
- $namespace = $reflection->getNamespaceName();
- $classname = $reflection->getShortName();
- $originalDoc = $reflection->getDocComment();
-
- if ($this->reset) {
- $phpdoc = new DocBlock('', new Context($namespace));
- } else {
- $phpdoc = new DocBlock($reflection, new Context($namespace));
- }
-
- if (!$phpdoc->getText()) {
- $phpdoc->setText($class);
- }
-
- $properties = array();
- $methods = array();
- foreach ($phpdoc->getTags() as $tag) {
- $name = $tag->getName();
- if ($name == "property" || $name == "property-read" || $name == "property-write") {
- $properties[] = $tag->getVariableName();
- } elseif ($name == "method") {
- $methods[] = $tag->getMethodName();
- }
- }
-
- foreach ($this->properties as $name => $property) {
- $name = "\$$name";
- if (in_array($name, $properties)) {
- continue;
- }
- if ($property['read'] && $property['write']) {
- $attr = 'property';
- } elseif ($property['write']) {
- $attr = 'property-write';
- } else {
- $attr = 'property-read';
- }
- $tag = Tag::createInstance("@{$attr} {$property['type']} {$name}", $phpdoc);
- $phpdoc->appendTag($tag);
- }
-
- foreach ($this->methods as $name => $method) {
- if (in_array($name, $methods)) {
- continue;
- }
- $arguments = implode(', ', $method['arguments']);
- $tag = Tag::createInstance("@method static {$method['type']} {$name}({$arguments}) ", $phpdoc);
- $phpdoc->appendTag($tag);
- }
-
- $serializer = new DocBlockSerializer();
- $serializer->getDocComment($phpdoc);
- $docComment = $serializer->getDocComment($phpdoc);
-
-
- if ($this->write) {
- $filename = $reflection->getFileName();
- $contents = \File::get($filename);
- if ($originalDoc) {
- $contents = str_replace($originalDoc, $docComment, $contents);
- } else {
- $needle = "class {$classname}";
- $replace = "{$docComment}\nclass {$classname}";
- $pos = strpos($contents, $needle);
- if ($pos !== false) {
- $contents = substr_replace($contents, $replace, $pos, strlen($needle));
- }
- }
- if (\File::put($filename, $contents)) {
- $this->info('Written new phpDocBlock to ' . $filename);
- }
- }
-
- $output = "namespace {$namespace}{\n{$docComment}\n\tclass {$classname} {}\n}\n\n";
- return $output;
- }
-
- /**
- * Get the parameters and format them correctly
- *
- * @param $method
- * @return array
- */
- public function getParameters($method)
- {
- //Loop through the default values for paremeters, and make the correct output string
- $params = array();
- $paramsWithDefault = array();
- foreach ($method->getParameters() as $param) {
- $paramStr = '$' . $param->getName();
- $params[] = $paramStr;
- if ($param->isOptional()) {
- $default = $param->getDefaultValue();
- if (is_bool($default)) {
- $default = $default ? 'true' : 'false';
- } elseif (is_array($default)) {
- $default = 'array()';
- } elseif (is_null($default)) {
- $default = 'null';
- } elseif (is_int($default)) {
- //$default = $default;
- } else {
- $default = "'" . trim($default) . "'";
- }
- $paramStr .= " = $default";
- }
- $paramsWithDefault[] = $paramStr;
- }
- return $paramsWithDefault;
- }
-
- }
|