ホームページ >バックエンド開発 >PHPチュートリアル >php xml-rpcリモート呼び出し_PHPチュートリアル

php xml-rpcリモート呼び出し_PHPチュートリアル

WBOY
WBOYオリジナル
2016-07-21 15:47:58843ブラウズ

コードをコピーします コードは次のとおりです:

/*
ネットワーク上に来ているXML-RPC库、公開小型の外部通讯インターフェイスに有用
*/

function & XML_serialize($data, $level = 0, $prior_key = NULL ){
#ハッシュを想定しており、キーは変数名です
$xml_serialized_string = "";
while(list($key, $value) = each($data)){
$inline = false;
$numeric_array = false;
$attributes = "";
#echo "私の現在のキーは '$key' で、前のキー '$prior_key' で呼び出されます
";
if(!strstr($key, " attr")){ #属性ではない場合
if(array_key_exists("$key attr", $data)){
while(list($attr_name, $attr_value) = each ($data["$key attr"])){
#echo "値 $attribute_value
を持つ属性 $attribute_name が見つかりました";
$attr_value = &htmlspecialchars($attr_value, ENT_QUOTES);
$attributes .= " $attr_name="$attr_value"";
}
}

if(is_numeric($key)){
#echo "現在のキー ($key) は数値です。親キーは '$prior_key'
";
$key = $prior_key;
}else{
#数値キーを 2 つのレベルに連続して使用することはできないので、これは問題ありません
#echo "データ内に数値キーが存在するかどうかを確認しています。";
if(is_array($value) and array_key_exists(0, $value)){
# echo " そうです! 数値配列の結果として自分自身を呼び出します。
";
$numeric_array = true;
$xml_serialized_string .= XML_serialize($value, $level, $key);
}
#echo "
";
}

if(!$numeric_array){
$xml_serialized_string .= str_repeat("t", $level) . "<$key$attributes>";

if(is_array($value)){
$xml_serialized_string .= "rn" . XML_serialize($value, $level+1);
}else{
$inline = true;
$xml_serialized_string .= htmlspecialchars($value);
}

$xml_serialized_string .= (!$inline ? str_repeat("t", $level) : "") 。 "rn";
}
}else{
#echo "キー $key
の属性レコードをスキップします";
}
}
if($level == 0){
$xml_serialized_string = "rn" . $xml_serialized_string;
$xml_serialized_string を返します;
}else{
return $xml_serialized_string;
}
}

クラス XML {
var $parser; #XML パーサーへの参照
var $document; #これまでに構築された XML 構造全体
var $current; #現在の項目へのポインター - これは何ですか
var $parent; #現在の親へのポインタ - 親は配列になります
var $parents; #各レベルの最新の親の配列

var $last_opened_tag;

関数 XML($data=null){
$this->parser = xml_parser_create();

xml_parser_set_option ($this->parser, XML_OPTION_CASE_FOLDING, 0);
xml_set_object($this->パーサー, $this);
xml_set_element_handler($this->parser, "open", "close");
xml_set_character_data_handler($this->parser, "データ");
# register_shutdown_function(array($this, 'destruct'));
}

function destruct(){
xml_parser_free($this->parser);
}

function parse($data){
$this->document = array();
$this->親 = $this->ドキュメント;
$this->parents = array();
$this->last_opened_tag = NULL;
xml_parse($this->parser, $data);
$this->ドキュメントを返す;
}

function open($parser, $tag, $attributes){
#echo "開始タグ $tag
n";
$this->data = "";
$this->last_opened_tag = $tag; #tag は文字列です
if(array_key_exists($tag, $this->parent)){
#echo "現在のレベル ($level) に '$tag' のインスタンスがすでにあります
n";
if(is_array($this->parent[$tag]) and array_key_exists(0, $this->parent[$tag])){ #キーが数値の場合
#数値であることを確認する必要がある(属性を考慮)
$key = count_numeric_items($this->parent[$tag]);
#echo "$key インスタンスがあります。キーは数値です。
n";
}else{
#echo "インスタンスは 1 つだけです。すべてを前後に移動します
n";
$temp = $this->親[$tag];
unset($this->親[$tag]);
$this->親[$tag][0] = $temp;

if(array_key_exists("$tag attr", $this->parent)){
#存在する場合は属性もシフトします
$temp = $this->parent["$tag attr"];
unset($this->parent["$tag attr"]);
$this->親[$tag]["0 属性"] = $temp;
}
$key = 1;
}
$this->parent = $this->parent[$tag];
}else{
$key = $tag;
}
if($attributes){
$this->parent["$key attr"] = $attributes;
}

$this->parent[$key] = array();
$this->parent = $this->parent[$key];
array_unshift($this->parents, $this->parent);
}

function data($parser, $data){
#echo "データは '", htmlspecialchars($data), "'
n";
if($this->last_opened_tag != NULL){
$this->data .= $data;
}
}

function close($parser, $tag){
#echo "タグ $tag
n";
if($this->last_opened_tag == $tag){
$this->parent = $this->data;
$this->last_opened_tag = NULL;
}
array_shift($this->parents);
$this->parent = $this->parents[0];
}
}

