WP REST API: Internals and Customization

In the previous part of the series, we learned about creating, updating, and deleting content remotely through the WP REST API. It allows us to create platform-independent applications that work seamlessly with a WordPress powered back-end, providing a rich experience to the user.

In the current part of the series, we will take a look at the internals of the WP REST API and how they work together to power the API. After that, we will learn to modify server responses for the default endpoints to include custom fields.

To be specific, in the current article, we will:

  • learn about the internal classes and methods of the WP REST API
  • modify the server response for default endpoints
  • learn how to make custom fields editable

So let’s begin by taking a look at the internals of the WP REST API.

Internal Classes and Methods of the WP REST API

Classes in the WP REST API can be divided into the following two categories:

  1. Infrastructure classes: These are the fundamental classes responsible for holding together the API. They don’t perform any data transformation.
  2. Endpoint classes: These classes are responsible for performing CRUD operations on resources like posts, pages, users, comments, etc.

Let’s take a look at the individual classes of each of the above two categories.

Infrastructure Classes

The three infrastructure classes that together power the REST API are as follows:

  1. WP_REST_Server
  2. WP_REST_Request
  3. WP_REST_Response

WP_REST_Server

This is the core class of the WP REST API which implements the REST server by registering routes, serving requests, and preparing responses. It formats the data to be passed to the client and in case of an error, it prepares the error by including the error code and message body. It also checks for authentication.

We have been working quite a lot with the /wp-json index endpoint for checking all the capabilities and supported routes for a site. The method get_index(), which is responsible for retrieving the site index, is also located in this class.

For serving requests and preparing responses, the WP_REST_Server class uses the WP_REST_Request and WP_REST_Response classes respectively.

WP_REST_Request

The WP_REST_Request class implements the request object for the WP REST API. It contains data from the request like headers and request body, and is passed to the callback function by the WP_REST_Server class. It also checks if the parameters that are passed along the request are valid and performs data sanitization when necessary.

WP_REST_Response

The WP_REST_Response class, as the name implies, implements the response object. It contains necessary data such as response status code and response body.

Let’s now take a look at the endpoint classes.

Endpoint Classes

The endpoint classes in the WP REST API are responsible for performing CRUD operations. These classes include WP_REST_Posts_Controller, WP_REST_Taxonomies_ControllerWP_REST_Users_Controller, etc. All of these endpoint classes extend a single abstract class WP_REST_Controller that provides a consistent pattern for modifying data.

The WP_REST_Controller class includes methods such as get_item(), create_item(), update_item()delete_item(), etc., for performing CRUD operations. These methods must be overridden by sub-classes by implementing their own abstraction for retrieving, creating, updating, and modifying data.

You can find more about these classes and their internal methods in the official documentation.

Having learned about the internal classes of the WP REST API, let’s take a look at how we can modify server responses for default endpoints to include custom fields.

Modifying Server Responses

In the previous section, we looked at the internal classes and methods that the API is built upon. Together these classes and methods drive the API as a whole and provide a way for developers to extend the API to account for different scenarios and use cases.

The WP REST API exposes data in a predictable manner. This includes various resources like posts, post meta, pages, and users, along with their standard properties. But this data can’t always conform to the needs of every single WordPress site or user. Therefore, the WP REST API provides a way of modifying the data that the server returns for each of the default routes.

The register_rest_field() method provides a way to add or update fields in the response for an object. However, changing a field from a response is never encouraged by the API since it might introduce compatibility issues for clients that expect a standard response from the server. So, if you need to change a field, you should consider duplicating the field with the desired value.

Similarly, deleting a default field is highly discouraged for the reason that a client might be expecting it. If you need a smaller subset of the response returned by the server, you should create additional contexts in addition to the default contexts like view or edit.

We can, however, safely add a field to the response returned by the server for one or multiple objects. These fields can contain any value ranging from post or user meta to any other arbitrary value.

In the next section, we will be working with the register_rest_field() method to add custom fields to the response returned by the server for the post object.

Working With the register_rest_field() Method

As mentioned previously, the register_rest_field() method can be used to add or update fields in the response returned by the server. This method accepts three arguments:

  1. $object_type
  2. $attribute
  3. $args

The $object_type argument can either be a string or an array containing the names of all the objects that we want to add the field for. These objects can be post, term, commentuser, etc. If we need to add a custom field to a custom post type, then the $object_type argument would be the name of the post type.

The $attribute argument is the name of the custom field. This name would appear in the server response as a key along with its value.

The $args array is an associative array that can contain the following three keys:

  1. $get_callback
  2. $update_callback
  3. $schema

The values of the first two keys are the names of the methods that are used to get or update the value of the custom field. The last $schema key defines the method or the variable that is used to define the schema for the custom field.

All of the above keys are optional, but if they are not added, the capability will not be added. For instance, if you define the $get_callback key but not the $update_callback key, the retrieval functionality will be added but the update functionality won’t be added. If you omit the $get_callback key, the field won’t be added to the response at all.

The register_rest_field() method works by modifying the $wp_rest_additional_fields variable. This array variable holds registered fields by object types to be returned in the response by the server. Whenever a field is registered by the register_rest_field() method, it gets added to the $wp_rest_additional_fields variable. However, modifying the $wp_rest_additional_fields variable manually is strongly discouraged.

Adding Custom Fields to the Response

