67c8f0
diff --git a/paramiko/common.py b/paramiko/common.py
67c8f0
index 0b0cc2a..50355f6 100644
67c8f0
--- a/paramiko/common.py
67c8f0
+++ b/paramiko/common.py
67c8f0
@@ -32,6 +32,7 @@ MSG_USERAUTH_INFO_REQUEST, MSG_USERAUTH_INFO_RESPONSE = range(60, 62)
67c8f0
 MSG_USERAUTH_GSSAPI_RESPONSE, MSG_USERAUTH_GSSAPI_TOKEN = range(60, 62)
67c8f0
 MSG_USERAUTH_GSSAPI_EXCHANGE_COMPLETE, MSG_USERAUTH_GSSAPI_ERROR,\
67c8f0
 MSG_USERAUTH_GSSAPI_ERRTOK, MSG_USERAUTH_GSSAPI_MIC = range(63, 67)
67c8f0
+HIGHEST_USERAUTH_MESSAGE_ID = 79
67c8f0
 MSG_GLOBAL_REQUEST, MSG_REQUEST_SUCCESS, MSG_REQUEST_FAILURE = range(80, 83)
67c8f0
 MSG_CHANNEL_OPEN, MSG_CHANNEL_OPEN_SUCCESS, MSG_CHANNEL_OPEN_FAILURE, \
67c8f0
     MSG_CHANNEL_WINDOW_ADJUST, MSG_CHANNEL_DATA, MSG_CHANNEL_EXTENDED_DATA, \
67c8f0
diff --git a/paramiko/transport.py b/paramiko/transport.py
67c8f0
index 7906c9f..31df82a 100644
67c8f0
--- a/paramiko/transport.py
67c8f0
+++ b/paramiko/transport.py
67c8f0
@@ -49,7 +49,8 @@ from paramiko.common import xffffffff, cMSG_CHANNEL_OPEN, cMSG_IGNORE, \
67c8f0
     MSG_CHANNEL_SUCCESS, MSG_CHANNEL_FAILURE, MSG_CHANNEL_DATA, \
67c8f0
     MSG_CHANNEL_EXTENDED_DATA, MSG_CHANNEL_WINDOW_ADJUST, MSG_CHANNEL_REQUEST, \
67c8f0
     MSG_CHANNEL_EOF, MSG_CHANNEL_CLOSE, MIN_WINDOW_SIZE, MIN_PACKET_SIZE, \
67c8f0
-    MAX_WINDOW_SIZE, DEFAULT_WINDOW_SIZE, DEFAULT_MAX_PACKET_SIZE
67c8f0
+    MAX_WINDOW_SIZE, DEFAULT_WINDOW_SIZE, DEFAULT_MAX_PACKET_SIZE, \
67c8f0
+    HIGHEST_USERAUTH_MESSAGE_ID
67c8f0
 from paramiko.compress import ZlibCompressor, ZlibDecompressor
67c8f0
 from paramiko.dsskey import DSSKey
67c8f0
 from paramiko.kex_gex import KexGex, KexGexSHA256
67c8f0
@@ -1720,6 +1721,43 @@ class Transport (threading.Thread, ClosingContextManager):
67c8f0
             max_packet_size = self.default_max_packet_size
67c8f0
         return clamp_value(MIN_PACKET_SIZE, max_packet_size, MAX_WINDOW_SIZE)
67c8f0
 
