Code Coverage
 
Lines
Functions and Methods
Classes and Traits
Total
27.88% covered (danger)
27.88%
373 / 1338
9.09% covered (danger)
9.09%
4 / 44
CRAP
0.00% covered (danger)
0.00%
0 / 1
WPCOM_JSON_API_Endpoint
27.98% covered (danger)
27.98%
373 / 1333
9.09% covered (danger)
9.09%
4 / 44
86540.87
0.00% covered (danger)
0.00%
0 / 1
 __construct
96.51% covered (success)
96.51%
83 / 86
0.00% covered (danger)
0.00%
0 / 1
12
 query_args
75.00% covered (warning)
75.00%
3 / 4
0.00% covered (danger)
0.00%
0 / 1
2.06
 input
71.43% covered (warning)
71.43%
25 / 35
0.00% covered (danger)
0.00%
0 / 1
23.74
 get_secure_body
0.00% covered (danger)
0.00%
0 / 9
0.00% covered (danger)
0.00%
0 / 1
6
 cast_and_filter
87.88% covered (warning)
87.88%
29 / 33
0.00% covered (danger)
0.00%
0 / 1
16.46
 cast_and_filter_item
10.28% covered (danger)
10.28%
33 / 321
0.00% covered (danger)
0.00%
0 / 1
4938.13
 parse_types
94.74% covered (success)
94.74%
18 / 19
0.00% covered (danger)
0.00%
0 / 1
5.00
 is_publicly_documentable
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
6
 document
0.00% covered (danger)
0.00%
0 / 55
0.00% covered (danger)
0.00%
0 / 1
132
 add_http_build_query_to_php_content_example
0.00% covered (danger)
0.00%
0 / 5
0.00% covered (danger)
0.00%
0 / 1
2
 generate_doc_description
0.00% covered (danger)
0.00%
0 / 5
0.00% covered (danger)
0.00%
0 / 1
12
 generate_documentation
0.00% covered (danger)
0.00%
0 / 60
0.00% covered (danger)
0.00%
0 / 1
240
 user_can_view_post
35.56% covered (danger)
35.56%
16 / 45
0.00% covered (danger)
0.00%
0 / 1
94.35
 get_author
71.17% covered (warning)
71.17%
79 / 111
0.00% covered (danger)
0.00%
0 / 1
72.60
 get_media_item
0.00% covered (danger)
0.00%
0 / 25
0.00% covered (danger)
0.00%
0 / 1
56
 get_media_item_v1_1
0.00% covered (danger)
0.00%
0 / 121
0.00% covered (danger)
0.00%
0 / 1
5700
 get_taxonomy
75.00% covered (warning)
75.00%
3 / 4
0.00% covered (danger)
0.00%
0 / 1
3.14
 format_taxonomy
77.78% covered (warning)
77.78%
21 / 27
0.00% covered (danger)
0.00%
0 / 1
8.70
 format_date
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 parse_date
0.00% covered (danger)
0.00%
0 / 19
0.00% covered (danger)
0.00%
0 / 1
20
 load_theme_functions
0.00% covered (danger)
0.00%
0 / 28
0.00% covered (danger)
0.00%
0 / 1
272
 copy_hooks
0.00% covered (danger)
0.00%
0 / 19
0.00% covered (danger)
0.00%
0 / 1
90
 get_reflection
0.00% covered (danger)
0.00%
0 / 11
0.00% covered (danger)
0.00%
0 / 1
72
 current_user_can_access_post_type
75.00% covered (warning)
75.00%
6 / 8
0.00% covered (danger)
0.00%
0 / 1
6.56
 is_post_type_allowed
53.85% covered (warning)
53.85%
7 / 13
0.00% covered (danger)
0.00%
0 / 1
11.82
 _get_whitelisted_post_types
100.00% covered (success)
100.00%
3 / 3
100.00% covered (success)
100.00%
1 / 1
1
 media_item_is_free_video_mobile_upload_and_too_long
0.00% covered (danger)
0.00%
0 / 31
0.00% covered (danger)
0.00%
0 / 1
90
 handle_media_creation_v1_1
0.00% covered (danger)
0.00%
0 / 84
0.00% covered (danger)
0.00%
0 / 1
1122
 handle_media_sideload
0.00% covered (danger)
0.00%
0 / 23
0.00% covered (danger)
0.00%
0 / 1
156
 is_file_supported_for_sideloading
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 allow_video_uploads
0.00% covered (danger)
0.00%
0 / 21
0.00% covered (danger)
0.00%
0 / 1
182
 is_current_site_multi_user
0.00% covered (danger)
0.00%
0 / 11
0.00% covered (danger)
0.00%
0 / 1
6
 allows_cross_origin_requests
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
6
 allows_unauthorized_requests
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
12
 accepts_site_based_authentication
100.00% covered (success)
100.00%
2 / 2
100.00% covered (success)
100.00%
1 / 1
2
 get_platform
100.00% covered (success)
100.00%
1 / 1
100.00% covered (success)
100.00%
1 / 1
1
 force_wpcom_request
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 get_amp_cache_origins
0.00% covered (danger)
0.00%
0 / 15
0.00% covered (danger)
0.00%
0 / 1
12
 create_rest_route_for_endpoint
0.00% covered (danger)
0.00%
0 / 9
0.00% covered (danger)
0.00%
0 / 1
2
 rest_callback
95.56% covered (success)
95.56%
43 / 45
0.00% covered (danger)
0.00%
0 / 1
16
 rest_permission_callback
0.00% covered (danger)
0.00%
0 / 15
0.00% covered (danger)
0.00%
0 / 1
90
 rest_permission_callback_custom
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 build_rest_route
0.00% covered (danger)
0.00%
0 / 2
0.00% covered (danger)
0.00%
0 / 1
6
 get_rest_min_jp_version
0.00% covered (danger)
0.00%
0 / 1
0.00% covered (danger)
0.00%
0 / 1
2
 callback
