<?php
namespace Lia;
/**
* Encaspulate multiple components, views, routes & other bits in a single directory as a package.
*
* @featured
* @tag setup
*/
class Package implements \LiaisonInterface\PackageLifeCycle {
/**
* The configured name of this package or uniqid() if not specified
*/
public $name;
/**
* The root dir for the package
* @featured
*/
public $dir;
/**
* the liaison instance
* @featured
*/
public $lia;
/**
* All compos added. `['name'=>$compoObject, 'name2'=>$otherCompo]`
* @featured
*/
protected $compos = [];
/**
* Components implementing `LiaisonInterface\PackageLifeCycle`
* @warning only initialized after first call to `lifecycle()`
*/
protected $lifecycleCompos = [];
/**
* Configurations for the package
* This implementation is going to change, but I don't know how that's gonna work yet. I just know I don't like it.
*/
//@export_start(Usage.Package.DefaultConfigs)
protected $config = [
'namespace'=>'',
'dir'=>
[
'component' => 'core',
// 'public' => 'public',
'autoload'=> 'class',
'view' => 'view',
'cache' => 'cache',
'public'=>[ // @TODO rethink config structure. public maybe shouldn't be nested under dir? Idk....
'dir' => 'public',
'baseUrl'=> '',
]
],
'route'=>
[
'force_trail_slash'=>true,
'index_names'=>
[
'index'
],
'hidden_extensions'=>
[
'php'
],
],
'class'=>
[
'view' => '\\Lia\\Obj\\View'
],
'views'=>
[
// 'conflict' => 'throw'
]
];
//@export_end(Package.DefaultConfigs)
/**
* Initialize package.
*
* - Configs are loaded from a config file in the package. Any keys set in the $config array will replace values set in the config file
* - Setup is only performed if dir is non-null
*
* @param \Liaison $lia a Liaison instance
* @param string $dir the root directory for the package or null to skip setup.
* @param array $configs Configs to set. Pass ['name'=>'SomeName'], to later use `$lia->package('SomeName')` to retrieve it
*
* @featured
*/
public function __construct(\Liaison $lia, ?string $dir=null, array $configs = []){
$this->dir = $dir;
$this->lia = $lia;
if (!isset($configs['name']))$this->name = uniqid();
else $this->name = strtolower($configs['name']);
echo "\nSHOULD arset lia:package\n";
$this->lia->arset('lia:package', $this->name, $this);
if ($dir!=null){
$this->setup($configs, $dir);
}
}
public function name(){
return $this->name;
}
/**
* Setup the package.
* @warning Can be called multiple times but shouldn't be.
*/
public function setup(array $configs, ?string $dir = null){
if ($dir!==null)$this->dir = $dir;
if (!is_dir($this->dir)){
$class = get_class($this);
throw new \Lia\Exception\Base("Package dir must be set via `new ${class}(\$lia, \$dir)` or ${class}->setup(\$configs, \$dir) call. '$dir' is not valid");
}
$lia = $this->lia;
$this->setup_config_file($this->dir.'/config.json');
$configs = \Liaison\Utility\DotNotation::nestedFromDotted($configs);
$this->config = array_replace_recursive($this->config, $configs);
$this->setup_components($this->dir('component'));
$this->lifecycle('onComponentsLoaded');
$this->setup_autoload($this->dir('autoload'));
$this->setup_views(realpath($this->dir('view')),
['lia'=>$lia, 'package'=>$this]
);
$this->setup_public_routes($this->dir('public'));
$this->lifecycle('onPrePackageReady');
$this->lifecycle('onPackageReady');
$name = $this->config['name'] ?? uniqid();
if ($this->get('namespace')==null)$this->set('namespace',$name);
$this->name = $name;
if ($lia->hasApi('lia:package.add')){
$lia->api('lia:package.add', $this, $this->get('namespace'));
}
}
/** @override */
public function onComponentsLoaded(){}
/** @override */
public function onPrePackageReady(){}
/** @override */
public function onPackageReady(){}
/**
* Execute the given lifecycle method on each component in this package
* @param $name the name of the lifecycle method
*/
protected function lifecycle($name){
if ($this->lifecycleCompos==null){
foreach ($this->compos as $c){
if ($c instanceof \LiaisonInterface\PackageLifeCycle)
$this->lifecycleCompos[] = $c;
}
}
$this->$name();
foreach ($this->lifecycleCompos as $compo){
$compo->$name();
}
}
//
// utility functions for processing/setup
//
/**
* Add a component to the package.
*
* @param string $name name to use for fetching the component
* @param mixed $component Generally a \Lia\Compo, but may be anything.
* @tag setup
*/
public function addComponent($name, $component){
$name = strtolower($name);
$this->compos[$name] = $component;
}
/**
* Load Config file. Currently `$package->dir.'/config.json'`.
* @param string $configFile the config file to load
* @tag setup
*/
protected function setup_config_file(string $configFile){
if (file_exists($configFile)){
$json = file_get_contents($configFile);
$configs = json_decode($json,true);
$configs = \Liaison\Utility\DotNotation::nestedFromDotted($configs);
$merged = array_replace_recursive($this->config,$configs);
$this->config = $merged;
}
}
/**
* Convert a relative file path into a pattern for the Router component
*
* @todo move file path => pattern conversion into router
* @tag utility, routing
*/
protected function fileToPattern($relFile){
$pattern = $relFile;
$ext = pathinfo($relFile,PATHINFO_EXTENSION);
$hidden = $this->config['route']['hidden_extensions'];
if (in_array($ext,$hidden)){
$pattern = substr($pattern,0,-(strlen($ext)+1));
$ext = '';
}
$indexNames = $this->config['route']['index_names'];
$base = basename($pattern);
if (in_array($base,$indexNames)){
$pattern = substr($pattern,0,-strlen($base));
}
if ($ext==''
&&$this->config['route']['force_trail_slash'])$pattern .= '/';
$pattern = str_replace(['///','//'], '/', $pattern);
return $pattern;
}
// get stuff/info from package
/**
* Set a config on the package (Does NOT propagate to Liaison)
* @param $dotKey
* @featured
*/
public function set($key, $value){
$array = \Liaison\Utility\DotNotation::nestedFromPair($key,$value);
$this->config = array_replace_recursive($this->config, $array);
}
/**
* Get a config from the package (NOT from Liaison)
* @featured
*/
public function get($key){
return \Liaison\Utility\DotNotation::getNestedValue($key, $this->config);
}
/**
* Get the configured class for the given key.
* Shorthand for `get('class.$key');`
* @featured
*/
public function class($forKey){
return $this->config['class'][$forKey];
}
/**
* Get the named component from the package
*
* @param $name usually the component's class name (without namespace)
* @featured
*/
public function compo($name){
$name = strtolower($name);
$compo = $this->compos[$name] ?? null;
return $compo;
if ($compo==null){
$packageName = $this->get('name') ?? '--unnamed-package--';
throw new \Lia\Exception\Base("A compo with name '{$name}' does not exist for package {$packageName}");
}
return $compo;
}
/**
* Get the path to a directory in this package, or the package root dir if `$forKey==null`
*
* @param string $dirName configured directory key/name.
* @return string directory path
* @featured
*/
public function dir($forKey=null){
if ($forKey==null)return $this->dir;
$dir = $this->config['dir'][$forKey] ?? null;
if ($dir===null)return null;
if (is_array($dir))$dir = $dir['dir'];
return $this->dir.'/'.$dir;
}
/**
* Prepends base url to the given path & sets $getparams as querystring. Replaces all double shashes (`//`) with single (`/`)
*
* @param $getParams key=>value array or a query string
* @return a url path (no domain)
* @todo create a version of this method in the Router component
* @featured
*/
public function url($path, $getParams=null){
$get = $getParams;
$qs = '';
if (is_array($get)){
$qs = http_build_query($get);
} else {
$qs = ''.$get;
}
$qs = substr($qs,0,1)=='?' ? $qs : '?'.$qs;
$base = $this->get('dir.public.baseUrl');
$url = '/'.$base.'/'.$path;
$url = str_replace(['////','///','//'], '/', $url);
$url = $qs=='?' ? $url : $url.$qs;
return $url;
}
//
//load all the pieces of a package
//
/**
* Adds configured 'autoload' dir to the Autoloader component
*
* @param $dir the directory to setup for autoloading
* @tag setup
*/
public function setup_autoload($dir){
if ($this->lia->hasApi('lia:autoload.addDir'))
$this->lia->api('lia:autoload.addDir', $dir);
}
/**
* Load all components from the configured 'component' dir
*
* @param $dir the directory components are in
* @tag setup
*/
protected function setup_components(string $dir){
$classes = \Lia\Utility\ClassFinder::classesFromDir($dir);
if (count($classes)==0)return;
$components = [];
foreach ($classes as $info){
// if (!in_array('Lia\\iCore\\Compo',$info['interfaces']))continue;
$className = $info['class'];
$compo = new $className($this);
}
}
/**
* Add all files in configured 'public' dir to the Router component
*
* @param $dir the directory to scan for public routed files
* @tag setup
*/
protected function setup_public_routes($dir){
$lia = $this->lia;
$baseUrl = $this->config['dir']['public']['baseUrl'];
$files = \Lia\Utility\Files::all($dir,$dir);
// print_r($files);
foreach ($files as $relFile){
$pattern = $this->fileToPattern($relFile);
$pattern = $baseUrl.$pattern;
$pattern = str_replace(['///','//'],'/',$pattern);
$path = $dir.$relFile;
$lia->api('lia:route.add', $pattern, $path, $this);
}
}
/**
* Add all views in configured 'view' dir to the View Component
*
* @param string $dir a view directory to setup
* @param array $args a key=>value array of args to extract to all views
* @todo add a way to configure $args for your package, without having to override the package class
* @tag setup
*/
public function setup_views(string $dir, array $args){
$lia = $this->lia;
$files = \Lia\Utility\Files::all($dir,$dir, '.php');
//set view conflict mode
$hasConfigApi = $lia->hasApi('lia:config.get');
$oldConflictMode = $hasConfigApi ? $lia->api('lia:config.get', 'lia:view.conflictMode') : null;
$conflictMode = $this->config['views']['conflict'] ?? $oldConflictMode ?? 'throw';
if ($hasConfigApi)$lia->api('lia:view.setConflictMode', $conflictMode);
foreach ($files as $f){
//remove leading `/` and trailing `.php` in a pretty simple, dumb way
$viewName = substr($f,1,-4);
$viewName = $this->get('namespace').':'.$viewName;
$class = $this->class('view');
$dir = $dir;
if ($hasConfigApi&&$viewName=='theme')$lia->api('lia:view.setConflictMode', 'overwrite');
$lia->api('lia:view.add', $class, $dir, $viewName, $args);
if ($hasConfigApi&&$viewName=='theme')$lia->api('lia:view.setConflictMode', $conflictMode);
}
$lia->api('lia:view.setConflictMode', $oldConflictMode);
//@export_end(Usage.View.AddView)
}
}