67c8f0
+    def _ensure_authed(self, ptype, message):
67c8f0
+        """
67c8f0
+        Checks message type against current auth state.
67c8f0
+
67c8f0
+        If server mode, and auth has not succeeded, and the message is of a
67c8f0
+        post-auth type (channel open or global request) an appropriate error
67c8f0
+        response Message is crafted and returned to caller for sending.
67c8f0
+
67c8f0
+        Otherwise (client mode, authed, or pre-auth message) returns None.
67c8f0
+        """
67c8f0
+        if (
67c8f0
+            not self.server_mode
67c8f0
+            or ptype <= HIGHEST_USERAUTH_MESSAGE_ID
67c8f0
+            or self.is_authenticated()
67c8f0
+        ):
67c8f0
+            return None
67c8f0
+        # WELP. We must be dealing with someone trying to do non-auth things
67c8f0
+        # without being authed. Tell them off, based on message class.
67c8f0
+        reply = Message()
67c8f0
+        # Global requests have no details, just failure.
67c8f0
+        if ptype == MSG_GLOBAL_REQUEST:
67c8f0
+            reply.add_byte(cMSG_REQUEST_FAILURE)
67c8f0
+        # Channel opens let us reject w/ a specific type + message.
67c8f0
+        elif ptype == MSG_CHANNEL_OPEN:
67c8f0
+            kind = message.get_text()
67c8f0
+            chanid = message.get_int()
67c8f0
+            reply.add_byte(cMSG_CHANNEL_OPEN_FAILURE)
67c8f0
+            reply.add_int(chanid)
67c8f0
+            reply.add_int(OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED)
67c8f0
+            reply.add_string('')
67c8f0
+            reply.add_string('en')
67c8f0
+        # NOTE: Post-open channel messages do not need checking; the above will
67c8f0
+        # reject attemps to open channels, meaning that even if a malicious
67c8f0
+        # user tries to send a MSG_CHANNEL_REQUEST, it will simply fall under
67c8f0
+        # the logic that handles unknown channel IDs (as the channel list will
67c8f0
+        # be empty.)
67c8f0
+        return reply
67c8f0
 
67c8f0
     def run(self):
67c8f0
         # (use the exposed "run" method, because if we specify a thread target
67c8f0
@@ -1779,7 +1817,11 @@ class Transport (threading.Thread, ClosingContextManager):
67c8f0
                             continue
67c8f0
 
67c8f0
                     if ptype in self._handler_table:
67c8f0
-                        self._handler_table[ptype](self, m)
67c8f0
+                        error_msg = self._ensure_authed(ptype, m)
67c8f0
+                        if error_msg:
67c8f0
+                            self._send_message(error_msg)
67c8f0
+                        else:
67c8f0
+                            self._handler_table[ptype](self, m)
67c8f0
                     elif ptype in self._channel_handler_table:
67c8f0
                         chanid = m.get_int()
67c8f0
                         chan = self._channels.get(chanid)
67c8f0
diff --git a/tests/test_transport.py b/tests/test_transport.py
67c8f0
index d81ad8f..1305cd5 100644
67c8f0
--- a/tests/test_transport.py
67c8f0
+++ b/tests/test_transport.py
67c8f0
@@ -32,7 +32,7 @@ from hashlib import sha1
67c8f0
 import unittest
67c8f0
 
67c8f0
 from paramiko import Transport, SecurityOptions, ServerInterface, RSAKey, DSSKey, \
67c8f0
-    SSHException, ChannelException, Packetizer
67c8f0
+    SSHException, ChannelException, Packetizer, Channel
67c8f0
 from paramiko import AUTH_FAILED, AUTH_SUCCESSFUL
67c8f0
 from paramiko import OPEN_SUCCEEDED, OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED
67c8f0
 from paramiko.common import MSG_KEXINIT, cMSG_CHANNEL_WINDOW_ADJUST, \
67c8f0
@@ -87,7 +87,11 @@ class NullServer (ServerInterface):
67c8f0
 
67c8f0
     def check_global_request(self, kind, msg):
67c8f0
         self._global_request = kind
67c8f0
-        return False
67c8f0
+        # NOTE: for w/e reason, older impl of this returned False always, even
67c8f0
+        # tho that's only supposed to occur if the request cannot be served.
67c8f0
+        # For now, leaving that the default unless test supplies specific
67c8f0
+        # 'acceptable' request kind
67c8f0
+        return kind == 'acceptable'
67c8f0
 