n/a
0 / 0
n/a
0 / 0
0
1<?php // phpcs:ignore WordPress.Files.FileName.InvalidClassFileName
2/**
3 * Jetpack API endpoint base class.
4 *
5 * @package automattic/jetpack
6 */
7
8use Automattic\Jetpack\Connection\Client;
9use Automattic\Jetpack\Connection\Manager;
10use Automattic\Jetpack\Connection\Rest_Authentication;
11use Automattic\Jetpack\Connection\Tokens;
12use Automattic\Jetpack\Status;
13use Automattic\Jetpack\Status\Host;
14
15require_once __DIR__ . '/json-api-config.php';
16require_once __DIR__ . '/sal/class.json-api-links.php';
17require_once __DIR__ . '/sal/class.json-api-metadata.php';
18require_once __DIR__ . '/sal/class.json-api-date.php';
19
20/**
21 * Endpoint.
22 */
23abstract class WPCOM_JSON_API_Endpoint {
24    /**
25     * The API Object
26     *
27     * @var WPCOM_JSON_API
28     */
29    public $api;
30
31    /**
32     * The link-generating utility class
33     *
34     * @var WPCOM_JSON_API_Links
35     */
36    public $links;
37
38    /**
39     * Whether to pass wpcom user details.
40     *
41     * @var bool
42     */
43    public $pass_wpcom_user_details = false;
44
45    /**
46     * One liner.
47     *
48     * @var string
49     */
50    public $description;
51
52    /**
53     * Object Grouping For Documentation (Users, Posts, Comments)
54     *
55     * @var string
56     */
57    public $group;
58
59    /**
60     * Stats extra value to bump
61     *
62     * @var mixed
63     */
64    public $stat;
65
66    /**
67     * HTTP Method
68     *
69     * @var string
70     */
71    public $method = 'GET';
72
73    /**
74     * Minimum version of the api for which to serve this endpoint
75     *
76     * @var string
77     */
78    public $min_version = '0';
79
80    /**
81     * Maximum version of the api for which to serve this endpoint
82     *
83     * @var string
84     */
85    public $max_version = WPCOM_JSON_API__CURRENT_VERSION;
86
87    /**
88     * Forced endpoint environment when running on WPCOM
89     *
90     * @var string '', 'wpcom', 'secure', or 'jetpack'
91     */
92    public $force = '';
93
94    /**
95     * Whether the endpoint is deprecated
96     *
97     * @var bool
98     */
99    public $deprecated = false;
100
101    /**
102     * Version of the endpoint this endpoint is deprecated in favor of.
103     *
104     * @var string
105     */
106    protected $new_version = WPCOM_JSON_API__CURRENT_VERSION;
107
108    /**
109     * Whether the endpoint is only available on WordPress.com hosted blogs
110     *
111     * @var bool
112     */
113    public $jp_disabled = false;
114
115    /**
116     * Path at which to serve this endpoint: sprintf() format.
117     *
118     * @var string
119     */
120    public $path = '';
121
122    /**
123     * Identifiers to fill sprintf() formatted $path
124     *
125     * @var array
126     */
127    public $path_labels = array();
128
129    /**
130     * The REST endpoint if available.
131     *
132     * @var string
133     */
134    public $rest_route;
135
136    /**
137     * Jetpack Version in which REST support was introduced.
138     *
139     * @var string
140     */
141    public $rest_min_jp_version;
142
143    /**
144     * Accepted query parameters
145     *
146     * @var array
147     */
148    public $query = array(
149        // Parameter name.
150        'context'       => array(
151            // Default value => description.
152            'display' => 'Formats the output as HTML for display.  Shortcodes are parsed, paragraph tags are added, etc..',
153            // Other possible values => description.
154            'edit'    => 'Formats the output for editing.  Shortcodes are left unparsed, significant whitespace is kept, etc..',
155        ),
156        'http_envelope' => array(
157            'false' => '',
158            'true'  => 'Some environments (like in-browser JavaScript or Flash) block or divert responses with a non-200 HTTP status code.  Setting this parameter will force the HTTP status code to always be 200.  The JSON response is wrapped in an "envelope" containing the "real" HTTP status code and headers.',
159        ),
160        'pretty'        => array(
161            'false' => '',
162            'true'  => 'Output pretty JSON',
163        ),
164        'meta'          => "(string) Optional. Loads data from the endpoints found in the 'meta' part of the response. Comma-separated list. Example: meta=site,likes",
165        'fields'        => '(string) Optional. Returns specified fields only. Comma-separated list. Example: fields=ID,title',
166        // Parameter name => description (default value is empty).
167        'callback'      => '(string) An optional JSONP callback function.',
168    );
169
170    /**
171     * Response format
172     *
173     * @var array
174     */
175    public $response_format = array();
176
177    /**
178     * Request format
179     *
180     * @var array
181     */
182    public $request_format = array();
183
184    /**
185     * Is this endpoint still in testing phase?  If so, not available to the public.
186     *
187     * @var bool
188     */
189    public $in_testing = false;
190
191    /**
192     * Is this endpoint still allowed if the site in question is flagged?
193     *
194     * @var bool
195     */
196    public $allowed_if_flagged = false;
197
198    /**
199     * Is this endpoint allowed if the site is red flagged?
200     *
201     * @var bool
202     */
203    public $allowed_if_red_flagged = false;
204
205    /**
206     * Is this endpoint allowed if the site is deleted?
207     *
208     * @var bool
209     */
210    public $allowed_if_deleted = false;
211
212    /**
213     * Version of the API
214     *
215     * @var string
216     */
217    public $version = '';
218
219    /**
220     * Example request to make
221     *
222     * @var string
223     */
224    public $example_request = '';
225
226    /**
227     * Example request data (for POST methods)
228     *
229     * @var string
230     */
231    public $example_request_data = '';
232
233    /**
234     * Example response from $example_request
235     *
236     * @var string
237     */
238    public $example_response = '';
239
240    /**
241     * OAuth2 scope required when running on WPCOM
242     *
243     * @var string
244     */
245    public $required_scope = '';
246
247    /**
248     * Set to true if the endpoint implements its own filtering instead of the standard `fields` query method
249     *
250     * @var bool
251     */
252    public $custom_fields_filtering = false;
253
254    /**
255     * Set to true if the endpoint accepts all cross origin requests. You probably should not set this flag.
256     *
257     * @var bool
258     */
259    public $allow_cross_origin_request = false;
260
261    /**
262     * Set to true if the endpoint can recieve unauthorized POST requests.
263     *
264     * @var bool
265     */
266    public $allow_unauthorized_request = false;
267
268    /**
269     * Set to true if the endpoint should accept site based (not user based) authentication.
270     *
271     * @var bool
272     */
273    public $allow_jetpack_site_auth = false;
274
275    /**
276     * Set to true if the endpoint should accept user based authentication.
277     *
278     * @var bool
279     */
280    public $allow_jetpack_token_auth = false;
281
282    /**
283     * Set to true if the endpoint should accept auth from an upload token.
284     *
285     * @var bool
286     */
287    public $allow_upload_token_auth = false;
288
289    /**
290     * Set to true if the endpoint should require auth from a Rewind auth token.
291     *
292     * @var bool
293     */
294    public $require_rewind_auth = false;
295
296    /**
297     * Whether this endpoint allows falling back to a blog token for making requests to remote Jetpack sites.
298     *
299     * @var bool
300     */
301    public $allow_fallback_to_jetpack_blog_token = false;
302
303    /**
304     * REST namespace.
305     */
306    const REST_NAMESPACE = 'jetpack/rest';
307
308    /**
309     * Post object format.
310     *
311     * @var array
312     */
313    public $post_object_format;
314
315    /**
316     * Comment object format.
317     *
318     * @var array
319     */
320    public $comment_object_format;
321
322    /**
323     * Dropdown page object format.
324     *
325     * @var array
326     */
327    public $dropdown_page_object_format;
328
329    /**
330     * Constructor.
331     *
332     * @param string|array|object $args Args.
333     */
334    public function __construct( $args ) {
335        $defaults = array(
336            'in_testing'                           => false,
337            'allowed_if_flagged'                   => false,
338            'allowed_if_red_flagged'               => false,
339            'allowed_if_deleted'                   => false,
340            'description'                          => '',
341            'group'                                => '',
342            'stat'                                 => '',
343            'method'                               => 'GET',
344            'path'                                 => '/',
345            'min_version'                          => '0',
346            'max_version'                          => WPCOM_JSON_API__CURRENT_VERSION,
347            'force'                                => '',
348            'deprecated'                           => false,
349            'new_version'                          => WPCOM_JSON_API__CURRENT_VERSION,
350            'jp_disabled'                          => false,
351            'path_labels'                          => array(),
352            'rest_route'                           => null,
353            'rest_min_jp_version'                  => null,
354            'request_format'                       => array(),
355            'response_format'                      => array(),
356            'query_parameters'                     => array(),
357            'version'                              => 'v1',
358            'example_request'                      => '',
359            'example_request_data'                 => '',
360            'example_response'                     => '',
361            'required_scope'                       => '',
362            'pass_wpcom_user_details'              => false,
363            'custom_fields_filtering'              => false,
364            'allow_cross_origin_request'           => false,
365            'allow_unauthorized_request'           => false,
366            'allow_jetpack_site_auth'              => false,
367            'allow_jetpack_token_auth'             => false,
368            'allow_upload_token_auth'              => false,
369            'allow_fallback_to_jetpack_blog_token' => false,
370        );
371
372        $args = wp_parse_args( $args, $defaults );
373
374        $this->in_testing = $args['in_testing'];
375
376        $this->allowed_if_flagged     = $args['allowed_if_flagged'];
377        $this->allowed_if_red_flagged = $args['allowed_if_red_flagged'];
378        $this->allowed_if_deleted     = $args['allowed_if_deleted'];
379
380        $this->description = $args['description'];
381        $this->group       = $args['group'];
382        $this->stat        = $args['stat'];
383        $this->force       = $args['force'];
384        $this->jp_disabled = $args['jp_disabled'];
385
386        $this->method      = $args['method'];
387        $this->path        = $args['path'];
388        $this->path_labels = $args['path_labels'];
389        $this->min_version = $args['min_version'];
390        $this->max_version = $args['max_version'];
391        $this->deprecated  = $args['deprecated'];
392        $this->new_version = $args['new_version'];
393
394        $this->rest_route          = $args['rest_route'];
395        $this->rest_min_jp_version = $args['rest_min_jp_version'];
396
397        // Ensure max version is not less than min version.
398        if ( version_compare( $this->min_version, $this->max_version, '>' ) ) {
399            $this->max_version = $this->min_version;
400        }
401
402        $this->pass_wpcom_user_details = $args['pass_wpcom_user_details'];
403        $this->custom_fields_filtering = (bool) $args['custom_fields_filtering'];
404
405        $this->allow_cross_origin_request           = (bool) $args['allow_cross_origin_request'];
406        $this->allow_unauthorized_request           = (bool) $args['allow_unauthorized_request'];
407        $this->allow_jetpack_site_auth              = (bool) $args['allow_jetpack_site_auth'];
408        $this->allow_jetpack_token_auth             = (bool) $args['allow_jetpack_token_auth'];
409        $this->allow_upload_token_auth              = (bool) $args['allow_upload_token_auth'];
410        $this->allow_fallback_to_jetpack_blog_token = (bool) $args['allow_fallback_to_jetpack_blog_token'];
411        $this->require_rewind_auth                  = isset( $args['require_rewind_auth'] ) ? (bool) $args['require_rewind_auth'] : false;
412
413        $this->version = $args['version'];
414
415        $this->required_scope = $args['required_scope'];
416
417        if ( $this->request_format ) {
418            $this->request_format = array_filter( array_merge( $this->request_format, $args['request_format'] ) );
419        } else {
420            $this->request_format = $args['request_format'];
421        }
422
423        if ( $this->response_format ) {
424            $this->response_format = array_filter( array_merge( $this->response_format, $args['response_format'] ) );
425        } else {
426            $this->response_format = $args['response_format'];
427        }
428
429        if ( false === $args['query_parameters'] ) {
430            $this->query = array();
431        } elseif ( is_array( $args['query_parameters'] ) ) {
432            $this->query = array_filter( array_merge( $this->query, $args['query_parameters'] ) );
433        }
434
435        $this->api   = WPCOM_JSON_API::init(); // Auto-add to WPCOM_JSON_API.
436        $this->links = WPCOM_JSON_API_Links::getInstance();
437
438        /** Example Request/Response */
439
440        // Examples for endpoint documentation request.
441        $this->example_request      = $args['example_request'];
442        $this->example_request_data = $args['example_request_data'];
443        $this->example_response     = $args['example_response'];
444
445        $this->api->add( $this );
446
447        if ( ( ! defined( 'IS_WPCOM' ) || ! IS_WPCOM ) && $this->rest_route && ( ! defined( 'XMLRPC_REQUEST' ) || ! XMLRPC_REQUEST ) ) {
448            $this->create_rest_route_for_endpoint();
449        }
450    }
451
452    /**
453     * Get all query args. Prefill with defaults.
454     *
455     * @param bool $return_default_values Whether to include default values in the response.
456     * @param bool $cast_and_filter Whether to cast and filter input according to the documentation.
457     * @return array
458     */
459    public function query_args( $return_default_values = true, $cast_and_filter = true ) {
460        $args = array_intersect_key( $this->api->query, $this->query );
461
462        if ( ! $cast_and_filter ) {
463            return $args;
464        }
465
466        return $this->cast_and_filter( $args, $this->query, $return_default_values );
467    }
468
469    /**
470     * Get POST body data.
471     *
472     * @param bool $return_default_values Whether to include default values in the response.
473     * @param bool $cast_and_filter Whether to cast and filter input according to the documentation.
474     * @return mixed
475     */
476    public function input( $return_default_values = true, $cast_and_filter = true ) {
477        $return       = null;
478        $input        = trim( (string) $this->api->post_body );
479        $content_type = (string) $this->api->content_type;
480        if ( $content_type ) {
481            list ( $content_type ) = explode( ';', $content_type );
482        }
483        $content_type = trim( $content_type );
484        switch ( $content_type ) {
485            case 'application/json':
486            case 'application/x-javascript':
487            case 'text/javascript':
488            case 'text/x-javascript':
489            case 'text/x-json':
490            case 'text/json':
491                $return = json_decode( $input, true );
492
493                if ( JSON_ERROR_NONE !== json_last_error() ) {
494                    return null;
495                }
496
497                break;
498            case 'multipart/form-data':
499                // phpcs:ignore WordPress.Security.NonceVerification.Missing
500                $return = array_merge( stripslashes_deep( $_POST ), $_FILES );
501                break;
502            case 'application/x-www-form-urlencoded':
503                // attempt JSON first, since probably a curl command.
504                $return = json_decode( $input, true );
505
506                if ( $return === null ) {
507                    wp_parse_str( $input, $return );
508                }
509
510                break;
511            default:
512                wp_parse_str( $input, $return );
513                break;
514        }
515
516        if ( isset( $this->api->query['force'] )
517            && 'secure' === $this->api->query['force']
518            && isset( $return['secure_key'] ) ) {
519            $this->api->post_body      = $this->get_secure_body( $return['secure_key'] );
520            $this->api->query['force'] = false;
521            return $this->input( $return_default_values, $cast_and_filter );
522        }
523
524        if ( $cast_and_filter ) {
525            $return = $this->cast_and_filter( $return, $this->request_format, $return_default_values );
526        }
527        return $return;
528    }
529
530    /**
531     * Fetch a body via secure request.
532     *
533     * @param string $secure_key Key for the request.
534     * @return mixed|null API response, or null if the request failed.
535     */
536    protected function get_secure_body( $secure_key ) {
537        $response = Client::wpcom_json_api_request_as_blog(
538            sprintf( '/sites/%d/secure-request', Jetpack_Options::get_option( 'id' ) ),
539            '1.1',
540            array( 'method' => 'POST' ),
541            array( 'secure_key' => $secure_key )
542        );
543        if ( 200 !== $response['response']['code'] ) {
544            return null;
545        }
546        return json_decode( $response['body'], true );
547    }
548
549    /**
550     * Cast and filter data.
551     *
552     * @param mixed $data Data to cast and filter.
553     * @param array $documentation Documentation for keys in `$data` to keep and cast.
554     * @param bool  $return_default_values Set default values from `$documentation` to process.
555     * @param bool  $for_output See `$this->cast_and_filter_item()`.
556     * @return mixed Filtered data.
557     */
558    public function cast_and_filter( $data, $documentation, $return_default_values = false, $for_output = false ) {
559        $return_as_object = false;
560        if ( is_object( $data ) ) {
561            // @todo this should probably be a deep copy if $data can ever have nested objects
562            $data             = (array) $data;
563            $return_as_object = true;
564        } elseif ( ! is_array( $data ) ) {
565            return $data;
566        }
567
568        $boolean_arg = array( 'false', 'true' );
569        $naeloob_arg = array( 'true', 'false' );
570
571        $return = array();
572
573        foreach ( $documentation as $key => $description ) {
574            if ( is_array( $description ) ) {
575                // String or boolean array keys only.
576                $whitelist = array_keys( $description );
577
578                if ( $whitelist === $boolean_arg || $whitelist === $naeloob_arg ) {
579                    // Truthiness.
580                    if ( isset( $data[ $key ] ) ) {
581                        $return[ $key ] = (bool) WPCOM_JSON_API::is_truthy( $data[ $key ] );
582                    } elseif ( $return_default_values ) {
583                        $return[ $key ] = $whitelist === $naeloob_arg; // Default to true for naeloob_arg and false for boolean_arg.
584                    }
585                } elseif ( isset( $data[ $key ] ) && isset( $description[ $data[ $key ] ] ) ) {
586                    // String Key.
587                    $return[ $key ] = (string) $data[ $key ];
588                } elseif ( $return_default_values ) {
589                    // Default value.
590                    $return[ $key ] = (string) current( $whitelist );
591                }
592
593                continue;
594            }
595
596            $types = $this->parse_types( $description );
597            $type  = array_shift( $types );
598
599            // Explicit default - string and int only for now.  Always set these reguardless of $return_default_values.
600            if ( isset( $type['default'] ) ) {
601                if ( ! isset( $data[ $key ] ) ) {
602                    $data[ $key ] = $type['default'];
603                }
604            }
605
606            if ( ! isset( $data[ $key ] ) ) {
607                continue;
608            }
609
610            $this->cast_and_filter_item( $return, $type, $key, $data[ $key ], $types, $for_output );
611        }
612
613        if ( $return_as_object ) {
614            return (object) $return;
615        }
616
617        return $return;
618    }
619
620    /**
621     * Casts $value according to $type.
622     * Handles fallbacks for certain values of $type when $value is not that $type
623     * Currently, only handles fallback between string <-> array (two way), from string -> false (one way), and from object -> false (one way),
624     * and string -> object (one way)
625     *
626     * Handles "child types" - array:URL, object:category
627     * array:URL means an array of URLs
628     * object:category means a hash of categories
629     *
630     * Handles object typing - object>post means an object of type post
631     *
632     * @param array        $return Array to assign the value into.
633     * @param string|array $type Type to cast.
634     * @param string|int   $key Key in `$return` to assign the value to.
635     * @param mixed        $value Value to cast.
636     * @param array        $types Fallback types.
637     * @param bool         $for_output Appears to affect formatting of 'date' types.
638     */
639    public function cast_and_filter_item( &$return, $type, $key, $value, $types = array(), $for_output = false ) {
640        if ( is_string( $type ) ) {
641            $type = compact( 'type' );
642        }
643
644        switch ( $type['type'] ) {
645            case 'false':
646                $return[ $key ] = false;
647                break;
648            case 'url':
649                if ( is_object( $value ) && isset( $value->url ) && str_contains( $value->url, 'https://videos.files.wordpress.com/' ) ) {
650                    $value = $value->url;
651                }
652                // Check for string since esc_url_raw() expects one.
653                if ( ! is_string( $value ) ) {
654                    break;
655                }
656                $return[ $key ] = (string) esc_url_raw( $value );
657                break;
658            case 'string':
659                // Fallback string -> array, or for string -> object.
660                if ( is_array( $value ) || is_object( $value ) ) {
661                    if ( ! empty( $types[0] ) ) {
662                        $next_type = array_shift( $types );
663                        return $this->cast_and_filter_item( $return, $next_type, $key, $value, $types, $for_output );
664                    }
665                }
666
667                // Fallback string -> false.
668                if ( ! is_string( $value ) ) {
669                    if ( ! empty( $types[0] ) && 'false' === $types[0]['type'] ) {
670                        $next_type = array_shift( $types );
671                        return $this->cast_and_filter_item( $return, $next_type, $key, $value, $types, $for_output );
672                    }
673                    if ( is_array( $value ) ) {
674                        // Give up rather than setting the value to the string 'Array'.
675                        break;
676                    }
677                }
678                $return[ $key ] = (string) $value;
679                break;
680            case 'html':
681                $return[ $key ] = (string) $value;
682                break;
683            case 'safehtml':
684                $return[ $key ] = wp_kses( (string) $value, wp_kses_allowed_html() );
685                break;
686            case 'zip':
687            case 'media':
688                if ( is_array( $value ) ) {
689                    if ( isset( $value['name'] ) && is_array( $value['name'] ) ) {
690                        // It's a $_FILES array
691                        // Reformat into array of $_FILES items.
692                        $files = array();
693
694                        foreach ( $value['name'] as $k => $v ) {
695                            $files[ $k ] = array();
696                            foreach ( array_keys( $value ) as $file_key ) {
697                                $files[ $k ][ $file_key ] = $value[ $file_key ][ $k ];
698                            }
699                        }
700
701                        foreach ( $files as $k => $file ) {
702                            if ( ! isset( $file['tmp_name'] ) || ! is_string( $file['tmp_name'] ) || ! is_uploaded_file( $file['tmp_name'] ) ) {
703                                unset( $files[ $k ] );
704                            }
705                        }
706                        if ( $files ) {
707                            $return[ $key ] = $files;
708                        }
709                    } elseif ( isset( $value['tmp_name'] ) && is_string( $value['tmp_name'] ) && is_uploaded_file( $value['tmp_name'] ) ) {
710                        $return[ $key ] = $value;
711                    }
712                }
713                break;
714            case 'array':
715                // Fallback array -> string.
716                if ( is_string( $value ) ) {
717                    if ( ! empty( $types[0] ) ) {
718                        $next_type = array_shift( $types );
719                        return $this->cast_and_filter_item( $return, $next_type, $key, $value, $types, $for_output );
720                    }
721                }
722
723                if ( isset( $type['children'] ) ) {
724                    $children = array();
725                    foreach ( (array) $value as $k => $child ) {
726                        $this->cast_and_filter_item( $children, $type['children'], $k, $child, array(), $for_output );
727                    }
728                    $return[ $key ] = (array) $children;
729                    break;
730                }
731
732                $return[ $key ] = (array) $value;
733                break;
734            case 'iso 8601 datetime':
735            case 'datetime':
736                // (string)s
737                $dates = $this->parse_date( (string) $value );
738                if ( $for_output ) {
739                    $return[ $key ] = $this->format_date( $dates[1], $dates[0] );
740                } else {
741                    list( $return[ $key ], $return[ "{$key}_gmt" ] ) = $dates;
742                }
743                break;
744            case 'float':
745                $return[ $key ] = (float) $value;
746                break;
747            case 'int':
748            case 'integer':
749                $return[ $key ] = (int) $value;
750                break;
751            case 'bool':
752            case 'boolean':
753                $return[ $key ] = (bool) WPCOM_JSON_API::is_truthy( $value );
754                break;
755            case 'object':
756                // Fallback object -> false.
757                if ( is_scalar( $value ) || $value === null ) {
758                    if ( ! empty( $types[0] ) && 'false' === $types[0]['type'] ) {
759                        return $this->cast_and_filter_item( $return, 'false', $key, $value, $types, $for_output );
760                    }
761                }
762
763                if ( isset( $type['children'] ) ) {
764                    $children = array();
765                    foreach ( (array) $value as $k => $child ) {
766                        $this->cast_and_filter_item( $children, $type['children'], $k, $child, array(), $for_output );
767                    }
768                    $return[ $key ] = (object) $children;
769                    break;
770                }
771
772                if ( isset( $type['subtype'] ) ) {
773                    return $this->cast_and_filter_item( $return, $type['subtype'], $key, $value, $types, $for_output );
774                }
775
776                $return[ $key ] = (object) $value;
777                break;
778            case 'post':
779                $return[ $key ] = (object) $this->cast_and_filter( $value, $this->post_object_format, false, $for_output );
780                break;
781            case 'comment':
782                $return[ $key ] = (object) $this->cast_and_filter( $value, $this->comment_object_format, false, $for_output );
783                break;
784            case 'tag':
785            case 'category':
786                $docs = array(
787                    'ID'          => '(int)',
788                    'name'        => '(string)',
789                    'slug'        => '(string)',
790                    'description' => '(HTML)',
791                    'post_count'  => '(int)',
792                    'feed_url'    => '(string)',
793                    'meta'        => '(object)',
794                );
795                if ( 'category' === $type['type'] ) {
796                    $docs['parent'] = '(int)';
797                }
798                $return[ $key ] = (object) $this->cast_and_filter( $value, $docs, false, $for_output );
799                break;
800            case 'post_reference':
801            case 'comment_reference':
802                $docs           = array(
803                    'ID'    => '(int)',
804                    'type'  => '(string)',
805                    'title' => '(string)',
806                    'link'  => '(URL)',
807                );
808                $return[ $key ] = (object) $this->cast_and_filter( $value, $docs, false, $for_output );
809                break;
810            case 'geo':
811                $docs           = array(
812                    'latitude'  => '(float)',
813                    'longitude' => '(float)',
814                    'address'   => '(string)',
815                );
816                $return[ $key ] = (object) $this->cast_and_filter( $value, $docs, false, $for_output );
817                break;
818            case 'author':
819                $docs           = array(
820                    'ID'             => '(int)',
821                    'user_login'     => '(string)',
822                    'login'          => '(string)',
823                    'email'          => '(string|false)',
824                    'name'           => '(string)',
825                    'first_name'     => '(string)',
826                    'last_name'      => '(string)',
827                    'nice_name'      => '(string)',
828                    'URL'            => '(URL)',
829                    'avatar_URL'     => '(URL)',
830                    'profile_URL'    => '(URL)',
831                    'is_super_admin' => '(bool)',
832                    'roles'          => '(array:string)',
833                    'ip_address'     => '(string|false)',
834                    'wpcom_id'       => '(int|null)',
835                    'wpcom_login'    => '(string|null)',
836                );
837                $return[ $key ] = (object) $this->cast_and_filter( $value, $docs, false, $for_output );
838                break;
839            case 'role':
840                $docs           = array(
841                    'name'         => '(string)',
842                    'display_name' => '(string)',
843                    'capabilities' => '(object:boolean)',
844                );
845                $return[ $key ] = (object) $this->cast_and_filter( $value, $docs, false, $for_output );
846                break;
847            case 'attachment':
848                $docs           = array(
849                    'ID'        => '(int)',
850                    'URL'       => '(URL)',
851                    'guid'      => '(string)',
852                    'mime_type' => '(string)',
853                    'width'     => '(int)',
854                    'height'    => '(int)',
855                    'duration'  => '(int)',
856                );
857                $return[ $key ] = (object) $this->cast_and_filter(
858                    $value,
859                    /**
860                    * Filter the documentation returned for a post attachment.
861                    *
862                    * @module json-api
863                    *
864                    * @since 1.9.0
865                    *
866                    * @param array $docs Array of documentation about a post attachment.
867                    */
868                    apply_filters( 'wpcom_json_api_attachment_cast_and_filter', $docs ),
869                    false,
870                    $for_output
871                );
872                break;
873            case 'metadata':
874                $docs           = array(
875                    'id'             => '(int)',
876                    'key'            => '(string)',
877                    'value'          => '(string|false|float|int|array|object)',
878                    'previous_value' => '(string)',
879                    'operation'      => '(string)',
880                );
881                $return[ $key ] = (object) $this->cast_and_filter(
882                    $value,
883                    /** This filter is documented in class.json-api-endpoints.php */
884                    apply_filters( 'wpcom_json_api_attachment_cast_and_filter', $docs ),
885                    false,
886                    $for_output
887                );
888                break;
889            case 'plugin':
890                $docs           = array(
891                    'id'           => '(safehtml) The plugin\'s ID',
892                    'slug'         => '(safehtml) The plugin\'s Slug',
893                    'active'       => '(boolean)  The plugin status.',
894                    'update'       => '(object)   The plugin update info.',
895                    'name'         => '(safehtml) The name of the plugin.',
896                    'plugin_url'   => '(url)      Link to the plugin\'s web site.',
897                    'version'      => '(safehtml) The plugin version number.',
898                    'description'  => '(safehtml) Description of what the plugin does and/or notes from the author',
899                    'author'       => '(safehtml) The plugin author\'s name',
900                    'author_url'   => '(url)      The plugin author web site address',
901                    'network'      => '(boolean)  Whether the plugin can only be activated network wide.',
902                    'autoupdate'   => '(boolean)  Whether the plugin is auto updated',
903                    'log'          => '(array:safehtml) An array of update log strings.',
904                    'action_links' => '(array) An array of action links that the plugin uses.',
905                );
906                $return[ $key ] = (object) $this->cast_and_filter(
907                    $value,
908                    /**
909                    * Filter the documentation returned for a plugin.
910                    *
911                    * @module json-api
912                    *
913                    * @since 3.1.0
914                    *
915                    * @param array $docs Array of documentation about a plugin.
916                    */
917                    apply_filters( 'wpcom_json_api_plugin_cast_and_filter', $docs ),
918                    false,
919                    $for_output
920                );
921                break;
922            case 'plugin_v1_2':
923                $docs           = class_exists( 'Jetpack_JSON_API_Get_Plugins_v1_2_Endpoint' )
924                ? Jetpack_JSON_API_Get_Plugins_v1_2_Endpoint::$_response_format
925                : Jetpack_JSON_API_Plugins_Endpoint::$_response_format_v1_2;
926                $return[ $key ] = (object) $this->cast_and_filter(
927                    $value,
928                    /**
929                    * Filter the documentation returned for a plugin.
930                    *
931                    * @module json-api
932                    *
933                    * @since 3.1.0
934                    *
935                    * @param array $docs Array of documentation about a plugin.
936                    */
937                    apply_filters( 'wpcom_json_api_plugin_cast_and_filter', $docs ),
938                    false,
939                    $for_output
940                );
941                break;
942            case 'file_mod_capabilities':
943                $docs           = array(
944                    'reasons_modify_files_unavailable' => '(array) The reasons why files can\'t be modified',
945                    'reasons_autoupdate_unavailable'   => '(array) The reasons why autoupdates aren\'t allowed',
946                    'modify_files'                     => '(boolean) true if files can be modified',
947                    'autoupdate_files'                 => '(boolean) true if autoupdates are allowed',
948                );
949                $return[ $key ] = (array) $this->cast_and_filter( $value, $docs, false, $for_output );
950                break;
951            case 'jetpackmodule':
952                $docs           = array(
953                    'id'          => '(string)   The module\'s ID',
954                    'active'      => '(boolean)  The module\'s status.',
955                    'name'        => '(string)   The module\'s name.',
956                    'description' => '(safehtml) The module\'s description.',
957                    'sort'        => '(int)      The module\'s display order.',
958                    'introduced'  => '(string)   The Jetpack version when the module was introduced.',
959                    'changed'     => '(string)   The Jetpack version when the module was changed.',
960                    'free'        => '(boolean)  The module\'s Free or Paid status.',
961                    'module_tags' => '(array)    The module\'s tags.',
962                    'override'    => '(string)   The module\'s override. Empty if no override, otherwise \'active\' or \'inactive\'',
963                );
964                $return[ $key ] = (object) $this->cast_and_filter(
965                    $value,
966                    /** This filter is documented in class.json-api-endpoints.php */
967                    apply_filters( 'wpcom_json_api_plugin_cast_and_filter', $docs ),
968                    false,
969                    $for_output
970                );
971                break;
972            case 'sharing_button':
973                $docs           = array(
974                    'ID'         => '(string)',
975                    'name'       => '(string)',
976                    'URL'        => '(string)',
977                    'icon'       => '(string)',
978                    'enabled'    => '(bool)',
979                    'visibility' => '(string)',
980                );
981                $return[ $key ] = (array) $this->cast_and_filter( $value, $docs, false, $for_output );
982                break;
983            case 'sharing_button_service':
984                $docs           = array(
985                    'ID'               => '(string) The service identifier',
986                    'name'             => '(string) The service name',
987                    'class_name'       => '(string) Class name for custom style sharing button elements',
988                    'genericon'        => '(string) The Genericon unicode character for the custom style sharing button icon',
989                    'preview_smart'    => '(string) An HTML snippet of a rendered sharing button smart preview',
990                    'preview_smart_js' => '(string) An HTML snippet of the page-wide initialization scripts used for rendering the sharing button smart preview',
991                );
992                $return[ $key ] = (array) $this->cast_and_filter( $value, $docs, false, $for_output );
993                break;
994            case 'site_keyring':
995                $docs           = array(
996                    'keyring_id'       => '(int) Keyring ID',
997                    'service'          => '(string) The service name',
998                    'external_user_id' => '(string) External user id for the service',
999                );
1000                $return[ $key ] = (array) $this->cast_and_filter( $value, $docs, false, $for_output );
1001                break;
1002            case 'taxonomy':
1003                $docs           = array(
1004                    'name'         => '(string) The taxonomy slug',
1005                    'label'        => '(string) The taxonomy human-readable name',
1006                    'labels'       => '(object) Mapping of labels for the taxonomy',
1007                    'description'  => '(string) The taxonomy description',
1008                    'hierarchical' => '(bool) Whether the taxonomy is hierarchical',
1009                    'public'       => '(bool) Whether the taxonomy is public',
1010                    'capabilities' => '(object) Mapping of current user capabilities for the taxonomy',
1011                );
1012                $return[ $key ] = (array) $this->cast_and_filter( $value, $docs, false, $for_output );
1013                break;
1014            case 'visibility':
1015                // This is needed to fix a bug in WPAndroid where `public: "PUBLIC"` is sent in place of `public: 1`.
1016                if ( 'public' === strtolower( $value ) ) {
1017                    $return[ $key ] = 1;
1018                } elseif ( 'private' === strtolower( $value ) ) {
1019                    $return[ $key ] = -1;
1020                } else {
1021                    $return[ $key ] = (int) $value;
1022                }
1023                break;
1024            case 'dropdown_page':
1025                $return[ $key ] = (array) $this->cast_and_filter( $value, $this->dropdown_page_object_format, false, $for_output );
1026                break;
1027            default:
1028                $method_name = $type['type'] . '_docs';
1029                if ( method_exists( 'WPCOM_JSON_API_Jetpack_Overrides', $method_name ) ) {
1030                    $docs = WPCOM_JSON_API_Jetpack_Overrides::$method_name();
1031                }
1032
1033                if ( ! empty( $docs ) ) {
1034                    $return[ $key ] = (object) $this->cast_and_filter(
1035                        $value,
1036                        /** This filter is documented in class.json-api-endpoints.php */
1037                        apply_filters( 'wpcom_json_api_plugin_cast_and_filter', $docs ),
1038                        false,
1039                        $for_output
1040                    );
1041                } else {
1042                    // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error, WordPress.Security.EscapeOutput.OutputNotEscaped
1043                    trigger_error( "Unknown API casting type {$type['type']}", E_USER_WARNING );
1044                }
1045        }
1046    }
1047
1048    /**
1049     * Parse types from text.
1050     *
1051     * @param string $text Text.
1052     * @return array Types.
1053     */
1054    public function parse_types( $text ) {
1055        if ( ! preg_match( '#^\(([^)]+)\)#', ltrim( $text ), $matches ) ) {
1056            return 'none';
1057        }
1058
1059        $types  = explode( '|', strtolower( $matches[1] ) );
1060        $return = array();
1061        foreach ( $types as $type ) {
1062            foreach ( array(
1063                ':' => 'children',
1064                '>' => 'subtype',
1065                '=' => 'default',
1066            ) as $operator => $meaning ) {
1067                if ( str_contains( $type, $operator ) ) {
1068                    $item     = explode( $operator, $type, 2 );
1069                    $return[] = array(
1070                        'type'   => $item[0],
1071                        $meaning => $item[1],
1072                    );
1073                    continue 2;
1074                }
1075            }
1076            $return[] = compact( 'type' );
1077        }
1078
1079        return $return;
1080    }
1081
1082    /**
1083     * Checks if the endpoint is publicly displayable
1084     *
1085     * @return bool
1086     */
1087    public function is_publicly_documentable() {
1088        return '__do_not_document' !== $this->group && true !== $this->in_testing;
1089    }
1090
1091    /**
1092     * Auto generates documentation based on description, method, path, path_labels, and query parameters.
1093     * Echoes HTML.
1094     *
1095     * @param bool $show_description Whether to show the description.
1096     */
1097    public function document( $show_description = true ) {
1098        global $wpdb;
1099        $original_post = isset( $GLOBALS['post'] ) ? $GLOBALS['post'] : 'unset';
1100        unset( $GLOBALS['post'] );
1101
1102        $doc = $this->generate_documentation();
1103
1104        if ( $show_description ) :
1105            ?>
1106<caption>
1107    <h1><?php echo wp_kses_post( $doc['method'] ); ?> <?php echo wp_kses_post( $doc['path_labeled'] ); ?></h1>
1108    <p><?php echo wp_kses_post( $doc['description'] ); ?></p>
1109</caption>
1110
1111<?php endif; ?>
1112
1113        <?php if ( true === $this->deprecated ) { ?>
1114<p><strong>This endpoint is deprecated in favor of version <?php echo (float) $this->new_version; ?></strong></p>
1115<?php } ?>
1116
1117<section class="resource-info">
1118    <h2 id="apidoc-resource-info">Resource Information</h2>
1119
1120    <table class="api-doc api-doc-resource-parameters api-doc-resource">
1121
1122    <thead>
1123        <tr>
1124            <th class="api-index-title" scope="column">&nbsp;</th>
1125            <th class="api-index-title" scope="column">&nbsp;</th>
1126        </tr>
1127    </thead>
1128    <tbody>
1129
1130        <tr class="api-index-item">
1131            <th scope="row" class="parameter api-index-item-title">Method</th>
1132            <td class="type api-index-item-title"><?php echo wp_kses_post( $doc['method'] ); ?></td>
1133        </tr>
1134
1135        <tr class="api-index-item">
1136            <th scope="row" class="parameter api-index-item-title">URL</th>
1137            <?php
1138            $version = WPCOM_JSON_API__CURRENT_VERSION;
1139            if ( ! empty( $this->max_version ) ) {
1140                $version = $this->max_version;
1141            }
1142            ?>
1143            <td class="type api-index-item-title">https://public-api.wordpress.com/rest/v<?php echo (float) $version; ?><?php echo wp_kses_post( $doc['path_labeled'] ); ?></td>
1144        </tr>
1145
1146        <tr class="api-index-item">
1147            <th scope="row" class="parameter api-index-item-title">Requires authentication?</th>
1148            <?php
1149            $requires_auth = $wpdb->get_row( $wpdb->prepare( 'SELECT requires_authentication FROM rest_api_documentation WHERE `version` = %s AND `path` = %s AND `method` = %s LIMIT 1', $version, untrailingslashit( $doc['path_labeled'] ), $doc['method'] ) );
1150            ?>
1151            <td class="type api-index-item-title"><?php echo ( ! empty( $requires_auth->requires_authentication ) ? 'Yes' : 'No' ); ?></td>
1152        </tr>
1153
1154    </tbody>
1155    </table>
1156
1157</section>
1158
1159        <?php
1160
1161        foreach ( array(
1162            'path'     => 'Method Parameters',
1163            'query'    => 'Query Parameters',
1164            'body'     => 'Request Parameters',
1165            'response' => 'Response Parameters',
1166        ) as $doc_section_key => $label ) :
1167            $doc_section = 'response' === $doc_section_key ? $doc['response']['body'] : $doc['request'][ $doc_section_key ];
1168            if ( ! $doc_section ) {
1169                continue;
1170            }
1171
1172            $param_label = strtolower( str_replace( ' ', '-', $label ) );
1173            ?>
1174
1175<section class="<?php echo esc_attr( $param_label ); ?>">
1176
1177<h2 id="apidoc-<?php echo esc_attr( $doc_section_key ); ?>"><?php echo wp_kses_post( $label ); ?></h2>
1178
1179<table class="api-doc api-doc-<?php echo esc_attr( $param_label ); ?>-parameters api-doc-<?php echo esc_attr( strtolower( str_replace( ' ', '-', $doc['group'] ) ) ); ?>">
1180
1181<thead>
1182    <tr>
1183        <th class="api-index-title" scope="column">Parameter</th>
1184        <th class="api-index-title" scope="column">Type</th>
1185        <th class="api-index-title" scope="column">Description</th>
1186    </tr>
1187</thead>
1188<tbody>
1189
1190            <?php foreach ( $doc_section as $key => $item ) : ?>
1191
1192    <tr class="api-index-item">
1193        <th scope="row" class="parameter api-index-item-title"><?php echo wp_kses_post( $key ); ?></th>
1194        <td class="type api-index-item-title"><?php echo wp_kses_post( $item['type'] ); // @todo auto-link? ?></td>
1195        <td class="description api-index-item-body">
1196                <?php
1197
1198                $this->generate_doc_description( $item['description'] );
1199
1200                ?>
1201        </td>
1202    </tr>
1203
1204<?php endforeach; ?>
1205</tbody>
1206</table>
1207</section>
1208<?php endforeach; ?>
1209
1210        <?php
1211        if ( 'unset' !== $original_post ) {
1212            $GLOBALS['post'] = $original_post; // phpcs:ignore WordPress.WP.GlobalVariablesOverride.Prohibited
1213        }
1214    }
1215
1216    /**
1217     * `preg_replace_callback` callback to add http_build_query to php content example.
1218     *
1219     * @todo Is this used anywhere?
1220     *
1221     * @param array $matches Matches.
1222     * @return string
1223     */
1224    public function add_http_build_query_to_php_content_example( $matches ) {
1225        $trimmed_match = ltrim( $matches[0] );
1226        $pad           = substr( $matches[0], 0, -1 * strlen( $trimmed_match ) );
1227        $pad           = ltrim( $pad, ' ' );
1228        $return        = '  ' . str_replace( "\n", "\n  ", $matches[0] );
1229        return " http_build_query({$return}{$pad})";
1230    }
1231
1232    /**
1233     * Recursively generates the <dl>'s to document item descriptions.
1234     * Echoes HTML.
1235     *
1236     * @param string|array $item Post data to output, or an array of key => data mappings.
1237     */
1238    public function generate_doc_description( $item ) {
1239        if ( is_array( $item ) ) :
1240            ?>
1241
1242        <dl>
1243            <?php    foreach ( $item as $description_key => $description_value ) : ?>
1244
1245            <dt><?php echo wp_kses_post( $description_key . ':' ); ?></dt>
1246            <dd><?php $this->generate_doc_description( $description_value ); ?></dd>
1247
1248            <?php    endforeach; ?>
1249
1250        </dl>
1251
1252            <?php
1253        else :
1254            echo wp_kses_post( $item );
1255        endif;
1256    }
1257
1258    /**
1259     * Auto generates documentation based on description, method, path, path_labels, and query parameters.
1260     * Echoes HTML.
1261     */
1262    public function generate_documentation() {
1263        $format       = str_replace( '%d', '%s', $this->path );
1264        $path_labeled = $format;
1265        if ( ! empty( $this->path_labels ) ) {
1266            $path_labeled = vsprintf( $format, array_keys( $this->path_labels ) );
1267        }
1268        $boolean_arg = array( 'false', 'true' );
1269        $naeloob_arg = array( 'true', 'false' );
1270
1271        $doc = array(
1272            'description'  => $this->description,
1273            'method'       => $this->method,
1274            'path_format'  => $this->path,
1275            'path_labeled' => $path_labeled,
1276            'group'        => $this->group,
1277            'request'      => array(
1278                'path'  => array(),
1279                'query' => array(),
1280                'body'  => array(),
1281            ),
1282            'response'     => array(
1283                'body' => array(),
1284            ),
1285        );
1286
1287        foreach ( array(
1288            'path_labels'     => 'path',
1289            'query'           => 'query',
1290            'request_format'  => 'body',
1291            'response_format' => 'body',
1292        ) as $_property => $doc_item ) {
1293            foreach ( (array) $this->$_property as $key => $description ) {
1294                if ( is_array( $description ) ) {
1295                    $description_keys = array_keys( $description );
1296                    if ( $boolean_arg === $description_keys || $naeloob_arg === $description_keys ) {
1297                        $type = '(bool)';
1298                    } else {
1299                        $type = '(string)';
1300                    }
1301
1302                    if ( 'response_format' !== $_property ) {
1303                        // hack - don't show "(default)" in response format.
1304                        reset( $description );
1305                        $description_key                 = key( $description );
1306                        $description[ $description_key ] = "(default) {$description[$description_key]}";
1307                    }
1308                } else {
1309                    $types   = $this->parse_types( $description );
1310                    $type    = array();
1311                    $default = '';
1312
1313                    if ( 'none' === $types ) {
1314                        $types           = array();
1315                        $types[]['type'] = 'none';
1316                    }
1317
1318                    foreach ( $types as $type_array ) {
1319                        $type[] = $type_array['type'];
1320                        if ( isset( $type_array['default'] ) ) {
1321                            $default = $type_array['default'];
1322                            if ( 'string' === $type_array['type'] ) {
1323                                $default = "'$default'";
1324                            }
1325                        }
1326                    }
1327                    $type = '(' . implode( '|', $type ) . ')';
1328                    if ( str_contains( $description, ')' ) ) {
1329                        list( , $description ) = explode( ')', $description, 2 );
1330                    }
1331                    $description = trim( $description );
1332                    if ( $default ) {
1333                        $description .= " Default: $default.";
1334                    }
1335                }
1336
1337                $item = compact( 'type', 'description' );
1338
1339                if ( 'response_format' === $_property ) {
1340                    $doc['response'][ $doc_item ][ $key ] = $item;
1341                } else {
1342                    $doc['request'][ $doc_item ][ $key ] = $item;
1343                }
1344            }
1345        }
1346
1347        return $doc;
1348    }
1349
1350    /**
1351     * Can the user view the post?
1352     *
1353     * @param int $post_id Post ID.
1354     * @return bool|WP_Error
1355     */
1356    public function user_can_view_post( $post_id ) {
1357        $post = get_post( $post_id );
1358        if ( ! $post || is_wp_error( $post ) ) {
1359            return false;
1360        }
1361
1362        if ( 'inherit' === $post->post_status ) {
1363            $parent_post     = get_post( $post->post_parent );
1364            $post_status_obj = get_post_status_object( $parent_post->post_status ?? $post->post_status );
1365        } else {
1366            $post_status_obj = get_post_status_object( $post->post_status );
1367        }
1368
1369        if ( empty( $post_status_obj->public ) ) {
1370            if ( is_user_logged_in() ) {
1371                if ( ! empty( $post_status_obj->protected ) ) {
1372                    if ( ! current_user_can( 'edit_post', $post->ID ) ) {
1373                        return new WP_Error( 'unauthorized', 'User cannot view post', 403 );
1374                    }
1375                } elseif ( ! empty( $post_status_obj->private ) ) {
1376                    if ( ! current_user_can( 'read_post', $post->ID ) ) {
1377                        return new WP_Error( 'unauthorized', 'User cannot view post', 403 );
1378                    }
1379                } elseif ( in_array( $post->post_status, array( 'inherit', 'trash' ), true ) ) {
1380                    if ( ! current_user_can( 'edit_post', $post->ID ) ) {
1381                        return new WP_Error( 'unauthorized', 'User cannot view post', 403 );
1382                    }
1383                } elseif ( 'auto-draft' === $post->post_status ) { // phpcs:ignore Generic.CodeAnalysis.EmptyStatement.DetectedElseif
1384                    // allow auto-drafts.
1385                } else {
1386                    return new WP_Error( 'unauthorized', 'User cannot view post', 403 );
1387                }
1388            } else {
1389                return new WP_Error( 'unauthorized', 'User cannot view post', 403 );
1390            }
1391        }
1392
1393        if (
1394            ( new Status() )->is_private_site() &&
1395            /**
1396             * Filter access to a specific post.
1397             *
1398             * @module json-api
1399             *
1400             * @since 3.4.0
1401             *
1402             * @param bool current_user_can( 'read_post', $post->ID ) Can the current user access the post.
1403             * @param WP_Post $post Post data.
1404             */
1405            ! apply_filters(
1406                'wpcom_json_api_user_can_view_post',
1407                current_user_can( 'read_post', $post->ID ),
1408                $post
1409            )
1410        ) {
1411            return new WP_Error(
1412                'unauthorized',
1413                'User cannot view post',
1414                array(
1415                    'status_code' => 403,
1416                    'error'       => 'private_blog',
1417                )
1418            );
1419        }
1420
1421        if ( strlen( $post->post_password ) && ! current_user_can( 'edit_post', $post->ID ) ) {
1422            return new WP_Error(
1423                'unauthorized',
1424                'User cannot view password protected post',
1425                array(
1426                    'status_code' => 403,
1427                    'error'       => 'password_protected',
1428                )
1429            );
1430        }
1431
1432        return true;
1433    }
1434
1435    /**
1436     * Returns author object.
1437     *
1438     * @param object $author user ID, user row, WP_User object, comment row, post row.
1439     * @param bool   $show_email_and_ip output the author's email address and IP address?.
1440     *
1441     * @return object
1442     */
1443    public function get_author( $author, $show_email_and_ip = false ) {
1444        $is_jetpack = null;
1445        $login      = null;
1446        $email      = null;
1447        $name       = null;
1448        $first_name = null;
1449        $last_name  = null;
1450        $nice       = null;
1451        $url        = null;
1452        $ip_address = isset( $author->comment_author_IP ) ? $author->comment_author_IP : '';
1453        $site_id    = -1;
1454
1455        if ( isset( $author->comment_author_email ) ) {
1456            $id         = empty( $author->user_id ) ? 0 : (int) $author->user_id;
1457            $login      = '';
1458            $email      = $author->comment_author_email;
1459            $name       = $author->comment_author;
1460            $first_name = '';
1461            $last_name  = '';
1462            $avatar_url = $this->api->get_avatar_url( $author );
1463            $nice       = '';
1464            $url        = $author->comment_author_url;
1465            // Convert Gravatar URLs containing an email address to the hashed version.
1466            if ( preg_match( '#^https?://(?:www\.)?gravatar\.com/([^/?]+)#i', $url, $matches ) && is_email( $matches[1] ) ) {
1467                $url = 'https://gravatar.com/' . md5( strtolower( trim( $matches[1] ) ) );
1468            }
1469
1470            // Add additional user data to the response if a valid user ID is available.
1471            if ( 0 < $id ) {
1472                $user = get_user_by( 'id', $id );
1473                if ( $user instanceof WP_User ) {
1474                    $login      = $user->user_login ?? '';
1475                    $first_name = $user->first_name ?? '';
1476                    $last_name  = $user->last_name ?? '';
1477                    $nice       = $user->user_nicename ?? '';
1478                } else {
1479                    trigger_error( 'Unknown user', E_USER_WARNING ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
1480                }
1481            }
1482
1483            // Comment author URLs and Emails are sent through wp_kses() on save, which replaces "&" with "&amp;"
1484            // "&" is the only email/URL character altered by wp_kses().
1485            foreach ( array( 'email', 'url' ) as $field ) {
1486                $$field = str_replace( '&amp;', '&', $$field );
1487            }
1488        } elseif ( $author instanceof WP_User || isset( $author->user_email ) ) {
1489            $author = $author->ID;
1490        } elseif ( isset( $author->user_id ) && $author->user_id ) {
1491            $author = $author->user_id;
1492        } elseif ( isset( $author->post_author ) ) {
1493            // then $author is a Post Object.
1494            if ( ! $author->post_author ) {
1495                return null;
1496            }
1497            /**
1498             * Filter whether the current site is a Jetpack site.
1499             *
1500             * @module json-api
1501             *
1502             * @since 3.3.0
1503             *
1504             * @param bool false Is the current site a Jetpack site. Default to false.
1505             * @param int get_current_blog_id() Blog ID.
1506             */
1507            $is_jetpack = true === apply_filters( 'is_jetpack_site', false, get_current_blog_id() );
1508            $post_id    = $author->ID;
1509            if ( $is_jetpack && ( defined( 'IS_WPCOM' ) && IS_WPCOM ) ) {
1510                $id         = get_post_meta( $post_id, '_jetpack_post_author_external_id', true );
1511                $email      = get_post_meta( $post_id, '_jetpack_author_email', true );
1512                $login      = '';
1513                $name       = get_post_meta( $post_id, '_jetpack_author', true );
1514                $first_name = '';
1515                $last_name  = '';
1516                $url        = '';
1517                $nice       = '';
1518            } else {
1519                $author = $author->post_author;
1520            }
1521        }
1522
1523        if ( ! isset( $id ) ) {
1524            $user = get_user_by( 'id', $author );
1525            if ( ! $user || is_wp_error( $user ) ) {
1526                trigger_error( 'Unknown user', E_USER_WARNING ); // phpcs:ignore WordPress.PHP.DevelopmentFunctions.error_log_trigger_error
1527
1528                return null;
1529            }
1530            $id         = $user->ID;
1531            $email      = $user->user_email;
1532            $login      = $user->user_login;
1533            $name       = $user->display_name;
1534            $first_name = $user->first_name;
1535            $last_name  = $user->last_name;
1536            $url        = $user->user_url;
1537            $nice       = $user->user_nicename;
1538        }
1539        if ( defined( 'IS_WPCOM' ) && IS_WPCOM && ! $is_jetpack && $id > 0 ) {
1540            /**
1541             * Allow customizing the blog ID returned with the author in WordPress.com REST API queries.
1542             *
1543             * @since 12.9
1544             *
1545             * @module json-api
1546             *
1547             * @param bool|int $active_blog  Blog ID, or false by default.
1548             * @param int      $id           User ID.
1549             */
1550            $active_blog = apply_filters( 'wpcom_api_pre_get_active_blog_author', false, $id );
1551            if ( false === $active_blog ) {
1552                $active_blog = get_active_blog_for_user( $id );
1553            }
1554            if ( ! empty( $active_blog ) ) {
1555                $site_id = $active_blog->blog_id;
1556            }
1557            if ( $site_id > - 1 ) {
1558                $site_visible = (
1559                    - 1 !== (int) $active_blog->public ||
1560                    is_private_blog_user( $site_id, get_current_user_id() )
1561                );
1562            }
1563            $profile_url = "https://gravatar.com/{$login}";
1564        } else {
1565            $profile_url = 'https://gravatar.com/' . md5( strtolower( trim( $email ) ) );
1566        }
1567
1568        if ( ! isset( $avatar_url ) ) {
1569            $avatar_url = $this->api->get_avatar_url( $email );
1570        }
1571
1572        if ( $show_email_and_ip ) {
1573            $email      = (string) $email;
1574            $ip_address = (string) $ip_address;
1575        } else {
1576            $email      = false;
1577            $ip_address = false;
1578        }
1579
1580        $author = array(
1581            'ID'          => (int) $id,
1582            'login'       => (string) $login,
1583            'email'       => $email, // string|bool.
1584            'name'        => (string) $name,
1585            'first_name'  => (string) $first_name,
1586            'last_name'   => (string) $last_name,
1587            'nice_name'   => (string) $nice,
1588            'URL'         => (string) esc_url_raw( $url ),
1589            'avatar_URL'  => (string) esc_url_raw( $avatar_url ),
1590            'profile_URL' => (string) esc_url_raw( $profile_url ),
1591            'ip_address'  => $ip_address, // string|bool.
1592        );
1593
1594        if ( $site_id > -1 ) {
1595            $author['site_ID']      = (int) $site_id;
1596            $author['site_visible'] = $site_visible ?? null;
1597        }
1598
1599        // Only include WordPress.com user data when author_wpcom_data is enabled.
1600        $args = $this->query_args();
1601
1602        if ( ! empty( $id ) && ! empty( $args['author_wpcom_data'] ) ) {
1603            if ( ( new Host() )->is_wpcom_simple() ) {
1604                $user                  = get_user_by( 'id', $id );
1605                $author['wpcom_id']    = isset( $user->ID ) ? (int) $user->ID : null;
1606                $author['wpcom_login'] = $user->user_login ?? '';
1607            } else {
1608                // If this is a Jetpack site, use the connection manager to get the user data.
1609                $wpcom_user_data = ( new Manager() )->get_connected_user_data( $id );
1610                if ( $wpcom_user_data && isset( $wpcom_user_data['ID'] ) ) {
1611                    $author['wpcom_id']    = (int) $wpcom_user_data['ID'];
1612                    $author['wpcom_login'] = $wpcom_user_data['login'] ?? '';
1613                }
1614            }
1615        }
1616
1617        return (object) $author;
1618    }
1619
1620    /**
1621     * Get a media item.
1622     *
1623     * @param int $media_id Media post ID.
1624     * @return object|WP_Error Media item data, or WP_Error.
1625     */
1626    public function get_media_item( $media_id ) {
1627        $media_item = get_post( $media_id );
1628
1629        if ( ! $media_item || is_wp_error( $media_item ) ) {
1630            return new WP_Error( 'unknown_media', 'Unknown Media', 404 );
1631        }
1632
1633        $response = array(
1634            'id'          => (string) $media_item->ID,
1635            'date'        => (string) $this->format_date( $media_item->post_date_gmt, $media_item->post_date ),
1636            'parent'      => $media_item->post_parent,
1637            'link'        => wp_get_attachment_url( $media_item->ID ),
1638            'title'       => $media_item->post_title,
1639            'caption'     => $media_item->post_excerpt,
1640            'description' => $media_item->post_content,
1641            'metadata'    => wp_get_attachment_metadata( $media_item->ID ),
1642        );
1643
1644        if ( defined( 'IS_WPCOM' ) && IS_WPCOM && is_array( $response['metadata'] ) && ! empty( $response['metadata']['file'] ) ) {
1645            remove_filter( '_wp_relative_upload_path', 'wpcom_wp_relative_upload_path', 10 );
1646            $response['metadata']['file'] = _wp_relative_upload_path( $response['metadata']['file'] );
1647            add_filter( '_wp_relative_upload_path', 'wpcom_wp_relative_upload_path', 10, 2 );
1648        }
1649
1650        $response['meta'] = (object) array(
1651            'links' => (object) array(
1652                'self' => (string) $this->links->get_media_link( $this->api->get_blog_id_for_output(), $media_id ),
1653                'help' => (string) $this->links->get_media_link( $this->api->get_blog_id_for_output(), $media_id, 'help' ),
1654                'site' => (string) $this->links->get_site_link( $this->api->get_blog_id_for_output() ),
1655            ),
1656        );
1657
1658        return (object) $response;
1659    }
1660
1661    /**
1662     * Get a v1.1 media item.
1663     *
1664     * @param int          $media_id Media post ID.
1665     * @param WP_Post|null $media_item Media item.
1666     * @param string|null  $file File path.
1667     * @return object|WP_Error Media item data, or WP_Error.
1668     */
1669    public function get_media_item_v1_1( $media_id, $media_item = null, $file = null ) {
1670        if ( ! $media_item ) {
1671            $media_item = get_post( $media_id );
1672        }
1673
1674        if ( ! $media_item || is_wp_error( $media_item ) ) {
1675            return new WP_Error( 'unknown_media', 'Unknown Media', 404 );
1676        }
1677
1678        $attachment_file = isset( $media_item->ID ) ? get_attached_file( $media_item->ID ) : null;
1679
1680        $file      = basename( $attachment_file ? $attachment_file : $file );
1681        $file_info = pathinfo( $file );
1682        $ext       = isset( $file_info['extension'] ) ? $file_info['extension'] : null;
1683
1684        // File operations are handled differently on WordPress.com.
1685        if ( defined( 'IS_WPCOM' ) && IS_WPCOM ) {
1686            $attachment_metadata = isset( $media_item->ID ) ? wp_get_attachment_metadata( $media_item->ID ) : array();
1687            $filesize            = ! empty( $attachment_metadata['filesize'] ) ? $attachment_metadata['filesize'] : 0;
1688        } else {
1689            // For VideoPress videos, $attachment_file is the video URL.
1690            $filesize = ( $attachment_file && file_exists( $attachment_file ) )
1691            ? filesize( $attachment_file )
1692            : 0;
1693        }
1694
1695        $response = array(
1696            'ID'          => isset( $media_item->ID ) ? $media_item->ID : null,
1697            'URL'         => isset( $media_item->ID ) ? wp_get_attachment_url( $media_item->ID ) : null,
1698            'guid'        => isset( $media_item->guid ) ? $media_item->guid : null,
1699            'date'        => ( isset( $media_item->post_date_gmt ) && isset( $media_item->post_date ) ) ?
1700            (string) $this->format_date( $media_item->post_date_gmt, $media_item->post_date ) : null,
1701            'post_ID'     => isset( $media_item->post_parent ) ? $media_item->post_parent : null,
1702            'author_ID'   => isset( $media_item->post_author ) ? (int) $media_item->post_author : null,
1703            'file'        => $file,
1704            'mime_type'   => isset( $media_item->post_mime_type ) ? $media_item->post_mime_type : null,
1705            'extension'   => $ext,
1706            'title'       => isset( $media_item->post_title ) ? $media_item->post_title : '',
1707            'caption'     => isset( $media_item->post_excerpt ) ? $media_item->post_excerpt : '',
1708            'description' => isset( $media_item->post_content ) ? $media_item->post_content : '',
1709            'alt'         => isset( $media_item->ID ) ? get_post_meta( $media_item->ID, '_wp_attachment_image_alt', true ) : '',
1710            'icon'        => isset( $media_item->ID ) ? wp_mime_type_icon( $media_item->ID ) : null,
1711            'size'        => size_format( (int) $filesize, 2 ),
1712            'thumbnails'  => array(),
1713        );
1714
1715        if ( in_array( $ext, array( 'jpg', 'jpeg', 'png', 'gif', 'webp' ), true ) && isset( $media_item->ID ) ) {
1716            $metadata = wp_get_attachment_metadata( $media_item->ID );
1717            if ( isset( $metadata['height'] ) ) {
1718                $response['height'] = $metadata['height'];
1719            }
1720            if ( isset( $metadata['width'] ) ) {
1721                $response['width'] = $metadata['width'];
1722            }
1723
1724            if ( isset( $metadata['sizes'] ) ) {
1725                /**
1726                 * Filter the thumbnail sizes available for each attachment ID.
1727                 *
1728                 * @module json-api
1729                 *
1730                 * @since 3.9.0
1731                 *
1732                 * @param array $metadata['sizes'] Array of thumbnail sizes available for a given attachment ID.
1733                 * @param string $media_id Attachment ID.
1734                 */
1735                $sizes = apply_filters( 'rest_api_thumbnail_sizes', $metadata['sizes'], $media_item->ID );
1736                if ( is_array( $sizes ) ) {
1737                    foreach ( $sizes as $size => $size_details ) {
1738                        $response['thumbnails'][ $size ] = dirname( $response['URL'] ) . '/' . $size_details['file'];
1739                    }
1740                    /**
1741                     * Filter the thumbnail URLs for attachment files.
1742                     *
1743                     * @module json-api
1744                     *
1745                     * @since 7.1.0
1746                     *
1747                     * @param array $metadata['sizes'] Array with thumbnail sizes as keys and URLs as values.
1748                     */
1749                    $response['thumbnails'] = apply_filters( 'rest_api_thumbnail_size_urls', $response['thumbnails'] );
1750                }
1751            }
1752
1753            if ( isset( $metadata['image_meta'] ) ) {
1754                $response['exif'] = $metadata['image_meta'];
1755            }
1756        }
1757
1758        if ( in_array( $ext, array( 'mp3', 'm4a', 'wav', 'ogg' ), true ) && isset( $media_item->ID ) ) {
1759            $metadata           = wp_get_attachment_metadata( $media_item->ID );
1760            $response['length'] = $metadata['length'];
1761            $response['exif']   = $metadata;
1762        }
1763
1764        $is_video = false;
1765
1766        if (
1767            in_array( $ext, array( 'ogv', 'mp4', 'mov', 'wmv', 'avi', 'mpg', '3gp', '3g2', 'm4v' ), true )
1768            || 'video/videopress' === $response['mime_type']
1769        ) {
1770            $is_video = true;
1771        }
1772
1773        if ( $is_video && isset( $media_item->ID ) ) {
1774            $metadata = wp_get_attachment_metadata( $media_item->ID );
1775
1776            if ( isset( $metadata['height'] ) ) {
1777                $response['height'] = $metadata['height'];
1778            }
1779            if ( isset( $metadata['width'] ) ) {
1780                $response['width'] = $metadata['width'];
1781            }
1782
1783            if ( isset( $metadata['length'] ) ) {
1784                $response['length'] = $metadata['length'];
1785            }
1786
1787            if ( empty( $response['length'] ) && isset( $metadata['duration'] ) ) {
1788                $response['length'] = (int) $metadata['duration'];
1789            }
1790
1791            if ( empty( $response['length'] ) && isset( $metadata['videopress']['duration'] ) ) {
1792                $response['length'] = ceil( $metadata['videopress']['duration'] / 1000 );
1793            }
1794
1795            // add VideoPress info.
1796            if ( function_exists( 'video_get_info_by_blogpostid' ) ) {
1797                $info = video_get_info_by_blogpostid( $this->api->get_blog_id_for_output(), $media_item->ID );
1798
1799                // If we failed to get VideoPress info, but it exists in the meta data (for some reason)
1800                // then let's use that.
1801                if ( false === $info && isset( $metadata['videopress'] ) ) {
1802                    $info = (object) $metadata['videopress'];
1803                }
1804
1805                if ( isset( $info->rating ) ) {
1806                    $response['rating'] = $info->rating;
1807                }
1808
1809                if ( isset( $info->display_embed ) ) {
1810                    $response['display_embed'] = (string) (int) $info->display_embed;
1811                    // If not, default to metadata (for WPCOM).
1812                } elseif ( isset( $metadata['videopress']['display_embed'] ) ) {
1813                    // We convert it to int then to string so that (bool) false to become "0".
1814                    $response['display_embed'] = (string) (int) $metadata['videopress']['display_embed'];
1815                }
1816
1817                if ( isset( $info->allow_download ) ) {
1818                    $response['allow_download'] = (string) (int) $info->allow_download;
1819                } elseif ( isset( $metadata['videopress']['allow_download'] ) ) {
1820                    // We convert it to int then to string so that (bool) false to become "0".
1821                    $response['allow_download'] = (string) (int) $metadata['videopress']['allow_download'];
1822                }
1823
1824                if ( isset( $info->thumbnail_generating ) ) {
1825                    $response['thumbnail_generating'] = (bool) intval( $info->thumbnail_generating );
1826                } elseif ( isset( $metadata['videopress']['thumbnail_generating'] ) ) {
1827                    $response['thumbnail_generating'] = (bool) intval( $metadata['videopress']['thumbnail_generating'] );
1828                }
1829
1830                if ( isset( $info->privacy_setting ) ) {
1831                    $response['privacy_setting'] = (int) $info->privacy_setting;
1832                } elseif ( isset( $metadata['videopress']['privacy_setting'] ) ) {
1833                    $response['privacy_setting'] = (int) $metadata['videopress']['privacy_setting'];
1834                }
1835
1836                $thumbnail_query_data = array();
1837                if ( ! empty( $info ) && function_exists( 'video_is_private' ) && video_is_private( $info ) ) {
1838                    $thumbnail_query_data['metadata_token'] = video_generate_auth_token( $info );
1839                }
1840
1841                // Thumbnails.
1842                if ( function_exists( 'video_format_done' ) && function_exists( 'video_image_url_by_guid' ) ) {
1843                    $response['thumbnails'] = array(
1844                        'fmt_hd'  => '',
1845                        'fmt_dvd' => '',
1846                        'fmt_std' => '',
1847                    );
1848                    foreach ( $response['thumbnails'] as $size => $thumbnail_url ) {
1849                        if ( video_format_done( $info, $size ) ) {
1850                            $response['thumbnails'][ $size ] = \add_query_arg( $thumbnail_query_data, \video_image_url_by_guid( $info->guid, $size ) );
1851                        } else {
1852                            unset( $response['thumbnails'][ $size ] );
1853                        }
1854                    }
1855                }
1856
1857                if ( isset( $info->title ) ) {
1858                    $response['title'] = $info->title;
1859                }
1860
1861                // If we didn't get VideoPress information (for some reason) then let's
1862                // not try and include it in the response.
1863                if ( isset( $info->guid ) ) {
1864                    $response['videopress_guid']            = $info->guid;
1865                    $response['videopress_processing_done'] = isset( $info->finish_date_gmt ) && '0000-00-00 00:00:00' !== $info->finish_date_gmt;
1866                }
1867            }
1868        }
1869
1870        $response['thumbnails'] = (object) $response['thumbnails'];
1871
1872        $response['meta'] = (object) array(
1873            'links' => (object) array(
1874                'self' => isset( $media_item->ID ) ? (string) $this->links->get_media_link( $this->api->get_blog_id_for_output(), $media_item->ID ) : null,
1875                'help' => isset( $media_item->ID ) ? (string) $this->links->get_media_link( $this->api->get_blog_id_for_output(), $media_item->ID, 'help' ) : null,
1876                'site' => (string) $this->links->get_site_link( $this->api->get_blog_id_for_output() ),
1877            ),
1878        );
1879
1880        // add VideoPress link to the meta.
1881        if ( isset( $response['videopress_guid'] ) ) {
1882            if ( function_exists( 'video_get_info_by_blogpostid' ) ) {
1883                $response['meta']->links->videopress = (string) $this->links->get_link( '/videos/%s', $response['videopress_guid'], '' );
1884            }
1885        }
1886
1887        if ( isset( $media_item->post_parent ) && $media_item->post_parent > 0 ) {
1888            $response['meta']->links->parent = (string) $this->links->get_post_link( $this->api->get_blog_id_for_output(), $media_item->post_parent );
1889        }
1890
1891        return (object) $response;
1892    }
1893
1894    /**
1895     * Get a formatted taxonomy.
1896     *
1897     * @param int    $taxonomy_id Taxonomy ID.
1898     * @param string $taxonomy_type Name of taxonomy.
1899     * @param string $context Context, 'edit' or 'display'.
1900     * @return object|WP_Error
1901     */
1902    public function get_taxonomy( $taxonomy_id, $taxonomy_type, $context ) {
1903
1904        $taxonomy = get_term_by( 'slug', $taxonomy_id, $taxonomy_type );
1905        // keep updating this function.
1906        if ( ! $taxonomy || is_wp_error( $taxonomy ) ) {
1907            return new WP_Error( 'unknown_taxonomy', 'Unknown taxonomy', 404 );
1908        }
1909
1910        return $this->format_taxonomy( $taxonomy, $taxonomy_type, $context );
1911    }
1912
1913    /**
1914     * Format a taxonomy.
1915     *
1916     * @param WP_Term $taxonomy Taxonomy.
1917     * @param string  $taxonomy_type Name of taxonomy.
1918     * @param string  $context Context, 'edit' or 'display'.
1919     * @return object|WP_Error
1920     */
1921    public function format_taxonomy( $taxonomy, $taxonomy_type, $context ) {
1922        // Permissions.
1923        switch ( $context ) {
1924            case 'edit':
1925                $tax = get_taxonomy( $taxonomy_type );
1926                if ( ! current_user_can( $tax->cap->edit_terms ) ) {
1927                    return new WP_Error( 'unauthorized', 'User cannot edit taxonomy', 403 );
1928                }
1929                break;
1930            case 'display':
1931                if ( ( new Status() )->is_private_site() && ! current_user_can( 'read' ) ) {
1932                    return new WP_Error( 'unauthorized', 'User cannot view taxonomy', 403 );
1933                }
1934                break;
1935            default:
1936                return new WP_Error( 'invalid_context', 'Invalid API CONTEXT', 400 );
1937        }
1938
1939        $response                = array();
1940        $response['ID']          = (int) $taxonomy->term_id;
1941        $response['name']        = (string) $taxonomy->name;
1942        $response['slug']        = (string) $taxonomy->slug;
1943        $response['description'] = (string) $taxonomy->description;
1944        $response['post_count']  = (int) $taxonomy->count;
1945        $response['feed_url']    = get_term_feed_link( $taxonomy->term_id, $taxonomy_type );
1946
1947        if ( is_taxonomy_hierarchical( $taxonomy_type ) ) {
1948            $response['parent'] = (int) $taxonomy->parent;
1949        }
1950
1951        $response['meta'] = (object) array(
1952            'links' => (object) array(
1953                'self' => (string) $this->links->get_taxonomy_link( $this->api->get_blog_id_for_output(), $taxonomy->slug, $taxonomy_type ),
1954                'help' => (string) $this->links->get_taxonomy_link( $this->api->get_blog_id_for_output(), $taxonomy->slug, $taxonomy_type, 'help' ),
1955                'site' => (string) $this->links->get_site_link( $this->api->get_blog_id_for_output() ),
1956            ),
1957        );
1958
1959        return (object) $response;
1960    }
1961
1962    /**
1963     * Returns ISO 8601 formatted datetime: 2011-12-08T01:15:36-08:00
1964     *
1965     * @param string $date_gmt GMT datetime string.
1966     * @param string $date Optional. Used to calculate the offset from GMT.
1967     * @return string
1968     */
1969    public function format_date( $date_gmt, $date = null ) {
1970        return WPCOM_JSON_API_Date::format_date( $date_gmt, $date );
1971    }
1972
1973    /**
1974     * Parses a date string and returns the local and GMT representations
1975     * of that date & time in 'YYYY-MM-DD HH:MM:SS' format without
1976     * timezones or offsets. If the parsed datetime was not localized to a
1977     * particular timezone or offset we will assume it was given in GMT
1978     * relative to now and will convert it to local time using either the
1979     * timezone set in the options table for the blog or the GMT offset.
1980     *
1981     * @param string $date_string Date to parse.
1982     *
1983     * @return array{string,string} ( $local_time_string, $gmt_time_string )
1984     */
1985    public function parse_date( $date_string ) {
1986        $date_string_info = date_parse( $date_string );
1987        if ( 0 === $date_string_info['error_count'] ) {
1988            // Check if it's already localized. Can't just check is_localtime because date_parse('oppossum') returns true; WTF, PHP.
1989            if ( isset( $date_string_info['zone'] ) && true === $date_string_info['is_localtime'] ) {
1990                $dt_utc   = new DateTime( $date_string );
1991                $dt_local = clone $dt_utc;
1992                $dt_utc->setTimezone( new DateTimeZone( 'UTC' ) );
1993                return array(
1994                    $dt_local->format( 'Y-m-d H:i:s' ),
1995                    $dt_utc->format( 'Y-m-d H:i:s' ),
1996                );
1997            }
1998
1999            // It's parseable but no TZ info so assume UTC.
2000            $dt_utc   = new DateTime( $date_string, new DateTimeZone( 'UTC' ) );
2001            $dt_local = clone $dt_utc;
2002        } else {
2003            // Could not parse time, use now in UTC.
2004            $dt_utc   = new DateTime( 'now', new DateTimeZone( 'UTC' ) );
2005            $dt_local = clone $dt_utc;
2006        }
2007
2008        $dt_local->setTimezone( wp_timezone() );
2009
2010        return array(
2011            $dt_local->format( 'Y-m-d H:i:s' ),
2012            $dt_utc->format( 'Y-m-d H:i:s' ),
2013        );
2014    }
2015
2016    /**
2017     * Load the functions.php file for the current theme to get its post formats, CPTs, etc.
2018     */
2019    public function load_theme_functions() {
2020        if ( false === defined( 'STYLESHEETPATH' ) ) {
2021            wp_templating_constants();
2022        }
2023
2024        // bail if we've done this already (can happen when calling /batch endpoint).
2025        if ( defined( 'REST_API_THEME_FUNCTIONS_LOADED' ) ) {
2026            return;
2027        }
2028
2029        // VIP context loading is handled elsewhere, so bail to prevent
2030        // duplicate loading. See `switch_to_blog_and_validate_user()`.
2031        if ( defined( 'WPCOM_IS_VIP_ENV' ) && WPCOM_IS_VIP_ENV ) {
2032            return;
2033        }
2034
2035        $do_check_theme =
2036            defined( 'REST_API_TEST_REQUEST' ) && REST_API_TEST_REQUEST ||
2037            defined( 'IS_WPCOM' ) && IS_WPCOM;
2038
2039        if ( $do_check_theme && ! wpcom_should_load_theme_files_on_rest_api() ) {
2040            return;
2041        }
2042
2043        define( 'REST_API_THEME_FUNCTIONS_LOADED', true );
2044
2045        // the theme info we care about is found either within functions.php or one of the jetpack files.
2046        $function_files = array( '/functions.php', '/inc/jetpack.compat.php', '/inc/jetpack.php', '/includes/jetpack.compat.php' );
2047
2048        $copy_dirs = array( get_template_directory() );
2049
2050        // Is this a child theme? Load the child theme's functions file.
2051        if ( get_stylesheet_directory() !== get_template_directory() && wpcom_is_child_theme() ) {
2052            foreach ( $function_files as $function_file ) {
2053                if ( file_exists( get_stylesheet_directory() . $function_file ) ) {
2054                    require_once get_stylesheet_directory() . $function_file;
2055                }
2056            }
2057            $copy_dirs[] = get_stylesheet_directory();
2058        }
2059
2060        foreach ( $function_files as $function_file ) {
2061            if ( file_exists( get_template_directory() . $function_file ) ) {
2062                require_once get_template_directory() . $function_file;
2063            }
2064        }
2065
2066        // add inc/wpcom.php and/or includes/wpcom.php.
2067        wpcom_load_theme_compat_file();
2068
2069        // Enable including additional directories or files in actions to be copied.
2070        $copy_dirs = apply_filters( 'restapi_theme_action_copy_dirs', $copy_dirs );
2071
2072        // since the stuff we care about (CPTS, post formats, are usually on setup or init hooks, we want to load those).
2073        $this->copy_hooks( 'after_setup_theme', 'restapi_theme_after_setup_theme', $copy_dirs );
2074
2075        /**
2076         * Fires functions hooked onto `after_setup_theme` by the theme for the purpose of the REST API.
2077         *
2078         * The REST API does not load the theme when processing requests.
2079         * To enable theme-based functionality, the API will load the '/functions.php',
2080         * '/inc/jetpack.compat.php', '/inc/jetpack.php', '/includes/jetpack.compat.php files
2081         * of the theme (parent and child) and copy functions hooked onto 'after_setup_theme' within those files.
2082         *
2083         * @module json-api
2084         *
2085         * @since 3.2.0
2086         */
2087        do_action( 'restapi_theme_after_setup_theme' );
2088        $this->copy_hooks( 'init', 'restapi_theme_init', $copy_dirs );
2089
2090        /**
2091         * Fires functions hooked onto `init` by the theme for the purpose of the REST API.
2092         *
2093         * The REST API does not load the theme when processing requests.
2094         * To enable theme-based functionality, the API will load the '/functions.php',
2095         * '/inc/jetpack.compat.php', '/inc/jetpack.php', '/includes/jetpack.compat.php files
2096         * of the theme (parent and child) and copy functions hooked onto 'init' within those files.
2097         *
2098         * @module json-api
2099         *
2100         * @since 3.2.0
2101         */
2102        do_action( 'restapi_theme_init' );
2103    }
2104
2105    /**
2106     * Copy hook functions.
2107     *
2108     * @param string $from_hook Hook to copy from.
2109     * @param string $to_hook Hook to copy to.
2110     * @param array  $base_paths Only copy hooks defined in the specified paths.
2111     */
2112    public function copy_hooks( $from_hook, $to_hook, $base_paths ) {
2113        global $wp_filter;
2114        foreach ( $wp_filter as $hook => $actions ) {
2115
2116            if ( $from_hook !== $hook ) {
2117                continue;
2118            }
2119            if ( ! has_action( $hook ) ) {
2120                continue;
2121            }
2122
2123            foreach ( $actions as $priority => $callbacks ) {
2124                foreach ( $callbacks as $callback_data ) {
2125                    $callback = $callback_data['function'];
2126
2127                    // use reflection api to determine filename where function is defined.
2128                    $reflection = $this->get_reflection( $callback );
2129
2130                    if ( false !== $reflection ) {
2131                        $file_name = $reflection->getFileName();
2132                        foreach ( $base_paths as $base_path ) {
2133
2134                            // only copy hooks with functions which are part of the specified files.
2135                            if ( str_starts_with( $file_name, $base_path ) ) {
2136                                add_action(
2137                                    $to_hook,
2138                                    $callback_data['function'],
2139                                    $priority,
2140                                    $callback_data['accepted_args']
2141                                );
2142                            }
2143                        }
2144                    }
2145                }
2146            }
2147        }
2148    }
2149
2150    /**
2151     * Get a ReflectionMethod or ReflectionFunction for the callback.
2152     *
2153     * @param callable $callback Callback.
2154     * @return ReflectionMethod|ReflectionFunction|false
2155     */
2156    public function get_reflection( $callback ) {
2157        if ( is_array( $callback ) ) {
2158            list( $class, $method ) = $callback;
2159            return new ReflectionMethod( $class, $method );
2160        }
2161
2162        if ( is_string( $callback ) && strpos( $callback, '::' ) !== false ) {
2163            list( $class, $method ) = explode( '::', $callback );
2164            return new ReflectionMethod( $class, $method );
2165        }
2166
2167        if ( method_exists( $callback, '__invoke' ) ) {
2168            return new ReflectionMethod( $callback, '__invoke' );
2169        }
2170
2171        if ( is_string( $callback ) && strpos( $callback, '::' ) === false && function_exists( $callback ) ) {
2172            return new ReflectionFunction( $callback );
2173        }
2174
2175        return false;
2176    }
2177
2178    /**
2179     * Check whether a user can view or edit a post type.
2180     *
2181     * @param string $post_type post type to check.
2182     * @param string $context   'display' or 'edit'.
2183     * @return bool
2184     */
2185    public function current_user_can_access_post_type( $post_type, $context = 'display' ) {
2186        $post_type_object = get_post_type_object( $post_type );
2187        if ( ! $post_type_object ) {
2188            return false;
2189        }
2190
2191        switch ( $context ) {
2192            case 'edit':
2193                return current_user_can( $post_type_object->cap->edit_posts );
2194            case 'display':
2195                return $post_type_object->public || current_user_can( $post_type_object->cap->read_private_posts );
2196            default:
2197                return false;
2198        }
2199    }
2200
2201    /**
2202     * Is the post type allowed?
2203     *
2204     * @param string $post_type Post type.
2205     * @return bool
2206     */
2207    public function is_post_type_allowed( $post_type ) {
2208        // if the post type is empty, that's fine, WordPress will default to post.
2209        if ( empty( $post_type ) ) {
2210            return true;
2211        }
2212
2213        // allow special 'any' type.
2214        if ( 'any' === $post_type ) {
2215            return true;
2216        }
2217
2218        // check for allowed types.
2219        if ( in_array( $post_type, $this->_get_whitelisted_post_types(), true ) ) {
2220            return true;
2221        }
2222
2223        $post_type_object = get_post_type_object( $post_type );
2224        if ( $post_type_object ) {
2225            if ( ! empty( $post_type_object->show_in_rest ) ) {
2226                return $post_type_object->show_in_rest;
2227            }
2228            if ( ! empty( $post_type_object->publicly_queryable ) ) {
2229                return $post_type_object->publicly_queryable;
2230            }
2231        }
2232
2233        return ! empty( $post_type_object->public );
2234    }
2235
2236    /**
2237     * Gets the whitelisted post types that JP should allow access to.
2238     *
2239     * @return array Whitelisted post types.
2240     */
2241    protected function _get_whitelisted_post_types() { // phpcs:ignore PSR2.Methods.MethodDeclaration.Underscore -- Legacy.
2242        $allowed_types = array( 'post', 'page', 'revision' );
2243
2244        /**
2245         * Filter the post types Jetpack has access to, and can synchronize with WordPress.com.
2246         *
2247         * @module json-api
2248         *
2249         * @since 2.2.3
2250         *
2251         * @param array $allowed_types Array of whitelisted post types. Default to `array( 'post', 'page', 'revision' )`.
2252         */
2253        $allowed_types = apply_filters( 'rest_api_allowed_post_types', $allowed_types );
2254
2255        return array_unique( $allowed_types );
2256    }
2257
2258    /**
2259     * Mobile apps are allowed free video uploads, but limited to 5 minutes in length.
2260     *
2261     * @param array $media_item the media item to evaluate.
2262     *
2263     * @return bool true if the media item is a video that was uploaded via the mobile
2264     * app that is longer than 5 minutes.
2265     */
2266    public function media_item_is_free_video_mobile_upload_and_too_long( $media_item ) {
2267        if ( ! $media_item ) {
2268            return false;
2269        }
2270
2271        // Verify file is a video.
2272        $is_video = preg_match( '@^video/@', $media_item['type'] );
2273        if ( ! $is_video ) {
2274            return false;
2275        }
2276
2277        // Check if the request is from a mobile app, where we allow free video uploads at limited length.
2278        if ( ! in_array( $this->api->token_details['client_id'], VIDEOPRESS_ALLOWED_REST_API_CLIENT_IDS, true ) ) {
2279            return false;
2280        }
2281
2282        // We're only worried about free sites.
2283        require_once WP_CONTENT_DIR . '/admin-plugins/wpcom-billing.php';
2284        $current_plan = WPCOM_Store_API::get_current_plan( get_current_blog_id() );
2285        if ( ! $current_plan['is_free'] ) {
2286            return false;
2287        }
2288
2289        // We don't know if this is an upload or a sideload, but in either case the tmp_name should be a path, not a URL.
2290        if ( wp_parse_url( $media_item['tmp_name'], PHP_URL_SCHEME ) !== null ) {
2291            return false;
2292        }
2293
2294        // Check if video is longer than 5 minutes.
2295        $video_meta = wp_read_video_metadata( $media_item['tmp_name'] );
2296        if (
2297            false !== $video_meta &&
2298            isset( $video_meta['length'] ) &&
2299            5 * MINUTE_IN_SECONDS < $video_meta['length']
2300        ) {
2301            videopress_log(
2302                'videopress_app_upload_length_block',
2303                'Mobile app upload on free site blocked because length was longer than 5 minutes.',
2304                null,
2305                null,
2306                null,
2307                null,
2308                array(
2309                    'blog_id' => get_current_blog_id(),
2310                    'user_id' => get_current_user_id(),
2311                )
2312            );
2313            return true;
2314        }
2315
2316        return false;
2317    }
2318
2319    /**
2320     * Handle a v1.1 media creation.
2321     *
2322     * Only one of $media_files and $media_urls should be non-empty.
2323     *
2324     * @param array     $media_files File upload data.
2325     * @param array     $media_urls URLs to fetch.
2326     * @param array     $media_attrs Attributes corresponding to each entry in `$media_files`/`$media_urls`.
2327     * @param int|false $force_parent_id Force the parent ID, overriding `$media_attrs[]['parent_id']`.
2328     * @return array Two items:
2329     *  - media_ids: IDs created, by index in `$media_files`/`$media_urls`.
2330     *  - errors: Errors encountered, by index in `$media_files`/`$media_urls`.
2331     */
2332    public function handle_media_creation_v1_1( $media_files, $media_urls, $media_attrs = array(), $force_parent_id = false ) {
2333
2334        add_filter( 'upload_mimes', array( $this, 'allow_video_uploads' ) );
2335
2336        $media_ids             = array();
2337        $errors                = array();
2338        $user_can_upload_files = current_user_can( 'upload_files' ) || $this->api->is_authorized_with_upload_token();
2339        $media_attrs           = array_values( $media_attrs ); // reset the keys.
2340        $i                     = 0;
2341
2342        if ( ! empty( $media_files ) ) {
2343            $this->api->trap_wp_die( 'upload_error' );
2344            foreach ( $media_files as $media_item ) {
2345                $_FILES['.api.media.item.'] = $media_item;
2346
2347                if ( ! $user_can_upload_files ) {
2348                    $media_id = new WP_Error( 'unauthorized', 'User cannot upload media.', 403 );
2349                } elseif ( ! is_array( $media_item ) ) {
2350                    $media_id   = new WP_Error( 'invalid_input', 'Unable to process request.', 400 );
2351                    $media_item = array(
2352                        'name' => 'invalid_file',
2353                    );
2354                } elseif ( $this->media_item_is_free_video_mobile_upload_and_too_long( $media_item ) ) {
2355                    $media_id = new WP_Error( 'upload_video_length', 'Video uploads longer than 5 minutes require a paid plan.', 400 );
2356                } else {
2357                    if ( $force_parent_id ) {
2358                        $parent_id = absint( $force_parent_id );
2359                    } elseif ( ! empty( $media_attrs[ $i ] ) && ! empty( $media_attrs[ $i ]['parent_id'] ) ) {
2360                        $parent_id = absint( $media_attrs[ $i ]['parent_id'] );
2361                    } else {
2362                        $parent_id = 0;
2363                    }
2364                    $media_id = media_handle_upload( '.api.media.item.', $parent_id );
2365                }
2366                if ( is_wp_error( $media_id ) ) {
2367                    $errors[ $i ]['file']    = $media_item['name'];
2368                    $errors[ $i ]['error']   = $media_id->get_error_code();
2369                    $errors[ $i ]['message'] = $media_id->get_error_message();
2370                } else {
2371                    $media_ids[ $i ] = $media_id;
2372                }
2373
2374                ++$i;
2375            }
2376            $this->api->trap_wp_die( null );
2377            unset( $_FILES['.api.media.item.'] );
2378        }
2379
2380        if ( ! empty( $media_urls ) ) {
2381            foreach ( $media_urls as $url ) {
2382                if ( ! $user_can_upload_files ) {
2383                    $media_id = new WP_Error( 'unauthorized', 'User cannot upload media.', 403 );
2384                } else {
2385                    if ( $force_parent_id ) {
2386                        $parent_id = absint( $force_parent_id );
2387                    } elseif ( ! empty( $media_attrs[ $i ] ) && ! empty( $media_attrs[ $i ]['parent_id'] ) ) {
2388                        $parent_id = absint( $media_attrs[ $i ]['parent_id'] );
2389                    } else {
2390                        $parent_id = 0;
2391                    }
2392                    $media_id = $this->handle_media_sideload( $url, $parent_id );
2393                }
2394                if ( is_wp_error( $media_id ) ) {
2395                    $errors[ $i ] = array(
2396                        'file'    => $url,
2397                        'error'   => $media_id->get_error_code(),
2398                        'message' => $media_id->get_error_message(),
2399                    );
2400                } elseif ( ! empty( $media_id ) ) {
2401                    $media_ids[ $i ] = $media_id;
2402                }
2403
2404                ++$i;
2405            }
2406        }
2407
2408        if ( ! empty( $media_attrs ) ) {
2409            foreach ( $media_ids as $index => $media_id ) {
2410                if ( empty( $media_attrs[ $index ] ) ) {
2411                    continue;
2412                }
2413
2414                $attrs  = $media_attrs[ $index ];
2415                $insert = array();
2416
2417                // Attributes: Title, Caption, Description.
2418
2419                if ( isset( $attrs['title'] ) ) {
2420                    $insert['post_title'] = $attrs['title'];
2421                }
2422
2423                if ( isset( $attrs['caption'] ) ) {
2424                    $insert['post_excerpt'] = $attrs['caption'];
2425                }
2426
2427                if ( isset( $attrs['description'] ) ) {
2428                    $insert['post_content'] = $attrs['description'];
2429                }
2430
2431                if ( ! empty( $insert ) ) {
2432                    $insert['ID'] = $media_id;
2433                    wp_update_post( (object) $insert );
2434                }
2435
2436                // Attributes: Alt.
2437
2438                if ( isset( $attrs['alt'] ) ) {
2439                    $alt = wp_strip_all_tags( $attrs['alt'], true );
2440                    update_post_meta( $media_id, '_wp_attachment_image_alt', $alt );
2441                }
2442
2443                // Attributes: Artist, Album.
2444
2445                $id3_meta = array();
2446
2447                foreach ( array( 'artist', 'album' ) as $key ) {
2448                    if ( isset( $attrs[ $key ] ) ) {
2449                        $id3_meta[ $key ] = wp_strip_all_tags( $attrs[ $key ], true );
2450                    }
2451                }
2452
2453                if ( ! empty( $id3_meta ) ) {
2454                    // Before updating metadata, ensure that the item is audio.
2455                    $item = $this->get_media_item_v1_1( $media_id );
2456                    if ( str_starts_with( $item->mime_type, 'audio/' ) ) {
2457                        wp_update_attachment_metadata( $media_id, $id3_meta );
2458                    }
2459                }
2460
2461                // Attributes: Meta
2462                if ( isset( $attrs['meta'] ) && isset( $attrs['meta']['vertical_id'] ) ) {
2463                    update_post_meta( $media_id, 'vertical_id', $attrs['meta']['vertical_id'] );
2464                }
2465            }
2466        }
2467
2468        return array(
2469            'media_ids' => $media_ids,
2470            'errors'    => $errors,
2471        );
2472    }
2473
2474    /**
2475     * Handle a media sideload.
2476     *
2477     * @param string $url URL.
2478     * @param int    $parent_post_id Parent post ID.
2479     * @param string $type Type.
2480     * @return int|WP_Error|false Media post ID, or error, or false if nothing was sideloaded.
2481     */
2482    public function handle_media_sideload( $url, $parent_post_id = 0, $type = 'any' ) {
2483        if ( ! function_exists( 'download_url' ) || ! function_exists( 'media_handle_sideload' ) ) {
2484            return false;
2485        }
2486
2487        // if we didn't get a URL, let's bail.
2488        $parsed = wp_parse_url( $url );
2489        if ( empty( $parsed ) ) {
2490            return false;
2491        }
2492
2493        $tmp = download_url( $url );
2494        if ( is_wp_error( $tmp ) ) {
2495            return $tmp;
2496        }
2497
2498        // First check to see if we get a mime-type match by file, otherwise, check to
2499        // see if WordPress supports this file as an image. If neither, then it is not supported.
2500        if ( ! $this->is_file_supported_for_sideloading( $tmp ) || 'image' === $type && ! file_is_displayable_image( $tmp ) ) {
2501            @unlink( $tmp ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged
2502            return new WP_Error( 'invalid_input', 'Invalid file type.', 403 );
2503        }
2504
2505        // emulate a $_FILES entry.
2506        $file_array = array(
2507            'name'     => basename( wp_parse_url( $url, PHP_URL_PATH ) ),
2508            'tmp_name' => $tmp,
2509        );
2510
2511        $id = media_handle_sideload( $file_array, $parent_post_id );
2512        if ( file_exists( $tmp ) ) {
2513            @unlink( $tmp ); // phpcs:ignore WordPress.PHP.NoSilencedErrors.Discouraged
2514        }
2515
2516        if ( is_wp_error( $id ) ) {
2517            return $id;
2518        }
2519
2520        if ( ! $id || ! is_int( $id ) ) {
2521            return false;
2522        }
2523
2524        return $id;
2525    }
2526
2527    /**
2528     * Checks that the mime type of the specified file is among those in a filterable list of mime types.
2529     *
2530     * @param string $file Path to file to get its mime type.
2531     *
2532     * @return bool
2533     */
2534    protected function is_file_supported_for_sideloading( $file ) {
2535        return jetpack_is_file_supported_for_sideloading( $file );
2536    }
2537
2538    /**
2539     * Filter for `upload_mimes`.
2540     *
2541     * @param array $mimes Allowed mime types.
2542     * @return array Allowed mime types.
2543     */
2544    public function allow_video_uploads( $mimes ) {
2545        // if we are on Jetpack, bail - Videos are already allowed.
2546        if ( ! defined( 'IS_WPCOM' ) || ! IS_WPCOM ) {
2547            return $mimes;
2548        }
2549
2550        // extra check that this filter is only ever applied during REST API requests.
2551        if ( ! defined( 'REST_API_REQUEST' ) || ! REST_API_REQUEST ) {
2552            return $mimes;
2553        }
2554
2555        // bail early if they already have video upload capability.
2556        if ( wpcom_site_can_upload_videos() ) {
2557            return $mimes;
2558        }
2559
2560        // lets whitelist to only specific clients right now.
2561        $clients_allowed_video_uploads = array();
2562        /**
2563         * Filter the list of whitelisted video clients.
2564         *
2565         * @module json-api
2566         *
2567         * @since 3.2.0
2568         *
2569         * @param array $clients_allowed_video_uploads Array of whitelisted Video clients.
2570         */
2571        $clients_allowed_video_uploads = apply_filters( 'rest_api_clients_allowed_video_uploads', $clients_allowed_video_uploads );
2572        if ( ! isset( $this->api->token_details['client_id'] ) || ! in_array( $this->api->token_details['client_id'], $clients_allowed_video_uploads, true ) ) {
2573            return $mimes;
2574        }
2575
2576        $mime_list = wp_get_mime_types();
2577
2578        $video_exts = explode( ' ', get_site_option( 'video_upload_filetypes', false, false ) );
2579        /**
2580         * Filter the video filetypes allowed on the site.
2581         *
2582         * @module json-api
2583         *
2584         * @since 3.2.0
2585         *
2586         * @param array $video_exts Array of video filetypes allowed on the site.
2587         */
2588        $video_exts  = apply_filters( 'video_upload_filetypes', $video_exts );
2589        $video_mimes = array();
2590
2591        if ( ! empty( $video_exts ) ) {
2592            foreach ( $video_exts as $ext ) {
2593                foreach ( $mime_list as $ext_pattern => $mime ) {
2594                    if ( '' !== $ext && str_contains( $ext_pattern, $ext ) ) {
2595                        $video_mimes[ $ext_pattern ] = $mime;
2596                    }
2597                }
2598            }
2599
2600            $mimes = array_merge( $mimes, $video_mimes );
2601        }
2602
2603        return $mimes;
2604    }
2605
2606    /**
2607     * Is the current site multi-user?
2608     *
2609     * @return bool
2610     */
2611    public function is_current_site_multi_user() {
2612        $users = wp_cache_get( 'site_user_count', 'WPCOM_JSON_API_Endpoint' );
2613        if ( false === $users ) {
2614            $user_query = new WP_User_Query(
2615                array(
2616                    'blog_id' => get_current_blog_id(),
2617                    'fields'  => 'ID',
2618                )
2619            );
2620            $users      = (int) $user_query->get_total();
2621            wp_cache_set( 'site_user_count', $users, 'WPCOM_JSON_API_Endpoint', DAY_IN_SECONDS );
2622        }
2623        return $users > 1;
2624    }
2625
2626    /**
2627     * Whether cross-origin requests are allowed.
2628     *
2629     * @return bool
2630     */
2631    public function allows_cross_origin_requests() {
2632        return 'GET' === $this->method || $this->allow_cross_origin_request;
2633    }
2634
2635    /**
2636     * Whether unauthorized requests are allowed.
2637     *
2638     * @param string   $origin Origin.
2639     * @param string[] $complete_access_origins Access origins.
2640     * @return bool
2641     */
2642    public function allows_unauthorized_requests( $origin, $complete_access_origins ) {
2643        return 'GET' === $this->method || ( $this->allow_unauthorized_request && in_array( $origin, $complete_access_origins, true ) );
2644    }
2645
2646    /**
2647     * Whether this endpoint accepts site based authentication for the current request.
2648     *
2649     * @since 9.1.0
2650     *
2651     * @return bool true, if Jetpack blog token is used and `allow_jetpack_site_auth` is true,
2652     * false otherwise.
2653     */
2654    public function accepts_site_based_authentication() {
2655        return $this->allow_jetpack_site_auth &&
2656            $this->api->is_jetpack_authorized_for_site();
2657    }
2658
2659    /**
2660     * Get platform.
2661     *
2662     * @return WPORG_Platform
2663     */
2664    public function get_platform() {
2665        return wpcom_get_sal_platform( $this->api->token_details );
2666    }
2667
2668    /**
2669     * Allows the endpoint to perform logic to allow it to decide whether-or-not it should force a
2670     * response from the WPCOM API, or potentially go to the Jetpack blog.
2671     *
2672     * Override this method if you want to do something different.
2673     *
2674     * @param int $blog_id Blog ID.
2675     * @return bool
2676     */
2677    public function force_wpcom_request( $blog_id ) { // phpcs:ignore VariableAnalysis.CodeAnalysis.VariableAnalysis.UnusedVariable
2678        return false;
2679    }
2680
2681    /**
2682     * Get an array of all valid AMP origins for a blog's siteurl.
2683     *
2684     * @param string $siteurl Origin url of the API request.
2685     * @return array
2686     */
2687    public function get_amp_cache_origins( $siteurl ) {
2688        $host = wp_parse_url( $siteurl, PHP_URL_HOST );
2689
2690        /*
2691         * From AMP docs:
2692         * "When possible, the Google AMP Cache will create a subdomain for each AMP document's domain by first converting it
2693         * from IDN (punycode) to UTF-8. The caches replaces every - (dash) with -- (2 dashes) and replace every . (dot) with
2694         * - (dash). For example, pub.com will map to pub-com.cdn.ampproject.org."
2695         */
2696        if ( function_exists( 'idn_to_utf8' ) ) {
2697            // The third parameter is set explicitly to prevent issues with newer PHP versions compiled with an old ICU version.
2698            $variant = defined( 'INTL_IDNA_VARIANT_UTS46' )
2699                ? INTL_IDNA_VARIANT_UTS46
2700                // phpcs:ignore PHPCompatibility.Constants.RemovedConstants.intl_idna_variant_2003Deprecated, PHPCompatibility.Constants.RemovedConstants.intl_idna_variant_2003DeprecatedRemoved
2701                : INTL_IDNA_VARIANT_2003; // @phan-suppress-current-line PhanUndeclaredConstant
2702            $host = idn_to_utf8( $host, IDNA_DEFAULT, $variant );
2703        }
2704        $subdomain = str_replace( array( '-', '.' ), array( '--', '-' ), $host );
2705        return array(
2706            $siteurl,
2707            // Google AMP Cache (legacy).
2708            'https://cdn.ampproject.org',
2709            // Google AMP Cache subdomain.
2710            sprintf( 'https://%s.cdn.ampproject.org', $subdomain ),
2711            // Cloudflare AMP Cache.
2712            sprintf( 'https://%s.amp.cloudflare.com', $subdomain ),
2713            // Bing AMP Cache.
2714            sprintf( 'https://%s.bing-amp.com', $subdomain ),
2715        );
2716    }
2717
2718    /**
2719     * Register a REST route for this jsonAPI endpoint.
2720     *
2721     * @return void
2722     * @throws Exception The exception if something goes wrong.
2723     */
2724    public function create_rest_route_for_endpoint() {
2725        register_rest_route(
2726            static::REST_NAMESPACE,
2727            $this->build_rest_route(),
2728            array(
2729                'methods'             => $this->method,
2730                'callback'            => array( $this, 'rest_callback' ),
2731                'permission_callback' => array( $this, 'rest_permission_callback' ),
2732            )
2733        );
2734    }
2735
2736    /**
2737     * Handle the rest call.
2738     *
2739     * @param WP_REST_Request $request The request object.
2740     *
2741     * @return mixed|WP_Error
2742     */
2743    public function rest_callback( WP_REST_Request $request ) {
2744        // phpcs:ignore WordPress.PHP.IniSet.display_errors_Disallowed -- Making sure random warnings don't break JSON.
2745        ini_set( 'display_errors', false );
2746
2747        $blog_id = Jetpack_Options::get_option( 'id' );
2748
2749        add_filter( 'user_can_richedit', '__return_true' );
2750        add_filter( 'comment_edit_pre', array( $this->api, 'comment_edit_pre' ) );
2751
2752        $this->api->initialize();
2753        $this->api->endpoint = $this;
2754
2755        $this->api->path    = $this->path;
2756        $this->api->version = $this->max_version;
2757
2758        $locale = $request->get_param( 'language' );
2759        if ( $locale ) {
2760            $this->api->init_locale( $locale );
2761        }
2762
2763        if ( $this->in_testing && ! WPCOM_JSON_API__DEBUG ) {
2764            return new WP_Error( 'endpoint_not_available' );
2765        }
2766
2767        $token_data = ( new Manager() )->verify_xml_rpc_signature();
2768        if ( ! $token_data || empty( $token_data['token_key'] ) || ! array_key_exists( 'user_id', $token_data ) ) {
2769            return new WP_Error( 'response_signature_error' );
2770        }
2771
2772        $token = ( new Tokens() )->get_access_token( $token_data['user_id'], $token_data['token_key'] );
2773        if ( is_wp_error( $token ) ) {
2774            return $token;
2775        }
2776        if ( ! $token ) {
2777            return new WP_Error( 'response_signature_error' );
2778        }
2779
2780        /** This action is documented in class.json-api.php */
2781        do_action( 'wpcom_json_api_output', $this->stat );
2782
2783        $response = call_user_func_array(
2784            array( $this, 'callback' ),
2785            array_values( array( $this->path, $blog_id ) + $request->get_url_params() )
2786        );
2787
2788        if ( ! $response && ! is_array( $response ) ) {
2789            // Dealing with empty non-array response.
2790            $response = new WP_Error( 'empty_response', 'Endpoint response is empty', 500 );
2791        }
2792
2793        $status_code = 200;
2794
2795        if ( is_wp_error( $response ) ) {
2796            $status_code = 500;
2797
2798            if ( $response->get_error_data() && is_scalar( $response->get_error_data() )
2799                && (string) (int) $response->get_error_data() === (string) $response->get_error_data()
2800            ) {
2801                $status_code = (int) $response->get_error_data();
2802            }
2803
2804            $response = WPCOM_JSON_API::serializable_error( $response );
2805        }
2806
2807        if ( $request->get_param( 'http_envelope' ) ) {
2808            $response = WPCOM_JSON_API::wrap_http_envelope( $status_code, $response, 'application/json' );
2809        }
2810
2811        $response = wp_json_encode( $response, JSON_UNESCAPED_SLASHES );
2812
2813        $nonce = wp_generate_password( 10, false );
2814        $hmac  = hash_hmac( 'sha1', $nonce . $response, $token->secret );
2815
2816        return array(
2817            $response,
2818            (string) $nonce,
2819            $hmac,
2820        );
2821    }
2822
2823    /**
2824     * The REST endpoint should only be available for requests signed with a valid blog or user token.
2825     * Declaring it "final" so individual endpoints couldn't remove this requirement.
2826     *
2827     * If you need to add custom permissions to individual endpoints, you can override method `rest_permission_callback_custom()`.
2828     *
2829     * @see self::rest_permission_callback_custom()
2830     *
2831     * @return true|WP_Error
2832     */
2833    final public function rest_permission_callback() {
2834        $manager = new Manager( 'jetpack' );
2835        if ( ! $manager->is_connected() ) {
2836            return new WP_Error( 'site_not_connected' );
2837        }
2838
2839        if ( ( ( $this->allow_jetpack_site_auth || $this->allow_fallback_to_jetpack_blog_token ) && Rest_Authentication::is_signed_with_blog_token() )
2840            || ( get_current_user_id() && Rest_Authentication::is_signed_with_user_token() )
2841        ) {
2842            $custom_permission_result = $this->rest_permission_callback_custom();
2843
2844            // Successful custom permission check.
2845            if ( $custom_permission_result === true ) {
2846                return true;
2847            }
2848
2849            // Custom permission check errored, returning the error.
2850            if ( is_wp_error( $custom_permission_result ) ) {
2851                return $custom_permission_result;
2852            }
2853
2854            // Custom permission check failed, but didn't return a specific error. Proceed to returning the generic error.
2855        }
2856
2857        $message = esc_html__(
2858            'You do not have the correct user permissions to perform this action. Please contact your site admin if you think this is a mistake.',
2859            'jetpack'
2860        );
2861        return new WP_Error( 'rest_api_invalid_permission', $message, array( 'status' => rest_authorization_required_code() ) );
2862    }
2863
2864    /**
2865     * You can override this method in individual endpoints to add custom permission checks.
2866     * This will run on top of `rest_permission_callback()`.
2867     *
2868     * @see self::rest_permission_callback()
2869     *
2870     * @return true|WP_Error
2871     */
2872    public function rest_permission_callback_custom() {
2873        return true;
2874    }
2875
2876    /**
2877     * Build the REST endpoint URL.
2878     *
2879     * @return string
2880     */
2881    public function build_rest_route() {
2882        $version_prefix = $this->max_version ? 'v' . $this->max_version : '';
2883        return $version_prefix . $this->rest_route;
2884    }
2885
2886    /**
2887     * Get Jetpack Version where support for the endpoint was introduced.
2888     *
2889     * @return string
2890     */
2891    public function get_rest_min_jp_version() {
2892        return $this->rest_min_jp_version;
2893    }
2894
2895    /**
2896     * Return endpoint response
2897     *
2898     * @param string $path ... determined by ->$path.
2899     *
2900     * @return array|WP_Error
2901     *  falsy: HTTP 500, no response body
2902     *  WP_Error( $error_code, $error_message, $http_status_code ): HTTP $status_code, json_encode( array( 'error' => $error_code, 'message' => $error_message ) ) response body
2903     *  $data: HTTP 200, json_encode( $data ) response body
2904     */
2905    abstract public function callback( $path = '' );
2906}
2907
2908require_once __DIR__ . '/json-endpoints.php';