Skip to content

Commit

Permalink
Add path loader to fluent driver
Browse files Browse the repository at this point in the history
  • Loading branch information
patrickbrouwers committed Jul 4, 2017
1 parent 99f1182 commit f6c1573
Show file tree
Hide file tree
Showing 2 changed files with 185 additions and 4 deletions.
94 changes: 92 additions & 2 deletions src/FluentDriver.php
Expand Up @@ -6,9 +6,15 @@
use Doctrine\Common\Persistence\Mapping\Driver\MappingDriver;
use Doctrine\ORM\Mapping\Builder\ClassMetadataBuilder;
use Doctrine\ORM\Mapping\MappingException;
use FilesystemIterator;
use InvalidArgumentException;
use LaravelDoctrine\Fluent\Builders\Builder;
use LaravelDoctrine\Fluent\Mappers\MapperSet;
use RecursiveDirectoryIterator;
use RecursiveIteratorIterator;
use RecursiveRegexIterator;
use ReflectionClass;
use RegexIterator;

class FluentDriver implements MappingDriver
{
Expand All @@ -22,20 +28,37 @@ class FluentDriver implements MappingDriver
*/
protected $fluentFactory;

/**
* The file extension of mapping documents.
*
* @var string
*/
protected $fileExtension = '.php';

/**
* Initializes a new FileDriver that looks in the given path(s) for mapping
* documents and operates in the specified operating mode.
*
* @param string[] $mappings
* @param array $paths
*
* @throws MappingException
*/
public function __construct(array $mappings = [])
public function __construct(array $mappings = [], array $paths = [])
{
$this->fluentFactory = function (ClassMetadata $metadata) {
return new Builder(new ClassMetadataBuilder($metadata));
};

$this->mappers = new MapperSet();
$this->addMappings($mappings);

if (!empty($paths)) {
$this->loadPaths($paths);
}

if (!empty($mappings)) {
$this->addMappings($mappings);
}
}

/**
Expand Down Expand Up @@ -78,8 +101,75 @@ public function isTransient($className)
$this->mappers->getMapperFor($className)->isTransient();
}

/**
* @param array $paths
*
* @throws MappingException
*/
public function loadPaths(array $paths)
{
$classes = [];
$includedFiles = [];

foreach ($paths as $path) {
if (!is_dir($path)) {
throw MappingException::fileMappingDriversRequireConfiguredDirectoryPath($path);
}

$iterator = new RegexIterator(
new RecursiveIteratorIterator(
new RecursiveDirectoryIterator($path, FilesystemIterator::SKIP_DOTS),
RecursiveIteratorIterator::LEAVES_ONLY
),
'/^.+'.preg_quote($this->fileExtension).'$/i',
RecursiveRegexIterator::GET_MATCH
);

foreach ($iterator as $file) {
$sourceFile = $file[0];

if (!preg_match('(^phar:)i', $sourceFile)) {
$sourceFile = realpath($sourceFile);
}

require_once $sourceFile;

$includedFiles[] = $sourceFile;
}

$declared = get_declared_classes();

foreach ($declared as $className) {
$rc = new ReflectionClass($className);
$sourceFile = $rc->getFileName();

if (!in_array($sourceFile, $includedFiles)) {
continue;
}

if ($rc->isAbstract() || $rc->isInterface()) {
continue;
}

if (!$rc->implementsInterface(Mapping::class)) {
continue;
}

if ($this->isTransient($className)) {

/** @var Mapping $mapping */
$mapping = $rc->newInstanceWithoutConstructor();

$this->addMapping($mapping);
}
}
}
}

/**
* @param string[] $mappings
*
* @throws InvalidArgumentException
*/
public function addMappings(array $mappings = [])
{
Expand Down
95 changes: 93 additions & 2 deletions tests/FluentDriverTest.php
Expand Up @@ -82,6 +82,7 @@ public function test_it_should_load_metadata_for_mappings_passed_as_constructor_
StubEntity::class,
new ClassMetadataInfo(StubEntity::class)
);

$this->assertInstanceOf(
EntityMapper::class,
$driver->getMappers()->getMapperFor(StubEntity::class)
Expand All @@ -91,6 +92,7 @@ public function test_it_should_load_metadata_for_mappings_passed_as_constructor_
StubEmbeddable::class,
new ClassMetadataInfo(StubEmbeddable::class)
);

$this->assertInstanceOf(
EmbeddableMapper::class,
$driver->getMappers()->getMapperFor(StubEmbeddable::class)
Expand All @@ -100,6 +102,7 @@ public function test_it_should_load_metadata_for_mappings_passed_as_constructor_
StubMappedSuperClass::class,
new ClassMetadataInfo(StubMappedSuperClass::class)
);

$this->assertInstanceOf(
MappedSuperClassMapper::class,
$driver->getMappers()->getMapperFor(StubMappedSuperClass::class)
Expand All @@ -126,9 +129,96 @@ public function test_can_add_array_of_new_mappings()
);
}

