Field Optionality & Nullability¶
In OpenAPI documents and JSON schemas, a field can specify its nullability and requiredness. It is crucial to accurately map the proto labels to the JSON schema to ensure a correct representation of your API. Each API can have its own unique set of patterns. In this document, you will learn how to configure the OpenAPI plug-in to adjust the way these specifications are mapped to proto labels.
Field Nullability¶
In OpenAPI and JSON schemas, a field can specify whether it accepts null values. In OpenAPI v3.1, the correct way to represent this is by using type arrays.
How does this map to proto fields? Let's consider the proto message below:
message User {
string id = 1;
string name = 2;
optional string email_address = 3;
optional Address address = 4;
PhoneNumber phone_number = 5;
}
Protobuf can handle null values, where a null value indicates an unspecified value. If you assigned null values to every key in a JSON object for the proto message above, the model would be mapped as follows:
id
is a string, a primitive type, and is not marked as optional. Therefore, it will be assigned its default value, which is an empty string.email_address
is a string, a primitive type, and is explicitly marked as optional. This means it can differentiate between having no value and being blank. Therefore, it is assignednull
.address
is an optional message type. Similar toemail_address
, because it is labeled asoptional
, it is nullable and thus receivesnull
.- Unlike
address
,phone_number
is not marked as an optional field. However, since it is a message type, it is inherently nullable and thus also receivesnull
.
That is how protobuf handles unmarshaling bytes into in-memory structures. How should this be represented in OpenAPI? If phone_number
is intended to be set and a null value is unacceptable, this requirement must be clearly documented in the OpenAPI specification.
This can be controlled via field_nullable_mode
plug-in option.
1. Disabled¶
Value of disabled
for field_nullable_mode
would enable this mode.
Disabled means that nullability will not be automatically documented. For the proto message above, this results in a JSON schema where none of the fields are marked as nullable.
You can manually adjust the nullability settings as needed:
Below is an example of explicitly setting the OpenAPI types via configuration files.
This can also be achieved directly within proto files as well:
2. Nullable If Optional (Default)¶
Value of optional
for field_nullable_mode
would enable this mode.
This mode treats any proto field marked as optional
as nullable.
In the proto message above, the email_address
and address
fields would be represented in the OpenAPI document as accepting null
values, while the other fields would not include null
as an accepted value.
3. Nullable Unless Required¶
Value of non_required
for field_nullable_mode
would enable this mode.
This mode treats all fields as nullable unless they are explicitly marked as required.
Follow the guidelines in Explicitly Define Required Fields to specify which fields should be marked as required.
Field Requiredness¶
In JSON Schema and OpenAPI, requiredness and nullability are distinct concepts:
- Requiredness: Determines whether a field must be present in the JSON object. If a field is required, it must be included in the object, but it can still have a null value unless specified otherwise.
- Nullability: Specifies whether a field can have a null value. A nullable field can explicitly be set to null, indicating the absence of a value.
In OpenAPI v3.1, nullability is indicated using type arrays (e.g., ["string", "null"]
), while requiredness is specified using the required
keyword at the object level.
We discussed field nullability in the previous section. Now, let's use the same object to explore the concept of requiredness.
Consider the proto message below:
message User {
string id = 1;
string name = 2;
optional string email_address = 3;
optional Address address = 4;
PhoneNumber phone_number = 5;
}
You can configure the OpenAPI plug-in to automatically mark certain fields as required using field_required_mode
. By default, this feature is disabled, meaning no fields are automatically marked as required.
1. Disabled (Default)¶
Value of disabled
for field_required_mode
would enable this mode.
As previously mentioned, this means the OpenAPI plug-in will not automatically determine the requiredness of any field. However, you can manually specify which fields are required.
Explicitly Define Required Fields¶
There are several ways to mark a field as required:
1. At the message level:
This approach mirrors how OpenAPI specifies required fields. Each model explicitly lists all properties that are required:
- Indicates that fields
id
,name
andphone_number
do not accept null values.
2. At the field level:
This approach sets the requiredness at the field.
2. Required If Not Optional¶
Value of non_optional
for field_required_mode
would enable this mode.
This mode designates all proto fields as required by default, except for those explicitly marked as optional.
Thus, in the proto message defined above, the fields id
, name
, and phone_number
would be considered required.
3. Required If Not Optional & Scalar¶
Value of non_optional_scalar
for field_required_mode
would enable this mode.
This mode designates proto fields as required if all of the following conditions are met:
- The field is a primitive type or a list of primitive types (e.x.
repeated string
). - The field is not a map.
- The field is not a message.
- The field is not marked as optional.
Thus, in the proto message defined above, the fields id
and name
would be considered required. Since phone_number
is a message type, it is not considered required in this mode.
Mix & Match¶
You have the flexibility to configure requiredness and nullability automatically using the OpenAPI plug-in, allowing you to combine different rules for these attributes. For example, by setting field_required_mode
to non_optional
and field_nullable_mode
to non_required
, any field that is not explicitly required will be considered nullable. In this configuration, requiredness is automatically determined based on whether your proto fields are marked as optional.
In this mode, proto message User
as defined below:
message User {
string id = 1;
string name = 2;
optional string email_address = 3;
optional Address address = 4;
PhoneNumber phone_number = 5;
}
would treat the fields id
, name
, and phone_number
as required and non-nullable by default (unless explicitly configured otherwise). Conversely, the fields email_address
and address
would be considered optional and nullable. This can be achieved without any custom annotations or configuration files. If your API follows a consistent pattern like this, adjusting these two modes can help avoid the need to configure these fields individually.