<?php
namespace Lia;
class Simple extends \Lia {
/**
* `require` a php file that instantiates a Lia instance & return that instance, without routing, sending a server response, or outputing any text.
* May not work with unconventional setups. Requires a variable named `$lia` at `$file_path`
* @param $file_path absolute path to your deliver script
* @return \Lia a lia instance
*/
static public function get_lia_instance(string $file_path): \Lia {
$_SERVER['DO_NOT_RESPOND'] = true;
$_SERVER['HTTP_HOST'] = 'localhost';
$_SERVER['REQUEST_URI'] = '/';
ob_start();
require($file_path);
ob_end_clean();
return $lia;
}
public array $phads = [];
public bool $debug = false;
/**
* The feature currently being enabled or the last feature enabled if one is not currently being enable.
*/
public ?string $last_feature = null;
/**
* array of `$feature_name=>true`
*/
public array $features = [];
public function __construct(){
parent::__construct();
$server_dir = \Lia\Package\Server::main_dir();
$server = new \Lia\Package\Server($this, 'server', $server_dir);
// some basic configs
$this->set('lia:server.cache.dir',$server->dir.'/cache/');
$this->set('lia:server.router.varDelim', '\\.\\/\\:'); //default includes a hyphen, which is dumb as hek
}
/**
* Check if current server is a production server
* @return false if `$_SERVER['HTTP_HOST'] == 'localhost'` else return true
*/
public function is_production(){
if (substr($_SERVER['HTTP_HOST'],0,9)=='localhost')return false;
return true;
}
public function require_feature(string $feature){
$last = $this->last_feature;
if (!isset($this->features[$feature]))throw new \Exception("'$last' requires feature '$feature'. Please add '$feature' to your config.json like `features:[\"$feature\", \"$last\"]`");
}
public function setup(){
foreach ($this->packages as $p){
$p->ready();
}
}
public function deliver_files(...$public_dirs){
// var_dump($public_dirs);
foreach ($public_dirs as $d){
\Lia\FastFileRouter::file($d);
}
}
public function load_apps(...$app_dirs){
$apps = array_map([$this,'load_app'], $app_dirs);
return $apps;
}
public function load_app($app_dir){
$name = strtolower(basename($app_dir));
$url_prefix = '/';
$app = new \Lia\Package\Server($this, $name, $app_dir, $url_prefix);
$app->base_url = $url_prefix;
$app->dir = $app_dir;
$config = [];
if (file_exists($app_dir.'/config.json')){
$config = json_decode(file_get_contents($app_dir.'/config.json'),true);
}
if (!isset($config['features']))$config['features'] = ['phad'];
$app->config = $config;
if (file_exists($init_php=$app_dir.'/init.php')){
$lia = $this;
require($init_php);
}
if (file_exists($fn_php = $app_dir.'/functions.php')){
$lia = $this;
require_once($fn_php);
}
foreach ($config['features'] as $feature_name){
$this->enable($feature_name, $app);
}
if (file_exists($post_init_php=$app_dir.'/post-init.php')){
$lia = $this;
$env = $this->env ?? null;
$user = $this->user ?? null;
require($post_init_php);
}
return $app;
}
public function enable(string $feature, $app=null){
if (substr($feature,0,2)=='--')return;
$this->last_feature = $feature;
$this->features[$feature] = true;
if (method_exists($this,$method='enable_'.$feature))return $this->$method($app);
switch ($feature){
case "seo_extractor":
\Lia\Addon\SeoExtractor::enable($this);
return;
case "kitchen_sink":
\Lia\App\KitchenSink::enable($this);
return;
case "R":
$r = \R::setup();
$dir = $this->root_dir;
if (file_exists($dir.'/RSettings.json'))$r->load($dir.'/RSettings.json');
if (file_exists($dir.'/.env/secret.json'))$r->load($dir.'/.env/secret.json');
return $r;
case "debug_empty_cache":
if ($this->debug!==false&&$this->env->is_bool(['host.local'])
&&substr(parse_url($_SERVER['REQUEST_URI'],PHP_URL_PATH),-1)=='/'
||$_SERVER['REQUEST_METHOD']=='POST'
){
// echo 'good';
// exit;
$this->cache->delete_all_cache_files();
$this->resources->cache->delete_all_cache_files();
}
return;
case "debug_show_errors":
if ($this->debug===true){
$this->env->showErrors();
}
return;
}
throw new \Exception("Feature '$feature' does not exist.");
}
public function enable_defaults($app){
$this->set('server.router.varDelim', '\.\\/\\:'); //default includes a hyphen, which is dumb as hek
}
public function enable_debug_fake_user_mail($app){
if ($this->env->is_bool(['host.local'])){
if (empty(@LIA_DEBUG_FAKE_USER_MAIL_OUT_DIR));
define('LIA_DEBUG_FAKE_USER_MAIL_OUT_DIR', $this->root_dir);
require_once(dirname(__DIR__).'/file/fake_user_mail.php');
}
}
public function enable_phad($app){
if (!class_exists('\Phad', true)){
throw new \Exception("\nPhad is not installed. Run `composer require taeluf/phad v0.4`");
}
$this->require_feature('pdo');
$dir = $app->dir;
$pdo = $this->pdo ?? $app->lia->pdo ?? null;
$debug = $this->debug;
$route_prefix = $app->base_url;
$options = [
'item_dir'=>$dir.'/phad/',
'cache_dir'=>$dir.'/cache/',
'sitemap_dir'=>$dir.'/sitemap/',
'pdo' => $pdo,// a pdo object
'router' => $this->addon('lia:server.router'),
'throw_on_query_failure'=>$debug,
'force_compile'=>$debug,
];
$phad_class = $app->config['phad.class'] ?? '\\Phad';
// var_dump($phad_class);
$phad = $phad_class::main($options);
$app->phad = $phad;
$phad->route_prefix = $route_prefix;
$phad->global_phad_args['lia'] = $this;
// $this->set('phad', $phad);
$this->addMethod('phad', [$this, 'phad_item']);
$phad->filters['markdown'] = [$this, 'markdown_to_html'];
$phad->integration->setup_liaison_routes($this);
$phad->integration->setup_liaison_route($this, '/sitemap.xml', $phad->sitemap_dir.'/sitemap.xml');
$phad->handlers['user_has_role'] =
function(string $role){
if (isset($this->user))return $this->user->has_role($role);
// echo 'generic user has role';
// exit;
return true;
}
;
$phad->handlers['can_read_row'] =
function(array $ItemRow,object $ItemInfo,string $ItemName){
return true;
};
if (isset($app->config['phad.handler']) && isset($this->user)){
$handler = $app->config['phad.handler'];
$phad->set_handler(new $handler($this->user));
// var_dump($handler);
// exit;
}
$this->phads[] = $phad;
return $phad;
}
/**
* Get a phad instance who's `$phad->item_dir == $target_dir``
* @param $target_dir the directory your target phad item should be in
* @return a Phad instance or null if no item dirs match.
* @note there is no path normalization
*/
public function phad_from_dir(string $target_dir): ?\Phad {
$len = strlen($target_dir);
foreach ($this->phads as $phad){
if (substr($phad->item_dir, 0,$len)==$target_dir)return $phad;
}
return null;
}
public function enable_user($app){
if (!class_exists('\Tlf\User\Lib', true)){
throw new \Exception("User lib not installed. Run `composer require taeluf/user-gui v0.3`");
}
$user_lib_dir = $this->root_dir.'/vendor/taeluf/user-gui/code/';
$pdo = $this->pdo;
// var_dump($user_lib_dir);
// exit;
$this->set('user.base_url', '/user/');
$user_package = new \Lia\Package\Server($this, 'user', $user_lib_dir);
$user_package->base_url = '/user/';
// load db settings from env file & make pdo
// $settings = json_decode(file_get_contents(dirname(__DIR__).'/db-env.json'),true);
// $pdo = new \PDO("mysql:dbname=".$settings['db'], $settings['user'], $settings['password']);
// configure the user login library
$lib = new \Tlf\User\Lib($pdo);
$lib->disabled_pages = $app->config['user.disabled_pages'] ?? [];
$user_lib = $lib;
$lib->user_class = $app->config['user.class'] ?? null;
$user_class = $lib->user_class;
$lib->config = [
'web_address'=>$app->config['user.web_address'],
'email_from'=>$app->config['user.email_from'],
];
// uncomment this line, run once, then re-comment this line
// $lib->init_db();
// log the user in
$current_user = $lib->user_from_cookie();
// if there was no cookie, we'll use an unregistered user
if ($current_user===false)$current_user = new $user_class($pdo);
$this->set('user', $current_user);
$this->addon('lia:server.view')->globalArgs['user'] = $current_user;
// passes the user & library to user pages (login, register, etc)
$user_package->public_file_params = [
'user'=>$current_user,
'lib'=>$lib
];
$user_package->lib = $lib;
// $this->dump($app);
$app->public_file_params = [
'user'=>$current_user,
'lib'=>$lib
];
}
public function enable_env($app){
$this->env = new \Env();
if (file_exists($env_file=$this->root_dir.'/.env/secret.json')){}
$this->env->load($env_file);
// $this->dump();
}
/**
*
*
* @deprecated in favor of pdo_mysql & pdo_sqlite to be more explicit
* @alias enable_pdo_mysql
*/
public function enable_pdo($app){
return $this->enable_pdo_mysql($app);
}
public function enable_pdo_mysql($app){
// echo "enable pdo mysql!";
// exit;
$this->features['pdo'] = true;
if (!isset($this->env)){
throw new \Exception("Feature 'env' is not enabled. Call `\$lia->enable_env()` or add `\"env\"` to the `\"features:[\"env\"]\"` of an app's confing.json");
}
$env = $this->env;
$this->pdo = $pdo = new \PDO(
'mysql:host='.$env->get('mysql.host').';dbname='.$env->get('mysql.dbname'),
$env->get('mysql.user'), $env->get('mysql.password')
);
$pdo->setAttribute(\PDO::ATTR_ERRMODE, \PDO::ERRMODE_EXCEPTION);
return $pdo;
}
public function enable_pdo_sqlite($app){
$this->features['pdo'] = true;
if (isset($this->env)&&$this->env->has('sqlite.path')){
$path = $this->root_dir.'/'.$this->env->get('sqlite.path');
} else {
$path = $this->root_dir.'/data.sqlite';
}
$this->pdo = $pdo = new \PDO('sqlite:'.$path);
$pdo->setAttribute(\PDO::ATTR_ERRMODE, \PDO::ERRMODE_EXCEPTION);
return $pdo;
}
public function enable_ldb($app){
if (!isset($this->pdo)){
throw new \Exception("Feature 'pdo' is not enabled. Call `\$lia->enable_pdo()` or add `\"pdo\"` to the `\"features:[\"pdo\"]\"` of an app's confing.json");
}
if (!class_exists('\Tlf\LilDb', true)){
throw new \Exception("\nLilDb is not installed. Run `composer require taeluf/lildb v0.1`");
}
require_once(__DIR__.'/ldb.php');
$this->addMethod('ldb', 'ldb');
return $this->ldb = \ldb($this->pdo);
}
/**
* 1. Save the current page to analytics
* 2. Enable the analytics route, if phad is enabled
*
* Set `$_SERVER['DO_NOT_RESPOND'] = true` to disable the analytics feature. Useful for cli environments
*/
public function enable_analytics($app){
if (($_SERVER['DO_NOT_RESPOND']??false)==true)return;
$pdo = $this->pdo;
// still need the phad view
$this->require_feature('pdo');
$this->require_feature('phad');
$url = @$_SERVER['REQUEST_URI'];
$pos = strpos($url, '?');
$url = str_replace(['<','>'],'-',$url);
if ($pos!==false)$url = substr($url,0,$pos);
// $ip_hash = md5(@$_SERVER["REMOTE_ADDR"]);
$ip_hash = 'na';
$datetime = new \DateTime();
$year_month = $datetime->format("Y-m")."-01";
$end = substr($url,-4);
// var_dump($end);
if (substr($url,-4)=='.css'||substr($url,-3)=='.js'){
} else {
$stmt = $pdo->prepare(
'INSERT INTO tlf_analytics (`url`, `ip_hash`, `year_month`)
VALUES (:url, :ip_hash, :year_month)
');
if ($stmt==false){
// print_r($pdo->errorInfo());
// exit;
return;
}
$stmt->execute( ['url'=>$url,'ip_hash'=>$ip_hash, 'year_month'=>$year_month]);
}
if (!isset($app->phad)||!is_object($app->phad))return;
$args = [
'pdo'=>$pdo,
];
$item = $app->phad->item_from_file(__DIR__.'/../phad/AnalyticsView.php', $args);
$this->addRoute(
'/analytics/',
function($route,$response) use ($item){
$response->content = $item->html();
}
);
}
/**
* enable the route /generic-error-page/ to display 'Unknown Error' message.
*/
public function enable_error_page($app){
// route to error page
if ($_SERVER['REQUEST_URI']=='/generic-error-page/'){ // save a small amount of overhead when this isn't the requested page
$this->addon('lia:server.seo')->html['meta-noindex'] = ['<meta name="robots" content="noindex" />'];
if (!$this->addon('lia:server.router')->has_static_route('/generic-error-page/')){
$this->addRoute('/generic-error-page/',function($route,$response){
$response->content = "Unknown Error. Please return to the <a href=\"/\">Home Page</a> or try again. If this persists, contact the website owner.";
});
}
// $this->hook('ResponseReady',
// function($response){
// if (!file_exists($file = $this->cache->dir.'/generic-error-page.html')){
// file_put_contents($file, $response->content);
// }
// }
// );
}
}
public function respond(){
// if (static::$IS_TLFSERV_CLI)return;
if (isset($_SERVER['DO_NOT_RESPOND']) && $_SERVER['DO_NOT_RESPOND']==true){
return;
}
$debug = $this->debug ?? false;
try {
if ($debug && $_SERVER['REQUEST_URI']=='/debug/'){
$this->debug();
exit;
}
$env = $this->env;
try {
$this->deliver();
} catch (\Exception $e){
if ($debug || !isset($this->generic_error_page)){
\Lia\ExceptionCatcher::throw($e,$debug);
} else if (!$debug && isset($this->generic_error_page)){
echo file_get_contents($this->generic_error_page);
}
}
} catch (\Throwable $e){
if ($debug){
echo nl2br("\n\n---------------------------------------\n\nError:\n");
throw $e;
} else {
ob_get_clean();
try {
$host = $_SERVER['HTTP_HOST'];
$file = $this->cache->dir.'/generic-error-page.html';
if (is_file($file)){
$i=0;
while(ob_get_level()>0&&$i++<10){
ob_end_clean();
}
echo file_get_contents($file);
}
exit;
} catch (\Throwable $e){
echo "There was an error. Try returning to the "
.'<a href="/">Home Page</a> at '.$host;
exit;
}
}
}
}
public function debug(){
$lia = $this;
$router = $lia->addon('lia:server.router');
$routes = $router->routeMap;
print_r(array_keys($routes['POST']));
exit;
}
/**
* get an array of all routes for the sitemap
*
* @param $sitemap_builder a var to reference the sitemap builder
*/
public function get_all_sitemap_routes(&$sm_builder){
$host = R("Site.Url") ?? false;
if ($host==false){
throw new \Exception("You must set 'Site.Url' in your RSettings.json file");
}
$full_list = [];
foreach ($this->phads as $phad){
$items = $phad->get_all_items();
$sm_builder = $phad->sitemap;
$sm_list = $sm_builder->get_sitemap_list($items, $phad);
$full_list = [...$full_list, ...$sm_list];
}
// return $this->sitemap_dir.'/sitemap.xml';
//
if (isset($this->methods['get_extra_sitemap_entries'])){
$full_list = array_merge(
$full_list,
$this->get_extra_sitemap_entries($full_list)
);
// print_r($full_list);
// exit;
}
return $full_list;
}
/**
* Convert markdown to html, if CommonMark is installed
*
* @param $markdown
* @return html, or if CommonMark is not installed, returns the markdown
*/
public function markdown_to_html(string $markdown){
$class = "League\\CommonMark\\CommonMarkConverter";
if (class_exists($class,true)){
$converter = new \League\CommonMark\CommonMarkConverter();
return $converter->convert($markdown).'';
}
throw new \Exception("\nCommonMark is not installed. Run `composer require league/commonmark 2.3`\n");
}
public function phad_item(string $item, array $args = []){
foreach ($this->phads as $phad){
if ($phad->has_item($item)){
$item = $phad->item($item, $args);
foreach ($item->resource_files()['css'] as $f){
$this->addResourceFile($f);
}
foreach ($item->resource_files()['js'] as $f){
$this->addResourceFile($f);
}
return $item;
}
}
throw new \Exception("Phad item '$item' not found");
}
/**
* Initialize your analytics table. Recommend using a local sqlite database.
*/
public function init_analytics_table(\PDO $pdo){
$pdo->exec(
'CREATE TABLE IF NOT EXISTS `tlf_analytics`(
`id` int AUTO_INCREMENT,
`url` varchar(255) NOT NULL,
`ip_hash` varchar(255) NOT NULL,
`year_month` varchar(12) NOT NULL,
PRIMARY KEY(`id`)
);
'
);
print_r($pdo->errorInfo());
// exit;
}
/**
* Generate an sql CREATE TABLE statement from a phad item containing a form.
*
* @param $phad
* @param $form_item the form-item to scan & use it's inputs to form a CREATE TABLE sql statement
* @return string sql CREATE TABLE statement
*/
public function create_phad_table(\Phad $phad, string $form_item): string {
$item = $phad->item($form_item);
$info = $item->info();
if (!is_object($info)||!isset($info->properties)){
throw new \Exception("'$form_item' does not appear to be a phad item or does not have properties.");
}
$properties = $info->properties;
// print_r($properties);
$table = strtolower($info->name);
$statement = "CREATE TABLE `$table` (\n ";
$did_first = false;
foreach ($properties as $name=>$details){
$col_str = col_str($name, $details);
if ($did_first)$statement .=",\n ";
$statement .= $col_str;
$did_first = true;
}
$statement .= "\n);";
return $statement;
}
/**
* Send an email through SMTP
* You MUST set the following settings to R: smtp.host, smtp.user, smtp.password, Business.Name
*/
function send_mail(string $to, string $reply_to, $subject, $body){
$env = $this->env;
if (!class_exists('PHPMailer\\PHPMailer\\PHPMailer', true)){
throw new \Exception("Please install PHPMailer to use send_mail. composer require phpmailer/phpmailer. Commit hash 'a04c4c2' is known to work.");
}
//Create an instance; passing `true` enables exceptions
$mail = new \PHPMailer\PHPMailer\PHPMailer(true);
try {
//Server settings
$mail->SMTPDebug = \PHPMailer\PHPMailer\SMTP::DEBUG_SERVER; //Enable verbose debug output
$mail->isSMTP(); //Send using SMTP
$mail->Host = R("smtp.host"); //Set the SMTP server to send through
$mail->SMTPAuth = true; //Enable SMTP authentication
$mail->Username = R("smtp.user"); //SMTP username
$mail->Password = R("smtp.password"); //SMTP password
$mail->SMTPSecure = \PHPMailer\PHPMailer\PHPMailer::ENCRYPTION_SMTPS; //Enable implicit TLS encryption
$mail->Port = 465; //TCP port to connect to; use 587 if you have set `SMTPSecure = PHPMailer::ENCRYPTION_STARTTLS`
//Recipients
$mail->setFrom(R("smtp.user"), R("Business.Name").' Help');
$mail->addAddress($to); //Name is optional
$mail->addReplyTo($reply_to);
// $mail->addCC('cc@example.com');
// $mail->addBCC('bcc@example.com');
//Attachments
// $mail->addAttachment('/var/tmp/file.tar.gz'); //Add attachments
// $mail->addAttachment('/tmp/image.jpg', 'new.jpg'); //Optional name
//Content
$mail->isHTML(true); //Set email format to HTML
$mail->Subject = $subject;
$mail->Body = $body;
// $mail->AltBody = 'This is the body in plain text for non-HTML mail clients';
ob_start();
$mail->send();
ob_get_clean();
$this->log("mail-success", "Email sent!");
} catch (\Exception $e) {
$this->log("mail-error", "Message could not be sent. Mailer Error: {$mail->ErrorInfo}");
}
}
public function log(){
// @TODO write log method
}
public function __isset($prop){
return isset($this->$prop) ? true : isset($this->props[$prop]);
}
/**
* get the named property or addon if property not found
* @param $addon the name of the addon
*/
public function __get($prop){
if (!isset($this->props[$prop])&&!isset($this->addons[$prop]))throw new \Exception("Property '$prop' not found");
$a = $this->props[$prop] ?? $this->addons[$prop];
return $a;
}
public function __set($prop, $value){
$this->props[$prop] = $value;
}
}