155 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
			
		
		
	
	
			155 lines
		
	
	
		
			5.3 KiB
		
	
	
	
		
			PHP
		
	
	
	
	
	
<?php
 | 
						|
 | 
						|
require_once "privilege_levels.php";
 | 
						|
 | 
						|
// response types
 | 
						|
const RESP_NONE = "none";
 | 
						|
const RESP_PLAIN = "plain";
 | 
						|
const RESP_JSON = "json";
 | 
						|
 | 
						|
// Request Handler Assignment class
 | 
						|
// Members get populated when instantiated by ReqHandler;
 | 
						|
// should not be used outside ReqHandler
 | 
						|
class ReqHandlerAssignment
 | 
						|
{
 | 
						|
    // members
 | 
						|
    public string $action; // action string
 | 
						|
    public array $params; // request parameters
 | 
						|
    public string $handler; // handler function
 | 
						|
    public string $hint; // human-readable hint about what this action performs
 | 
						|
    public string $level; // required command authentication level
 | 
						|
    public string $resp_type; // response type
 | 
						|
    public ReqHandler $rh; // reference to request handler
 | 
						|
 | 
						|
    function __construct(string $action, array $params, string $level, string $handler, string $resp_type, string $hint, ReqHandler &$rh)
 | 
						|
    {
 | 
						|
        $this->action = $action;
 | 
						|
        $this->params = $params;
 | 
						|
        $this->level = $level;
 | 
						|
        $this->handler = $handler;
 | 
						|
        $this->resp_type = $resp_type;
 | 
						|
        $this->hint = $hint;
 | 
						|
        $this->rh = &$rh; // assign be reference
 | 
						|
    }
 | 
						|
 | 
						|
    function dump() : string {
 | 
						|
        return "<i><font color='#2f4f4f'>" . $this->action . "</font></i>(<code>" . join(", ", $this->params) . "</code>) <font color='#008b8b'>" . $this->hint . "</font> [" . $this->level . "]";
 | 
						|
    }
 | 
						|
}
 | 
						|
 | 
						|
// Request Handler class
 | 
						|
class ReqHandler
 | 
						|
{
 | 
						|
    private array $mapping; // action-handler mapping
 | 
						|
    private array $request; // reference to $_REQUEST
 | 
						|
    private array $files; // reference to $_FILES
 | 
						|
    private string $level; // max privilege level
 | 
						|
 | 
						|
    // Constructor
 | 
						|
    function __construct()
 | 
						|
    {
 | 
						|
        // create reference bindings
 | 
						|
        $this->request = &$_REQUEST;
 | 
						|
        $this->files = &$_FILES;
 | 
						|
 | 
						|
        $this->mapping = [PRIVILEGE_NONE => [], PRIVILEGE_PLAYER => [], PRIVILEGE_CREATOR => [], PRIVILEGE_QUIZMASTER => []]; // create assignment categories
 | 
						|
        $this->level = PRIVILEGE_NONE;
 | 
						|
    }
 | 
						|
 | 
						|
    // Add new handler
 | 
						|
    function add(string|array $actions, array $params, string $level, string $handler, string $resp_type, string $hint): bool
 | 
						|
    {
 | 
						|
        // convert single actions to array
 | 
						|
        $actions = is_string($actions) ? [$actions] : $actions;
 | 
						|
 | 
						|
        // cannot assign more than one handler to the same action
 | 
						|
        foreach ($actions as $action) {
 | 
						|
            if (in_array($action, $this->mapping[$level])) {
 | 
						|
                return false;
 | 
						|
            }
 | 
						|
 | 
						|
            // create the assignment object
 | 
						|
            $assignment = new ReqHandlerAssignment($action, $params, $level, $handler, $resp_type, $hint, $this);
 | 
						|
            $this->mapping[$level][$action] = $assignment;
 | 
						|
        }
 | 
						|
 | 
						|
        return true;
 | 
						|
    }
 | 
						|
 | 
						|
    const ACTION_KEY = "action";
 | 
						|
 | 
						|
    // Process possible pending request. Also check access privilege.
 | 
						|
    function process(string $level = null, string $action_key = self::ACTION_KEY) : array {
 | 
						|
        // get action
 | 
						|
        $action = $this->request[$action_key] ?? "";
 | 
						|
 | 
						|
        // don't process empty actions
 | 
						|
        if (trim($action) === "") {
 | 
						|
            return ["", false];
 | 
						|
        }
 | 
						|
 | 
						|
        // if privilege level is specified, then store
 | 
						|
        $this->level = $level ?? $this->level;
 | 
						|
 | 
						|
        $resp = "";
 | 
						|
        $success = false;
 | 
						|
 | 
						|
        // look up action
 | 
						|
        $categories = array_keys($this->mapping);
 | 
						|
        $highest_category_index = array_search($this->level, $categories);
 | 
						|
        for ($i = 0; ($i < count($categories)) && ($i <= $highest_category_index); $i++) {
 | 
						|
            // get category
 | 
						|
            $category = &$this->mapping[$categories[$i]];
 | 
						|
 | 
						|
            // find the command level category that includes this action
 | 
						|
            if (array_key_exists($action, $category)) {
 | 
						|
                // get assignment
 | 
						|
                $assignment = &$category[$action];
 | 
						|
 | 
						|
                // check that all required parameters were passed
 | 
						|
                $all_params_passed = true;
 | 
						|
                foreach ($assignment->params as $param) {
 | 
						|
                    $all_params_passed &= array_key_exists($param, $this->request);
 | 
						|
                }
 | 
						|
 | 
						|
                // if params were provided, then invoke the callback
 | 
						|
                if ($all_params_passed) {
 | 
						|
                    $handler = $assignment->handler;
 | 
						|
                    $raw_resp = $handler($this, $this->request);
 | 
						|
 | 
						|
                    // convert response to preset response type
 | 
						|
                    if ($assignment->resp_type !== RESP_NONE) { // if command returns with anything at all
 | 
						|
                        if ($assignment->resp_type === RESP_JSON) { // JSON
 | 
						|
                            $resp = json_encode($raw_resp);
 | 
						|
                        } else if ($assignment->resp_type === RESP_PLAIN) { // PLAIN
 | 
						|
                            $resp = $raw_resp;
 | 
						|
                        }
 | 
						|
                    }
 | 
						|
 | 
						|
                    // processing successful
 | 
						|
                    $success = true;
 | 
						|
 | 
						|
                    break; // break the loop
 | 
						|
                }
 | 
						|
            }
 | 
						|
        }
 | 
						|
 | 
						|
        return [$resp, $success];
 | 
						|
    }
 | 
						|
 | 
						|
    // Set access privilege level
 | 
						|
    function set_privilege_level(string $level) {
 | 
						|
        $this->level = $level;
 | 
						|
    }
 | 
						|
 | 
						|
    // Dump commands
 | 
						|
    function dump_actions() : string {
 | 
						|
        $dump = "";
 | 
						|
        foreach ($this->mapping as $level => $actions) {
 | 
						|
            foreach ($actions as $action) {
 | 
						|
                $dump .= $action->dump() . "<br>";
 | 
						|
            }
 | 
						|
        }
 | 
						|
        return $dump;
 | 
						|
    }
 | 
						|
} |