function & XML_unserialize($xml){
$xml_parser = new XML();
$data = $xml_parser->parse($xml);
$xml_parser->destruct();
$data を返す;
}

function & XMLRPC_parse($request){
if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_parse', "

次の生のリクエストを受信しました:

" 。 XMLRPC_show($request, 'print_r', true));
}
$data = &XML_unserialize($request);
if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_parse', "

次の解析されたリクエストを返します:

" . XMLRPC_show($data, 'print_r', true)) ;
}
$data を返します。
}

function & XMLRPC_prepare($data, $type = NULL){
if(is_array($data)){
$num_elements = count($data);
if((array_key_exists(0, $data) または !$num_elements) and $type != 'struct'){ #配列です
if(!$num_elements){ #配列が空の場合
$returnvalue = array( '配列' => 配列('データ' => NULL));
}else{
$returnvalue['array']['data']['value'] = array();
$temp = $returnvalue['配列']['データ']['値'];
$count = count_numeric_items($data);
for($n=0; $n$type = NULL;
if(array_key_exists("$n 型", $data)){
$type = $data["$n 型"];
}
$temp[$n] = XMLRPC_prepare($data[$n], $type);
}
}
}else{ #これは構造体です
if(!$num_elements){ #構造体が空の場合
$returnvalue = array('struct' => NULL);
}else{
$returnvalue['struct']['member'] = array();
$temp = $returnvalue['struct']['member'];
while(list($key, $value) = each($data)){
if(substr($key, -5) != ' type'){ #型指定子でない場合
$type = NULL;
if(array_key_exists("$key タイプ", $data)){
$type = $data["$key タイプ"];
}
$temp[] = array('name' => $key, 'value' => XMLRPC_prepare($value, $type));
}
}
}
}
}else{ #これはスカラーです
if(!$type){
if(is_int($data)){
$returnvalue['int'] = $data;
$returnvalue を返す;
}elseif(is_float($data)){
$returnvalue['double'] = $data;
$returnvalue を返す;
}elseif(is_bool($data)){
$returnvalue['boolean'] = ($data ? 1 : 0);
$returnvalue を返す;
}elseif(preg_match('/^d{8}Td{2}:d{2}:d{2}$/', $data, $matches)){ #日付です
$returnvalue['dateTime. iso8601'] = $data;
$returnvalue を返す;
}elseif(is_string($data)){
$returnvalue['string'] = htmlspecialchars($data);
$returnvalue を返す;
}
}else{
$returnvalue[$type] = htmlspecialchars($data);
}
}
return $returnvalue;
}

function & XMLRPC_adjustValue($current_node){
if(is_array($current_node)){
if(isset($current_node['array'])){
if(!is_array($current_node['array']) ['data'])){
#要素がない場合は、空の空の配列を返します
return array();
}else{
#echo "配列 -> データ -> 値
n の削除";
$temp = $current_node['配列']['データ']['値'];
if(is_array($temp) および array_key_exists(0, $temp)){
$count = count($temp);
for($n=0;$n$temp2[$n] = &XMLRPC_adjustValue($temp[$n]);
}
$temp = $temp2;
}else{
$temp2 = &XMLRPC_adjustValue($temp);
$temp = 配列($temp2);
#コピーを避けるため一時代入を行います。
#配列に参照を入れることができるためです。
#PHP の参照モデルは少しばかげており、単純に次のように言うことはできません:
# $temp = array(&XMLRPC_adjustValue( $temp));
}
}
}elseif(isset($current_node['struct'])){
if(!is_array($current_node['struct'])){
#メンバーが存在しない場合は、空の空の配列を返します
return配列();
}else{
#echo "構造体 -> メンバー
n の削除";
$temp = $current_node['struct']['member'];
if(is_array($temp) および array_key_exists(0, $temp)){
$count = count($temp);
for($n=0;$n#echo "名前 {$temp[$n][name]} を渡します。値は: " . show($temp[$n][value], var_dump, true) 。 "
n";
$temp2[$temp[$n]['name']] = &XMLRPC_adjustValue($temp[$n]['value']);
#echo "adjustValue(): 割り当て後の値は " です。 show($temp2[$temp[$n]['name']], var_dump, true) 。 "
n";
}
}else{
#echo "名前 $temp[name]
n を渡します";
$temp2[$temp['name']] = &XMLRPC_adjustValue($temp['value']);
}
$temp = $temp2;
}
}else{
$types = array('string', 'int', 'i4', 'double', 'dateTime.iso8601', 'base64', 'boolean');
$fell_through = true;
foreach($types as $type){
if(array_key_exists($type, $current_node)){
#echo "'$type'
n の削除";
$temp = $current_node[$type];
#echo "adjustValue(): 現在のノードは $type
n"; のタイプで設定されます。
$fell_through = false;
休憩;
}
}
if($fell_through){
$type = '文字列';
#echo "失敗しました! タイプは $type
n";
}
switch ($type){
case 'int': case 'i4': $temp = (int)$temp;壊す;
case 'string': $temp = (string)$temp;壊す;
case 'double': $temp = (double)$temp;壊す;
case 'boolean': $temp = (bool)$temp;壊す;
}
}
}else{
$temp = (string)$current_node;
}
$temp を返します。
}

