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 * * @param string $url The URL endpoint * @param bool|int $cached If the result should be cached, or its duration. Default: 1hr. * * @return array|WP_Error The GET Response or an WP_ERROR */ public static function get( $url, $cached = false ) { \do_action( 'activitypub_pre_http_get', $url ); 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; } } $date = \gmdate( 'D, d M Y H:i:s T' ); $signature = Signature::generate_signature( Users::APPLICATION_USER_ID, 'get', $url, $date ); $wp_version = get_masked_wp_version(); /** * 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 ); if ( $cached ) { $cache_duration = $cached; if ( ! is_int( $cache_duration ) ) { $cache_duration = HOUR_IN_SECONDS; } \set_transient( $transient_key, $response, $cache_duration ); } return $response; } /** * 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; } public static function generate_cache_key( $url ) { return 'activitypub_http_' . \md5( $url ); } /** * 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; } }