Query Parameter Binding¶
Understanding how query parameters are processed by the gRPC Gateway is crucial for customizing your API to meet specific needs.
Default Behavior¶
In each EndpointConfig, any proto field from the request message that is not already bound to path parameters or the HTTP body is automatically bound to query parameters.
Naming¶
By default, the names of these query parameters are derived from the proto field names. For nested fields, the name is constructed by concatenating the parent field name with the nested field name, separated by a dot.
Example
Consider the following proto messages where Request
is the request message:
message Options {
bool case_sensitive = 1;
}
message Request {
Options options = 1;
string some_input = 2;
}
Assuming neither are captured by path parameters or the HTTP body,
query parameters some_input
and options.case_sensitive
will be
bound to the corresponding fields in the proto message.
Customization¶
There are a number of customizations available.
Disable Automatic Discovery¶
In EndpointConfig, you have the option to disable the automatic discovery and binding of query parameters. When this feature is disabled for a specific endpoint, only the query parameter bindings that are explicitly defined will be considered.
Some practical uses of this setting include:
- Assigning custom names to all query parameters.
- Restricting the exposure of certain parts of the proto message to the HTTP request.
Additional Query Parameter Binding and Aliases¶
To use custom names for query parameters or to allow multiple names (aliases) for the same message field, you can explicitly define the query parameter to request proto message field binding.
By utilizing query_params
in EndpointConfig, you can add multiple QueryParameterBinding objects.
Example
Consider request proto message below:
message PageOptions {
uint32 per_page = 1;
}
message QueryRequest {
string term = 1;
string language = 2;
PageOptions pagination = 3;
}
- Allow both
language
andlang
as query parameters, withlanguage
taking precedence. - Use
per_page
as the query parameter name instead of the defaultpagination.per_page
.
query.proto | |
---|---|
With the configuration above:
- Both
language
andlang
are accepted and bound to thelanguage
field in the proto request, withlanguage
taking precedence. - The
per_page
query parameter is accepted and bound topagination.per_page
. - The
term
field does not have an explicit binding but is automatically discovered and added.
Warning
Once you add an explicit binding for a proto field, that specific field will no longer receive automatic bindings.
Only the explicitly defined bindings will be applied.
For instance, in the example above, pagination.per_page
is no longer automatically bound because an
explicit binding for pagination.per_page
is defined.
Prioritization¶
If multiple names are defined for the same proto field, the most recently defined binding takes precedence over earlier ones.
In the example above, language
takes precedence over lang
.
Ignoring Parameters¶
You may want to exclude certain proto fields from being bound to any query parameters. To achieve this, you can use the ignore
attribute in QueryParameterBinding. This will prevent the specified proto fields from being included in the query parameters.
Example
Proto field language
is excluded during the automatic discovery phase and, as a result, is not bound to any query parameter.
Data Types¶
This section explains how query parameter data is mapped to proto types. While mapping scalar types is straightforward, handling more complex data types like repeated values and maps can be less intuitive. Here, we clarify how various types are parsed from query parameters.
Throughout this section, references to scalar types include all numerical types, booleans, strings, and enums.
Repeated Fields¶
Repeated fields can hold multiple values. Query parameters support repeated fields only for scalar types.
Example
?names=value1&names=value2&names=value3
gets mapped to ["value1", "value2", "value3"]
.
Info
Commas in the value do not act as a separator and are instead read as part of the value.
In the example above, ?names=value1,value2
gets mapped to ["value1,value2"]
.
Maps¶
Map types are also supported if both the key and the value are scalar types. The HTTP query parameter format is field_name[key]=value
.
Example
?metadata[key1]=value1&metadata[key2]=value2
gets mapped to {"key1": "value1", "key2": "value2"}
.
Unbound Query Parameters¶
All unbound query parameters are ignored without generating any errors.