Having familiarized ourselves with the register_rest_field() method, we can now modify the response for the post object. A typical use case here would be the addition of an author display name field, which is commonly needed when listing posts on an index page. Since the standard response doesn’t include this field, we can use the register_rest_field() method to include it in the response.

We begin by creating a simple plugin. So create a new folder named rest-response-modifier in your /wp-content/plugins directory. Create an empty index.php file and paste in the following plugin definition:

The register_rest_field() method should be registered in the rest_api_init action. Hence, we create a function named bs_add_custom_rest_fields() and bind it to the rest_api_init hook:

Note that the opening PHP tags <?php are not required here, but I’ve included them so the syntax is highlighted correctly.

Inside the bs_add_custom_rest_fields() function, we can use the register_rest_field() method to include a field for author name:

As mentioned in the previous section, the first argument in the register_rest_field() method is the name of the object for which we are modifying the response. Since we need to modify the response for the post object, we pass the same as the first argument.

The second argument in the above code is the name of the field which will appear in the response. It’s always a good practice to prefix the name of a custom field in the response to ensure maximum forward compatibility and that it doesn’t get overridden in the future by other plugins. Hence, we pass bs_author_name in the second argument as the $attribute of the custom field.

The third and last argument in the above code is an array for callback methods and the schema. This array holds the name of the callback methods for the retrieval and update of the custom field in the $get_callback and $update_callback keys respectively. We pass the bs_get_author_name function as the retrieval callback method. We will define this function shortly.

For the $update_callback key, we pass null since this is a read-only field and we don’t need to update the author name for a post.

For the $schema key, we pass an array named $bs_author_name_schema. This array holds various properties for the field like the data type, the context, and the description.

The only thing we need to define now is the bs_get_author_name() function that will act as the $get_callback method for our custom field. Below is the code for this function:

The $get_callback method receives three arguments for the following:

  1. $object: The current object. In our case, it’s the current post.
  2. $field_name: The name of the custom field being added.
  3. $request: The request object.

We are using the $author property of the $object argument which holds the id of the post author. And by using the get_the_author_meta() function, we retrieve and return the display name of the author for the current post.

Now that the field is registered, we can send a GET request to the /wp/v2/posts route to see if it works properly:

Here is the response in Postman:

Response

This newly registered custom field will also appear in the server response, along with its schema, when we send an OPTIONS request to the /wp/v2/posts route:

schema

Hence a custom field for the author name property has been successfully registered. But this field is read-only as we can’t update it by sending a POST request. In the following section, we will register an editable field for post views count.

Registering an Editable Field

We will now register a custom field for the post views count. We will only deal with the actual registration of the field with WP REST API, leaving out the implementation for incrementing the count number.

Below is the code for bs_post_views custom field registration along with its schema:

The code is similar to the one we wrote in the previous section except that it now includes a callback method bs_update_post_views for the $update_callback key. This function is responsible for updating the value of the field.

The $context property in the $bs_post_views_schema schema array includes two values for view and edit. Inclusion of edit value in the $context argument ensures that the bs_post_views field is returned in the server response after it has been updated.

The retrieval and update callback methods are as follows:

The code is pretty simple as it uses the get_post_meta() and update_post_meta() methods for retrieving and updating the values respectively.

The bs_get_post_views() method first retrieves the meta value for the bs_post_views meta key and casts it into an integer before returning it.

The callback method passed in $update_callback receives three arguments for the following:

  1. $value: The new value for the field.
  2. $object: Current object from the response.
  3. $field_name: The name of the field being updated.

In the bs_update_post_views() method, we first check if the value being passed is not empty and is a numeric value. If not, we return without doing anything.

If the value is numeric, we pass it to the update_post_meta() function which saves it to the database after type casting it into a valid integer.

Having registered the field successfully, let’s test it by sending a GET request:

Below is a sample response for the above request:

response

As we can see in the image above, the current value of the bs_post_views field is 0 for a given post. This is because the get_post_meta() method is returning an empty string since it couldn’t find a meta value for the bs_post_views meta key and type-casting an empty string into an integer in PHP results in 0.

We can update the bs_post_views field by sending a POST request to the /wp/v2/posts/<id> endpoint. The JSON body for the request is as follows:

If the request is successful, the server returns a 200 – OK status code along with the updated post object that also includes the bs_post_views field:

response

The bs_post_views custom field is now updated.

Note that we sent a JSON body along the request to update the field. The JSON body included the field name—bs_post_views—with an integer value of 4050. If we try to send a non-numeric value, say “abc1234”, the field will not be updated since we have a condition checking for a numeric value in the bs_update_post_views() callback method.

Below is the full source code for the plugin:

That’s all for modifying server responses for the default API endpoints. We have barely scratched the surface for modifying the REST API since it provides much more flexibility than just modifying server responses. This includes adding support for the custom content type via custom controllers and namespaces, and registering custom routes for exposing and modifying data. We will try to cover these advanced topics in future articles.

Just the Beginning…

Here we conclude our journey of introducing ourselves to the WP REST API. In this series, we have covered pretty basic concepts like authentication methods and retrieving, creating, and updating data. In this last part of the series, we briefly looked at the internal classes of the WP REST API and then learned to modify server responses for the default endpoints.

It was never the purpose of this series to cover each and every aspect of the WP REST API—in fact, it can never be achieved in a single series. But rather, the purpose of this series was to get you up and running with this new fantastic addition and to encourage you to play around and experiment on your own. I hope that you have found this series fulfilling its ultimate objective.