Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
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
2 changes: 1 addition & 1 deletion src/BlueprintServiceProvider.php
Original file line number Diff line number Diff line change
Expand Up @@ -65,7 +65,7 @@ function ($app) {
$this->app->bind(
'command.blueprint.trace',
function ($app) {
return new TraceCommand($app['files']);
return new TraceCommand($app['files'], app(Tracer::class));
}
);
$this->app->bind(
Expand Down
243 changes: 12 additions & 231 deletions src/Commands/TraceCommand.php
Original file line number Diff line number Diff line change
Expand Up @@ -3,10 +3,8 @@
namespace Blueprint\Commands;

use Blueprint\Blueprint;
use Blueprint\EnumType;
use Doctrine\DBAL\Types\Type;
use Blueprint\Tracer;
use Illuminate\Console\Command;
use Illuminate\Database\Eloquent\Model;
use Illuminate\Filesystem\Filesystem;
use Illuminate\Support\Str;

Expand All @@ -29,15 +27,19 @@ class TraceCommand extends Command
/** @var Filesystem $files */
protected $files;

/** @var Tracer */
private $tracer;

/**
* @param Filesystem $files
* @param \Illuminate\Contracts\View\Factory $view
* @param Tracer $tracer
*/
public function __construct(Filesystem $files)
public function __construct(Filesystem $files, Tracer $tracer)
{
parent::__construct();

$this->files = $files;
$this->tracer = $tracer;
}

/**
Expand All @@ -47,236 +49,15 @@ public function __construct(Filesystem $files)
*/
public function handle()
{
$definitions = [];
foreach ($this->appClasses() as $class) {
$model = $this->loadModel($class);
if (is_null($model)) {
continue;
}

$definitions[$this->relativeClassName($model)] = $this->translateColumns($this->mapColumns($this->extractColumns($model)));
}
$blueprint = resolve(Blueprint::class);
$definitions = $this->tracer->execute($blueprint, $this->files);

if (empty($definitions)) {
$this->error('No models found');

return;
}

$blueprint = new Blueprint();

$cache = [];
if ($this->files->exists('.blueprint')) {
$cache = $blueprint->parse($this->files->get('.blueprint'));
}

$cache['models'] = $definitions;

$this->files->put('.blueprint', $blueprint->dump($cache));

$this->info('Traced ' . count($definitions) . ' ' . Str::plural('model', count($definitions)));
}

private function appClasses()
{
$dir = Blueprint::appPath();

if (config('blueprint.models_namespace')) {
$dir .= '/' . str_replace('\\', '/', config('blueprint.models_namespace'));
}

if (!$this->files->exists($dir)) {
return [];
}

return array_map(function (\SplFIleInfo $file) {
return str_replace(
[Blueprint::appPath() . '/', '/'],
[config('blueprint.namespace') . '\\', '\\'],
$file->getPath() . '/' . $file->getBasename('.php')
);
}, $this->files->allFiles($dir));
}

private function loadModel(string $class)
{
if (!class_exists($class)) {
return null;
}

$reflectionClass = new \ReflectionClass($class);
if (
!$reflectionClass->isSubclassOf(\Illuminate\Database\Eloquent\Model::class) ||
(class_exists('Jenssegers\Mongodb\Eloquent\Model') &&
$reflectionClass->isSubclassOf('Jenssegers\Mongodb\Eloquent\Model'))
) {
return null;
}

return $this->laravel->make($class);
}

private function extractColumns(Model $model)
{
$table = $model->getConnection()->getTablePrefix() . $model->getTable();
$schema = $model->getConnection()->getDoctrineSchemaManager();

if (!Type::hasType('enum')) {
Type::addType('enum', EnumType::class);
$databasePlatform = $schema->getDatabasePlatform();
$databasePlatform->registerDoctrineTypeMapping('enum', 'enum');
}

$database = null;
if (strpos($table, '.')) {
[$database, $table] = explode('.', $table);
}

$columns = $schema->listTableColumns($table, $database);

$uses_enums = collect($columns)->contains(function ($column) {
return $column->getType() instanceof \Blueprint\EnumType;
});

if ($uses_enums) {
$definitions = $model->getConnection()->getDoctrineConnection()->fetchAll($schema->getDatabasePlatform()->getListTableColumnsSQL($table, $database));

collect($columns)->filter(function ($column) {
return $column->getType() instanceof \Blueprint\EnumType;
})->each(function (&$column, $key) use ($definitions) {
$definition = collect($definitions)->where('Field', $key)->first();

$column->options = \Blueprint\EnumType::extractOptions($definition['Type']);
});
}

return $columns;
}

/**
* @param \Doctrine\DBAL\Schema\Column[] $columns
*/
private function mapColumns($columns)
{
return collect($columns)
->map([self::class, 'columns'])
->toArray();
}

public static function columns(\Doctrine\DBAL\Schema\Column $column, string $key)
{
$attributes = [];

$type = self::translations($column->getType()->getName());

if (in_array($type, ['decimal', 'float'])) {
if ($column->getPrecision()) {
$type .= ':' . $column->getPrecision();
}
if ($column->getScale()) {
$type .= ',' . $column->getScale();
}
} elseif ($type === 'string' && $column->getLength()) {
if ($column->getLength() !== 255) {
$type .= ':' . $column->getLength();
}
} elseif ($type === 'text') {
if ($column->getLength() > 65535) {
$type = 'longtext';
}
} elseif ($type === 'enum' && !empty($column->options)) {
$type .= ':' . implode(',', $column->options);
}

// TODO: guid/uuid

$attributes[] = $type;

if ($column->getUnsigned()) {
$attributes[] = 'unsigned';
}

if (!$column->getNotnull()) {
$attributes[] = 'nullable';
}

if ($column->getAutoincrement()) {
$attributes[] = 'autoincrement';
}

if (!is_null($column->getDefault())) {
$attributes[] = 'default:' . $column->getDefault();
}

return implode(' ', $attributes);
}

private static function translations(string $type)
{
static $mappings = [
'array' => 'string',
'bigint' => 'biginteger',
'binary' => 'binary',
'blob' => 'binary',
'boolean' => 'boolean',
'date' => 'date',
'date_immutable' => 'date',
'dateinterval' => 'date',
'datetime' => 'datetime',
'datetime_immutable' => 'datetime',
'datetimetz' => 'datetimetz',
'datetimetz_immutable' => 'datetimetz',
'decimal' => 'decimal',
'enum' => 'enum',
'float' => 'float',
'guid' => 'string',
'integer' => 'integer',
'json' => 'json',
'object' => 'string',
'simple_array' => 'string',
'smallint' => 'smallinteger',
'string' => 'string',
'text' => 'text',
'time' => 'time',
'time_immutable' => 'time',
];

return $mappings[$type] ?? 'string';
}

private function translateColumns(array $columns)
{
if (isset($columns['id']) && strpos($columns['id'], 'autoincrement') !== false && strpos($columns['id'], 'integer') !== false) {
unset($columns['id']);
}

if (isset($columns[Model::CREATED_AT]) && isset($columns[Model::UPDATED_AT])) {
if (strpos($columns[Model::CREATED_AT], 'datetimetz') !== false) {
$columns['timestampstz'] = 'timestampsTz';
}

unset($columns[Model::CREATED_AT]);
unset($columns[Model::UPDATED_AT]);
}

if (isset($columns['deleted_at'])) {
if (strpos($columns['deleted_at'], 'datetimetz') !== false) {
$columns['softdeletestz'] = 'softDeletesTz';
}

unset($columns['deleted_at']);
}

return $columns;
}

private function relativeClassName($model)
{
$name = Blueprint::relativeNamespace(get_class($model));
if (config('blueprint.models_namespace')) {
return $name;
} else {
$this->info('Traced ' . count($definitions) . ' ' . Str::plural('model', count($definitions)));
}

return ltrim(str_replace(config('blueprint.models_namespace'), '', $name), '\\');
return 0;
}
}
Loading