tests: Add createVHBAByStoragePool-by-parent to fchosttest

Add a new test to fchosttest in order to test creation of our vHBA
via the Storage Pool logic.  Unlike the real code, we cannot yet use
the virVHBA* API's because they (currently) traverse the file system
in order to get the parent vport capable scsi_host. Besides there's
no "real" NPIV device here - so we have to take some liberties, at
least for now.

Instead, we'll follow the node device tests partially in order to
create and destroy the vHBA with the test node devices.

Signed-off-by: John Ferlan <jferlan@redhat.com>
This commit is contained in:
John Ferlan 2017-01-29 11:12:48 -05:00
parent e915942b05
commit 0623945c40
2 changed files with 156 additions and 1 deletions

View File

@ -4362,6 +4362,34 @@ testConnectFindStoragePoolSources(virConnectPtr conn ATTRIBUTE_UNUSED,
} }
static virNodeDeviceObjPtr
testNodeDeviceMockCreateVport(testDriverPtr driver,
const char *wwnn,
const char *wwpn);
static int
testCreateVport(testDriverPtr driver,
const char *wwnn,
const char *wwpn)
{
virNodeDeviceObjPtr obj = NULL;
/* The storage_backend_scsi createVport() will use the input adapter
* fields parent name, parent_wwnn/parent_wwpn, or parent_fabric_wwn
* in order to determine whether the provided parent can be used to
* create a vHBA or will find "an available vport capable" to create
* a vHBA. In order to do this, it uses the virVHBA* API's which traverse
* the sysfs looking at various fields (rather than going via nodedev).
*
* Since the test environ doesn't have the sysfs for the storage pool
* test, at least for now use the node device test infrastructure to
* create the vHBA. In the long run the result is the same. */
if (!(obj = testNodeDeviceMockCreateVport(driver, wwnn, wwpn)))
return -1;
virNodeDeviceObjUnlock(obj);
return 0;
}
static virStoragePoolPtr static virStoragePoolPtr
testStoragePoolCreateXML(virConnectPtr conn, testStoragePoolCreateXML(virConnectPtr conn,
const char *xml, const char *xml,
@ -4392,6 +4420,21 @@ testStoragePoolCreateXML(virConnectPtr conn,
goto cleanup; goto cleanup;
def = NULL; def = NULL;
if (pool->def->source.adapter.type ==
VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST) {
/* In the real code, we'd call virVHBAManageVport followed by
* find_new_device, but we cannot do that here since we're not
* mocking udev. The mock routine will copy an existing vHBA and
* rename a few fields to mock that. */
if (testCreateVport(privconn,
pool->def->source.adapter.data.fchost.wwnn,
pool->def->source.adapter.data.fchost.wwpn) < 0) {
virStoragePoolObjRemove(&privconn->pools, pool);
pool = NULL;
goto cleanup;
}
}
if (testStoragePoolObjSetDefaults(pool) == -1) { if (testStoragePoolObjSetDefaults(pool) == -1) {
virStoragePoolObjRemove(&privconn->pools, pool); virStoragePoolObjRemove(&privconn->pools, pool);
pool = NULL; pool = NULL;
@ -4522,6 +4565,44 @@ testStoragePoolBuild(virStoragePoolPtr pool,
} }
static int
testDestroyVport(testDriverPtr privconn,
const char *wwnn ATTRIBUTE_UNUSED,
const char *wwpn ATTRIBUTE_UNUSED)
{
int ret = -1;
virNodeDeviceObjPtr obj = NULL;
virObjectEventPtr event = NULL;
/* NB: Cannot use virVHBAGetHostByWWN (yet) like the storage_backend_scsi
* deleteVport() helper since that traverses the file system looking for
* the wwnn/wwpn. So our choice short term is to cheat and use the name
* (scsi_host12) we know was created.
*
* Reaching across the boundaries of space and time into the
* Node Device in order to remove */
if (!(obj = virNodeDeviceObjFindByName(&privconn->devs, "scsi_host12"))) {
virReportError(VIR_ERR_NO_NODE_DEVICE, "%s",
_("no node device with matching name 'scsi_host12'"));
goto cleanup;
}
event = virNodeDeviceEventLifecycleNew("scsi_host12",
VIR_NODE_DEVICE_EVENT_DELETED,
0);
virNodeDeviceObjRemove(&privconn->devs, &obj);
ret = 0;
cleanup:
if (obj)
virNodeDeviceObjUnlock(obj);
testObjectEventQueue(privconn, event);
return ret;
}
static int static int
testStoragePoolDestroy(virStoragePoolPtr pool) testStoragePoolDestroy(virStoragePoolPtr pool)
{ {
@ -4540,7 +4621,17 @@ testStoragePoolDestroy(virStoragePoolPtr pool)
} }
privpool->active = 0; privpool->active = 0;
event = virStoragePoolEventLifecycleNew(privpool->def->name, privpool->def->uuid,
if (privpool->def->source.adapter.type ==
VIR_STORAGE_POOL_SOURCE_ADAPTER_TYPE_FC_HOST) {
if (testDestroyVport(privconn,
privpool->def->source.adapter.data.fchost.wwnn,
privpool->def->source.adapter.data.fchost.wwpn) < 0)
goto cleanup;
}
event = virStoragePoolEventLifecycleNew(privpool->def->name,
privpool->def->uuid,
VIR_STORAGE_POOL_EVENT_STOPPED, VIR_STORAGE_POOL_EVENT_STOPPED,
0); 0);

View File

@ -75,6 +75,19 @@ static const char test10_xml[] =
" </capability>" " </capability>"
"</device>"; "</device>";
/* virStoragePoolCreateXML using parent='%s' to find the vport capable HBA */
static const char test11_xml[] =
"<pool type='scsi'>"
" <name>vhba_pool</name>"
" <source>"
" <adapter type='fc_host' parent='scsi_host1' wwnn='20000000c9831b4b' wwpn='10000000c9831b4b'/>"
" </source>"
" <target>"
" <path>/dev/disk/by-path</path>"
" </target>"
"</pool>";
/* Test virIsVHBACapable */ /* Test virIsVHBACapable */
static int static int
test1(const void *data ATTRIBUTE_UNUSED) test1(const void *data ATTRIBUTE_UNUSED)
@ -275,6 +288,54 @@ manageVHBAByNodeDevice(const void *data)
} }
/* Test manageVHBAByStoragePool
* - Test both virStoragePoolCreateXML and virStoragePoolDestroy
* - Create a storage pool vHBA allowing usage of various different
* methods based on the input data/xml argument.
* - Be sure that it's possible to destroy the storage pool as well.
*/
static int
manageVHBAByStoragePool(const void *data)
{
const char *expect_hostname = "scsi_host12";
virConnectPtr conn = NULL;
virStoragePoolPtr pool = NULL;
virNodeDevicePtr dev = NULL;
int ret = -1;
const char *vhba = data;
if (!(conn = virConnectOpen("test:///default")))
return -1;
if (!(pool = virStoragePoolCreateXML(conn, vhba, 0)))
goto cleanup;
if (!(dev = virNodeDeviceLookupByName(conn, expect_hostname))) {
VIR_DEBUG("Failed to find expected_hostname '%s'", expect_hostname);
ignore_value(virStoragePoolDestroy(pool));
goto cleanup;
}
if (virStoragePoolDestroy(pool) < 0)
goto cleanup;
if ((dev = virNodeDeviceLookupByName(conn, expect_hostname))) {
VIR_DEBUG("Found expected_hostname '%s' after destroy",
expect_hostname);
goto cleanup;
}
ret = 0;
cleanup:
if (pool)
virStoragePoolFree(pool);
if (conn)
virConnectClose(conn);
return ret;
}
static int static int
mymain(void) mymain(void)
{ {
@ -310,6 +371,9 @@ mymain(void)
if (virTestRun("manageVHBAByNodeDevice-parent-fabric-wwn", if (virTestRun("manageVHBAByNodeDevice-parent-fabric-wwn",
manageVHBAByNodeDevice, test10_xml) < 0) manageVHBAByNodeDevice, test10_xml) < 0)
ret = -1; ret = -1;
if (virTestRun("manageVHBAByStoragePool-by-parent", manageVHBAByStoragePool,
test11_xml) < 0)
ret = -1;
cleanup: cleanup:
VIR_FREE(fchost_prefix); VIR_FREE(fchost_prefix);