123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278 |
- <?php
- // Copyright (C) 2010-2013 Combodo SARL
- //
- // This file is part of iTop.
- //
- // iTop is free software; you can redistribute it and/or modify
- // it under the terms of the GNU Affero General Public License as published by
- // the Free Software Foundation, either version 3 of the License, or
- // (at your option) any later version.
- //
- // iTop 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 Affero General Public License for more details.
- //
- // You should have received a copy of the GNU Affero General Public License
- // along with iTop. If not, see <http://www.gnu.org/licenses/>
- /**
- * Entry point for all the REST services
- *
- * --------------------------------------------------
- * Create an object
- * --------------------------------------------------
- * POST itop/webservices/rest.php
- * {
- * operation: 'object_create',
- * comment: 'Synchronization from blah...',
- * class: 'UserRequest',
- * results: 'id, friendlyname',
- * fields:
- * {
- * org_id: 'SELECT Organization WHERE name = "Demo"',
- * caller_id:
- * {
- * name: 'monet',
- * first_name: 'claude',
- * }
- * title: 'Houston, got a problem!',
- * description: 'The fridge is empty'
- * contacts_list:
- * [
- * {
- * role: 'pizza delivery',
- * contact_id:
- * {
- * finalclass: 'Person',
- * name: 'monet',
- * first_name: 'claude'
- * }
- * }
- * ]
- * }
- * }
- *
- *
- * @copyright Copyright (C) 2010-2013 Combodo SARL
- * @license http://opensource.org/licenses/AGPL-3.0
- */
- if (!defined('__DIR__')) define('__DIR__', dirname(__FILE__));
- require_once(__DIR__.'/../approot.inc.php');
- require_once(APPROOT.'/application/application.inc.php');
- require_once(APPROOT.'/application/loginwebpage.class.inc.php');
- require_once(APPROOT.'/application/ajaxwebpage.class.inc.php');
- require_once(APPROOT.'/application/startup.inc.php');
- require_once(APPROOT.'core/restservices.class.inc.php');
- /**
- * Result structure that is specific to the hardcoded verb 'list_operations'
- */
- class RestResultListOperations extends RestResult
- {
- public $version;
- public $operations;
- public function AddOperation($sVerb, $sDescription, $sServiceProviderClass)
- {
- $this->operations[] = array(
- 'verb' => $sVerb,
- 'description' => $sDescription,
- 'extension' => $sServiceProviderClass
- );
- }
- }
- if (!function_exists('json_last_error_msg')) {
- function json_last_error_msg() {
- static $ERRORS = array(
- JSON_ERROR_NONE => 'No error',
- JSON_ERROR_DEPTH => 'Maximum stack depth exceeded',
- JSON_ERROR_STATE_MISMATCH => 'State mismatch (invalid or malformed JSON)',
- JSON_ERROR_CTRL_CHAR => 'Control character error, possibly incorrectly encoded',
- JSON_ERROR_SYNTAX => 'Syntax error',
- JSON_ERROR_UTF8 => 'Malformed UTF-8 characters, possibly incorrectly encoded'
- );
- $error = json_last_error();
- return isset($ERRORS[$error]) ? $ERRORS[$error] : 'Unknown error';
- }
- }
- ////////////////////////////////////////////////////////////////////////////////
- //
- // Main
- //
- $oP = new ajax_page('rest');
- $oCtx = new ContextTag('REST/JSON');
- $sVersion = utils::ReadParam('version', null, false, 'raw_data');
- $sOperation = utils::ReadParam('operation', null);
- $sJsonString = utils::ReadParam('json_data', null, false, 'raw_data');
- $sProvider = '';
- try
- {
- utils::UseParamFile();
- $iRet = LoginWebPage::DoLogin(false, false, LoginWebPage::EXIT_RETURN); // Starting with iTop 2.2.0 portal users are no longer allowed to access the REST/JSON API
- if ($iRet != LoginWebPage::EXIT_CODE_OK)
- {
- switch($iRet)
- {
- case LoginWebPage::EXIT_CODE_MISSINGLOGIN:
- throw new Exception("Missing parameter 'auth_user'", RestResult::MISSING_AUTH_USER);
- break;
-
- case LoginWebPage::EXIT_CODE_MISSINGPASSWORD:
- throw new Exception("Missing parameter 'auth_pwd'", RestResult::MISSING_AUTH_PWD);
- break;
-
- case LoginWebPage::EXIT_CODE_WRONGCREDENTIALS:
- throw new Exception("Invalid login", RestResult::UNAUTHORIZED);
- break;
-
- case LoginWebPage::EXIT_CODE_PORTALUSERNOTAUTHORIZED:
- throw new Exception("Portal user is not allowed", RestResult::UNAUTHORIZED);
- break;
-
- default:
- throw new Exception("Unknown authentication error (retCode=$iRet)", RestResult::UNAUTHORIZED);
- }
- }
- if ($sVersion == null)
- {
- throw new Exception("Missing parameter 'version' (e.g. '1.0')", RestResult::MISSING_VERSION);
- }
-
- if ($sJsonString == null)
- {
- throw new Exception("Missing parameter 'json_data", RestResult::MISSING_JSON);
- }
- $aJsonData = @json_decode($sJsonString);
- if ($aJsonData == null)
- {
- throw new Exception("Parameter json_data is not a valid JSON structure", RestResult::INVALID_JSON);
- }
- $aProviders = array();
- foreach(get_declared_classes() as $sPHPClass)
- {
- $oRefClass = new ReflectionClass($sPHPClass);
- if ($oRefClass->implementsInterface('iRestServiceProvider'))
- {
- $aProviders[] = new $sPHPClass;
- }
- }
- $aOpToRestService = array(); // verb => $oRestServiceProvider
- foreach ($aProviders as $oRestSP)
- {
- $aOperations = $oRestSP->ListOperations($sVersion);
- foreach ($aOperations as $aOpData)
- {
- $aOpToRestService[$aOpData['verb']] = array
- (
- 'service_provider' => $oRestSP,
- 'description' => $aOpData['description'],
- );
- }
- }
- if (count($aOpToRestService) == 0)
- {
- throw new Exception("There is no service available for version '$sVersion'", RestResult::UNSUPPORTED_VERSION);
- }
- $sOperation = RestUtils::GetMandatoryParam($aJsonData, 'operation');
- if ($sOperation == 'list_operations')
- {
- $oResult = new RestResultListOperations();
- $oResult->message = "Operations: ".count($aOpToRestService);
- $oResult->version = $sVersion;
- foreach ($aOpToRestService as $sVerb => $aOpData)
- {
- $oResult->AddOperation($sVerb, $aOpData['description'], get_class($aOpData['service_provider']));
- }
- }
- else
- {
- if (!array_key_exists($sOperation, $aOpToRestService))
- {
- throw new Exception("Unknown verb '$sOperation' in version '$sVersion'", RestResult::UNKNOWN_OPERATION);
- }
- $oRS = $aOpToRestService[$sOperation]['service_provider'];
- $sProvider = get_class($oRS);
-
- CMDBObject::SetTrackOrigin('webservice-rest');
- $oResult = $oRS->ExecOperation($sVersion, $sOperation, $aJsonData);
- }
- }
- catch(Exception $e)
- {
- $oResult = new RestResult();
- if ($e->GetCode() == 0)
- {
- $oResult->code = RestResult::INTERNAL_ERROR;
- }
- else
- {
- $oResult->code = $e->GetCode();
- }
- $oResult->message = "Error: ".$e->GetMessage();
- }
- // Output the results
- //
- $sResponse = json_encode($oResult);
- if ($sResponse === false)
- {
- $oJsonIssue = new RestResult();
- $oJsonIssue->code = RestResult::INTERNAL_ERROR;
- $oJsonIssue->message = 'json encoding failed with message: '.json_last_error_msg().'. Full response structure for debugging purposes (print_r+bin2hex): '.bin2hex(print_r($oResult, true));
- $sResponse = json_encode($oJsonIssue);
- }
- $oP->add_header('Access-Control-Allow-Origin: *');
- $sCallback = utils::ReadParam('callback', null);
- if ($sCallback == null)
- {
- $oP->SetContentType('application/json');
- $oP->add($sResponse);
- }
- else
- {
- $oP->SetContentType('application/javascript');
- $oP->add($sCallback.'('.$sResponse.')');
- }
- $oP->Output();
- // Log usage
- //
- if (MetaModel::GetConfig()->Get('log_rest_service'))
- {
- $oLog = new EventRestService();
- $oLog->SetTrim('userinfo', UserRights::GetUser());
- $oLog->Set('version', $sVersion);
- $oLog->Set('operation', $sOperation);
- $oLog->SetTrim('json_input', $sJsonString);
- $oLog->Set('provider', $sProvider);
- $sMessage = $oResult->message;
- if (empty($oResult->message))
- {
- $sMessage = 'Ok';
- }
- $oLog->SetTrim('message', $sMessage);
- $oLog->Set('code', $oResult->code);
- $oLog->SetTrim('json_output', $sResponse);
- $oLog->DBInsertNoReload();
- }
|