All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
3 from unittest.mock import patch
5 from click.testing import CliRunner
8 from tests.util import read_data, DETERMINISTIC_HEADER
12 from aiohttp.test_utils import AioHTTPTestCase
13 from aiohttp import web
14 except ImportError as e:
15 raise RuntimeError("Please install Black with the 'd' extra") from e
18 from aiohttp.test_utils import unittest_run_loop
20 # unittest_run_loop is unnecessary and a no-op since aiohttp 3.8, and aiohttp 4
21 # removed it. To maintain compatibility we can make our own no-op decorator.
22 def unittest_run_loop(func: Any, *args: Any, **kwargs: Any) -> Any:
27 class BlackDTestCase(AioHTTPTestCase):
28 def test_blackd_main(self) -> None:
29 with patch("blackd.web.run_app"):
30 result = CliRunner().invoke(blackd.main, [])
31 if result.exception is not None:
32 raise result.exception
33 self.assertEqual(result.exit_code, 0)
35 async def get_application(self) -> web.Application:
36 return blackd.make_app()
39 async def test_blackd_request_needs_formatting(self) -> None:
40 response = await self.client.post("/", data=b"print('hello world')")
41 self.assertEqual(response.status, 200)
42 self.assertEqual(response.charset, "utf8")
43 self.assertEqual(await response.read(), b'print("hello world")\n')
46 async def test_blackd_request_no_change(self) -> None:
47 response = await self.client.post("/", data=b'print("hello world")\n')
48 self.assertEqual(response.status, 204)
49 self.assertEqual(await response.read(), b"")
52 async def test_blackd_request_syntax_error(self) -> None:
53 response = await self.client.post("/", data=b"what even ( is")
54 self.assertEqual(response.status, 400)
55 content = await response.text()
57 content.startswith("Cannot parse"),
58 msg=f"Expected error to start with 'Cannot parse', got {repr(content)}",
62 async def test_blackd_unsupported_version(self) -> None:
63 response = await self.client.post(
64 "/", data=b"what", headers={blackd.PROTOCOL_VERSION_HEADER: "2"}
66 self.assertEqual(response.status, 501)
69 async def test_blackd_supported_version(self) -> None:
70 response = await self.client.post(
71 "/", data=b"what", headers={blackd.PROTOCOL_VERSION_HEADER: "1"}
73 self.assertEqual(response.status, 200)
76 async def test_blackd_invalid_python_variant(self) -> None:
77 async def check(header_value: str, expected_status: int = 400) -> None:
78 response = await self.client.post(
79 "/", data=b"what", headers={blackd.PYTHON_VARIANT_HEADER: header_value}
81 self.assertEqual(response.status, expected_status)
84 await check("ruby3.5")
93 await check("pypy3.0")
94 await check("jython3.4")
97 async def test_blackd_pyi(self) -> None:
98 source, expected = read_data("stub.pyi")
99 response = await self.client.post(
100 "/", data=source, headers={blackd.PYTHON_VARIANT_HEADER: "pyi"}
102 self.assertEqual(response.status, 200)
103 self.assertEqual(await response.text(), expected)
106 async def test_blackd_diff(self) -> None:
107 diff_header = re.compile(
108 r"(In|Out)\t\d\d\d\d-\d\d-\d\d \d\d:\d\d:\d\d\.\d\d\d\d\d\d \+\d\d\d\d"
111 source, _ = read_data("blackd_diff.py")
112 expected, _ = read_data("blackd_diff.diff")
114 response = await self.client.post(
115 "/", data=source, headers={blackd.DIFF_HEADER: "true"}
117 self.assertEqual(response.status, 200)
119 actual = await response.text()
120 actual = diff_header.sub(DETERMINISTIC_HEADER, actual)
121 self.assertEqual(actual, expected)
124 async def test_blackd_python_variant(self) -> None:
127 " and_has_a_bunch_of,\n"
128 " very_long_arguments_too,\n"
129 " and_lots_of_them_as_well_lol,\n"
130 " **and_very_long_keyword_arguments\n"
135 async def check(header_value: str, expected_status: int) -> None:
136 response = await self.client.post(
137 "/", data=code, headers={blackd.PYTHON_VARIANT_HEADER: header_value}
140 response.status, expected_status, msg=await response.text()
143 await check("3.6", 200)
144 await check("py3.6", 200)
145 await check("3.6,3.7", 200)
146 await check("3.6,py3.7", 200)
147 await check("py36,py37", 200)
148 await check("36", 200)
149 await check("3.6.4", 200)
150 await check("3.4", 204)
151 await check("py3.4", 204)
152 await check("py34,py36", 204)
153 await check("34", 204)
156 async def test_blackd_line_length(self) -> None:
157 response = await self.client.post(
158 "/", data=b'print("hello")\n', headers={blackd.LINE_LENGTH_HEADER: "7"}
160 self.assertEqual(response.status, 200)
163 async def test_blackd_invalid_line_length(self) -> None:
164 response = await self.client.post(
165 "/", data=b'print("hello")\n', headers={blackd.LINE_LENGTH_HEADER: "NaN"}
167 self.assertEqual(response.status, 400)
170 async def test_blackd_response_black_version_header(self) -> None:
171 response = await self.client.post("/")
172 self.assertIsNotNone(response.headers.get(blackd.BLACK_VERSION_HEADER))
175 async def test_cors_preflight(self) -> None:
176 response = await self.client.options(
179 "Access-Control-Request-Method": "POST",
181 "Access-Control-Request-Headers": "Content-Type",
184 self.assertEqual(response.status, 200)
185 self.assertIsNotNone(response.headers.get("Access-Control-Allow-Origin"))
186 self.assertIsNotNone(response.headers.get("Access-Control-Allow-Headers"))
187 self.assertIsNotNone(response.headers.get("Access-Control-Allow-Methods"))
190 async def test_cors_headers_present(self) -> None:
191 response = await self.client.post("/", headers={"Origin": "*"})
192 self.assertIsNotNone(response.headers.get("Access-Control-Allow-Origin"))
193 self.assertIsNotNone(response.headers.get("Access-Control-Expose-Headers"))