hanze/iwa-panda2

updating Lollipop (bcc9a36bb5488725bb9f0022b396e738d9264c78)
Repositories | README.md

commit bcc9a36bb5488725bb9f0022b396e738d9264c78
parent 65149823272f79f28ac6defeeabf38ee1b53ba95
Author: Friedel Schön <[email protected]>
Date:   Tue, 27 Jun 2023 23:58:53 +0200

updating Lollipop

Diffstat:
MLollipop/DatabaseObject.php463++++++++++++++++++++++++++++++++++++++++++++++++++-----------------------------
MLollipop/Router.php197+++++++++++++++++++++++++++++++++++++++++++++----------------------------------
MLollipop/SQLDatabase.php182++++++++++++++++++++++++++++++++++++++++---------------------------------------
MLollipop/Template.php243+++++++++++++++++++++++++++++--------------------------------------------------
ALollipop/TemplateMethods.php119+++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
ALollipop/Utils.php34++++++++++++++++++++++++++++++++++
6 files changed, 739 insertions(+), 499 deletions(-)

diff --git a/Lollipop/DatabaseObject.php b/Lollipop/DatabaseObject.php @@ -1,129 +1,146 @@ <?php namespace Lollipop { - require_once "SQLDatabase.php"; - - abstract class DatabaseObject - { - protected string $table; - protected string $primary; - - protected SQLDatabase $db; - protected array $data = []; - protected array $changed_keys = []; - - function __construct(SQLDatabase $db) - { - $this->db = $db; - $this->primary = $this->get_primary(); - $this->table = $this->get_table(); - $this->notNullable(); - } - - abstract static function get_primary(): string; - abstract static function get_table(): string; - - public function setData($data) - { - $this->data = $data; - } - public function where(string $key, string $value) - { - $sql = "SELECT * FROM {$this->table} WHERE $key = ?"; - $value = array($value); - $stmt = $this->db->conn->prepare($sql); - $stmt->execute($value); - $result = $stmt->get_result(); - if ($result->num_rows == 0) { - return false; - } - $this->data = $result->fetch_assoc(); - return true; - } - - public function where_array(array $values) : bool - { - $sql = "SELECT * FROM {$this->table} WHERE "; - $params = []; - $i = 0; - foreach($values as $key => $param){ - if($i > 0) - $sql .= " and "; - $sql .= "{$key} = ?"; - $params[] = $param; - } - - $stmt = $this->db->conn->prepare($sql); - $stmt->execute($params); - $result = $stmt->get_result(); - - if ($result->num_rows == 0) { - return false; - } - - $this->data = $result->fetch_assoc(); - return true; - } - public function load(string $id): bool - { - /*this fuction accepts an $id value for the primary key - * loads the row into data[] - * returns bool if row is found - */ - $sql = "SELECT * FROM {$this->table} WHERE {$this->primary} = ?"; - - $stmt = $this->db->conn->prepare($sql); - $stmt->execute([$id]); - $result = $stmt->get_result(); - - if ($result->num_rows == 0) { - return false; - } - - $this->data = $result->fetch_assoc(); - return true; - } - - public function save() : bool - { - if (!$this->changed_keys) - return false; - - $sql = "UPDATE {$this->table} SET "; - - $values = []; - foreach ($this->changed_keys as $index => $key) { - if ($index > 0) - $sql .= ', '; - $sql .= "$key = ?"; - $values[] = $this->data[$key]; - } - - $sql .= " WHERE {$this->primary} = ?"; - $values[] = $this->data[$this->primary]; - - $stmt = $this->db->conn->prepare($sql); - - $this->changed_keys = []; - - if($stmt->execute($values)) - return true; - else - return false; - } - - public function add() : bool - /* this function add the set variables to the database */ - { - if (!$this->changed_keys) + require_once "SQLDatabase.php"; + + /// Lollipop\DatabaseObject is an abstract class, a TableClass like Model\User should extends this + abstract class DatabaseObject + { + protected string $table; + protected string $primary; + protected array $column_names; + protected array $not_nullable; + protected SQLDatabase $db; + protected array $data = []; + protected array $changed_keys = []; + private string $schema; + + public function __construct(SQLDatabase $db) + { + $this->db = $db; + $this->primary = $this->get_primary(); + $this->table = $this->get_table(); + $this->schema = $this->get_schema(); + } + + abstract public static function get_primary(): string; + abstract public static function get_table(): string; + abstract public static function get_schema(): string; + + /// setData is to bulk-set the row instead of one-for-one + public function setData($data) + { + $this->data = $data; + } + + /// select row by key + public function where(string $key, string $value) + { + $sql = "SELECT * FROM {$this->table} WHERE $key = ?"; + $value = array($value); + $stmt = $this->db->conn->prepare($sql); + $stmt->execute($value); + $result = $stmt->get_result(); + if ($result->num_rows == 0) { + return false; + } + $this->data = $result->fetch_assoc(); + return true; + } + + /// select rows by multiple values + public function where_array(array $values): bool + { + $sql = "SELECT * FROM {$this->table} WHERE "; + $params = []; + $i = 0; + foreach($values as $key => $param) { + if($i > 0) { + $sql .= " and "; + } + $sql .= "{$key} = ?"; + $params[] = $param; + } + + $stmt = $this->db->conn->prepare($sql); + $stmt->execute($params); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return false; + } + + $this->data = $result->fetch_assoc(); + return true; + } + + /// select row by id (and key is $this->primary_key()) + public function load(string $id): bool + { + /*this fuction accepts an $id value for the primary key + * loads the row into data[] + * returns bool if row is found + */ + $sql = "SELECT * FROM {$this->table} WHERE {$this->primary} = ?"; + + $stmt = $this->db->conn->prepare($sql); + $stmt->execute([$id]); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return false; + } + + $this->data = $result->fetch_assoc(); + return true; + } + + /// update the row + public function save(): bool + { + if (!$this->changed_keys) { + return false; + } + + $sql = "UPDATE {$this->table} SET "; + + $values = []; + foreach ($this->changed_keys as $index => $key) { + if ($index > 0) { + $sql .= ', '; + } + $sql .= "$key = ?"; + $values[] = $this->data[$key]; + } + + $sql .= " WHERE {$this->primary} = ?"; + $values[] = $this->data[$this->primary]; + + $stmt = $this->db->conn->prepare($sql); + + $this->changed_keys = []; + + if($stmt->execute($values)) { + return true; + } else { + return false; + } + } + + /// insert row into database if not existent + public function add(): bool + /* this function add the set variables to the database */ + { + if (!$this->changed_keys) { return false; + } $sql = "INSERT INTO {$this->table} ("; $sql_val = ") VALUES ("; $values = []; foreach ($this->changed_keys as $index => $key) { - if ($index > 0){ + if ($index > 0) { $sql .= ', '; $sql_val .= ', '; } @@ -137,61 +154,166 @@ namespace Lollipop { $this->changed_keys = []; - if($stmt->execute($values)) + if($stmt->execute($values)) { return true; - else + } else { return false; - } - public function delete() - { - $sql = "DELETE FROM {$this->table} WHERE {$this->primary} = ?"; - $stmt = $this->db->conn->prepare($sql); - $stmt->execute([$this->data[$this->primary]]); - $this->data = []; - $this->changed_keys = []; - } - - public function __get(string $name) - { - return $this->data[$name]; - } - - public function __set(string $name, $value) - { - $this->data[$name] = $value; - $this->changed_keys[] = $name; - } - - public function getData() - { - return $this->data; - } - private function notNullable(){ - //non-auto-increment not-nullable collumn names query - $not_null = []; - $col_names = []; - $sql = " SELECT column_name, is_nullable, extra + } + } + + /// deletes the row + public function delete() + { + $sql = "DELETE FROM {$this->table} WHERE {$this->primary} = ?"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute([$this->data[$this->primary]]); + $this->data = []; + $this->changed_keys = []; + } + + /// magic method: echo $obj->column + public function __get(string $name) + { + return $this->data[$name]; + } + + /// magic method: $obj->column = "value" + public function __set(string $name, $value) + { + $this->data[$name] = $value; + $this->changed_keys[] = $name; + } + + /// get row as array + public function getData() + { + return $this->data; + } + + /// get not-nullable fields of this table + public function notNullable() + { + //non-auto-increment not-nullable collumn names query + $col_names = []; + $sql = " SELECT column_name, is_nullable, extra FROM INFORMATION_SCHEMA.COLUMNS WHERE TABLE_NAME = '{$this->table}' - AND TABLE_SCHEMA = 'panda'"; - $stmt = $this->db->conn->prepare($sql); - $stmt->execute(); - $result = $stmt->get_result(); - - if ($result->num_rows == 0) { - return false; - } - while($tmp = $result->fetch_assoc()){ - if($tmp["is_nullable"] == 'NO'){ - if($tmp["extra"] == "auto_increment") - continue; - $not_null[] = $tmp["column_name"]; - } - $col_names[] = $tmp["column_name"]; - } - $this->data["not_nullable"] = $not_null; - $this->data["column_names"] = $col_names; - return true; - } - } -} -\ No newline at end of file + AND TABLE_SCHEMA = '{$this->schema}'"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute(); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return []; + } + while($tmp = $result->fetch_assoc()) { + if($tmp["is_nullable"] == 'NO') { + if(!$tmp["extra"] == "auto_increment") { + $col_names[] = $tmp["column_name"]; + } + } + } + return $col_names; + } + + /// get column names of table + public function get_column_names(): array + { + $column_names = []; + $sql = " SELECT column_name + FROM INFORMATION_SCHEMA.COLUMNS + WHERE TABLE_NAME = '$this->table' + AND TABLE_SCHEMA = '$this->schema'"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute(); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return []; + } + while($tmp = $result->fetch_assoc()) { + $column_names[] = $tmp["column_name"]; + } + return $column_names; + } + + /// get column names without auto-increments + public function get_col_names_no_ai(): array + { + $column_names = []; + $sql = " SELECT column_name, extra + FROM INFORMATION_SCHEMA.COLUMNS + WHERE TABLE_NAME = '$this->table' + AND TABLE_SCHEMA = '$this->schema' + AND EXTRA not like '%auto_increment%'"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute(); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return []; + } + while($tmp = $result->fetch_assoc()) { + $column_names[] = $tmp["column_name"]; + } + return $column_names; + } + + /// get auto-incremented columns + public function get_col_names_ai(): array + { + $column_names = []; + $sql = " SELECT column_name, extra + FROM INFORMATION_SCHEMA.COLUMNS + WHERE TABLE_NAME = '$this->table' + AND TABLE_SCHEMA = '$this->schema' + AND EXTRA like '%auto_increment%'"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute(); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return []; + } + while($tmp = $result->fetch_assoc()) { + $column_names[] = $tmp["column_name"]; + } + return $column_names; + } + + /// get column infos + public function get_col_info(): array + { + $column_names = []; + $sql = " SELECT column_name, extra, data_type + FROM INFORMATION_SCHEMA.COLUMNS + WHERE TABLE_NAME = '$this->table' + AND TABLE_SCHEMA = '$this->schema'"; + $stmt = $this->db->conn->prepare($sql); + $stmt->execute(); + $result = $stmt->get_result(); + + if ($result->num_rows == 0) { + return []; + } + while($tmp = $result->fetch_assoc()) { + if(str_contains($tmp['data_type'], "varchar") || str_contains($tmp['data_type'], "text")) { + $column_names[$tmp["column_name"]]["input_type"] = "text"; + } elseif(str_contains($tmp['data_type'], "date")) { + $column_names[$tmp["column_name"]]["input_type"] = "date"; + } elseif(str_contains($tmp['data_type'], "int")) { + $column_names[$tmp["column_name"]]["input_type"] = "number"; + } elseif(str_contains($tmp['data_type'], "double")) { + $column_names[$tmp["column_name"]]["input_type"] = "number"; + } + if(str_contains($tmp['extra'], "auto_increment")) { + $column_names[$tmp["column_name"]]['extra'] = "auto_increment"; + } + if(str_contains($tmp['column_name'], "password")) { + $column_names[$tmp["column_name"]]['extra'] = "password"; + } + } + return $column_names; + } + } +} diff --git a/Lollipop/Router.php b/Lollipop/Router.php @@ -1,86 +1,116 @@ <?php namespace Lollipop { - class Router - { - protected array $routes = []; - protected string $path; - protected Template $temp; - - public function __construct($temp) { - $this->temp = $temp; - } - - protected function match(string $match, array &$route_vars): bool - { - $route_split = explode('/', trim($this->path, '/ ')); - $match_split = explode('/', trim($match, '/ ')); - - if (sizeof($route_split) != sizeof($match_split)) { - return false; - } - - foreach ($match_split as $index => $m) { - if (str_starts_with($m, ':')) { - $route_vars[substr($m, 1)] = $route_split[$index]; - } else if ($m != $route_split[$index]) { - return false; - } - } - return true; - } - - - function addRoute(string|array $method, string $match, string|callable $func) - { - if (is_string($method)) - $method = [$method]; - - - $this->routes[] = array( - "method" => $method, - "match" => $match, - "func" => $func, - ); - } - - function includeRoute(string $path, array $_PARAM) - { - include $path; - } - - function route(string $base = null) - { - $this->path = $_SERVER["REQUEST_URI"]; - - if (strpos($this->path, '?')) - $this->path = explode('?', $this->path)[0]; - - if ($base && strpos($this->path, $base)) - $this->path = explode($base, $this->path)[1]; - - $method = $_SERVER["REQUEST_METHOD"]; - - foreach ($this->routes as $route) { - if ($route["method"] != null && !in_array($method, $route["method"])) - continue; - - $vars = []; - if ($this->match($route["match"], $vars)) { - if (is_callable($route["func"])) { - $fil = $route["func"]($vars); - if (!is_null($fil)) - echo $this->temp->template($fil, $vars); - return; - } else { - echo $this->temp->template($route["func"], $vars); - return; - } - } - } - - echo $this->temp->template("views/login.html", $vars); - return null; - } - } -} -\ No newline at end of file + const MIME_TYPES = array( + "css" => "text/css", + "js" => "text/javascript" + ); + + /// this is the basic router, implementing an automatic templater + class Router + { + protected array $routes = []; + protected string $path; + protected Template $temp; + + public function __construct($temp) + { + $this->temp = $temp; + } + + /// set content-type header + protected function set_mime($file) + { + if (!is_null($file)) { + $ext = pathinfo($file, PATHINFO_EXTENSION); + } else { + $ext = null; + } + + if ($ext != null && array_key_exists($ext, MIME_TYPES)) { + $mime = MIME_TYPES[$ext]; + } else { + $mime = "text/html"; + } + + header("Content-Type: $mime"); + } + + protected function match(string $match, array &$route_vars): bool + { + $route_split = explode('/', trim($this->path, '/ ')); + $match_split = explode('/', trim($match, '/ ')); + + if (sizeof($route_split) != sizeof($match_split)) { + return false; + } + + foreach ($match_split as $index => $m) { + if (str_starts_with($m, ':')) { + $route_vars[substr($m, 1)] = $route_split[$index]; + } elseif ($m != $route_split[$index]) { + return false; + } + } + return true; + } + + /// add route + /// $func can be a path to an template or a function which returns the path to an template and modifies $vars + public function addRoute(string|array $method, string $match, string|callable $func) + { + if (is_string($method)) { + $method = [$method]; + } + + $this->routes[] = array( + "method" => $method, + "match" => $match, + "func" => $func, + ); + } + + /// final routing + public function route(string $base = null) + { + $this->path = $_SERVER["REQUEST_URI"]; + + if (strpos($this->path, '?')) { + $this->path = explode('?', $this->path)[0]; + } + + if ($base && strpos($this->path, $base)) { + $this->path = explode($base, $this->path)[1]; + } + + $method = $_SERVER["REQUEST_METHOD"]; + + foreach ($this->routes as $route) { + if ($route["method"] != null && !in_array($method, $route["method"])) { + continue; + } + + $vars = []; + if ($this->match($route["match"], $vars)) { + if (is_callable($route["func"])) { + $fil = $route["func"]($vars); + if (is_null($fil)) { + return; + } + $this->set_mime($fil); + echo $this->temp->template($fil, $vars); + return; + } else { + $this->set_mime($route["func"]); + echo $this->temp->template($route["func"], $vars); + return; + } + } + } + + echo "404 '$this->path' not found!"; + header("Location: /"); + return null; + } + } +} diff --git a/Lollipop/SQLDatabase.php b/Lollipop/SQLDatabase.php @@ -1,94 +1,98 @@ <?php namespace Lollipop { - use mysqli; - - class SQLDatabase - { - public mysqli $conn; - - function __construct(string $host, string $username, string $password, string $database = null, int $port = null) - { - $this->conn = new mysqli($host, $username, $password, $database, $port); - } - - function get(string $table_class) - { - /* this function accepts a $table_name creates a Database object with the class $table_name - * retuns a Database object - */ - $cls = new $table_class($this); - return $cls; - } - - function all_where(string $table_name, array $vars) - { - /* this function accepts a table name and an array[$column_name => $value] - * statement is select * from $table_name where $column_name = $value AND etc... - * returns an array of classes - */ - if (!sizeof($vars)) { - return []; - } - $cls = new $table_name($this); - - $sql = "SELECT * FROM {$cls->get_table()} WHERE "; - $params = []; - - $i = 0; - foreach ($vars as $key => $value) { - if ($i > 0) { - $sql .= ' AND '; - } - $sql .= " $key LIKE ?"; - $params[] = $value; - $i++; - } - - $stmt = $this->conn->prepare($sql); - $stmt->execute($params); - $result = $stmt->get_result(); - - if (!$result || $result->num_rows == 0) { - return []; - } - - $objects = []; - while ($row = $result->fetch_assoc()) { - $o = new $table_name($this); - $o->setData($row); - $objects[] = $o; - } - return $objects; - } - - function all(string $table_name) - { - /* loads whole table $table_name - * returns array of objects - */ - $cls = new $table_name($this); - - $sql = "SELECT * FROM {$cls->get_table()}"; - - $result = $this->conn->query($sql); - - if (!$result || $result->num_rows == 0) { - return []; - } - - $objects = []; - while ($row = $result->fetch_assoc()) { - $o = new $table_name($this); - $o->setData($row); - $objects[] = $o; - } - return $objects; - } + use mysqli; + + /// this is the main database engine + class SQLDatabase + { + public mysqli $conn; + + public function __construct(string $host, string $username, string $password, string $database = null, int $port = null) + { + $this->conn = new mysqli($host, $username, $password, $database, $port); + } + + /// get a table by passing the desired class + public function get(string $table_class) + { + /* this function accepts a $table_name creates a Database object with the class $table_name + * retuns a Database object + */ + $cls = new $table_class($this); + return $cls; + } + + /// this function accepts a table name and an array[$column_name => $value] + /// statement is select * from $table_name where $column_name = $value AND etc... + /// returns an array of classes + public function all_where(string $table_name, array $vars) + { + if (!sizeof($vars)) { + return []; + } + $cls = new $table_name($this); + + $sql = "SELECT * FROM {$cls->get_table()} WHERE "; + $params = []; + + $i = 0; + foreach ($vars as $key => $value) { + if ($i > 0) { + $sql .= ' AND '; + } + $sql .= " $key LIKE ?"; + $params[] = $value; + $i++; + } + + $stmt = $this->conn->prepare($sql); + $stmt->execute($params); + $result = $stmt->get_result(); + + if (!$result || $result->num_rows == 0) { + return []; + } + + $objects = []; + while ($row = $result->fetch_assoc()) { + $o = new $table_name($this); + $o->setData($row); + $objects[] = $o; + } + return $objects; + } + + /// returns every row in database of table + public function all(string $table_name) + { + /* loads whole table $table_name + * returns array of objects + */ + $cls = new $table_name($this); + + $sql = "SELECT * FROM {$cls->get_table()}"; + + $result = $this->conn->query($sql); + + if (!$result || $result->num_rows == 0) { + return []; + } + + $objects = []; + while ($row = $result->fetch_assoc()) { + $o = new $table_name($this); + $o->setData($row); + $objects[] = $o; + } + return $objects; + } + public function getDateRange(string $table_name, array $query, $order) { - if($query == null) + if($query == null) { return []; + } $cls = new $table_name($this); @@ -106,8 +110,8 @@ namespace Lollipop { } } - $sql .= " ORDER BY date_time " . $order; - $sql .= " LIMIT 1000"; + $sql .= " ORDER BY date_time " . $order; + $sql .= " LIMIT 1000"; $stmt = $this->conn->prepare($sql); $stmt->execute($values); $result = $stmt->get_result(); @@ -124,6 +128,5 @@ namespace Lollipop { } return $objects; } - } + } } -?> -\ No newline at end of file diff --git a/Lollipop/Template.php b/Lollipop/Template.php @@ -1,167 +1,103 @@ -<?php +<?php + namespace Lollipop { -use ErrorException; - Class Template{ - function template(string $uri, array $data) : string{ + use ErrorException; + + /// this is the templating engine + /// syntax: + /// {{ $var }} will be replaces with the variable or nothing if not existend + /// {{ "value" !func }} func will be called, popping things from the stack and pushing the result + /// {{ $var "exist: %%" "doesnt exist" !format_if }} is an example + class Template + { + private TemplateMethods $methods; + + public function __construct(TemplateMethods $methods) + { + $this->methods = $methods; + } + + public function template(string $uri, array $data): string + { /* this function takes a uri and a string array data */ /* opens a stream to the uri specified file and stores the content in $file*/ - return $this->insert_data(file_get_contents($uri), $data); + return $this->insert_data(file_get_contents($uri), $data); } - - private function insert_data(string $file, array $data):string{ + + private function insert_data(string $file, array $data): string + { $html = ""; $filesize = strlen($file); - for($i = 0; $i < $filesize-1; $i++){ - if ($file[$i] == '{' && $file[$i + 1] == '{') { - for ($j = $i; $j < $filesize-1; $j++) { - if ($file[$j] == '}' && $file[$j + 1] == '}') { - $html .= $this->parse_template(trim(substr($file, $i + 2, $j - $i - 2)), $data); - $i = $j + 1; - break; - } - } - } else { - $html .= $file[$i]; - } - } - return $html; - } - - private function parse_template(string $expr, array $data) { - $tokens = []; - $in_string = false; - $buffer = ''; - - foreach (str_split($expr) as $c) { - if ($c == '"' && !$in_string) { // string start - $in_string = true; - } else if ($c == '"') { // string end - $tokens[] = $buffer; - $buffer = ''; - $in_string = false; - } else if ($c == ' ' && !$in_string) { - if ($buffer) { - $tokens[] = $buffer; - $buffer = ''; - } - } else { - $buffer .= $c; - } - } - if ($buffer) - $tokens[] = $buffer; - - return $this->eval_tokens($tokens, $data); - } - - private function eval_tokens(array $tokens, array $data) { - $funcs = [ - "add" => function(array &$tokens) { - $right = array_pop($tokens); - $left = array_pop($tokens); - - if (is_null($left) || is_null($right)) - throw new ErrorException("Stack is empty"); + for($i = 0; $i < $filesize-1; $i++) { + if ($file[$i] == '{' && $file[$i + 1] == '{') { + for ($j = $i; $j < $filesize-1; $j++) { + if ($file[$j] == '}' && $file[$j + 1] == '}') { + $html .= $this->parse_template(trim(substr($file, $i + 2, $j - $i - 2)), $data); + $i = $j + 1; + break; + } + } + } else { + $html .= $file[$i]; + } + } + return $html; + } - return $left + $right; - }, - "sub" => function(array &$tokens) { - $right = array_pop($tokens); - $left = array_pop($tokens); - - if (is_null($left) || is_null($right)) - throw new ErrorException("Stack is empty"); + private function parse_template(string $expr, array $data) + { + $tokens = []; + $in_string = false; + $buffer = ''; - return intval($left) - intval($right); - }, - "mul" => function(array &$tokens) { - $right = array_pop($tokens); - $left = array_pop($tokens); - - if (is_null($left) || is_null($right)) - throw new ErrorException("Stack is empty"); + foreach (str_split($expr) as $c) { + if ($c == '"' && !$in_string) { // string start + $in_string = true; + } elseif ($c == '"') { // string end + $tokens[] = $buffer; + $buffer = ''; + $in_string = false; + } elseif ($c == ' ' && !$in_string) { + if ($buffer) { + $tokens[] = $buffer; + $buffer = ''; + } + } else { + $buffer .= $c; + } + } + if ($buffer) { + $tokens[] = $buffer; + } - return intval($left) * intval($right); - }, - "div" => function(array &$tokens) { - $right = array_pop($tokens); - $left = array_pop($tokens); - - if (is_null($left) || is_null($right)) - throw new ErrorException("Stack is empty"); + return $this->eval_tokens($tokens, $data); + } - return intval($left) / intval($right); - }, - "cat" => function(array &$tokens) { - $right = array_pop($tokens); - $left = array_pop($tokens); - - if (is_null($left) || is_null($right)) - throw new ErrorException("Stack is empty"); + private function eval_tokens(array $tokens, array $data) + { + $stack = []; + foreach ($tokens as $token) { + if ($token && $token[0] == '!') { + $val = $this->methods->{substr($token, 1)}($stack); + if (!is_null($val)) { + $stack[] = $val; + } + } elseif ($token && $token[0] == '$') { + $stack[] = array_key_exists(substr($token, 1), $data) ? $data[substr($token, 1)] : ""; + } else { + $stack[] = $token; + } + } - return $left . $right; - }, - - "to_int" => function(array &$tokens) { - $val = array_pop($val); - - if (is_null($val)) - throw new ErrorException("Stack is empty"); - - return inval($val); - }, - - "include" => function (array &$tokens) { - $name = array_pop($tokens); - - if ($name == null) - throw new ErrorException("Stack is empty"); - - include($name); - }, - "eval" => function (array &$tokens) { - $expr = array_pop($tokens); - - if (is_null($expr)) - throw new ErrorException("Stack is empty"); - - return eval("return ($expr);"); - }, - "format_if" => function (array &$stack) { - $format_false = array_pop($stack); - $format_true = array_pop($stack); - $expr = array_pop($stack); - - if (is_null($expr) || is_null($format_true) || is_null($format_false)) - throw new ErrorException("Stack is empty"); - - if ($expr == "") - return $format_false; - else - return str_replace("%%", $expr, $format_true); - }, - ]; - - $stack = []; - foreach ($tokens as $token) { - if ($token && $token[0] == '!') { - $val = $funcs[substr($token, 1)]($stack); - if (!is_null($val)) - $stack[] = $val; - } else if ($token && $token[0] == '$') { - $stack[] = array_key_exists(substr($token, 1), $data) ? $data[substr($token, 1)] : ""; - } else { - $stack[] = $token; - } - } - - if (sizeof($stack) > 1) - throw new ErrorException("Stack-size is not 1"); - if (sizeof($stack) == 0) - return ""; - return $stack[0]; - } + if (sizeof($stack) > 1) { + throw new ErrorException("Stack-size is not 1"); + } + if (sizeof($stack) == 0) { + return ""; + } + return $stack[0]; + } } -} -\ No newline at end of file +} diff --git a/Lollipop/TemplateMethods.php b/Lollipop/TemplateMethods.php @@ -0,0 +1,119 @@ +<?php + +namespace Lollipop { + + /// this class contains all built-in functions for the templater + class TemplateMethods + { + public static function add(array &$tokens) + { + $right = array_pop($tokens); + $left = array_pop($tokens); + + if (is_null($left) || is_null($right)) { + throw new ErrorException("Stack is empty"); + } + + return $left + $right; + + } + + public static function sub(array &$tokens) + { + $right = array_pop($tokens); + $left = array_pop($tokens); + + if (is_null($left) || is_null($right)) { + throw new ErrorException("Stack is empty"); + } + + return intval($left) - intval($right); + } + + public static function mul(array &$tokens) + { + $right = array_pop($tokens); + $left = array_pop($tokens); + + if (is_null($left) || is_null($right)) { + throw new ErrorException("Stack is empty"); + } + + return intval($left) * intval($right); + } + + public static function div(array &$tokens) + { + $right = array_pop($tokens); + $left = array_pop($tokens); + + if (is_null($left) || is_null($right)) { + throw new ErrorException("Stack is empty"); + } + + return intval($left) / intval($right); + } + + public static function cat(array &$tokens) + { + $right = array_pop($tokens); + $left = array_pop($tokens); + + if (is_null($left) || is_null($right)) { + throw new ErrorException("Stack is empty"); + } + + return $left . $right; + } + + public static function to_int(array &$tokens) + { + $val = array_pop($val); + + if (is_null($val)) { + throw new ErrorException("Stack is empty"); + } + + return inval($val); + } + + public static function include(array &$tokens) + { + $name = array_pop($tokens); + + if ($name == null) { + throw new ErrorException("Stack is empty"); + } + + include($name); + } + + public static function eval(array &$tokens) + { + $expr = array_pop($tokens); + + if (is_null($expr)) { + throw new ErrorException("Stack is empty"); + } + + return eval("return ($expr);"); + } + + public static function format_if(array &$stack) + { + $format_false = array_pop($stack); + $format_true = array_pop($stack); + $expr = array_pop($stack); + + if (is_null($expr) || is_null($format_true) || is_null($format_false)) { + throw new ErrorException("Stack is empty"); + } + + if ($expr == "") { + return $format_false; + } else { + return str_replace("%%", $expr, $format_true); + } + } + } +} diff --git a/Lollipop/Utils.php b/Lollipop/Utils.php @@ -0,0 +1,34 @@ +<?php + +namespace Lollipop{ + + /// this class is a collection of utilities + class Utils + { + public static function missing_fields($not_nullable) + { + $missing = []; + foreach($not_nullable as $column) { + if($_POST[$column] == null || $_POST[$column] == "") { + $key = 'missing_' . $column; + $missing[$key] = "This field cannot be empty!"; + } + } + return $missing; + } + + public static function missing_fields_sans_pw($not_nullable) + { + $missing = []; + foreach($not_nullable as $column) { + if($_POST[$column] == null || $_POST[$column] == "") { + if($column != "password") { + $key = 'missing_' . $column; + $missing[$key] = "This field cannot be empty!"; + } + } + } + return $missing; + } + } +}