6eb9f58c87
This adds the ability to store all of the data collected during introspection. The configuration option "[processing] store_data" (defaults to 'none'), determines this behavior. Initially, only 'none' and 'swift' are supported. If 'swift' is used, the data is stored in Swift with the object name of "inspector_data-<UUID>". Adds an endpoint /v1/introspection/<UUID>/data which retrieves the data according to the method in "[processing] store_data". Returns 404 if this option is disabled. There is a further option to store the location of the data in the Ironic Node.extra column. For 'swift', this will be the name of the swift object. The option, "[processing] store_data_location" determines the key name in the Node.extra column. (defaults to not storing the location). Change-Id: Ibc38064f7ea56f85b9f5a77ef6f62a50f0381ff4 Implements: blueprint store-introspection-data
168 lines
7.5 KiB
Python
168 lines
7.5 KiB
Python
# Copyright 2013 Hewlett-Packard Development Company, L.P.
|
|
#
|
|
# 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.
|
|
|
|
# Mostly copied from ironic/tests/test_swift.py
|
|
|
|
import sys
|
|
|
|
try:
|
|
from unittest import mock
|
|
except ImportError:
|
|
import mock
|
|
from oslo_config import cfg
|
|
from six.moves import reload_module
|
|
from swiftclient import client as swift_client
|
|
from swiftclient import exceptions as swift_exception
|
|
|
|
from ironic_inspector.common import swift
|
|
from ironic_inspector.test import base as test_base
|
|
from ironic_inspector import utils
|
|
|
|
CONF = cfg.CONF
|
|
|
|
|
|
class BaseTest(test_base.NodeTest):
|
|
def setUp(self):
|
|
super(BaseTest, self).setUp()
|
|
self.all_macs = self.macs + ['DE:AD:BE:EF:DE:AD']
|
|
self.pxe_mac = self.macs[1]
|
|
self.data = {
|
|
'ipmi_address': self.bmc_address,
|
|
'cpus': 2,
|
|
'cpu_arch': 'x86_64',
|
|
'memory_mb': 1024,
|
|
'local_gb': 20,
|
|
'interfaces': {
|
|
'em1': {'mac': self.macs[0], 'ip': '1.2.0.1'},
|
|
'em2': {'mac': self.macs[1], 'ip': '1.2.0.2'},
|
|
'em3': {'mac': self.all_macs[2]},
|
|
},
|
|
'boot_interface': '01-' + self.pxe_mac.replace(':', '-'),
|
|
}
|
|
|
|
|
|
@mock.patch.object(swift_client, 'Connection', autospec=True)
|
|
class SwiftTestCase(BaseTest):
|
|
|
|
def setUp(self):
|
|
super(SwiftTestCase, self).setUp()
|
|
self.swift_exception = swift_exception.ClientException('', '')
|
|
|
|
CONF.set_override('username', 'swift', 'swift')
|
|
CONF.set_override('tenant_name', 'tenant', 'swift')
|
|
CONF.set_override('password', 'password', 'swift')
|
|
CONF.set_override('os_auth_url', 'http://authurl/v2.0', 'swift')
|
|
CONF.set_override('os_auth_version', '2', 'swift')
|
|
CONF.set_override('max_retries', 2, 'swift')
|
|
CONF.set_override('os_service_type', 'object-store', 'swift')
|
|
CONF.set_override('os_endpoint_type', 'internalURL', 'swift')
|
|
|
|
# The constructor of SwiftAPI accepts arguments whose
|
|
# default values are values of some config options above. So reload
|
|
# the module to make sure the required values are set.
|
|
reload_module(sys.modules['ironic_inspector.common.swift'])
|
|
|
|
def test___init__(self, connection_mock):
|
|
swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
params = {'retries': 2,
|
|
'user': 'swift',
|
|
'tenant_name': 'tenant',
|
|
'key': 'password',
|
|
'authurl': 'http://authurl/v2.0',
|
|
'auth_version': '2',
|
|
'os_options': {'service_type': 'object-store',
|
|
'endpoint_type': 'internalURL'}}
|
|
connection_mock.assert_called_once_with(**params)
|
|
|
|
def test_create_object(self, connection_mock):
|
|
swiftapi = swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
connection_obj_mock = connection_mock.return_value
|
|
|
|
connection_obj_mock.put_object.return_value = 'object-uuid'
|
|
|
|
object_uuid = swiftapi.create_object('object', 'some-string-data')
|
|
|
|
connection_obj_mock.put_container.assert_called_once_with('ironic-'
|
|
'inspector')
|
|
connection_obj_mock.put_object.assert_called_once_with(
|
|
'ironic-inspector', 'object', 'some-string-data', headers=None)
|
|
self.assertEqual('object-uuid', object_uuid)
|
|
|
|
def test_create_object_create_container_fails(self, connection_mock):
|
|
swiftapi = swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
connection_obj_mock = connection_mock.return_value
|
|
connection_obj_mock.put_container.side_effect = self.swift_exception
|
|
self.assertRaises(utils.Error, swiftapi.create_object, 'object',
|
|
'some-string-data')
|
|
connection_obj_mock.put_container.assert_called_once_with('ironic-'
|
|
'inspector')
|
|
self.assertFalse(connection_obj_mock.put_object.called)
|
|
|
|
def test_create_object_put_object_fails(self, connection_mock):
|
|
swiftapi = swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
connection_obj_mock = connection_mock.return_value
|
|
connection_obj_mock.put_object.side_effect = self.swift_exception
|
|
self.assertRaises(utils.Error, swiftapi.create_object, 'object',
|
|
'some-string-data')
|
|
connection_obj_mock.put_container.assert_called_once_with('ironic-'
|
|
'inspector')
|
|
connection_obj_mock.put_object.assert_called_once_with(
|
|
'ironic-inspector', 'object', 'some-string-data', headers=None)
|
|
|
|
def test_get_object(self, connection_mock):
|
|
swiftapi = swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
connection_obj_mock = connection_mock.return_value
|
|
|
|
expected_obj = self.data
|
|
connection_obj_mock.get_object.return_value = ('headers', expected_obj)
|
|
|
|
swift_obj = swiftapi.get_object('object')
|
|
|
|
connection_obj_mock.get_object.assert_called_once_with(
|
|
'ironic-inspector', 'object')
|
|
self.assertEqual(expected_obj, swift_obj)
|
|
|
|
def test_get_object_fails(self, connection_mock):
|
|
swiftapi = swift.SwiftAPI(user=CONF.swift.username,
|
|
tenant_name=CONF.swift.tenant_name,
|
|
key=CONF.swift.password,
|
|
auth_url=CONF.swift.os_auth_url,
|
|
auth_version=CONF.swift.os_auth_version)
|
|
connection_obj_mock = connection_mock.return_value
|
|
connection_obj_mock.get_object.side_effect = self.swift_exception
|
|
self.assertRaises(utils.Error, swiftapi.get_object,
|
|
'object')
|
|
connection_obj_mock.get_object.assert_called_once_with(
|
|
'ironic-inspector', 'object')
|