add a tool for recording notifications and replaying them
Change-Id: I852a6fbef7b9bf02309f699419da0a2537ce7a90
This commit is contained in:
parent
11a8416190
commit
8f4ba1656c
5
.gitignore
vendored
5
.gitignore
vendored
@ -1 +1,4 @@
|
||||
*.pyc
|
||||
*.pyc
|
||||
*.dat
|
||||
TAGS
|
||||
*.egg-info
|
||||
|
39
tests/test_tools_notificationclient.py
Normal file
39
tests/test_tools_notificationclient.py
Normal file
@ -0,0 +1,39 @@
|
||||
import os
|
||||
import cPickle as pickle
|
||||
from StringIO import StringIO
|
||||
import sys
|
||||
import types
|
||||
|
||||
import mox
|
||||
|
||||
from nova.rpc import impl_kombu
|
||||
|
||||
# The module being tested is part of the tools directory,
|
||||
# so make sure it is in our import path.
|
||||
sys.path.insert(0, os.path.normpath(os.path.join(os.path.dirname(__file__),
|
||||
'..', 'tools')))
|
||||
import notificationclient
|
||||
|
||||
|
||||
def test_send_messages():
|
||||
message = {'timestamp': 'date goes here',
|
||||
'event_type': 'compute.instance.exists',
|
||||
# real messages have more fields...
|
||||
}
|
||||
input = StringIO(pickle.dumps(message))
|
||||
conn = mox.MockObject(impl_kombu.Connection)
|
||||
conn.topic_send('notifications.info', message)
|
||||
mox.Replay(conn)
|
||||
notificationclient.send_messages(conn, input)
|
||||
mox.Verify(conn)
|
||||
return
|
||||
|
||||
|
||||
def test_record_messages():
|
||||
conn = mox.MockObject(impl_kombu.Connection)
|
||||
conn.declare_topic_consumer('notifications.info', mox.IsA(types.FunctionType))
|
||||
conn.consume()
|
||||
mox.Replay(conn)
|
||||
notificationclient.record_messages(conn, StringIO())
|
||||
mox.Verify(conn)
|
||||
return
|
124
tools/notificationclient.py
Executable file
124
tools/notificationclient.py
Executable file
@ -0,0 +1,124 @@
|
||||
#!/usr/bin/env python
|
||||
# -*- encoding: utf-8 -*-
|
||||
#
|
||||
# Copyright © 2012 New Dream Network, LLC (DreamHost)
|
||||
#
|
||||
# Author: Doug Hellmann <doug.hellmann@dreamhost.com>
|
||||
#
|
||||
# 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.
|
||||
|
||||
"""Command line tool for recording notification messages and replaying
|
||||
them later.
|
||||
"""
|
||||
|
||||
import argparse
|
||||
import logging
|
||||
import cPickle as pickle
|
||||
import sys
|
||||
|
||||
from nova import flags
|
||||
from nova import rpc
|
||||
from nova import utils
|
||||
from nova.openstack.common import cfg
|
||||
|
||||
FLAGS = flags.FLAGS
|
||||
LOG = logging.getLogger(__name__)
|
||||
|
||||
NOTIFICATION_TOPIC = 'notifications.info'
|
||||
|
||||
|
||||
def record_messages(connection, output):
|
||||
"""Listen to notification.info messages and pickle them to output."""
|
||||
def process_event(body):
|
||||
print ('%s: %s' %
|
||||
(body.get('timestamp'),
|
||||
body.get('event_type', 'unknown event'),
|
||||
))
|
||||
pickle.dump(body, output)
|
||||
|
||||
connection.declare_topic_consumer(NOTIFICATION_TOPIC, process_event)
|
||||
try:
|
||||
connection.consume()
|
||||
# for i in connection.iterconsume(5):
|
||||
# print 'iteration', i
|
||||
except KeyboardInterrupt:
|
||||
pass
|
||||
|
||||
|
||||
def send_messages(connection, input):
|
||||
"""Read messages from the input and send them to the AMQP queue."""
|
||||
while True:
|
||||
try:
|
||||
body = pickle.load(input)
|
||||
except EOFError:
|
||||
break
|
||||
print('%s: %s' %
|
||||
(body.get('timestamp'),
|
||||
body.get('event_type', 'unknown event'),
|
||||
))
|
||||
connection.topic_send(NOTIFICATION_TOPIC, body)
|
||||
|
||||
|
||||
def main():
|
||||
rpc.register_opts(FLAGS)
|
||||
FLAGS.register_opts([
|
||||
cfg.StrOpt('datafile',
|
||||
default=None,
|
||||
help='Data file to read or write',
|
||||
),
|
||||
cfg.BoolOpt('record',
|
||||
help='Record events',
|
||||
),
|
||||
cfg.BoolOpt('replay',
|
||||
help='Replay events',
|
||||
),
|
||||
])
|
||||
|
||||
remaining_args = FLAGS(sys.argv)
|
||||
utils.monkey_patch()
|
||||
|
||||
parser = argparse.ArgumentParser(
|
||||
description='record or play back notification events',
|
||||
)
|
||||
parser.add_argument('mode',
|
||||
choices=('record', 'replay'),
|
||||
help='operating mode',
|
||||
)
|
||||
parser.add_argument('data_file',
|
||||
help='the data file to read or write',
|
||||
)
|
||||
args = parser.parse_args(remaining_args[1:])
|
||||
|
||||
console = logging.StreamHandler(sys.stderr)
|
||||
console.setLevel(logging.DEBUG)
|
||||
formatter = logging.Formatter('%(message)s')
|
||||
console.setFormatter(formatter)
|
||||
root_logger = logging.getLogger('')
|
||||
root_logger.addHandler(console)
|
||||
root_logger.setLevel(logging.DEBUG)
|
||||
|
||||
connection = rpc.create_connection()
|
||||
try:
|
||||
if args.mode == 'replay':
|
||||
with open(args.data_file, 'rb') as input:
|
||||
send_messages(connection, input)
|
||||
elif args.mode == 'record':
|
||||
with open(args.data_file, 'wb') as output:
|
||||
record_messages(connection, output)
|
||||
finally:
|
||||
connection.close()
|
||||
|
||||
return 0
|
||||
|
||||
if __name__ == '__main__':
|
||||
main()
|
Loading…
Reference in New Issue
Block a user