Skip to content

Commit

Permalink
Merge pull request #52 from cakephp/fileloader-extends
Browse files Browse the repository at this point in the history
Manually search template paths for files loaded outside View functions
  • Loading branch information
othercorey authored May 1, 2020
2 parents e2f557a + ea1405a commit c5d6f03
Show file tree
Hide file tree
Showing 6 changed files with 76 additions and 30 deletions.
65 changes: 52 additions & 13 deletions src/Twig/FileLoader.php
Original file line number Diff line number Diff line change
Expand Up @@ -18,8 +18,8 @@

namespace Cake\TwigView\Twig;

use Cake\TwigView\View\TwigView;
use Cake\View\Exception\MissingTemplateException;
use Cake\Core\App;
use Cake\Core\Plugin;
use Twig\Error\LoaderError;
use Twig\Loader\LoaderInterface;
use Twig\Source;
Expand All @@ -32,16 +32,16 @@
class FileLoader implements LoaderInterface
{
/**
* @var \Cake\TwigView\View\TwigView
* @var string[]
*/
protected $twigView;
protected $extensions;

/**
* @param \Cake\TwigView\View\TwigView $twigView TwigView instance.
* @param string[] $extensions Template file extensions
*/
public function __construct(TwigView $twigView)
public function __construct(array $extensions)
{
$this->twigView = $twigView;
$this->extensions = $extensions;
}

/**
Expand Down Expand Up @@ -79,7 +79,7 @@ public function exists(string $name)
{
try {
$this->findTemplate($name);
} catch (MissingTemplateException $e) {
} catch (LoaderError $e) {
return false;
}

Expand All @@ -96,12 +96,51 @@ public function findTemplate(string $name): string
return $name;
}

try {
$path = $this->twigView->resolveTemplatePath($name);
} catch (MissingTemplateException $e) {
throw new LoaderError($e->getMessage());
[$plugin, $name] = pluginSplit($name);
$name = str_replace('/', DIRECTORY_SEPARATOR, $name);

if ($plugin !== null) {
$templatePath = Plugin::templatePath($plugin);
$path = $this->checkExtensions($templatePath . $name);
if ($path !== null) {
return $path;
}

$error = "Could not find template `{$name}` in plugin `{$plugin}` in these paths:\n\n"
. "- `{$templatePath}`\n";
throw new LoaderError($error);
}

foreach (App::path('templates') as $templatePath) {
$path = $this->checkExtensions($templatePath . $name);
if ($path !== null) {
return $path;
}
}

$error = "Could not find template `{$name}` in these paths:\n\n";
foreach (App::path('templates') as $templatePath) {
$error .= "- `{$templatePath}`\n";
}
throw new LoaderError($error);
}

/**
* Check partial path with all template file extensions to see
* which file exists.
*
* @param string $partial Template path excluding extension
* @return string|null
*/
public function checkExtensions(string $partial): ?string
{
foreach ($this->extensions as $extension) {
$path = $partial . $extension;
if (file_exists($path)) {
return $path;
}
}

return $path;
return null;
}
}
14 changes: 1 addition & 13 deletions src/View/TwigView.php
Original file line number Diff line number Diff line change
Expand Up @@ -150,26 +150,14 @@ public function getExtensions(): array
return $this->extensions;
}

/**
* Gets the full path for template.
*
* @param string $name Template name
* @return string
* @throws \Cake\View\Exception\MissingTemplateException When template is not found
*/
public function resolveTemplatePath(string $name): string
{
return $this->_getTemplateFileName($name);
}

/**
* Creates the Twig LoaderInterface instance.
*
* @return \Twig\Loader\LoaderInterface
*/
protected function createLoader(): LoaderInterface
{
return new FileLoader($this);
return new FileLoader($this->extensions);
}

/**
Expand Down
13 changes: 9 additions & 4 deletions tests/TestCase/Twig/FileLoaderTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -20,7 +20,6 @@

use Cake\TestSuite\TestCase;
use Cake\TwigView\Twig\FileLoader;
use Cake\TwigView\View\TwigView;
use Twig\Error\LoaderError;

/**
Expand All @@ -39,7 +38,7 @@ public function setUp(): void

$this->loadPlugins(['TestTwigView']);

$this->loader = new FileLoader(new TwigView());
$this->loader = new FileLoader(['.twig']);
}

public function tearDown(): void
Expand All @@ -60,8 +59,9 @@ public function testGetSource()
public function testGetSourceNonExistingFile()
{
$this->expectException(LoaderError::class);
$this->expectExceptionMessage("Could not find template `missing` in plugin `TestTwigView`");

$this->loader->getSourceContext('TestTwigView.no_twig');
$this->loader->getSourceContext('TestTwigView.missing');
}

public function testGetCacheKey()
Expand Down Expand Up @@ -93,7 +93,12 @@ public function testIsFresh()
public function testIsFreshNonExistingFile()
{
$this->expectException(LoaderError::class);

$this->loader->isFresh(TMP . 'foobar' . time(), time());
}

public function testExistsNonExistingFile()
{
$exists = $this->loader->exists(TMP . 'foobar' . time(), time());
$this->assertSame(false, $exists);
}
}
12 changes: 12 additions & 0 deletions tests/TestCase/View/TwigViewTest.php
Original file line number Diff line number Diff line change
Expand Up @@ -169,6 +169,18 @@ public function testTwigInclude()
$this->removePlugins(['TestTwigView']);
}

/**
* Tests extends loads templates from root templates paths.
*
* @return void
*/
public function testTwigExtendsRootPath()
{
$view = new AppView(null, null, null, ['templatePath' => 'Blog']);
$output = $view->render('blog_with_extends');
$this->assertSame('base from subdir/base', $output);
}

/**
* Tests deprecated element and cell tags render.
*
Expand Down
1 change: 1 addition & 0 deletions tests/test_app/templates/Blog/blog_with_extends.twig
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
{% extends 'subdir_with_base/base' %}
1 change: 1 addition & 0 deletions tests/test_app/templates/subdir_with_base/base.twig
Original file line number Diff line number Diff line change
@@ -0,0 +1 @@
base from subdir/base

0 comments on commit c5d6f03

Please sign in to comment.