public function test_can_load_mappings_through_file_path()
{
$driver = new FluentDriver([], [__DIR__ . '/' . 'Stubs/Mappings']);

$this->assertContains(
StubEntity::class,
$driver->getAllClassNames()
);

$this->assertContains(
StubEmbeddable::class,
$driver->getAllClassNames()
);

$this->assertContains(
StubMappedSuperClass::class,
$driver->getAllClassNames()
);

$driver->loadMetadataForClass(
StubEntity::class,
new ClassMetadataInfo(StubEntity::class)
);

$this->assertInstanceOf(
EntityMapper::class,
$driver->getMappers()->getMapperFor(StubEntity::class)
);

$driver->loadMetadataForClass(
StubEmbeddable::class,
new ClassMetadataInfo(StubEmbeddable::class)
);

$this->assertInstanceOf(
EmbeddableMapper::class,
$driver->getMappers()->getMapperFor(StubEmbeddable::class)
);

$driver->loadMetadataForClass(
StubMappedSuperClass::class,
new ClassMetadataInfo(StubMappedSuperClass::class)
);

$this->assertInstanceOf(
MappedSuperClassMapper::class,
$driver->getMappers()->getMapperFor(StubMappedSuperClass::class)
);
}

public function test_can_load_paths()
{
$driver = new FluentDriver();
$driver->loadPaths([__DIR__ . '/' . 'Stubs/Mappings']);

$this->assertContains(
StubEntity::class,
$driver->getAllClassNames()
);

$this->assertContains(
StubEmbeddable::class,
$driver->getAllClassNames()
);

$this->assertContains(
StubMappedSuperClass::class,
$driver->getAllClassNames()
);
}

public function test_loading_by_paths_throws_exception_if_dir_not_exists()
{
$folder = __DIR__ . '/' . 'Stubs/non-existing-folder';

$this->setExpectedException(
MappingException::class,
'File mapping drivers must have a valid directory path, however the given path [' . $folder . '] seems to be incorrect!'
);

$driver = new FluentDriver();
$driver->loadPaths([$folder]);

$driver->getAllClassNames();
}

public function test_the_given_mapping_class_should_exist()
{
$this->setExpectedException(\InvalidArgumentException::class, 'Mapping class [Tests\DoesnExist] does not exist');
$this->setExpectedException(\InvalidArgumentException::class,
'Mapping class [Tests\DoesnExist] does not exist');

$driver = new FluentDriver;

Expand All @@ -139,7 +229,8 @@ public function test_the_given_mapping_class_should_exist()

public function test_the_given_mapping_class_should_implement_mapping()
{
$this->setExpectedException(\InvalidArgumentException::class, 'Mapping class [Tests\Stubs\Entities\StubEntity] should implement LaravelDoctrine\Fluent\Mapping');
$this->setExpectedException(\InvalidArgumentException::class,
'Mapping class [Tests\Stubs\Entities\StubEntity] should implement LaravelDoctrine\Fluent\Mapping');

$driver = new FluentDriver;

Expand Down

0 comments on commit f6c1573

Please sign in to comment.