<?php
+/**
+ * @copyright Copyright (C) 2020, Friendica
+ *
+ * @license GNU AGPL version 3 or any later version
+ *
+ * This program 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.
+ *
+ * This program 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 this program. If not, see <https://www.gnu.org/licenses/>.
+ *
+ */
namespace Friendica\Network;
-
+use Friendica\Core\Logger;
use Friendica\Network\HTTPException\InternalServerErrorException;
+use Friendica\Util\Network;
/**
* A content class for Curl call results
*/
private $header;
+ /**
+ * @var array the HTTP headers of the Curl call
+ */
+ private $header_fields;
+
/**
* @var boolean true (if HTTP 2xx result) or false
*/
* @param string $url optional URL
*
* @return CurlResult a CURL with error response
+ * @throws InternalServerErrorException
*/
public static function createErrorCurl($url = '')
{
$this->errorNumber = $errorNumber;
$this->error = $error;
- logger($url . ': ' . $this->returnCode . " " . $result, LOGGER_DATA);
+ Logger::log($url . ': ' . $this->returnCode . " " . $result, Logger::DATA);
$this->parseBodyHeader($result);
$this->checkSuccess();
$header = '';
$base = $result;
- while (preg_match('/^HTTP\/[1-2].+? [1-5][0-9][0-9]/', $base)) {
+ while (preg_match('/^HTTP\/.+? \d+/', $base)) {
$chunk = substr($base, 0, strpos($base, "\r\n\r\n") + 4);
$header .= $chunk;
$base = substr($base, strlen($chunk));
$this->body = substr($result, strlen($header));
$this->header = $header;
+ $this->header_fields = []; // Is filled on demand
}
private function checkSuccess()
{
$this->isSuccess = ($this->returnCode >= 200 && $this->returnCode <= 299) || $this->errorNumber == 0;
+ // Everything higher or equal 400 is not a success
+ if ($this->returnCode >= 400) {
+ $this->isSuccess = false;
+ }
+
if (!$this->isSuccess) {
- logger('error: ' . $this->url . ': ' . $this->returnCode . ' - ' . $this->error, LOGGER_INFO);
- logger('debug: ' . print_r($this->info, true), LOGGER_DATA);
+ Logger::log('error: ' . $this->url . ': ' . $this->returnCode . ' - ' . $this->error, Logger::INFO);
+ Logger::log('debug: ' . print_r($this->info, true), Logger::DATA);
}
if (!$this->isSuccess && $this->errorNumber == CURLE_OPERATION_TIMEDOUT) {
}
if ($this->returnCode == 301 || $this->returnCode == 302 || $this->returnCode == 303 || $this->returnCode== 307) {
- $new_location_info = (!array_key_exists('redirect_url', $this->info) ? '' : @parse_url($this->info['redirect_url']));
- $old_location_info = (!array_key_exists('url', $this->info) ? '' : @parse_url($this->info['url']));
-
- $this->redirectUrl = $new_location_info;
-
- if (empty($new_location_info['path']) && !empty($new_location_info['host'])) {
- $this->redirectUrl = $new_location_info['scheme'] . '://' . $new_location_info['host'] . $old_location_info['path'];
+ $redirect_parts = parse_url($this->info['redirect_url'] ?? '');
+ if (empty($redirect_parts)) {
+ $redirect_parts = [];
}
- $matches = [];
-
if (preg_match('/(Location:|URI:)(.*?)\n/i', $this->header, $matches)) {
- $this->redirectUrl = trim(array_pop($matches));
+ $redirect_parts2 = parse_url(trim(array_pop($matches)));
+ if (!empty($redirect_parts2)) {
+ $redirect_parts = array_merge($redirect_parts, $redirect_parts2);
+ }
}
- if (strpos($this->redirectUrl, '/') === 0) {
- $this->redirectUrl = $old_location_info["scheme"] . "://" . $old_location_info["host"] . $this->redirectUrl;
+
+ $parts = parse_url($this->info['url'] ?? '');
+ if (empty($parts)) {
+ $parts = [];
}
- $old_location_query = @parse_url($this->url, PHP_URL_QUERY);
- if ($old_location_query != '') {
- $this->redirectUrl .= '?' . $old_location_query;
+ /// @todo Checking the corresponding RFC which parts of a redirect can be ommitted.
+ $components = ['scheme', 'host', 'path', 'query', 'fragment'];
+ foreach ($components as $component) {
+ if (empty($redirect_parts[$component]) && !empty($parts[$component])) {
+ $redirect_parts[$component] = $parts[$component];
+ }
}
- $this->isRedirectUrl = filter_var($this->redirectUrl, FILTER_VALIDATE_URL);
+ $this->redirectUrl = Network::unparseURL($redirect_parts);
+
+ $this->isRedirectUrl = true;
} else {
$this->isRedirectUrl = false;
}
/**
* Returns the Curl headers
*
- * @return string the Curl headers
+ * @param string $field optional header field. Return all fields if empty
+ *
+ * @return string the Curl headers or the specified content of the header variable
+ */
+ public function getHeader(string $field = '')
+ {
+ if (empty($field)) {
+ return $this->header;
+ }
+
+ $field = strtolower(trim($field));
+
+ $headers = $this->getHeaderArray();
+
+ if (isset($headers[$field])) {
+ return $headers[$field];
+ }
+
+ return '';
+ }
+
+ /**
+ * Check if a specified header exists
+ *
+ * @param string $field header field
+ *
+ * @return boolean "true" if header exists
+ */
+ public function inHeader(string $field)
+ {
+ $field = strtolower(trim($field));
+
+ $headers = $this->getHeaderArray();
+
+ return array_key_exists($field, $headers);
+ }
+
+ /**
+ * Returns the Curl headers as an associated array
+ *
+ * @return array associated header array
*/
- public function getHeader()
+ public function getHeaderArray()
{
- return $this->header;
+ if (!empty($this->header_fields)) {
+ return $this->header_fields;
+ }
+
+ $this->header_fields = [];
+
+ $lines = explode("\n", trim($this->header));
+ foreach ($lines as $line) {
+ $parts = explode(':', $line);
+ $headerfield = strtolower(trim(array_shift($parts)));
+ $headerdata = trim(implode(':', $parts));
+ $this->header_fields[$headerfield] = $headerdata;
+ }
+
+ return $this->header_fields;
}
/**