67c8f0
     def check_channel_x11_request(self, channel, single_connection, auth_protocol, auth_cookie, screen_number):
67c8f0
         self._x11_single_connection = single_connection
67c8f0
@@ -125,7 +129,9 @@ class TransportTest(unittest.TestCase):
67c8f0
         self.socks.close()
67c8f0
         self.sockc.close()
67c8f0
 
67c8f0
-    def setup_test_server(self, client_options=None, server_options=None):
67c8f0
+    def setup_test_server(
67c8f0
+        self, client_options=None, server_options=None, connect_kwargs=None,
67c8f0
+    ):
67c8f0
         host_key = RSAKey.from_private_key_file(test_path('test_rsa.key'))
67c8f0
         public_host_key = RSAKey(data=host_key.asbytes())
67c8f0
         self.ts.add_server_key(host_key)
67c8f0
@@ -139,8 +145,13 @@ class TransportTest(unittest.TestCase):
67c8f0
         self.server = NullServer()
67c8f0
         self.assertTrue(not event.is_set())
67c8f0
         self.ts.start_server(event, self.server)
67c8f0
-        self.tc.connect(hostkey=public_host_key,
67c8f0
-                        username='slowdive', password='pygmalion')
67c8f0
+        if connect_kwargs is None:
67c8f0
+            connect_kwargs = dict(
67c8f0
+                hostkey=public_host_key,
67c8f0
+                username='slowdive',
67c8f0
+                password='pygmalion',
67c8f0
+            )
67c8f0
+        self.tc.connect(**connect_kwargs)
67c8f0
         event.wait(1.0)
67c8f0
         self.assertTrue(event.is_set())
67c8f0
         self.assertTrue(self.ts.is_active())
67c8f0
@@ -846,3 +857,37 @@ class TransportTest(unittest.TestCase):
67c8f0
         self.assertEqual([chan], r)
67c8f0
         self.assertEqual([], w)
67c8f0
         self.assertEqual([], e)
67c8f0
+
67c8f0
+    def test_server_rejects_open_channel_without_auth(self):
67c8f0
+        try:
67c8f0
+            self.setup_test_server(connect_kwargs={})
67c8f0
+            self.tc.open_session()
67c8f0
+        except ChannelException as e:
67c8f0
+            assert e.code == OPEN_FAILED_ADMINISTRATIVELY_PROHIBITED
67c8f0
+        else:
67c8f0
+            assert False, "Did not raise ChannelException!"
67c8f0
+
67c8f0
+    def test_server_rejects_arbitrary_global_request_without_auth(self):
67c8f0
+        self.setup_test_server(connect_kwargs={})
67c8f0
+        # NOTE: this dummy global request kind would normally pass muster
67c8f0
+        # from the test server.
67c8f0
+        self.tc.global_request('acceptable')
67c8f0
+        # Global requests never raise exceptions, even on failure (not sure why
67c8f0
+        # this was the original design...ugh.) Best we can do to tell failure
67c8f0
+        # happened is that the client transport's global_response was set back
67c8f0
+        # to None; if it had succeeded, it would be the response Message.
67c8f0
+        err = "Unauthed global response incorrectly succeeded!"
67c8f0
+        assert self.tc.global_response is None, err
67c8f0
+
67c8f0
+    def test_server_rejects_port_forward_without_auth(self):
67c8f0
+        # NOTE: at protocol level port forward requests are treated same as a
67c8f0
+        # regular global request, but Paramiko server implements a special-case
67c8f0
+        # method for it, so it gets its own test. (plus, THAT actually raises
67c8f0
+        # an exception on the client side, unlike the general case...)
67c8f0
+        self.setup_test_server(connect_kwargs={})
67c8f0
+        try:
67c8f0
+            self.tc.request_port_forward('localhost', 1234)
67c8f0
+        except SSHException as e:
67c8f0
+            assert "forwarding request denied" in str(e)
67c8f0
+        else:
67c8f0
+            assert False, "Did not raise SSHException!"