|
35 | 35 | import pytest
|
36 | 36 |
|
37 | 37 | from pymongo import AsyncMongoClient, monitoring
|
38 |
| -from pymongo.asynchronous.auth import HAVE_KERBEROS |
| 38 | +from pymongo.asynchronous.auth import HAVE_KERBEROS, _canonicalize_hostname |
39 | 39 | from pymongo.auth_shared import _build_credentials_tuple
|
40 | 40 | from pymongo.errors import OperationFailure
|
41 | 41 | from pymongo.hello import HelloCompat
|
@@ -96,10 +96,11 @@ def setUpClass(cls):
|
96 | 96 | cls.service_realm_required = (
|
97 | 97 | GSSAPI_SERVICE_REALM is not None and GSSAPI_SERVICE_REALM not in GSSAPI_PRINCIPAL
|
98 | 98 | )
|
99 |
| - mech_properties = f"SERVICE_NAME:{GSSAPI_SERVICE_NAME}" |
100 |
| - mech_properties += f",CANONICALIZE_HOST_NAME:{GSSAPI_CANONICALIZE}" |
| 99 | + mech_properties = dict( |
| 100 | + SERVICE_NAME=GSSAPI_SERVICE_NAME, CANONICALIZE_HOST_NAME=GSSAPI_CANONICALIZE |
| 101 | + ) |
101 | 102 | if GSSAPI_SERVICE_REALM is not None:
|
102 |
| - mech_properties += f",SERVICE_REALM:{GSSAPI_SERVICE_REALM}" |
| 103 | + mech_properties["SERVICE_REALM"] = GSSAPI_SERVICE_REALM |
103 | 104 | cls.mech_properties = mech_properties
|
104 | 105 |
|
105 | 106 | async def test_credentials_hashing(self):
|
@@ -167,7 +168,10 @@ async def test_gssapi_simple(self):
|
167 | 168 | await client[GSSAPI_DB].collection.find_one()
|
168 | 169 |
|
169 | 170 | # Log in using URI, with authMechanismProperties.
|
170 |
| - mech_uri = uri + f"&authMechanismProperties={self.mech_properties}" |
| 171 | + mech_properties_str = "" |
| 172 | + for key, value in self.mech_properties.items(): |
| 173 | + mech_properties_str += f"{key}:{value}," |
| 174 | + mech_uri = uri + f"&authMechanismProperties={mech_properties_str[:-1]}" |
171 | 175 | client = self.simple_client(mech_uri)
|
172 | 176 | await client[GSSAPI_DB].collection.find_one()
|
173 | 177 |
|
@@ -268,6 +272,58 @@ async def test_gssapi_threaded(self):
|
268 | 272 | thread.join()
|
269 | 273 | self.assertTrue(thread.success)
|
270 | 274 |
|
| 275 | + async def test_gssapi_canonicalize_host_name(self): |
| 276 | + # Test the low level method. |
| 277 | + assert GSSAPI_HOST is not None |
| 278 | + result = _canonicalize_hostname(GSSAPI_HOST, "forward") |
| 279 | + if "compute-1.amazonaws.com" not in result: |
| 280 | + self.assertEqual(result, GSSAPI_HOST) |
| 281 | + result = _canonicalize_hostname(GSSAPI_HOST, "forwardAndReverse") |
| 282 | + self.assertEqual(result, GSSAPI_HOST) |
| 283 | + |
| 284 | + # Use the equivalent named CANONICALIZE_HOST_NAME. |
| 285 | + props = self.mech_properties.copy() |
| 286 | + if props["CANONICALIZE_HOST_NAME"] == "true": |
| 287 | + props["CANONICALIZE_HOST_NAME"] = "forwardAndReverse" |
| 288 | + else: |
| 289 | + props["CANONICALIZE_HOST_NAME"] = "none" |
| 290 | + client = self.simple_client( |
| 291 | + GSSAPI_HOST, |
| 292 | + GSSAPI_PORT, |
| 293 | + username=GSSAPI_PRINCIPAL, |
| 294 | + password=GSSAPI_PASS, |
| 295 | + authMechanism="GSSAPI", |
| 296 | + authMechanismProperties=props, |
| 297 | + ) |
| 298 | + await client.server_info() |
| 299 | + |
| 300 | + async def test_gssapi_host_name(self): |
| 301 | + props = self.mech_properties |
| 302 | + props["SERVICE_HOST"] = "example.com" |
| 303 | + |
| 304 | + # Authenticate with authMechanismProperties. |
| 305 | + client = self.simple_client( |
| 306 | + GSSAPI_HOST, |
| 307 | + GSSAPI_PORT, |
| 308 | + username=GSSAPI_PRINCIPAL, |
| 309 | + password=GSSAPI_PASS, |
| 310 | + authMechanism="GSSAPI", |
| 311 | + authMechanismProperties=self.mech_properties, |
| 312 | + ) |
| 313 | + with self.assertRaises(OperationFailure): |
| 314 | + await client.server_info() |
| 315 | + |
| 316 | + props["SERVICE_HOST"] = GSSAPI_HOST |
| 317 | + client = self.simple_client( |
| 318 | + GSSAPI_HOST, |
| 319 | + GSSAPI_PORT, |
| 320 | + username=GSSAPI_PRINCIPAL, |
| 321 | + password=GSSAPI_PASS, |
| 322 | + authMechanism="GSSAPI", |
| 323 | + authMechanismProperties=self.mech_properties, |
| 324 | + ) |
| 325 | + await client.server_info() |
| 326 | + |
271 | 327 |
|
272 | 328 | class TestSASLPlain(AsyncPyMongoTestCase):
|
273 | 329 | @classmethod
|
|
0 commit comments