daux.io/libs/Daux.php

282 lines
8.0 KiB
PHP
Raw Normal View History

<?php namespace Todaymade\Daux;
2015-07-17 18:34:00 +02:00
use Symfony\Component\Console\Output\NullOutput;
2018-06-05 20:31:51 +02:00
use Symfony\Component\Console\Output\OutputInterface;
2015-07-29 22:31:41 +02:00
use Todaymade\Daux\ContentTypes\ContentTypeHandler;
use Todaymade\Daux\Tree\Builder;
use Todaymade\Daux\Tree\Root;
2015-04-22 12:23:57 +02:00
2015-04-23 00:32:30 +02:00
class Daux
{
const STATIC_MODE = 'DAUX_STATIC';
const LIVE_MODE = 'DAUX_LIVE';
2018-06-05 20:31:51 +02:00
public static $output;
2020-04-22 22:24:52 +02:00
/** @var Tree\Root */
public $tree;
/** @var Config */
public $config;
/** @var \Todaymade\Daux\Format\Base\Generator */
protected $generator;
2015-07-29 22:31:41 +02:00
/** @var ContentTypeHandler */
protected $typeHandler;
/** @var string[] */
protected $validExtensions;
2015-07-18 21:45:38 +02:00
/** @var Processor */
2015-07-17 18:34:00 +02:00
protected $processor;
2015-07-19 14:05:12 +02:00
/** @var bool */
private $merged_tree = false;
public function __construct(Config $config, OutputInterface $output)
{
2018-06-05 20:31:51 +02:00
Daux::$output = $output;
$this->config = $config;
2015-04-23 00:32:30 +02:00
}
2015-07-18 21:45:38 +02:00
/**
2020-04-22 22:24:52 +02:00
* Generate the tree that will be used.
2015-07-18 21:45:38 +02:00
*/
2015-07-19 23:17:57 +02:00
public function generateTree()
2015-04-23 00:32:30 +02:00
{
$this->config->setValidContentExtensions($this->getContentExtensions());
$this->tree = new Root($this->getConfig());
Builder::build($this->tree, $this->config->getIgnore());
// Apply the language name as Section title
if ($this->config->isMultilanguage()) {
foreach ($this->config->getLanguages() as $key => $node) {
2015-07-18 21:45:38 +02:00
$this->tree->getEntries()[$key]->setTitle($node);
}
}
// Enhance the tree with processors
$this->getProcessor()->manipulateTree($this->tree);
// Sort the tree one last time before it is finalized
2017-11-07 22:44:27 +01:00
Builder::sortTree($this->tree);
2017-11-07 22:44:27 +01:00
Builder::finalizeTree($this->tree);
2015-04-23 00:32:30 +02:00
}
/**
2015-07-14 23:35:47 +02:00
* @return Config
*/
public function getConfig()
2015-04-23 00:32:30 +02:00
{
2015-07-19 14:05:12 +02:00
if ($this->tree && !$this->merged_tree) {
$this->config->setTree($this->tree);
$this->config->setIndex($this->tree->getIndexPage() ?: $this->tree->getFirstPage());
$entry_page = null;
if ($this->config->isMultilanguage()) {
$entry_page = [];
foreach ($this->config->getLanguages() as $key => $name) {
$entry_page[$key] = $this->tree->getEntries()[$key]->getFirstPage();
2015-04-22 12:23:57 +02:00
}
} else {
2019-12-07 16:32:38 +01:00
$entry_page = $this->tree->getFirstPage();
}
$this->config->setEntryPage($entry_page);
2015-07-19 14:05:12 +02:00
$this->merged_tree = true;
}
return $this->config;
}
/**
* @return Config
2020-04-22 22:24:52 +02:00
*
* @deprecated Use getConfig instead
*/
public function getParams()
{
return $this->getConfig();
2015-04-22 12:23:57 +02:00
}
2015-07-17 18:34:00 +02:00
/**
* @return Processor
*/
public function getProcessor()
{
if (!$this->processor) {
2018-06-05 20:31:51 +02:00
$this->processor = new Processor($this, Daux::getOutput(), 0);
2015-07-17 18:34:00 +02:00
}
return $this->processor;
}
public function setProcessor(Processor $processor)
{
$this->processor = $processor;
// This is not the cleanest but it's
// the best I've found to use the
// processor in very remote places
$this->config->setProcessorInstance($processor);
}
public function getGenerators()
{
$default = [
'confluence' => '\Todaymade\Daux\Format\Confluence\Generator',
'html-file' => '\Todaymade\Daux\Format\HTMLFile\Generator',
'html' => '\Todaymade\Daux\Format\HTML\Generator',
];
$extended = $this->getProcessor()->addGenerators();
return array_replace($default, $extended);
2015-07-17 18:34:00 +02:00
}
public function getProcessorClass()
{
$processor = $this->getConfig()->getProcessor();
if (empty($processor)) {
return null;
}
2016-07-27 21:32:51 +02:00
$class = '\\Todaymade\\Daux\\Extension\\' . $processor;
if (!class_exists($class)) {
throw new \RuntimeException("Class '$class' not found. We cannot use it as a Processor");
}
2016-07-27 21:32:51 +02:00
if (!array_key_exists('Todaymade\\Daux\\Processor', class_parents($class))) {
2016-01-31 11:59:35 +01:00
throw new \RuntimeException("Class '$class' invalid, should extend '\\Todaymade\\Daux\\Processor'");
}
return $class;
}
2020-04-22 21:55:53 +02:00
protected function findAlternatives($input, $words)
{
$alternatives = [];
foreach ($words as $word) {
$lev = levenshtein($input, $word);
if ($lev <= \strlen($word) / 3) {
$alternatives[] = $word;
}
}
return $alternatives;
}
/**
* @return \Todaymade\Daux\Format\Base\Generator
*/
public function getGenerator()
{
if ($this->generator) {
return $this->generator;
}
$generators = $this->getGenerators();
$format = $this->getConfig()->getFormat();
if (!array_key_exists($format, $generators)) {
$message = "The format '$format' doesn't exist, did you forget to set your processor ?";
$alternatives = $this->findAlternatives($format, array_keys($generators));
if (0 == \count($alternatives)) {
$message .= "\n\nAvailable formats are \n " . implode("\n ", array_keys($generators));
} elseif (1 == \count($alternatives)) {
$message .= "\n\nDid you mean this?\n " . implode("\n ", $alternatives);
} else {
$message .= "\n\nDid you mean one of these?\n " . implode("\n ", $alternatives);
}
throw new \RuntimeException($message);
}
$class = $generators[$format];
if (!class_exists($class)) {
throw new \RuntimeException("Class '$class' not found. We cannot use it as a Generator");
}
$interface = 'Todaymade\Daux\Format\Base\Generator';
if (!in_array('Todaymade\Daux\Format\Base\Generator', class_implements($class))) {
throw new \RuntimeException("The class '$class' does not implement the '$interface' interface");
}
return $this->generator = new $class($this);
}
public function getContentTypeHandler()
{
if ($this->typeHandler) {
return $this->typeHandler;
}
$base_types = $this->getGenerator()->getContentTypes();
$extended = $this->getProcessor()->addContentType();
$types = array_merge($base_types, $extended);
return $this->typeHandler = new ContentTypeHandler($types);
}
/**
2020-04-22 22:24:52 +02:00
* Get all content file extensions.
*
* @return string[]
*/
public function getContentExtensions()
{
2016-07-29 22:38:03 +02:00
if (!empty($this->validExtensions)) {
return $this->validExtensions;
}
return $this->validExtensions = $this->getContentTypeHandler()->getContentExtensions();
}
2018-06-05 20:31:51 +02:00
2020-04-22 21:55:53 +02:00
public static function getOutput()
{
2018-06-05 20:31:51 +02:00
if (!Daux::$output) {
2018-06-07 20:40:38 +02:00
Daux::$output = new NullOutput();
2018-06-05 20:31:51 +02:00
}
return Daux::$output;
}
/**
* Writes a message to the output.
*
2020-04-22 22:24:52 +02:00
* @param array|string $messages The message as an array of lines or a single string
2018-06-05 20:31:51 +02:00
* @param bool $newline Whether to add a newline
* @param int $options A bitmask of options (one of the OUTPUT or VERBOSITY constants), 0 is considered the same as self::OUTPUT_NORMAL | self::VERBOSITY_NORMAL
*/
2020-04-22 21:55:53 +02:00
public static function write($messages, $newline = false, $options = 0)
{
2018-06-07 20:40:38 +02:00
Daux::getOutput()->write($messages, $newline, $options);
2018-06-05 20:31:51 +02:00
}
/**
* Writes a message to the output and adds a newline at the end.
*
2020-04-22 22:24:52 +02:00
* @param array|string $messages The message as an array of lines of a single string
2018-06-05 20:31:51 +02:00
* @param int $options A bitmask of options (one of the OUTPUT or VERBOSITY constants), 0 is considered the same as self::OUTPUT_NORMAL | self::VERBOSITY_NORMAL
*/
2020-04-22 21:55:53 +02:00
public static function writeln($messages, $options = 0)
{
2018-06-07 20:40:38 +02:00
Daux::getOutput()->write($messages, true, $options);
2018-06-05 20:31:51 +02:00
}
2020-04-22 21:55:53 +02:00
public static function getVerbosity()
{
2018-06-05 20:31:51 +02:00
return Daux::getOutput()->getVerbosity();
}
2015-04-23 00:32:30 +02:00
}