function XMLRPC_getParams($request){
if(!is_array($request['methodCall']['params'])){
#パラメータがない場合は、空の空の配列を返します
return array();
}else{
#echo "methodCall -> params -> param
n の削除";
$temp = $request['methodCall']['params']['param'];
if(is_array($temp) および array_key_exists(0, $temp)){
$count = count($temp);
for($n = 0; $n #echo "パラメータ $n
をシリアル化します";
$temp2[$n] = &XMLRPC_adjustValue($temp[$n]['value']);
}
}else{
$temp2[0] = &XMLRPC_adjustValue($temp['value']);
}
$temp = $temp2;
$temp を返します;
}
}

function XMLRPC_getMethodName($methodCall){
#メソッド名を返します
return $methodCall['methodCall']['methodName'];
}

function XMLRPC_request($site, $location, $methodName, $params = NULL, $user_agent = NULL){
$site =explode(':', $site);
if(isset($site[1]) および is_numeric($site[1])){
$port = $site[1];
}else{
$port = 80;
}
$site = $site[0];

$data["メソッド呼び出し"]["メソッド名"] = $メソッド名;
$param_count = カウント($params);
if(!$param_count){
$data["methodCall"]["params"] = NULL;
}else{
for($n = 0; $n$data["methodCall"]["params"]["param"][$n]["value"] = $params[$n];
}
}
$data = XML_serialize($data);

if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

送信する次のパラメータ リストを受信しました:

" . XMLRPC_show($params, 'print_r',真実));
}
$conn = fsockopen ($site, $port); #接続を開きます
if(!$conn){ #接続が正常に開かなかった場合
if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

接続に失敗しました: Couldn' $site に接続しません。

");
}
return array(false, array('faultCode'=>10532, 'faultString'=>"接続に失敗しました: $site に接続できませんでした。"));
}else{
$headers =
"POST $location HTTP/1.0rn" .
「ホスト: $sitern」 .
「つながり: 近づく」 .
($user_agent ? "ユーザーエージェント: $user_agentrn" : '') 。
「コンテンツタイプ: text/xmlrn」 。
「コンテンツの長さ: " . strlen($data) 。 「ルンルン」;

fputs($conn, "$headers");
fputs($conn, $data);

if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

次のリクエストを送信しました:

nn" . XMLRPC_show($headers . $data, 'print_r' 、 真実));
}

#socket_set_blocking ($conn, false);
$response = "";
while(!feof($conn)){
$response .= fgets($conn, 1024);
}
fclose($conn);

#レスポンスからヘッダーを削除します
$data = XML_unserialize(substr($response, strpos($response, "rnrn")+4));

if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

次の応答を受信しました:

nn" . XMLRPC_show($response, 'print_r', true) . "

次のデータにシリアル化されました:

nn" . XMLRPC_show($data, 'print_r', true);
}
if(isset($data['methodResponse']['fault'])){
$return = array(false, XMLRPC_adjustValue($data['methodResponse']['fault']['value']) );
if(define('XMLRPC_DEBUG') および XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

Returning:

nn" . XMLRPC_show($return, 'var_dump', true));
}
return $return;
}else{
$return = array(true, XMLRPC_adjustValue($data['methodResponse']['params']['param']['value']));
if(define('XMLRPC_DEBUG') および XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_request', "

Returning:

nn" . XMLRPC_show($return, 'var_dump', true));
}
return $return;
}
}
}

