6e3f1a1194
* Upgrade syntax with pyupgrade --py38-plus Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> * Convert to f-strings with flynt Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> * Format with Black Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> * Remove redundant mock backport dependency Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> * isort imports Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> * Add changelog entry Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com> --------- Signed-off-by: Hugo van Kemenade <1324225+hugovk@users.noreply.github.com>
199 lines
7.0 KiB
Python
199 lines
7.0 KiB
Python
# SPDX-License-Identifier: Apache-2.0
|
|
#
|
|
# The OpenSearch Contributors require contributions made to
|
|
# this file be licensed under the Apache-2.0 license or a
|
|
# compatible open source license.
|
|
#
|
|
# Modifications Copyright OpenSearch Contributors. See
|
|
# GitHub history for details.
|
|
#
|
|
# Licensed to Elasticsearch B.V. under one or more contributor
|
|
# license agreements. See the NOTICE file distributed with
|
|
# this work for additional information regarding copyright
|
|
# ownership. Elasticsearch B.V. licenses this file to you under
|
|
# the Apache License, Version 2.0 (the "License"); you may
|
|
# not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing,
|
|
# software distributed under the License is distributed on an
|
|
# "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
|
|
# KIND, either express or implied. See the License for the
|
|
# specific language governing permissions and limitations
|
|
# under the License.
|
|
|
|
|
|
from typing import Any
|
|
|
|
from opensearchpy.client.utils import _bulk_body, _escape, _make_path, query_params
|
|
|
|
from ..test_cases import TestCase
|
|
|
|
|
|
class TestQueryParams(TestCase):
|
|
def setup_method(self, _: Any) -> None:
|
|
self.calls: Any = []
|
|
|
|
@query_params("simple_param")
|
|
def func_to_wrap(self, *args: Any, **kwargs: Any) -> None:
|
|
self.calls.append((args, kwargs))
|
|
|
|
def test_handles_params(self) -> None:
|
|
self.func_to_wrap(params={"simple_param_2": "2"}, simple_param="3")
|
|
self.assertEqual(
|
|
self.calls,
|
|
[
|
|
(
|
|
(),
|
|
{
|
|
"params": {"simple_param": b"3", "simple_param_2": "2"},
|
|
"headers": {},
|
|
},
|
|
)
|
|
],
|
|
)
|
|
|
|
def test_handles_headers(self) -> None:
|
|
self.func_to_wrap(headers={"X-Opaque-Id": "app-1"})
|
|
self.assertEqual(
|
|
self.calls, [((), {"params": {}, "headers": {"x-opaque-id": "app-1"}})]
|
|
)
|
|
|
|
def test_handles_opaque_id(self) -> None:
|
|
self.func_to_wrap(opaque_id="request-id")
|
|
self.assertEqual(
|
|
self.calls, [((), {"params": {}, "headers": {"x-opaque-id": "request-id"}})]
|
|
)
|
|
|
|
def test_handles_empty_none_and_normalization(self) -> None:
|
|
self.func_to_wrap(params=None)
|
|
self.assertEqual(self.calls[-1], ((), {"params": {}, "headers": {}}))
|
|
|
|
self.func_to_wrap(headers=None)
|
|
self.assertEqual(self.calls[-1], ((), {"params": {}, "headers": {}}))
|
|
|
|
self.func_to_wrap(headers=None, params=None)
|
|
self.assertEqual(self.calls[-1], ((), {"params": {}, "headers": {}}))
|
|
|
|
self.func_to_wrap(headers={}, params={})
|
|
self.assertEqual(self.calls[-1], ((), {"params": {}, "headers": {}}))
|
|
|
|
self.func_to_wrap(headers={"X": "y"})
|
|
self.assertEqual(self.calls[-1], ((), {"params": {}, "headers": {"x": "y"}}))
|
|
|
|
def test_non_escaping_params(self) -> None:
|
|
# the query_params decorator doesn't validate "timeout" it simply avoids escaping as it did
|
|
self.func_to_wrap(simple_param="x", timeout="4s")
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
((), {"params": {"simple_param": b"x", "timeout": "4s"}, "headers": {}}),
|
|
)
|
|
|
|
self.func_to_wrap(simple_param="x", timeout=4, ignore=5, request_timeout=6)
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
(
|
|
(),
|
|
{
|
|
"params": {
|
|
"simple_param": b"x",
|
|
"timeout": 4,
|
|
"ignore": 5,
|
|
"request_timeout": 6,
|
|
},
|
|
"headers": {},
|
|
},
|
|
),
|
|
)
|
|
|
|
def test_per_call_authentication(self) -> None:
|
|
self.func_to_wrap(api_key=("name", "key"))
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
((), {"headers": {"authorization": "ApiKey bmFtZTprZXk="}, "params": {}}),
|
|
)
|
|
|
|
self.func_to_wrap(http_auth=("user", "password"))
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
(
|
|
(),
|
|
{
|
|
"headers": {"authorization": "Basic dXNlcjpwYXNzd29yZA=="},
|
|
"params": {},
|
|
},
|
|
),
|
|
)
|
|
|
|
self.func_to_wrap(http_auth="abcdef")
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
((), {"headers": {"authorization": "Basic abcdef"}, "params": {}}),
|
|
)
|
|
|
|
# If one or the other is 'None' it's all good!
|
|
self.func_to_wrap(http_auth=None, api_key=None)
|
|
self.assertEqual(self.calls[-1], ((), {"headers": {}, "params": {}}))
|
|
|
|
self.func_to_wrap(http_auth="abcdef", api_key=None)
|
|
self.assertEqual(
|
|
self.calls[-1],
|
|
((), {"headers": {"authorization": "Basic abcdef"}, "params": {}}),
|
|
)
|
|
|
|
# If both are given values an error is raised.
|
|
with self.assertRaises(ValueError) as e:
|
|
self.func_to_wrap(http_auth="key", api_key=("1", "2"))
|
|
self.assertEqual(
|
|
str(e.exception),
|
|
"Only one of 'http_auth' and 'api_key' may be passed at a time",
|
|
)
|
|
|
|
|
|
class TestMakePath(TestCase):
|
|
def test_handles_unicode(self) -> None:
|
|
id = "中文"
|
|
self.assertEqual(
|
|
"/some-index/type/%E4%B8%AD%E6%96%87", _make_path("some-index", "type", id)
|
|
)
|
|
|
|
|
|
class TestEscape(TestCase):
|
|
def test_handles_ascii(self) -> None:
|
|
string = "abc123"
|
|
self.assertEqual(b"abc123", _escape(string))
|
|
|
|
def test_handles_unicode(self) -> None:
|
|
string = "中文"
|
|
self.assertEqual(b"\xe4\xb8\xad\xe6\x96\x87", _escape(string))
|
|
|
|
def test_handles_bytestring(self) -> None:
|
|
string = b"celery-task-meta-c4f1201f-eb7b-41d5-9318-a75a8cfbdaa0"
|
|
self.assertEqual(string, _escape(string))
|
|
|
|
|
|
class TestBulkBody(TestCase):
|
|
def test_proper_bulk_body_as_string_is_not_modified(self) -> None:
|
|
string_body = '"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"\n'
|
|
self.assertEqual(string_body, _bulk_body(None, string_body))
|
|
|
|
def test_proper_bulk_body_as_bytestring_is_not_modified(self) -> None:
|
|
bytestring_body = b'"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"\n'
|
|
self.assertEqual(bytestring_body, _bulk_body(None, bytestring_body))
|
|
|
|
def test_bulk_body_as_string_adds_trailing_newline(self) -> None:
|
|
string_body = '"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"'
|
|
self.assertEqual(
|
|
'"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"\n',
|
|
_bulk_body(None, string_body),
|
|
)
|
|
|
|
def test_bulk_body_as_bytestring_adds_trailing_newline(self) -> None:
|
|
bytestring_body = b'"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"'
|
|
self.assertEqual(
|
|
b'"{"index":{ "_index" : "test"}}\n{"field1": "value1"}"\n',
|
|
_bulk_body(None, bytestring_body),
|
|
)
|