Skip to content

Commit 1c5eb4d

Browse files
kbandesKenneth Bandes
and
Kenneth Bandes
authored
feat: Add helper function to format query_params for rest transport. (#275)
Co-authored-by: Kenneth Bandes <[email protected]>
1 parent afe0fa1 commit 1c5eb4d

File tree

2 files changed

+171
-0
lines changed

2 files changed

+171
-0
lines changed

google/api_core/rest_helpers.py

+94
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,94 @@
1+
# Copyright 2021 Google LLC
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
"""Helpers for rest transports."""
16+
17+
import functools
18+
import operator
19+
20+
21+
def flatten_query_params(obj):
22+
"""Flatten a nested dict into a list of (name,value) tuples.
23+
24+
The result is suitable for setting query params on an http request.
25+
26+
.. code-block:: python
27+
28+
>>> obj = {'a':
29+
... {'b':
30+
... {'c': ['x', 'y', 'z']} },
31+
... 'd': 'uvw', }
32+
>>> flatten_query_params(obj)
33+
[('a.b.c', 'x'), ('a.b.c', 'y'), ('a.b.c', 'z'), ('d', 'uvw')]
34+
35+
Note that, as described in
36+
https://github.com/googleapis/googleapis/blob/48d9fb8c8e287c472af500221c6450ecd45d7d39/google/api/http.proto#L117,
37+
repeated fields (i.e. list-valued fields) may only contain primitive types (not lists or dicts).
38+
This is enforced in this function.
39+
40+
Args:
41+
obj: a nested dictionary (from json), or None
42+
43+
Returns: a list of tuples, with each tuple having a (possibly) multi-part name
44+
and a scalar value.
45+
46+
Raises:
47+
TypeError if obj is not a dict or None
48+
ValueError if obj contains a list of non-primitive values.
49+
"""
50+
51+
if obj is not None and not isinstance(obj, dict):
52+
raise TypeError("flatten_query_params must be called with dict object")
53+
54+
return _flatten(obj, key_path=[])
55+
56+
57+
def _flatten(obj, key_path):
58+
if obj is None:
59+
return []
60+
if isinstance(obj, dict):
61+
return _flatten_dict(obj, key_path=key_path)
62+
if isinstance(obj, list):
63+
return _flatten_list(obj, key_path=key_path)
64+
return _flatten_value(obj, key_path=key_path)
65+
66+
67+
def _is_primitive_value(obj):
68+
if obj is None:
69+
return False
70+
71+
if isinstance(obj, (list, dict)):
72+
raise ValueError("query params may not contain repeated dicts or lists")
73+
74+
return True
75+
76+
77+
def _flatten_value(obj, key_path):
78+
return [(".".join(key_path), obj)]
79+
80+
81+
def _flatten_dict(obj, key_path):
82+
items = (_flatten(value, key_path=key_path + [key]) for key, value in obj.items())
83+
return functools.reduce(operator.concat, items, [])
84+
85+
86+
def _flatten_list(elems, key_path):
87+
# Only lists of scalar values are supported.
88+
# The name (key_path) is repeated for each value.
89+
items = (
90+
_flatten_value(elem, key_path=key_path)
91+
for elem in elems
92+
if _is_primitive_value(elem)
93+
)
94+
return functools.reduce(operator.concat, items, [])

tests/unit/test_rest_helpers.py

+77
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,77 @@
1+
# Copyright 2021 Google LLC
2+
#
3+
# Licensed under the Apache License, Version 2.0 (the "License");
4+
# you may not use this file except in compliance with the License.
5+
# You may obtain a copy of the License at
6+
#
7+
# http://www.apache.org/licenses/LICENSE-2.0
8+
#
9+
# Unless required by applicable law or agreed to in writing, software
10+
# distributed under the License is distributed on an "AS IS" BASIS,
11+
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12+
# See the License for the specific language governing permissions and
13+
# limitations under the License.
14+
15+
import pytest
16+
17+
from google.api_core import rest_helpers
18+
19+
20+
def test_flatten_simple_value():
21+
with pytest.raises(TypeError):
22+
rest_helpers.flatten_query_params("abc")
23+
24+
25+
def test_flatten_list():
26+
with pytest.raises(TypeError):
27+
rest_helpers.flatten_query_params(["abc", "def"])
28+
29+
30+
def test_flatten_none():
31+
assert rest_helpers.flatten_query_params(None) == []
32+
33+
34+
def test_flatten_empty_dict():
35+
assert rest_helpers.flatten_query_params({}) == []
36+
37+
38+
def test_flatten_simple_dict():
39+
assert rest_helpers.flatten_query_params({"a": "abc", "b": "def"}) == [
40+
("a", "abc"),
41+
("b", "def"),
42+
]
43+
44+
45+
def test_flatten_repeated_field():
46+
assert rest_helpers.flatten_query_params({"a": ["x", "y", "z", None]}) == [
47+
("a", "x"),
48+
("a", "y"),
49+
("a", "z"),
50+
]
51+
52+
53+
def test_flatten_nested_dict():
54+
obj = {"a": {"b": {"c": ["x", "y", "z"]}}, "d": {"e": "uvw"}}
55+
expected_result = [("a.b.c", "x"), ("a.b.c", "y"), ("a.b.c", "z"), ("d.e", "uvw")]
56+
57+
assert rest_helpers.flatten_query_params(obj) == expected_result
58+
59+
60+
def test_flatten_repeated_dict():
61+
obj = {
62+
"a": {"b": {"c": [{"v": 1}, {"v": 2}]}},
63+
"d": "uvw",
64+
}
65+
66+
with pytest.raises(ValueError):
67+
rest_helpers.flatten_query_params(obj)
68+
69+
70+
def test_flatten_repeated_list():
71+
obj = {
72+
"a": {"b": {"c": [["e", "f"], ["g", "h"]]}},
73+
"d": "uvw",
74+
}
75+
76+
with pytest.raises(ValueError):
77+
rest_helpers.flatten_query_params(obj)

0 commit comments

Comments
 (0)