phalconplus Documentation v1.1.0
Class PhalconPlus Rpc Server

AbstractServer

    namespace PhalconPlus\Rpc\Server;
use PhalconPlus\Base\ProtoBuffer;
use PhalconPlus\Logger\Processor\LogId;

abstract class AbstractServer
{
    protected di = null {
        get
    };

    protected phpOnly = false {
        get
    };

    protected eventsManager = null;

    abstract public function __construct(<\Phalcon\DI> di);

    protected function callByParams(string! service, string! method, request)
    {
        var serviceClass = "";
        let serviceClass = service->upperfirst() . "Service";

        if is_object(request) && request instanceof ProtoBuffer {
            let this->phpOnly = true;
        } elseif is_array(request) {
            var tmp = [], param, paramClass;
            let tmp = request;
            let param = new \ReflectionParameter([serviceClass, method], 0);
            if param->getClass() {
                let paramClass = param->getClass()->getName();
                let request = new {paramClass}();
                request->softClone(tmp);
            } else {
                throw new \PhalconPlus\Base\Exception("Service class:method definition is invalid. Detail: " . service . " : " . method . ". Request: " . json_encode(request));
            }
        } else {
            throw new \PhalconPlus\Base\Exception("Your input is not allowed. Request: " . json_encode(request));
        }

        var serviceObj, response, e;
        let serviceObj = new {serviceClass}(this->di);

        if is_callable([serviceObj, method]) {
            this->eventsManager->fire("backend-server:beforeExecute", $this, [service, method, request]);
            try {
                let response =  call_user_func_array([serviceObj, method], [request]);
            } catch \Exception, e {
                throw e;
            }
            this->eventsManager->fire("backend-server:afterExecute", $this, [service, method, request]);
            // We do not allow to return #Resource type. And if an object returned, we expected type 
            if is_object(response) && !(response instanceof ProtoBuffer) {
                throw new \PhalconPlus\Base\Exception("Your output is not allowed. Response: " . get_class(response) . ". We expect scalar type or ");
            } elseif is_resource(response) {
                throw new \PhalconPlus\Base\Exception("Your output is not allowed. Response: #Resource.");
            }
            // Distinguish object/scalar types when there are non-php clients
            if this->phpOnly == false && is_object(response) {
                return response->toArray();
            }
            return response;
        } else {
            throw new \PhalconPlus\Base\Exception("Service:method not found. Detail: " . service . " : " . method);
        }
    }

    /**
     *
     * @param array rawData
     * 
     *     rawData = ["service":"Demo", "method":"demo", "args": , "logId": "234fdfaf3334"]
     * 
     * @return 
     * @throw \Exception
     *
     */
    public function callByObject(array rawData)
    {
        var service, method, request, response, logId = "", message = "";

        if !fetch service, rawData["service"] {
            throw new \PhalconPlus\Base\Exception("service " . service . " not exists");
        }

        if !fetch method, rawData["method"] {
            throw new \PhalconPlus\Base\Exception("method " . method . " not exists");
        }

        if !fetch request, rawData["args"] {
            throw new \PhalconPlus\Base\Exception("args not exists");
        }

        this->eventsManager->fire("backend-server:requestCheck", $this, [service, method, rawData]);

        let service = trim(service);
        let method = trim(method);

        if empty service || empty method {
            throw new \PhalconPlus\Base\Exception("service:method(args) must exists. All of them!!!");
        }

        if fetch logId, rawData["logId"] {
            // set logId, same as request api
            LogId::setId(logId);
        }

        if this->di->has("logger") {
            let message = "RPC Request - logId: " . logId . ", invoke: " . service . "::" . method . ", args: " . json_encode(request);           
            this->di->get("logger")->log(message);
        }

        this->eventsManager->fire("backend-server:beforeDispatch", $this, [service, method, request]);
        let response = this->callByParams(service, method, request);
        this->eventsManager->fire("backend-server:afterDispatch", $this, [service, method, request, response]);

        if this->di->has("logger") {
            let message = "RPC Response - logId: " . logId . ", invoke: " . service . "::" . method . ", response: " . json_encode(response);
            this->di->get("logger")->log(message);
        }

        return response;
    }
}