function XMLRPC_response($return_value, $server = NULL){
$data["methodResponse"]["params"]["param"]["value"] = $return_value;
$return = XML_serialize($data);

if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_response', "

返す次のデータを受信しました:

nn" . XMLRPC_show($return_value, 'print_r',真実));
}

header("接続: 閉じる");
header("Content-Length: " .strlen($return));
header("Content-Type: text/xml");
header("日付: " . date("r"));
if($server){
header("サーバー: $server");
}

if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_response', "

次の応答を送信しました:

nn" . XMLRPC_show($return, 'print_r',真実));
}
echo $return;
}

function XMLRPC_error($faultCode, $faultString, $server = NULL){
$array["methodResponse"]["fault"]["value"]["struct"]["member"] = array( );
$temp = $array["methodResponse"]["fault"]["value"]["struct"]["member"];
$temp[0]["名前"] = "障害コード";
$temp[0]["value"]["int"] = $faultCode;
$temp[1]["名前"] = "障害文字列";
$temp[1]["値"]["文字列"] = $faultString;

$return = XML_serialize($array);

header("接続: 閉じる");
header("Content-Length: " .strlen($return));
header("Content-Type: text/xml");
header("日付: " . date("r"));
if($server){
header("サーバー: $server");
}
if(define('XMLRPC_DEBUG') and XMLRPC_DEBUG){
XMLRPC_debug('XMLRPC_error', "

次のエラー応答を送信しました:

nn" . XMLRPC_show($return, 'print_r',真実));
}
echo $return;
} function XMLRPC_CONVERT_TIMESTAMP_TO_ISO8601($ TIMESTAMP){
#######unixタイムスタンプを取得し、XMLRPCに必要なISO8601に変換します。 、$タイムスタンプ);
}

function XMLRPC_convert_iso8601_to_timestamp($iso8601){
return strtotime($iso8601);
}

function count_numeric_items($array){
return is_array($array) ? count(array_filter(array_keys($array), 'is_numeric')) : 0;
}

function XMLRPC_debug($function_name, $debug_message){
$GLOBALS['XMLRPC_DEBUG_INFO'][] = array($function_name, $debug_message);
}

function XMLRPC_debug_print(){
if($GLOBALS['XMLRPC_DEBUG_INFO']){
echo "n";
foreach($GLOBALS['XMLRPC_DEBUG_INFO'] as $debug){
echo "n";
}
echo "
$debug[0] $debug[1]
n";
unset($GLOBALS['XMLRPC_DEBUG_INFO']);
}else{
echo "

デバッグ情報はまだありません。

";
}
}

function XMLRPC_show($data, $func = "print_r", $return_str = false){
ob_start();
$func($data);
$output = ob_get_contents();
ob_end_clean();
if($return_str){
return "
" 。 htmlspecialchars($output) 。 "
n";
}else{
echo "
", htmlspecialchars($output), "
n";
}
}

?>


服务端程序例子,server.php


复制代码
代码如下:「xml-rpc.inc.php」を含める;
//プロセスで使用できるメソッドを決定する
$xmlrpc_methods=array();
$xmlrpc_methods['insertRecords']='insertRecords';

//ユーザーが入力したメソッド名とパラメータを取得します
$xmlrpc_request = XMLRPC_parse($HTTP_RAW_POST_DATA);
$methodName = XMLRPC_getMethodName($xmlrpc_request);
$params = XMLRPC_getParams($xmlrpc_request);

if (!isset($xmlrpc_methods[$methodName])){
XMLRPC_error('1',"你所调用のメソッドが存在しません");
}else {
$xmlrpc_methods[$methodName]($params);
}
function insertRecords($params){
if (emptyempty($params)){
XMLRPC_error('2',"パラメータ出错");
}
XMLRPC_response(XMLRPC_prepare('http://www.emtit.com'));
}
?>


PHP 客户端调用サービス务端方法例子


复制代
代码如下:include_once 'xml-rpc.inc';
$params=配列(2,3);
$result=XMLRPC_request("127.0.0.1","/services/server.php","insertRecords",$params);//服务端文件放在services文件夹下
print_r($result);
?>


結果には www.emtiit.com が表示されます

www.bkjia.comtru​​ehttp://www.bkjia.com/PHPjc/319869.html技術記事次のようにコードをコピーします。 ?php /* インターネットから見つかった XML-RPC ライブラリ。小規模な外部通信インターフェイスの開発に非常に役立ちます*/ function while(list($key, $value) = each($data)){ $ inline = ファ...
声明:
この記事の内容はネチズンが自主的に寄稿したものであり、著作権は原著者に帰属します。このサイトは、それに相当する法的責任を負いません。盗作または侵害の疑いのあるコンテンツを見つけた場合は、admin@php.cn までご連絡ください。