123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233 |
- """
- :codeauthor: Nicole Thomas <nicole@saltstack.com>
- """
- import socket
- from contextlib import closing
- import salt.utils.http as http
- from tests.support.helpers import MirrorPostHandler, Webserver, slowTest
- from tests.support.unit import TestCase
- class HTTPTestCase(TestCase):
- """
- Unit TestCase for the salt.utils.http module.
- """
- @classmethod
- def setUpClass(cls):
- cls.post_webserver = Webserver(handler=MirrorPostHandler)
- cls.post_webserver.start()
- cls.post_web_root = cls.post_webserver.web_root
- @classmethod
- def tearDownClass(cls):
- cls.post_webserver.stop()
- del cls.post_webserver
- # sanitize_url tests
- def test_sanitize_url_hide_fields_none(self):
- """
- Tests sanitizing a url when the hide_fields kwarg is None.
- """
- mock_url = "https://api.testing.com/?&foo=bar&test=testing"
- ret = http.sanitize_url(mock_url, hide_fields=None)
- self.assertEqual(ret, mock_url)
- def test_sanitize_url_no_elements(self):
- """
- Tests sanitizing a url when no elements should be sanitized.
- """
- mock_url = "https://api.testing.com/?&foo=bar&test=testing"
- ret = http.sanitize_url(mock_url, [""])
- self.assertEqual(ret, mock_url)
- def test_sanitize_url_single_element(self):
- """
- Tests sanitizing a url with only a single element to be sanitized.
- """
- mock_url = (
- "https://api.testing.com/?&keep_it_secret=abcdefghijklmn"
- "&api_action=module.function"
- )
- mock_ret = (
- "https://api.testing.com/?&keep_it_secret=XXXXXXXXXX&"
- "api_action=module.function"
- )
- ret = http.sanitize_url(mock_url, ["keep_it_secret"])
- self.assertEqual(ret, mock_ret)
- def test_sanitize_url_multiple_elements(self):
- """
- Tests sanitizing a url with multiple elements to be sanitized.
- """
- mock_url = (
- "https://api.testing.com/?rootPass=badpassword%21"
- "&skipChecks=True&api_key=abcdefghijklmn"
- "&NodeID=12345&api_action=module.function"
- )
- mock_ret = (
- "https://api.testing.com/?rootPass=XXXXXXXXXX"
- "&skipChecks=True&api_key=XXXXXXXXXX"
- "&NodeID=12345&api_action=module.function"
- )
- ret = http.sanitize_url(mock_url, ["api_key", "rootPass"])
- self.assertEqual(ret, mock_ret)
- # _sanitize_components tests
- def test_sanitize_components_no_elements(self):
- """
- Tests when zero elements need to be sanitized.
- """
- mock_component_list = ["foo=bar", "bar=baz", "hello=world"]
- mock_ret = "foo=bar&bar=baz&hello=world&"
- ret = http._sanitize_url_components(mock_component_list, "api_key")
- self.assertEqual(ret, mock_ret)
- def test_sanitize_components_one_element(self):
- """
- Tests a single component to be sanitized.
- """
- mock_component_list = ["foo=bar", "api_key=abcdefghijklmnop"]
- mock_ret = "foo=bar&api_key=XXXXXXXXXX&"
- ret = http._sanitize_url_components(mock_component_list, "api_key")
- self.assertEqual(ret, mock_ret)
- def test_sanitize_components_multiple_elements(self):
- """
- Tests two componenets to be sanitized.
- """
- mock_component_list = ["foo=bar", "foo=baz", "api_key=testing"]
- mock_ret = "foo=XXXXXXXXXX&foo=XXXXXXXXXX&api_key=testing&"
- ret = http._sanitize_url_components(mock_component_list, "foo")
- self.assertEqual(ret, mock_ret)
- @slowTest
- def test_query_null_response(self):
- """
- This tests that we get a null response when raise_error=False and the
- host/port cannot be reached.
- """
- host = "127.0.0.1"
- # Find unused port
- with closing(socket.socket(socket.AF_INET, socket.SOCK_STREAM)) as sock:
- sock.bind((host, 0))
- port = sock.getsockname()[1]
- url = "http://{host}:{port}/".format(host=host, port=port)
- result = http.query(url, raise_error=False)
- assert result == {"body": None}, result
- def test_query_error_handling(self):
- ret = http.query("http://127.0.0.1:0")
- self.assertTrue(isinstance(ret, dict))
- self.assertTrue(isinstance(ret.get("error", None), str))
- ret = http.query("http://myfoobardomainthatnotexist")
- self.assertTrue(isinstance(ret, dict))
- self.assertTrue(isinstance(ret.get("error", None), str))
- def test_parse_cookie_header(self):
- header = "; ".join(
- [
- "foo=bar",
- "expires=Mon, 03-Aug-20 14:26:27 GMT",
- "path=/",
- "domain=.mydomain.tld",
- "HttpOnly",
- "SameSite=Lax",
- "Secure",
- ]
- )
- ret = http.parse_cookie_header(header)
- cookie = ret.pop(0)
- assert cookie.name == "foo", cookie.name
- assert cookie.value == "bar", cookie.value
- assert cookie.expires == 1596464787, cookie.expires
- assert cookie.path == "/", cookie.path
- assert cookie.domain == ".mydomain.tld", cookie.domain
- assert cookie.secure
- # Only one cookie should have been returned, if anything is left in the
- # parse_cookie_header return then something went wrong.
- assert not ret
- class HTTPPostTestCase(TestCase):
- """
- Unit TestCase for the salt.utils.http module when
- using POST method
- """
- @classmethod
- def setUpClass(cls):
- cls.post_webserver = Webserver(handler=MirrorPostHandler)
- cls.post_webserver.start()
- cls.post_web_root = cls.post_webserver.web_root
- @classmethod
- def tearDownClass(cls):
- cls.post_webserver.stop()
- del cls.post_webserver
- def test_requests_multipart_formdata_post(self):
- """
- Test handling of a multipart/form-data POST using the requests backend
- """
- match_this = '{0}\r\nContent-Disposition: form-data; name="fieldname_here"\r\n\r\nmydatahere\r\n{0}--\r\n'
- ret = http.query(
- self.post_web_root,
- method="POST",
- data="mydatahere",
- formdata=True,
- formdata_fieldname="fieldname_here",
- backend="requests",
- )
- body = ret.get("body", "")
- boundary = body[: body.find("\r")]
- self.assertEqual(body, match_this.format(boundary))
- class HTTPGetTestCase(TestCase):
- """
- Unit TestCase for the salt.utils.http module when
- using Get method
- """
- @classmethod
- def setUpClass(cls):
- cls.get_webserver = Webserver()
- cls.get_webserver.start()
- @classmethod
- def tearDownClass(cls):
- cls.get_webserver.stop()
- del cls.get_webserver
- def test_backends_decode_body_false(self):
- """
- test all backends when using
- decode_body=False that it returns
- bytes and does not try to decode
- """
- for backend in ["tornado", "requests", "urllib2"]:
- ret = http.query(
- self.get_webserver.url("custom.tar.gz"),
- backend=backend,
- decode_body=False,
- )
- body = ret.get("body", "")
- assert isinstance(body, bytes)
- def test_backends_decode_body_true(self):
- """
- test all backends when using
- decode_body=True that it returns
- string and decodes it.
- """
- for backend in ["tornado", "requests", "urllib2"]:
- ret = http.query(self.get_webserver.url("core.sls"), backend=backend)
- body = ret.get("body", "")
- assert isinstance(body, str)
|