mirror of https://github.com/OpenKMIP/PyKMIP.git
Merge pull request #110 from viktorTarasov/feature/services-server-use-EOF
services/server: use EOF exception
This commit is contained in:
commit
abd11c6327
|
@ -36,8 +36,15 @@ class KMIPProtocol(object):
|
||||||
self.socket.sendall(sbuffer)
|
self.socket.sendall(sbuffer)
|
||||||
|
|
||||||
def read(self):
|
def read(self):
|
||||||
|
try:
|
||||||
header = self._recv_all(self.HEADER_SIZE)
|
header = self._recv_all(self.HEADER_SIZE)
|
||||||
|
except RequestLengthMismatch as e:
|
||||||
|
if e.received == 0:
|
||||||
|
raise EOFError("No request to process")
|
||||||
|
else:
|
||||||
|
raise
|
||||||
msg_size = unpack('!I', header[4:])[0]
|
msg_size = unpack('!I', header[4:])[0]
|
||||||
|
|
||||||
payload = self._recv_all(msg_size)
|
payload = self._recv_all(msg_size)
|
||||||
data = BytearrayStream(header + payload)
|
data = BytearrayStream(header + payload)
|
||||||
self.logger.debug('KMIPProtocol.read: {0}'.format(
|
self.logger.debug('KMIPProtocol.read: {0}'.format(
|
||||||
|
@ -54,8 +61,10 @@ class KMIPProtocol(object):
|
||||||
bytes_read += len(msg)
|
bytes_read += len(msg)
|
||||||
total_msg += msg
|
total_msg += msg
|
||||||
if bytes_read != total_bytes_to_be_read:
|
if bytes_read != total_bytes_to_be_read:
|
||||||
raise Exception("Expected {0} bytes, Received {1} bytes"
|
msg = "expected {0}, received {1} bytes".format(
|
||||||
.format(total_bytes_to_be_read, bytes_read))
|
total_bytes_to_be_read, bytes_read)
|
||||||
|
raise RequestLengthMismatch(total_bytes_to_be_read, bytes_read)
|
||||||
|
|
||||||
return total_msg
|
return total_msg
|
||||||
|
|
||||||
|
|
||||||
|
@ -63,3 +72,22 @@ class KMIPProtocolFactory(object):
|
||||||
|
|
||||||
def getProtocol(self, socket):
|
def getProtocol(self, socket):
|
||||||
return KMIPProtocol(socket)
|
return KMIPProtocol(socket)
|
||||||
|
|
||||||
|
|
||||||
|
class RequestLengthMismatch(Exception):
|
||||||
|
"""
|
||||||
|
This exception raised when the request read from stream has unexpected
|
||||||
|
length.
|
||||||
|
"""
|
||||||
|
def __init__(self, expected, received, message="KMIPProtocol read error"):
|
||||||
|
super(RequestLengthMismatch, self).__init__(message)
|
||||||
|
self.message = message
|
||||||
|
self.expected = expected
|
||||||
|
self.received = received
|
||||||
|
|
||||||
|
def __str__(self):
|
||||||
|
return "{0}: expected {1}, received {2}".format(
|
||||||
|
self.message, self.expected, self.received)
|
||||||
|
|
||||||
|
def __repr__(self):
|
||||||
|
return self.__str__()
|
||||||
|
|
|
@ -53,6 +53,7 @@ class KMIPServer(object):
|
||||||
self.socket.listen(0)
|
self.socket.listen(0)
|
||||||
while True:
|
while True:
|
||||||
connection, address = self.socket.accept()
|
connection, address = self.socket.accept()
|
||||||
|
self.logger.info("Connected by {0}".format(address))
|
||||||
connection = ssl.wrap_socket(
|
connection = ssl.wrap_socket(
|
||||||
connection,
|
connection,
|
||||||
keyfile=self.keyfile,
|
keyfile=self.keyfile,
|
||||||
|
@ -70,9 +71,13 @@ class KMIPServer(object):
|
||||||
try:
|
try:
|
||||||
while True:
|
while True:
|
||||||
self._processor.process(protocol, protocol)
|
self._processor.process(protocol, protocol)
|
||||||
|
except EOFError as e:
|
||||||
|
self.logger.warning("KMIPServer {0} {1}".format(type(e), e))
|
||||||
except Exception as e:
|
except Exception as e:
|
||||||
self.logger.error('KMIPServer {0} {1}'.format(type(e), e))
|
self.logger.error('KMIPServer {0} {1}'.format(type(e), e))
|
||||||
|
finally:
|
||||||
connection.close()
|
connection.close()
|
||||||
|
self.logger.info('Connection closed')
|
||||||
|
|
||||||
def _set_variables(self, host, port, keyfile, certfile, cert_reqs,
|
def _set_variables(self, host, port, keyfile, certfile, cert_reqs,
|
||||||
ssl_version, ca_certs, do_handshake_on_connect,
|
ssl_version, ca_certs, do_handshake_on_connect,
|
||||||
|
|
|
@ -0,0 +1,128 @@
|
||||||
|
# Copyright (c) 2014 The Johns Hopkins University/Applied Physics Laboratory
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from mock import call, MagicMock
|
||||||
|
from testtools import TestCase
|
||||||
|
|
||||||
|
import binascii
|
||||||
|
|
||||||
|
from kmip.services.server.kmip_protocol import KMIPProtocol
|
||||||
|
from kmip.services.server.kmip_protocol import RequestLengthMismatch
|
||||||
|
from kmip.services.server.kmip_protocol import KMIPProtocolFactory
|
||||||
|
|
||||||
|
|
||||||
|
class TestKMIPProtocol(TestCase):
|
||||||
|
|
||||||
|
request = binascii.unhexlify(
|
||||||
|
'42007801000000b04200770100000088420069010000002042006a02000000040'
|
||||||
|
'00000010000000042006b0200000004000000010000000042000c010000004842'
|
||||||
|
'00230100000040420024050000000400000001000000004200250100000028420'
|
||||||
|
'099070000000a4b6d6970436c69656e740000000000004200a10700000006436f'
|
||||||
|
'75636f75000042000d0200000004000000010000000042000f010000001842005'
|
||||||
|
'c05000000040000001e000000004200790100000000')
|
||||||
|
response = binascii.unhexlify(
|
||||||
|
'42007b01000000d042007a0100000048420069010000002042006a02000000040'
|
||||||
|
'00000010000000042006b02000000040000000100000000420092090000000800'
|
||||||
|
'00000056bda8eb42000d0200000004000000010000000042000f0100000078420'
|
||||||
|
'05c05000000040000001e0000000042007f050000000400000000000000004200'
|
||||||
|
'7c0100000050420069010000002042006a0200000004000000010000000042006'
|
||||||
|
'b02000000040000000100000000420069010000002042006a0200000004000000'
|
||||||
|
'010000000042006b02000000040000000000000000')
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(TestKMIPProtocol, self).setUp()
|
||||||
|
self.factory = KMIPProtocolFactory()
|
||||||
|
|
||||||
|
def tearDown(self):
|
||||||
|
super(TestKMIPProtocol, self).tearDown()
|
||||||
|
|
||||||
|
def test_init(self):
|
||||||
|
"""
|
||||||
|
Test that a KmipProtocol can be created without errors.
|
||||||
|
"""
|
||||||
|
socket = MagicMock()
|
||||||
|
KMIPProtocol(socket)
|
||||||
|
|
||||||
|
def test_protocol_factory(self):
|
||||||
|
mock_name = 'test_protocol_factory'
|
||||||
|
socket = MagicMock(mock_name=mock_name)
|
||||||
|
protocol = self.factory.getProtocol(socket)
|
||||||
|
|
||||||
|
base = "expected {0}, received {1}"
|
||||||
|
msg = base.format(KMIPProtocol, protocol)
|
||||||
|
self.assertIsInstance(protocol, KMIPProtocol, msg)
|
||||||
|
self.assertEqual(protocol.socket.mock_name, mock_name, msg)
|
||||||
|
|
||||||
|
def test_IO_write(self):
|
||||||
|
socket = MagicMock()
|
||||||
|
protocol = self.factory.getProtocol(socket)
|
||||||
|
protocol.logger = MagicMock()
|
||||||
|
protocol.write(self.request)
|
||||||
|
|
||||||
|
protocol.logger.debug.assert_any_call(
|
||||||
|
"KMIPProtocol.write: {0}".format(binascii.hexlify(self.request)))
|
||||||
|
protocol.socket.sendall.assert_called_once_with(self.request)
|
||||||
|
|
||||||
|
def test_IO_read(self):
|
||||||
|
socket = MagicMock()
|
||||||
|
socket.recv = MagicMock(
|
||||||
|
side_effect=[self.response[:8], self.response[8:]])
|
||||||
|
protocol = self.factory.getProtocol(socket)
|
||||||
|
|
||||||
|
received = protocol.read()
|
||||||
|
|
||||||
|
socket.recv.assert_any_call(8)
|
||||||
|
socket.recv.assert_any_call(len(self.response) - 8)
|
||||||
|
|
||||||
|
self.assertEqual(self.response, received.peek())
|
||||||
|
|
||||||
|
def test_IO_read_EOF(self):
|
||||||
|
socket = MagicMock()
|
||||||
|
socket.recv = MagicMock(side_effect=[[]])
|
||||||
|
protocol = self.factory.getProtocol(socket)
|
||||||
|
|
||||||
|
try:
|
||||||
|
protocol.read()
|
||||||
|
except Exception as e:
|
||||||
|
self.assertIsInstance(e, EOFError, "Invalid exception")
|
||||||
|
else:
|
||||||
|
self.assertTrue(False, "Unexpected error")
|
||||||
|
|
||||||
|
socket.recv.assert_any_call(8)
|
||||||
|
|
||||||
|
def test_IO_read_request_length_mismatch(self):
|
||||||
|
socket = MagicMock()
|
||||||
|
socket.recv = MagicMock(
|
||||||
|
side_effect=[self.response[:8], self.response[8:16], []])
|
||||||
|
protocol = self.factory.getProtocol(socket)
|
||||||
|
resp_len = len(self.response)
|
||||||
|
|
||||||
|
try:
|
||||||
|
protocol.read()
|
||||||
|
except Exception as e:
|
||||||
|
self.assertIsInstance(
|
||||||
|
e, RequestLengthMismatch, "Invalid exception")
|
||||||
|
self.assertEqual(e.expected, resp_len - 8, "Unexpected expected")
|
||||||
|
self.assertEqual(e.received, 8, "Unexpected received")
|
||||||
|
self.assertEqual(
|
||||||
|
"{0}".format(e),
|
||||||
|
"{0}: expected {1}, received {2}".format(
|
||||||
|
"KMIPProtocol read error", resp_len - 8, 8),
|
||||||
|
"Invalid RequestLengthMismatch attributes")
|
||||||
|
else:
|
||||||
|
self.assertTrue(False, "Unexpected error")
|
||||||
|
|
||||||
|
calls = [call(8), call(resp_len - 8), call(resp_len - 16)]
|
||||||
|
socket.recv.assert_has_calls(calls)
|
Loading…
Reference in New Issue