Skip to content

Commit

Permalink
Merge pull request #41 from rackerlabs/new_agent_hearbeat
Browse files Browse the repository at this point in the history
RFR: New-Style Agent Heartbeat
  • Loading branch information
russellhaering committed Jan 28, 2014
2 parents 727feb2 + d371de6 commit c821fd7
Show file tree
Hide file tree
Showing 5 changed files with 68 additions and 19 deletions.
4 changes: 2 additions & 2 deletions teeth_agent/agent.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,7 @@ class TeethAgentHeartbeater(threading.Thread):
def __init__(self, agent):
super(TeethAgentHeartbeater, self).__init__()
self.agent = agent
self.hardware = hardware.get_manager()
self.api = overlord_agent_api.APIClient(agent.api_url)
self.log = structlog.get_logger(api_url=agent.api_url)
self.stop_event = threading.Event()
Expand All @@ -87,8 +88,7 @@ def run(self):
def do_heartbeat(self):
try:
deadline = self.api.heartbeat(
mac_addr=self.agent.get_agent_mac_addr(),
url=self.agent.get_agent_url(),
hardware_info=self.hardware.list_hardware_info(),
version=self.agent.version,
mode=self.agent.get_mode_name())
self.error_delay = self.initial_delay
Expand Down
26 changes: 26 additions & 0 deletions teeth_agent/hardware.py
Original file line number Diff line number Diff line change
Expand Up @@ -15,12 +15,15 @@
"""

import abc
import collections
import os

import plumbum
import stevedore
import structlog

from teeth_rest import encoding

_global_manager = None


Expand All @@ -37,6 +40,22 @@ class HardwareSupport(object):
SERVICE_PROVIDER = 3


class HardwareType(object):
MAC_ADDRESS = 'mac_address'


class HardwareInfo(encoding.Serializable):
def __init__(self, type, id):
self.type = type
self.id = id

def serialize(self, view):
return collections.OrderedDict([
('type', self.type),
('id', self.id),
])


class BlockDevice(object):
def __init__(self, name, size, start_sector):
self.name = name
Expand Down Expand Up @@ -66,6 +85,13 @@ def list_network_interfaces(self):
def get_os_install_device(self):
pass

def list_hardware_info(self):
hardware_info = []
for interface in self.list_network_interfaces():
hardware_info.append(HardwareInfo(HardwareType.MAC_ADDRESS,
interface.mac_address))
return hardware_info


class GenericHardwareManager(HardwareManager):
def __init__(self):
Expand Down
8 changes: 3 additions & 5 deletions teeth_agent/overlord_agent_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -47,13 +47,11 @@ def _request(self, method, path, data=None):
headers=request_headers,
data=data)

def heartbeat(self, mac_addr, url, mode, version):
path = '/{api_version}/agents/{mac_addr}'.format(
api_version=self.api_version,
mac_addr=mac_addr)
def heartbeat(self, hardware_info, mode, version):
path = '/{api_version}/agents'.format(api_version=self.api_version)

data = {
'url': url,
'hardware': hardware_info,
'mode': mode,
'version': version,
}
Expand Down
14 changes: 14 additions & 0 deletions teeth_agent/tests/hardware.py
Original file line number Diff line number Diff line change
Expand Up @@ -53,3 +53,17 @@ def test_get_os_install_device(self):
self.assertEqual(self.hardware.get_os_install_device(), '/dev/sdb')
self.hardware._cmd.assert_called_once_with('blockdev')
blockdev.assert_called_once_with('--report')

def test_list_hardwre_info(self):
self.hardware.list_network_interfaces = mock.Mock()
self.hardware.list_network_interfaces.return_value = [
hardware.NetworkInterface('eth0', '00:0c:29:8c:11:b1'),
hardware.NetworkInterface('eth1', '00:0c:29:8c:11:b2'),
]

hardware_info = self.hardware.list_hardware_info()
self.assertEqual(len(hardware_info), 2)
self.assertEqual(hardware_info[0].type, 'mac_address')
self.assertEqual(hardware_info[1].type, 'mac_address')
self.assertEqual(hardware_info[0].id, '00:0c:29:8c:11:b1')
self.assertEqual(hardware_info[1].id, '00:0c:29:8c:11:b2')
35 changes: 23 additions & 12 deletions teeth_agent/tests/overlord_agent_api.py
Original file line number Diff line number Diff line change
Expand Up @@ -21,6 +21,7 @@
import unittest

from teeth_agent import errors
from teeth_agent import hardware
from teeth_agent import overlord_agent_api

API_URL = 'http://agent-api.overlord.example.org/'
Expand All @@ -29,6 +30,12 @@
class TestBaseTeethAgent(unittest.TestCase):
def setUp(self):
self.api_client = overlord_agent_api.APIClient(API_URL)
self.hardware_info = [
hardware.HardwareInfo(hardware.HardwareType.MAC_ADDRESS,
'a:b:c:d'),
hardware.HardwareInfo(hardware.HardwareType.MAC_ADDRESS,
'0:1:2:3'),
]

def test_successful_heartbeat(self):
expected_heartbeat_before = time.time() + 120
Expand All @@ -40,31 +47,38 @@ def test_successful_heartbeat(self):
self.api_client.session.request.return_value = response

heartbeat_before = self.api_client.heartbeat(
url='http://1.2.3.4:9999/',
mac_addr='a:b:c:d',
hardware_info=self.hardware_info,
version='15',
mode='STANDBY')

self.assertEqual(heartbeat_before, expected_heartbeat_before)

request_args = self.api_client.session.request.call_args[0]
self.assertEqual(request_args[0], 'PUT')
self.assertEqual(request_args[1], API_URL + 'v1/agents/a:b:c:d')
self.assertEqual(request_args[1], API_URL + 'v1/agents')

data = self.api_client.session.request.call_args[1]['data']
content = json.loads(data)
self.assertEqual(content['url'], 'http://1.2.3.4:9999/')
self.assertEqual(content['mode'], 'STANDBY')
self.assertEqual(content['version'], '15')
self.assertEqual(content['hardware'], [
{
'type': 'mac_address',
'id': 'a:b:c:d',
},
{
'type': 'mac_address',
'id': '0:1:2:3',
},
])

def test_heartbeat_requests_exception(self):
self.api_client.session.request = mock.Mock()
self.api_client.session.request.side_effect = Exception('api is down!')

self.assertRaises(errors.HeartbeatError,
self.api_client.heartbeat,
url='http://1.2.3.4:9999/',
mac_addr='a:b:c:d',
hardware_info=self.hardware_info,
version='15',
mode='STANDBY')

Expand All @@ -75,8 +89,7 @@ def test_heartbeat_invalid_status_code(self):

self.assertRaises(errors.HeartbeatError,
self.api_client.heartbeat,
url='http://1.2.3.4:9999/',
mac_addr='a:b:c:d',
hardware_info=self.hardware_info,
version='15',
mode='STANDBY')

Expand All @@ -87,8 +100,7 @@ def test_heartbeat_missing_heartbeat_before_header(self):

self.assertRaises(errors.HeartbeatError,
self.api_client.heartbeat,
url='http://1.2.3.4:9999/',
mac_addr='a:b:c:d',
hardware_info=self.hardware_info,
version='15',
mode='STANDBY')

Expand All @@ -101,7 +113,6 @@ def test_heartbeat_invalid_heartbeat_before_header(self):

self.assertRaises(errors.HeartbeatError,
self.api_client.heartbeat,
url='http://1.2.3.4:9999/',
mac_addr='a:b:c:d',
hardware_info=self.hardware_info,
version='15',
mode='STANDBY')

0 comments on commit c821fd7

Please sign in to comment.