2023-10-22 22:20:53 +00:00
|
|
|
<?php
|
|
|
|
namespace Activitypub;
|
|
|
|
|
|
|
|
use WP_Error;
|
|
|
|
use Activitypub\Collection\Users;
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
use function Activitypub\get_masked_wp_version;
|
|
|
|
|
2023-10-22 22:20:53 +00:00
|
|
|
/**
|
|
|
|
* ActivityPub HTTP Class
|
|
|
|
*
|
|
|
|
* @author Matthias Pfefferle
|
|
|
|
*/
|
|
|
|
class Http {
|
|
|
|
/**
|
|
|
|
* Send a POST Request with the needed HTTP Headers
|
|
|
|
*
|
|
|
|
* @param string $url The URL endpoint
|
|
|
|
* @param string $body The Post Body
|
|
|
|
* @param int $user_id The WordPress User-ID
|
|
|
|
*
|
|
|
|
* @return array|WP_Error The POST Response or an WP_ERROR
|
|
|
|
*/
|
|
|
|
public static function post( $url, $body, $user_id ) {
|
2023-12-08 23:23:11 +00:00
|
|
|
\do_action( 'activitypub_pre_http_post', $url, $body, $user_id );
|
2023-10-22 22:20:53 +00:00
|
|
|
|
|
|
|
$date = \gmdate( 'D, d M Y H:i:s T' );
|
|
|
|
$digest = Signature::generate_digest( $body );
|
|
|
|
$signature = Signature::generate_signature( $user_id, 'post', $url, $date, $digest );
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
$wp_version = get_masked_wp_version();
|
2023-10-22 22:20:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter the HTTP headers user agent.
|
|
|
|
*
|
|
|
|
* @param string $user_agent The user agent string.
|
|
|
|
*/
|
|
|
|
$user_agent = \apply_filters( 'http_headers_useragent', 'WordPress/' . $wp_version . '; ' . \get_bloginfo( 'url' ) );
|
|
|
|
$args = array(
|
|
|
|
'timeout' => 100,
|
|
|
|
'limit_response_size' => 1048576,
|
|
|
|
'redirection' => 3,
|
|
|
|
'user-agent' => "$user_agent; ActivityPub",
|
|
|
|
'headers' => array(
|
|
|
|
'Accept' => 'application/activity+json',
|
|
|
|
'Content-Type' => 'application/activity+json',
|
|
|
|
'Digest' => $digest,
|
|
|
|
'Signature' => $signature,
|
|
|
|
'Date' => $date,
|
|
|
|
),
|
|
|
|
'body' => $body,
|
|
|
|
);
|
|
|
|
|
|
|
|
$response = \wp_safe_remote_post( $url, $args );
|
|
|
|
$code = \wp_remote_retrieve_response_code( $response );
|
|
|
|
|
|
|
|
if ( $code >= 400 ) {
|
|
|
|
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ), array( 'status' => $code ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
\do_action( 'activitypub_safe_remote_post_response', $response, $url, $body, $user_id );
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Send a GET Request with the needed HTTP Headers
|
|
|
|
*
|
2024-05-09 15:26:55 +00:00
|
|
|
* @param string $url The URL endpoint
|
|
|
|
* @param bool|int $cached If the result should be cached, or its duration. Default: 1hr.
|
2023-10-22 22:20:53 +00:00
|
|
|
*
|
|
|
|
* @return array|WP_Error The GET Response or an WP_ERROR
|
|
|
|
*/
|
2024-05-09 15:26:55 +00:00
|
|
|
public static function get( $url, $cached = false ) {
|
2023-12-08 23:23:11 +00:00
|
|
|
\do_action( 'activitypub_pre_http_get', $url );
|
2023-10-22 22:20:53 +00:00
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
if ( $cached ) {
|
|
|
|
$transient_key = self::generate_cache_key( $url );
|
|
|
|
|
|
|
|
$response = \get_transient( $transient_key );
|
|
|
|
|
|
|
|
if ( $response ) {
|
|
|
|
\do_action( 'activitypub_safe_remote_get_response', $response, $url );
|
|
|
|
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-10-22 22:20:53 +00:00
|
|
|
$date = \gmdate( 'D, d M Y H:i:s T' );
|
|
|
|
$signature = Signature::generate_signature( Users::APPLICATION_USER_ID, 'get', $url, $date );
|
|
|
|
|
2024-04-19 10:49:31 +00:00
|
|
|
$wp_version = get_masked_wp_version();
|
2023-10-22 22:20:53 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Filter the HTTP headers user agent.
|
|
|
|
*
|
|
|
|
* @param string $user_agent The user agent string.
|
|
|
|
*/
|
|
|
|
$user_agent = \apply_filters( 'http_headers_useragent', 'WordPress/' . $wp_version . '; ' . \get_bloginfo( 'url' ) );
|
|
|
|
|
|
|
|
$args = array(
|
|
|
|
'timeout' => apply_filters( 'activitypub_remote_get_timeout', 100 ),
|
|
|
|
'limit_response_size' => 1048576,
|
|
|
|
'redirection' => 3,
|
|
|
|
'user-agent' => "$user_agent; ActivityPub",
|
|
|
|
'headers' => array(
|
|
|
|
'Accept' => 'application/activity+json',
|
|
|
|
'Content-Type' => 'application/activity+json',
|
|
|
|
'Signature' => $signature,
|
|
|
|
'Date' => $date,
|
|
|
|
),
|
|
|
|
);
|
|
|
|
|
|
|
|
$response = \wp_safe_remote_get( $url, $args );
|
|
|
|
$code = \wp_remote_retrieve_response_code( $response );
|
|
|
|
|
|
|
|
if ( $code >= 400 ) {
|
|
|
|
$response = new WP_Error( $code, __( 'Failed HTTP Request', 'activitypub' ), array( 'status' => $code ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
\do_action( 'activitypub_safe_remote_get_response', $response, $url );
|
|
|
|
|
2024-05-09 15:26:55 +00:00
|
|
|
if ( $cached ) {
|
|
|
|
$cache_duration = $cached;
|
|
|
|
if ( ! is_int( $cache_duration ) ) {
|
|
|
|
$cache_duration = HOUR_IN_SECONDS;
|
|
|
|
}
|
|
|
|
\set_transient( $transient_key, $response, $cache_duration );
|
|
|
|
}
|
|
|
|
|
2023-10-22 22:20:53 +00:00
|
|
|
return $response;
|
|
|
|
}
|
2023-12-08 23:23:11 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Check for URL for Tombstone.
|
|
|
|
*
|
|
|
|
* @param string $url The URL to check.
|
|
|
|
*
|
|
|
|
* @return bool True if the URL is a tombstone.
|
|
|
|
*/
|
|
|
|
public static function is_tombstone( $url ) {
|
|
|
|
\do_action( 'activitypub_pre_http_is_tombstone', $url );
|
|
|
|
|
|
|
|
$response = \wp_safe_remote_get( $url );
|
|
|
|
$code = \wp_remote_retrieve_response_code( $response );
|
|
|
|
|
|
|
|
if ( in_array( (int) $code, array( 404, 410 ), true ) ) {
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
return false;
|
|
|
|
}
|
2024-05-09 15:26:55 +00:00
|
|
|
|
|
|
|
public static function generate_cache_key( $url ) {
|
|
|
|
return 'activitypub_http_' . \md5( $url );
|
|
|
|
}
|
2024-06-27 12:10:38 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Requests the Data from the Object-URL or Object-Array
|
|
|
|
*
|
|
|
|
* @param array|string $url_or_object The Object or the Object URL.
|
|
|
|
* @param bool $cached If the result should be cached.
|
|
|
|
*
|
|
|
|
* @return array|WP_Error The Object data as array or WP_Error on failure.
|
|
|
|
*/
|
|
|
|
public static function get_remote_object( $url_or_object, $cached = true ) {
|
|
|
|
if ( is_array( $url_or_object ) ) {
|
|
|
|
if ( array_key_exists( 'id', $url_or_object ) ) {
|
|
|
|
$url = $url_or_object['id'];
|
|
|
|
} elseif ( array_key_exists( 'url', $url_or_object ) ) {
|
|
|
|
$url = $url_or_object['url'];
|
|
|
|
} else {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_no_valid_actor_identifier',
|
|
|
|
\__( 'The "actor" identifier is not valid', 'activitypub' ),
|
|
|
|
array(
|
|
|
|
'status' => 404,
|
|
|
|
'object' => $url_or_object,
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
$url = $url_or_object;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( preg_match( '/^@?' . ACTIVITYPUB_USERNAME_REGEXP . '$/i', $url ) ) {
|
|
|
|
$url = Webfinger::resolve( $url );
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! $url ) {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_no_valid_actor_identifier',
|
|
|
|
\__( 'The "actor" identifier is not valid', 'activitypub' ),
|
|
|
|
array(
|
|
|
|
'status' => 404,
|
|
|
|
'object' => $url,
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( is_wp_error( $url ) ) {
|
|
|
|
return $url;
|
|
|
|
}
|
|
|
|
|
|
|
|
$transient_key = self::generate_cache_key( $url );
|
|
|
|
|
|
|
|
// only check the cache if needed.
|
|
|
|
if ( $cached ) {
|
|
|
|
$data = \get_transient( $transient_key );
|
|
|
|
|
|
|
|
if ( $data ) {
|
|
|
|
return $data;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if ( ! \wp_http_validate_url( $url ) ) {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_no_valid_object_url',
|
|
|
|
\__( 'The "object" is/has no valid URL', 'activitypub' ),
|
|
|
|
array(
|
|
|
|
'status' => 400,
|
|
|
|
'object' => $url,
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
$response = self::get( $url );
|
|
|
|
|
|
|
|
if ( \is_wp_error( $response ) ) {
|
|
|
|
return $response;
|
|
|
|
}
|
|
|
|
|
|
|
|
$data = \wp_remote_retrieve_body( $response );
|
|
|
|
$data = \json_decode( $data, true );
|
|
|
|
|
|
|
|
if ( ! $data ) {
|
|
|
|
return new WP_Error(
|
|
|
|
'activitypub_invalid_json',
|
|
|
|
\__( 'No valid JSON data', 'activitypub' ),
|
|
|
|
array(
|
|
|
|
'status' => 400,
|
|
|
|
'object' => $url,
|
|
|
|
)
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
\set_transient( $transient_key, $data, WEEK_IN_SECONDS );
|
|
|
|
|
|
|
|
return $data;
|
|
|
|
}
|
2023-10-22 22:20:53 +00:00
|
|
|
}
|