2009-12-29 90 views
1

我正在爲一個正在運行的站點設置信用卡處理。 PHP沒有使用CURL支持進行編譯,我不想讓該站點重新編譯PHP,所以我嘗試使用與示例代碼不同的方法。將CURL轉換爲PHP中的fosckopen

例捲曲代碼:

function send($packet, $url) { 
    $header = array("MIME-Version: 1.0","Content-type: application/x-www-form-urlencoded","Contenttransfer-encoding: text"); 
    $ch = curl_init(); 

    // set URL and other appropriate options 
    curl_setopt($ch, CURLOPT_URL, $url); 
    curl_setopt($ch, CURLOPT_VERBOSE, 1); 
    curl_setopt ($ch, CURLOPT_PROXYTYPE, CURLPROXY_HTTP); 
    // Uncomment for host with proxy server 
    // curl_setopt ($ch, CURLOPT_PROXY, "http://proxyaddress:port"); 
    curl_setopt($ch, CURLOPT_HTTPHEADER, $header); 
    curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE); 
    curl_setopt($ch, CURLOPT_POST, true); 
    curl_setopt($ch, CURLOPT_POSTFIELDS, $packet); 
    curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); 
    curl_setopt ($ch, CURLOPT_TIMEOUT, 10); 

    // send packet and receive response 
    $response = curl_exec($ch); 
    curl_close($ch); 
    return($response); 
} 

我用不同的方法嘗試:

function send($packet, $host, $path) { 
    $content = ''; 
    $flag = false; 
    $post_query = urlencode($packet) . "\r\n"; 
    $fp = fsockopen($host, '80'); 
    if ($fp) { 
     fputs($fp, "POST $path HTTP/1.0\r\n"); 
     fputs($fp, "Host: $host\r\n"); 
     fputs($fp, "Content-length: ". strlen($post_query) ."\r\n\r\n"); 
     fputs($fp, $post_query); 

     while (!feof($fp)) { 
      $line = fgets($fp, 10240); 
      if ($flag) { 
       $content .= $line; 
      } else { 
       $headers .= $line; 
       if (strlen(trim($line)) == 0) { 
        $flag = true; 
       } 
      } 
     } 
     fclose($fp); 
    } 

    return $content; 
} 

雖然這並真正給我從我打電話的URL返回時,它是不正確的,因爲我收到一個錯誤,說它格式不正確。我對CURL或其他方法不是很熟悉,所以我不確定我做錯了什麼。

回答

0

雖然你特別要求fsockopen()你可能也想嘗試stream apicontext options

function send($packet, $url) { 
    $ctx = stream_context_create(
    array(
     'http'=>array(
     'header'=>"Content-type: application/x-www-form-urlencoded", 
     'method'=>'POST', 
     'content'=>$packet 
    ) 
    ) 
); 
    return file_get_contents($url, 0, $ctx); 
} 

,如果你需要更多的錯誤處理,而不是file_get_contents()

+0

謝謝!這工作完美。 – 2009-12-29 01:21:13

+0

嗯,實際上這不適用於HTTPS,有沒有辦法使這項工作? – 2009-12-29 16:18:40

+0

哦https。是的......但你需要一個提供https的ssl部分的模塊。這就是你想避免的:重新啓動網絡服務器,以便讓php加載另一個模塊。大多數「標準」版本的php使用openssl-extension,http://docs.php.net/openssl,但它也可以是... taadaa ...捲曲擴展。我有點懷疑你會發現ssl傳輸層的生產級「純」php實現。可能是錯誤的。 – VolkerK 2009-12-29 17:38:39

1

使用fopen()stream_get_meta_data()我試圖修改後的版本您的解決方案 - 一旦一切似乎都奏效,我發現多個fputs()導致目標服務器返回間歇性的HTTP 505錯誤。

爲了解決這個問題,我不得不預先構建完整的請求,只是做一個的fputs這樣調用

$post_vars = ""; 

$post_vars .= "&somekey=somevalue"; 

$header = ""; 

$header .= "POST /my.php HTTP/1.0\r\n"; 

$header .= "Content-Type: application/x-www-form-urlencoded\r\n"; 

$header .= "Host: www.myhost.com \r\n"; 

$header .= "Content-Length: " . strlen($post_vars) . "\r\n\r\n"; 

$fp = ""; 

@$fp = fsockopen ('ssl://www.myhost.com', 443, $errno, $errstr,30); 

fputs($fp, $header . $post_vars); 

while(!feof($fp)) { 

    // Read the data returned 

    $response .= @fgets($fp, 4096); 

} 

fclose ($fp);