Passed
Pull Request — master (#2172)
by Franck
05:02
created

Render::getName()   A

Complexity

Conditions 1
Paths 1

Size

Total Lines 3
Code Lines 1

Duplication

Lines 0
Ratio 0 %

Code Coverage

Tests 2
CRAP Score 1

Importance

Changes 0
Metric Value
cc 1
eloc 1
nc 1
nop 0
dl 0
loc 3
ccs 2
cts 2
cp 1
crap 1
rs 10
c 0
b 0
f 0
1
<?php
2
3
declare(strict_types=1);
4
5
/*
6
 * This file is part of Cecil.
7
 *
8
 * Copyright (c) Arnaud Ligny <[email protected]>
9
 *
10
 * For the full copyright and license information, please view the LICENSE
11
 * file that was distributed with this source code.
12
 */
13
14
namespace Cecil\Step\Pages;
15
16
use Cecil\Builder;
17
use Cecil\Collection\Page\Collection;
18
use Cecil\Collection\Page\Page;
19
use Cecil\Exception\RuntimeException;
20
use Cecil\Renderer\Config;
21
use Cecil\Renderer\Layout;
22
use Cecil\Renderer\Site;
23
use Cecil\Renderer\Twig;
24
use Cecil\Step\AbstractStep;
25
use Cecil\Util;
26
27
/**
28
 * Pages rendering.
29
 */
30
class Render extends AbstractStep
31
{
32
    public const TMP_DIR = '.cecil';
33
34
    protected $renderSubset = [];
35
36
    /**
37
     * {@inheritdoc}
38
     */
39 1
    public function getName(): string
40
    {
41 1
        return 'Rendering pages';
42
    }
43
44
    /**
45
     * {@inheritdoc}
46
     */
47 1
    public function init(array $options): void
48
    {
49 1
        if (!is_dir($this->config->getLayoutsPath()) && !$this->config->hasTheme()) {
50
            $message = \sprintf('"%s" is not a valid layouts directory', $this->config->getLayoutsPath());
51
            $this->builder->getLogger()->debug($message);
52
        }
53
54 1
        if (!empty($options['render-subset'])) {
55
            $subset = 'pages.subsets.' . (string) $options['render-subset'];
56
            if ($this->config->has($subset)) {
57
                $this->renderSubset = (array) $this->config->get($subset);
58
            }
59
        }
60
61 1
        $this->canProcess = true;
62
    }
63
64
    /**
65
     * {@inheritdoc}
66
     *
67
     * @throws RuntimeException
68
     */
69 1
    public function process(): void
70
    {
71
        // prepares renderer
72 1
        $this->builder->setRenderer(new Twig($this->builder, $this->getAllLayoutsPaths()));
73
74
        // adds global variables
75 1
        $this->addGlobals();
76
77 1
        $renderSubset = $this->renderSubset;
78
79
        /** @var Collection $pages */
80 1
        $pages = $this->builder->getPages()
81 1
            // published only
82 1
            ->filter(function (Page $page) {
83 1
                return (bool) $page->getVariable('published');
84 1
            })
85 1
            ->filter(function (Page $page) use ($renderSubset) {
86 1
                if (empty($renderSubset)) {
87 1
                    return true;
88
                }
89
                if (!empty($renderSubset['path'])
90
                    && !(
91
                        (bool) preg_match('/' . (string) $renderSubset['path'] . '/', $page->getPath())
92
                    )
93
                ) {
94
                    return false;
95
                }
96
                if (!empty($renderSubset['language'])) {
97
                    $language = $page->getVariable('language', $this->config->getLanguageDefault());
98
                    if ($language !== (string) $renderSubset['language']) {
99
                        return false;
100
                    }
101
                }
102
                return true;
103 1
            })
104 1
            // enrichs some variables
105 1
            ->map(function (Page $page) {
106 1
                $formats = $this->getOutputFormats($page);
107
                // output formats
108 1
                $page->setVariable('output', $formats);
109
                // alternates formats
110 1
                $page->setVariable('alternates', $this->getAlternates($formats));
111
                // translations
112 1
                $page->setVariable('translations', $this->getTranslations($page));
113
114 1
                return $page;
115 1
            });
116 1
        $total = \count($pages);
117
118
        // renders each page
119 1
        $count = 0;
120 1
        $postprocessors = [];
121 1
        foreach ((array) $this->config->get('output.postprocessors') as $name => $postprocessor) {
122
            try {
123 1
                if (!class_exists($postprocessor)) {
124 1
                    throw new RuntimeException(\sprintf('Class "%s" not found', $postprocessor));
125
                }
126 1
                $postprocessors[] = new $postprocessor($this->builder);
127 1
                $this->builder->getLogger()->debug(\sprintf('Output post processor "%s" loaded', $name));
128 1
            } catch (\Exception $e) {
129 1
                $this->builder->getLogger()->error(\sprintf('Unable to load output post processor "%s": %s', $name, $e->getMessage()));
130
            }
131
        }
132
133 1
        $cacheLocale = $cacheSite = $cacheConfig = [];
134
135
        /** @var Page $page */
136 1
        foreach ($pages as $page) {
137 1
            $count++;
138 1
            $rendered = [];
139
140
            // l10n
141 1
            $language = $page->getVariable('language', $this->config->getLanguageDefault());
142 1
            if (!isset($cacheLocale[$language])) {
143 1
                $cacheLocale[$language] = $this->config->getLanguageProperty('locale', $language);
144
            }
145 1
            $this->builder->getRenderer()->setLocale($cacheLocale[$language]);
146
147
            // global site variables
148 1
            if (!isset($cacheSite[$language])) {
149 1
                $cacheSite[$language] = new Site($this->builder, $language);
150
            }
151 1
            $this->builder->getRenderer()->addGlobal('site', $cacheSite[$language]);
152
153
            // global config raw variables
154 1
            if (!isset($cacheConfig[$language])) {
155 1
                $cacheConfig[$language] = new Config($this->builder, $language);
156
            }
157 1
            $this->builder->getRenderer()->addGlobal('config', $cacheConfig[$language]);
158
159
            // excluded format(s)?
160 1
            $formats = (array) $page->getVariable('output');
161 1
            foreach ($formats as $key => $format) {
162 1
                if ($exclude = $this->config->getOutputFormatProperty($format, 'exclude')) {
163
                    // ie:
164
                    //   formats:
165
                    //     atom:
166
                    //       [...]
167
                    //       exclude: [paginated]
168 1
                    if (!\is_array($exclude)) {
169
                        $exclude = [$exclude];
170
                    }
171 1
                    foreach ($exclude as $variable) {
172 1
                        if ($page->hasVariable($variable)) {
173 1
                            unset($formats[$key]);
174
                        }
175
                    }
176
                }
177
            }
178
179
            // renders each output format
180 1
            foreach ($formats as $format) {
181
                // search for the template
182 1
                $layout = Layout::finder($page, $format, $this->config);
183
                // renders with Twig
184
                try {
185 1
                    $deprecations = [];
186 1
                    set_error_handler(function ($type, $msg) use (&$deprecations) {
187 1
                        if (E_USER_DEPRECATED === $type) {
188 1
                            $deprecations[] = $msg;
189
                        }
190 1
                    });
191 1
                    $output = $this->builder->getRenderer()->render($layout['file'], ['page' => $page]);
192 1
                    foreach ($deprecations as $value) {
193 1
                        $this->builder->getLogger()->warning($value);
194
                    }
195 1
                    foreach ($postprocessors as $postprocessor) {
196 1
                        $output = $postprocessor->process($page, $output, $format);
197
                    }
198 1
                    $rendered[$format] = [
199 1
                        'output'   => $output,
200 1
                        'template' => [
201 1
                            'scope' => $layout['scope'],
202 1
                            'file'  => $layout['file'],
203 1
                        ],
204 1
                    ];
205 1
                    $page->addRendered($rendered);
206
                } catch (\Twig\Error\Error $e) {
207
                    throw new RuntimeException(
208
                        \sprintf(
209
                            'Can\'t render template "%s" for page "%s".',
210
                            $e->getSourceContext()->getName(),
211
                            $page->getFileName() ?? $page->getId()
212
                        ),
213
                        previous: $e,
214
                        file: $e->getSourceContext()->getPath(),
215
                        line: $e->getTemplateLine(),
216
                    );
217
                } catch (\Exception $e) {
218
                    throw new RuntimeException($e->getMessage(), previous: $e);
219
                }
220
            }
221 1
            $this->builder->getPages()->replace($page->getId(), $page);
222
223 1
            $templates = array_column($rendered, 'template');
224 1
            $message = \sprintf(
225 1
                'Page "%s" rendered with [%s]',
226 1
                $page->getId() ?: 'index',
227 1
                Util\Str::combineArrayToString($templates, 'scope', 'file')
228 1
            );
229 1
            $this->builder->getLogger()->info($message, ['progress' => [$count, $total]]);
230
        }
231
        // profiler
232 1
        if ($this->builder->isDebug()) {
233
            try {
234
                // HTML
235 1
                $htmlDumper = new \Twig\Profiler\Dumper\HtmlDumper();
236 1
                $profileHtmlFile = Util::joinFile($this->config->getDestinationDir(), self::TMP_DIR, 'twig_profile.html');
237 1
                Util\File::getFS()->dumpFile($profileHtmlFile, $htmlDumper->dump($this->builder->getRenderer()->getDebugProfile()));
238
                // TXT
239 1
                $textDumper = new \Twig\Profiler\Dumper\TextDumper();
240 1
                $profileTextFile = Util::joinFile($this->config->getDestinationDir(), self::TMP_DIR, 'twig_profile.txt');
241 1
                Util\File::getFS()->dumpFile($profileTextFile, $textDumper->dump($this->builder->getRenderer()->getDebugProfile()));
242
                // log
243 1
                $this->builder->getLogger()->debug(\sprintf('Twig profile dumped in "%s"', Util::joinFile($this->config->getDestinationDir(), self::TMP_DIR)));
244
            } catch (\Symfony\Component\Filesystem\Exception\IOException $e) {
245
                throw new RuntimeException($e->getMessage());
246
            }
247
        }
248
    }
249
250
    /**
251
     * Returns an array of layouts directories.
252
     */
253 1
    protected function getAllLayoutsPaths(): array
254
    {
255 1
        $paths = [];
256
257
        // layouts/
258 1
        if (is_dir($this->config->getLayoutsPath())) {
259 1
            $paths[] = $this->config->getLayoutsPath();
260
        }
261
        // <theme>/layouts/
262 1
        if ($this->config->hasTheme()) {
263 1
            foreach ($this->config->getTheme() ?? [] as $theme) {
264 1
                $paths[] = $this->config->getThemeDirPath($theme);
265
            }
266
        }
267
        // resources/layouts/
268 1
        if (is_dir($this->config->getLayoutsInternalPath())) {
269 1
            $paths[] = $this->config->getLayoutsInternalPath();
270
        }
271
272 1
        return $paths;
273
    }
274
275
    /**
276
     * Adds global variables.
277
     */
278 1
    protected function addGlobals()
279
    {
280 1
        $this->builder->getRenderer()->addGlobal('cecil', [
281 1
            'url'       => \sprintf('https://cecil.app/#%s', Builder::getVersion()),
282 1
            'version'   => Builder::getVersion(),
283 1
            'poweredby' => \sprintf('Cecil v%s', Builder::getVersion()),
284 1
        ]);
285
    }
286
287
    /**
288
     * Get available output formats.
289
     *
290
     * @throws RuntimeException
291
     */
292 1
    protected function getOutputFormats(Page $page): array
293
    {
294
        // Get page output format(s) if defined.
295
        // ie:
296
        // ```yaml
297
        // output: txt
298
        // ```
299 1
        if ($page->getVariable('output')) {
300 1
            $formats = $page->getVariable('output');
301 1
            if (!\is_array($formats)) {
302 1
                $formats = [$formats];
303
            }
304
305 1
            return $formats;
306
        }
307
308
        // Get available output formats for the page type.
309
        // ie:
310
        // ```yaml
311
        // page: [html, json]
312
        // ```
313 1
        $formats = $this->config->get('output.pagetypeformats.' . $page->getType());
314 1
        if (empty($formats)) {
315
            throw new RuntimeException('Configuration key "pagetypeformats" can\'t be empty.');
316
        }
317 1
        if (!\is_array($formats)) {
318
            $formats = [$formats];
319
        }
320
321
        // Render specific output formats from subset
322 1
        if (!empty($this->renderSubset['output']) && \in_array((string) $this->renderSubset['output'], $formats)) {
323
            return [(string) $this->renderSubset['output']];
324
        }
325
326 1
        return array_unique($formats);
327
    }
328
329
    /**
330
     * Get alternates.
331
     */
332 1
    protected function getAlternates(array $formats): array
333
    {
334 1
        $alternates = [];
335
336 1
        if (\count($formats) > 1 || \in_array('html', $formats)) {
337 1
            foreach ($formats as $format) {
338 1
                $format == 'html' ? $rel = 'canonical' : $rel = 'alternate';
339 1
                $alternates[] = [
340 1
                    'rel'    => $rel,
341 1
                    'type'   => $this->config->getOutputFormatProperty($format, 'mediatype'),
342 1
                    'title'  => strtoupper($format),
343 1
                    'format' => $format,
344 1
                ];
345
            }
346
        }
347
348 1
        return $alternates;
349
    }
350
351
    /**
352
     * Returns the collection of translated pages for a given page.
353
     */
354 1
    protected function getTranslations(Page $refPage): Collection
355
    {
356 1
        $pages = $this->builder->getPages()->filter(function (Page $page) use ($refPage) {
357 1
            return $page->getId() !== $refPage->getId()
358 1
                && $page->getVariable('langref') == $refPage->getVariable('langref')
359 1
                && $page->getType() == $refPage->getType()
360 1
                && !empty($page->getVariable('published'))
361 1
                && !$page->getVariable('paginated');
362 1
        });
363
364 1
        return $pages;
365
    }
366
}
367