2090b3
diff --git a/src/urllib3/connection.py b/src/urllib3/connection.py
2090b3
index 02b3665..1ab1890 100644
2090b3
--- a/src/urllib3/connection.py
2090b3
+++ b/src/urllib3/connection.py
2090b3
@@ -1,4 +1,5 @@
2090b3
 from __future__ import absolute_import
2090b3
+import re
2090b3
 import datetime
2090b3
 import logging
2090b3
 import os
2090b3
@@ -61,6 +62,8 @@ port_by_scheme = {
2090b3
 # after 2016-01-01 (today - 2 years) AND before 2017-07-01 (today - 6 months)
2090b3
 RECENT_DATE = datetime.date(2017, 6, 30)
2090b3
 
2090b3
+_CONTAINS_CONTROL_CHAR_RE = re.compile(r"[^-!#$%&'*+.^_`|~0-9a-zA-Z]")
2090b3
+
2090b3
 
2090b3
 class DummyConnection(object):
2090b3
     """Used to detect a failed ConnectionCls import."""
2090b3
@@ -181,6 +184,17 @@ class HTTPConnection(_HTTPConnection, object):
2090b3
         conn = self._new_conn()
2090b3
         self._prepare_conn(conn)
2090b3
 
2090b3
+    def putrequest(self, method, url, *args, **kwargs):
2090b3
+        """Send a request to the server"""
2090b3
+        match = _CONTAINS_CONTROL_CHAR_RE.search(method)
2090b3
+        if match:
2090b3
+            raise ValueError(
2090b3
+                "Method cannot contain non-token characters %r (found at least %r)"
2090b3
+                % (method, match.group())
2090b3
+            )
2090b3
+
2090b3
+        return _HTTPConnection.putrequest(self, method, url, *args, **kwargs)
2090b3
+
2090b3
     def request_chunked(self, method, url, body=None, headers=None):
2090b3
         """
2090b3
         Alternative to the common request method, which sends the