mirror of
https://github.com/OpenXE-org/OpenXE.git
synced 2025-01-27 21:11:12 +01:00
367 lines
9.5 KiB
PHP
367 lines
9.5 KiB
PHP
<?php
|
|
|
|
/**
|
|
* Copyright (C) 2015 Datto, Inc.
|
|
*
|
|
* This file is part of PHP JSON-RPC.
|
|
*
|
|
* PHP JSON-RPC is free software: you can redistribute it and/or modify
|
|
* it under the terms of the GNU Lesser General Public License, version 3,
|
|
* as published by the Free Software Foundation.
|
|
*
|
|
* PHP JSON-RPC is distributed in the hope that it will be useful,
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
* GNU Lesser General Public License for more details.
|
|
*
|
|
* You should have received a copy of the GNU Lesser General Public License
|
|
* along with PHP JSON-RPC. If not, see <http://www.gnu.org/licenses/>.
|
|
*
|
|
* @author Spencer Mortensen <smortensen@datto.com>
|
|
* @license http://www.gnu.org/licenses/lgpl-3.0.html LGPL-3.0
|
|
* @copyright 2015 Datto, Inc.
|
|
*/
|
|
|
|
namespace Datto\JsonRpc;
|
|
|
|
use Datto\JsonRpc\Responses\ErrorResponse;
|
|
use Datto\JsonRpc\Responses\Response;
|
|
use Datto\JsonRpc\Responses\ResultResponse;
|
|
use ErrorException;
|
|
|
|
/**
|
|
* @link http://www.jsonrpc.org/specification JSON-RPC 2.0 Specifications
|
|
*
|
|
* @package Datto\JsonRpc
|
|
*/
|
|
class Client
|
|
{
|
|
/** @var string */
|
|
const VERSION = '2.0';
|
|
|
|
/** @var array */
|
|
private $requests;
|
|
|
|
public function __construct()
|
|
{
|
|
$this->reset();
|
|
}
|
|
|
|
/**
|
|
* Forget any unsent queries or notifications
|
|
*/
|
|
public function reset()
|
|
{
|
|
$this->requests = [];
|
|
}
|
|
|
|
/**
|
|
* @param mixed $id
|
|
* @param string $method
|
|
* @param array $arguments|null
|
|
*
|
|
* @return self
|
|
* Returns the object handle (so you can chain method calls, if you like)
|
|
*/
|
|
public function query($id, string $method, array $arguments = null): self
|
|
{
|
|
$request = self::getRequest($method, $arguments);
|
|
$request['id'] = $id;
|
|
|
|
$this->requests[] = $request;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* @param string $method
|
|
* @param array $arguments
|
|
*
|
|
* @return self
|
|
* Returns the object handle (so you can chain method calls, if you like)
|
|
*/
|
|
public function notify($method, array $arguments = null): self
|
|
{
|
|
$request = self::getRequest($method, $arguments);
|
|
|
|
$this->requests[] = $request;
|
|
|
|
return $this;
|
|
}
|
|
|
|
/**
|
|
* Encodes the request(s) as a valid JSON-RPC 2.0 string
|
|
*
|
|
* This also resets the Client, so you can use the same Client object
|
|
* to perform more queries.
|
|
*
|
|
* @return string|null
|
|
* Returns a valid JSON-RPC 2.0 message string
|
|
* Returns null if there is nothing to encode
|
|
*/
|
|
public function encode()
|
|
{
|
|
$input = $this->preEncode();
|
|
|
|
if ($input === null) {
|
|
return null;
|
|
}
|
|
|
|
return json_encode($input);
|
|
}
|
|
|
|
/**
|
|
* Encodes the request(s) as a JSON-RPC 2.0 array, but does NOT perform
|
|
* the final "json_encode" step which is necessary to turn the array
|
|
* into a valid JSON-RPC 2.0 string. This gives you the opportunity
|
|
* to inspect or modify the raw data, or to alter the encoding algorithm.
|
|
*
|
|
* When you're finished manipulating the request, you are responsible for
|
|
* JSON-encoding the value to construct the final JSON-RPC 2.0 string.
|
|
* @see self::encode()
|
|
*
|
|
* This also resets the Client, so you can use the same Client object
|
|
* to perform more queries.
|
|
*
|
|
* @return array|null
|
|
* Returns a JSON-RPC 2.0 request array
|
|
* Returns null if no requests have been queued
|
|
*/
|
|
public function preEncode()
|
|
{
|
|
$n = count($this->requests);
|
|
|
|
if ($n === 0) {
|
|
return null;
|
|
}
|
|
|
|
if ($n === 1) {
|
|
$input = $this->requests[0];
|
|
} else {
|
|
$input = $this->requests;
|
|
}
|
|
|
|
$this->reset();
|
|
|
|
return $input;
|
|
}
|
|
|
|
/**
|
|
* Translates a JSON-RPC 2.0 server reply into an array of "Response"
|
|
* objects.
|
|
*
|
|
* @param string $json
|
|
* String reply from a JSON-RPC 2.0 server
|
|
*
|
|
* @return Response[]
|
|
* Returns a zero-indexed array of "Response" objects
|
|
*
|
|
* @throws ErrorException
|
|
* Throws an "ErrorException" if the reply was not well-formed
|
|
*/
|
|
public function decode(string $json)
|
|
{
|
|
$input = json_decode($json, true);
|
|
|
|
$errorCode = json_last_error();
|
|
|
|
if ($errorCode !== 0) {
|
|
$errorMessage = json_last_error_msg();
|
|
$jsonException = new ErrorException($errorMessage, $errorCode);
|
|
|
|
$valueText = self::getValueText($json);
|
|
throw new ErrorException("Invalid JSON: {$valueText}", 0, E_ERROR, __FILE__, __LINE__, $jsonException);
|
|
}
|
|
|
|
return $this->postDecode($input);
|
|
}
|
|
|
|
/**
|
|
* Translates a JSON-decoded server reply into an array of "Response"
|
|
* objects.
|
|
*
|
|
* This gives you the opportunity to use your own modified "json_decode"
|
|
* algorithm, or to inspect or modify the server response before it is
|
|
* processed under the JSON-RPC 2.0 specifications. This can be handy
|
|
* if you're tweaking or extending the JSON-RPC 2.0 format.
|
|
*
|
|
* Before calling this method, you are responsible for JSON-decoding
|
|
* the server reply string. You should have that decoded array value
|
|
* to use as the input here.
|
|
* @see self::decode()
|
|
*
|
|
* @param mixed $input
|
|
* An array containing the JSON-decoded server reply
|
|
*
|
|
* @return Response[]
|
|
* Returns a zero-indexed array of "Response" objects
|
|
*
|
|
* @throws ErrorException
|
|
* Throws an "ErrorException" if the reply was not well-formed
|
|
*/
|
|
public function postDecode($input)
|
|
{
|
|
if (!$this->getResponses($input, $responses)) {
|
|
$valueText = self::getValueText($input);
|
|
throw new ErrorException("Invalid JSON-RPC 2.0 response: {$valueText}");
|
|
}
|
|
|
|
return $responses;
|
|
}
|
|
|
|
private static function getRequest(string $method, array $arguments = null): array
|
|
{
|
|
$request = [
|
|
'jsonrpc' => self::VERSION,
|
|
'method' => $method
|
|
];
|
|
|
|
if ($arguments !== null) {
|
|
$request['params'] = $arguments;
|
|
}
|
|
|
|
return $request;
|
|
}
|
|
|
|
private static function getValueText($value): string
|
|
{
|
|
if (is_null($value)) {
|
|
return 'null';
|
|
}
|
|
|
|
if (is_resource($value)) {
|
|
$type = get_resource_type($value);
|
|
$id = (int)$value;
|
|
return "{$type}#{$id}";
|
|
}
|
|
|
|
return var_export($value, true);
|
|
}
|
|
|
|
private function getResponses($input, array &$responses = null): bool
|
|
{
|
|
if ($this->getResponse($input, $response)) {
|
|
$responses = [$response];
|
|
return true;
|
|
}
|
|
|
|
return $this->getBatchResponses($input, $responses);
|
|
}
|
|
|
|
private function getResponse($input, &$response)
|
|
{
|
|
return $this->getResultResponse($input, $response) ||
|
|
$this->getErrorResponse($input, $response);
|
|
}
|
|
|
|
private function getResultResponse($input, &$response)
|
|
{
|
|
if (
|
|
is_array($input) &&
|
|
!array_key_exists('error', $input) &&
|
|
$this->getVersion($input) &&
|
|
$this->getId($input, $id) &&
|
|
$this->getResult($input, $value)
|
|
) {
|
|
$response = new ResultResponse($id, $value);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function getVersion(array $input)
|
|
{
|
|
return isset($input['jsonrpc']) && ($input['jsonrpc'] === self::VERSION);
|
|
}
|
|
|
|
private function getId(array $input, &$id)
|
|
{
|
|
if (array_key_exists('id', $input)) {
|
|
$id = $input['id'];
|
|
return is_null($id) || is_int($id) || is_float($id) || is_string($id);
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function getResult(array $input, &$value)
|
|
{
|
|
if (array_key_exists('result', $input)) {
|
|
$value = $input['result'];
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function getErrorResponse(array &$input, &$response)
|
|
{
|
|
if (
|
|
is_array($input) &&
|
|
!array_key_exists('result', $input) &&
|
|
$this->getVersion($input) &&
|
|
$this->getId($input, $id) &&
|
|
$this->getError($input, $code, $message, $data)
|
|
) {
|
|
$response = new ErrorResponse($id, $message, $code, $data);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
private function getError(array $input, &$code, &$message, &$data)
|
|
{
|
|
$error = $input['error'] ?? null;
|
|
|
|
return is_array($error) &&
|
|
$this->getErrorCode($error, $code) &&
|
|
$this->getErrorMessage($error, $message) &&
|
|
$this->getErrorData($error, $data);
|
|
}
|
|
|
|
private function getErrorCode(array $input, &$code)
|
|
{
|
|
$code = $input['code'] ?? null;
|
|
|
|
return is_int($code);
|
|
}
|
|
|
|
private function getErrorMessage(array $input, &$message)
|
|
{
|
|
$message = $input['message'] ?? null;
|
|
|
|
return is_string($message);
|
|
}
|
|
|
|
private function getErrorData(array $input, &$data)
|
|
{
|
|
$data = $input['data'] ?? null;
|
|
|
|
return true;
|
|
}
|
|
|
|
private function getBatchResponses($input, &$responses)
|
|
{
|
|
if (!is_array($input)) {
|
|
return false;
|
|
}
|
|
|
|
$responses = [];
|
|
$i = 0;
|
|
|
|
foreach ($input as $key => $value) {
|
|
if ($key !== $i++) {
|
|
return false;
|
|
}
|
|
|
|
if (!$this->getResponse($value, $responses[])) {
|
|
return false;
|
|
}
|
|
}
|
|
|
|
return true;
|
|
}
|
|
}
|