VirtualBox

source: vbox/trunk/src/VBox/ValidationKit/tests/storage/tdStorageStress1.py@ 79449

Last change on this file since 79449 was 79092, checked in by vboxsync, 6 years ago

ValKit,++: Pylint 2.3.1 adjustments.

  • Property svn:eol-style set to LF
  • Property svn:executable set to *
  • Property svn:keywords set to Author Date Id Revision
File size: 22.8 KB
Line 
1#!/usr/bin/env python
2# -*- coding: utf-8 -*-
3
4"""
5Storage testcase using xfstests.
6"""
7
8__copyright__ = \
9"""
10Copyright (C) 2012-2019 Oracle Corporation
11
12This file is part of VirtualBox Open Source Edition (OSE), as
13available from http://www.virtualbox.org. This file is free software;
14you can redistribute it and/or modify it under the terms of the GNU
15General Public License (GPL) as published by the Free Software
16Foundation, in version 2 as it comes in the "COPYING" file of the
17VirtualBox OSE distribution. VirtualBox OSE is distributed in the
18hope that it will be useful, but WITHOUT ANY WARRANTY of any kind.
19
20The contents of this file may alternatively be used under the terms
21of the Common Development and Distribution License Version 1.0
22(CDDL) only, as it comes in the "COPYING.CDDL" file of the
23VirtualBox OSE distribution, in which case the provisions of the
24CDDL are applicable instead of those of the GPL.
25
26You may elect to license modified versions of this file under the
27terms and conditions of either the GPL or the CDDL or both.
28"""
29__version__ = "$Id: tdStorageStress1.py 79092 2019-06-11 15:26:40Z vboxsync $"
30
31
32# Standard Python imports.
33import os;
34import sys;
35
36# Only the main script needs to modify the path.
37try: __file__
38except: __file__ = sys.argv[0];
39g_ksValidationKitDir = os.path.dirname(os.path.dirname(os.path.dirname(os.path.abspath(__file__))));
40sys.path.append(g_ksValidationKitDir);
41
42# Validation Kit imports.
43from testdriver import reporter;
44from testdriver import base;
45from testdriver import vbox;
46from testdriver import vboxcon;
47
48
49class tdStorageStress(vbox.TestDriver): # pylint: disable=too-many-instance-attributes
50 """
51 Storage testcase.
52 """
53
54 def __init__(self):
55 vbox.TestDriver.__init__(self);
56 self.asRsrcs = None;
57 self.oGuestToGuestVM = None;
58 self.oGuestToGuestSess = None;
59 self.oGuestToGuestTxs = None;
60 self.asTestVMsDef = ['tst-debian'];
61 self.asTestVMs = self.asTestVMsDef;
62 self.asSkipVMs = [];
63 self.asVirtModesDef = ['hwvirt', 'hwvirt-np', 'raw',]
64 self.asVirtModes = self.asVirtModesDef
65 self.acCpusDef = [1, 2,]
66 self.acCpus = self.acCpusDef;
67 self.asStorageCtrlsDef = ['AHCI', 'IDE', 'LsiLogicSAS', 'LsiLogic', 'BusLogic'];
68 self.asStorageCtrls = self.asStorageCtrlsDef;
69 self.asDiskFormatsDef = ['VDI', 'VMDK', 'VHD', 'QED', 'Parallels', 'QCOW'];
70 self.asDiskFormats = self.asDiskFormatsDef;
71 self.asTestsDef = ['xfstests'];
72 self.asTests = self.asTestsDef;
73 self.asGuestFs = ['xfs', 'ext4', 'btrfs'];
74 self.asGuestFsDef = self.asGuestFs;
75 self.asIscsiTargetsDef = ['aurora|iqn.2011-03.home.aurora:aurora.storagebench|1'];
76 self.asIscsiTargets = self.asIscsiTargetsDef;
77 self.asDirsDef = ['/run/media/alexander/OWCSSD/alexander', \
78 '/run/media/alexander/CrucialSSD/alexander', \
79 '/run/media/alexander/HardDisk/alexander', \
80 '/home/alexander'];
81 self.asDirs = self.asDirsDef;
82
83 #
84 # Overridden methods.
85 #
86 def showUsage(self):
87 rc = vbox.TestDriver.showUsage(self);
88 reporter.log('');
89 reporter.log('tdStorageBenchmark1 Options:');
90 reporter.log(' --virt-modes <m1[:m2[:]]');
91 reporter.log(' Default: %s' % (':'.join(self.asVirtModesDef)));
92 reporter.log(' --cpu-counts <c1[:c2[:]]');
93 reporter.log(' Default: %s' % (':'.join(str(c) for c in self.acCpusDef)));
94 reporter.log(' --storage-ctrls <type1[:type2[:...]]>');
95 reporter.log(' Default: %s' % (':'.join(self.asStorageCtrls)));
96 reporter.log(' --disk-formats <type1[:type2[:...]]>');
97 reporter.log(' Default: %s' % (':'.join(self.asDiskFormats)));
98 reporter.log(' --disk-dirs <path1[:path2[:...]]>');
99 reporter.log(' Default: %s' % (':'.join(self.asDirs)));
100 reporter.log(' --iscsi-targets <target1[:target2[:...]]>');
101 reporter.log(' Default: %s' % (':'.join(self.asIscsiTargets)));
102 reporter.log(' --tests <test1[:test2[:...]]>');
103 reporter.log(' Default: %s' % (':'.join(self.asTests)));
104 reporter.log(' --guest-fs <fs1[:fs2[:...]]>');
105 reporter.log(' Default: %s' % (':'.join(self.asGuestFs)));
106 reporter.log(' --test-vms <vm1[:vm2[:...]]>');
107 reporter.log(' Test the specified VMs in the given order. Use this to change');
108 reporter.log(' the execution order or limit the choice of VMs');
109 reporter.log(' Default: %s (all)' % (':'.join(self.asTestVMsDef)));
110 reporter.log(' --skip-vms <vm1[:vm2[:...]]>');
111 reporter.log(' Skip the specified VMs when testing.');
112 return rc;
113
114 def parseOption(self, asArgs, iArg): # pylint: disable=too-many-branches,too-many-statements
115 if asArgs[iArg] == '--virt-modes':
116 iArg += 1;
117 if iArg >= len(asArgs): raise base.InvalidOption('The "--virt-modes" takes a colon separated list of modes');
118 self.asVirtModes = asArgs[iArg].split(':');
119 for s in self.asVirtModes:
120 if s not in self.asVirtModesDef:
121 raise base.InvalidOption('The "--virt-modes" value "%s" is not valid; valid values are: %s' \
122 % (s, ' '.join(self.asVirtModesDef)));
123 elif asArgs[iArg] == '--cpu-counts':
124 iArg += 1;
125 if iArg >= len(asArgs): raise base.InvalidOption('The "--cpu-counts" takes a colon separated list of cpu counts');
126 self.acCpus = [];
127 for s in asArgs[iArg].split(':'):
128 try: c = int(s);
129 except: raise base.InvalidOption('The "--cpu-counts" value "%s" is not an integer' % (s,));
130 if c <= 0: raise base.InvalidOption('The "--cpu-counts" value "%s" is zero or negative' % (s,));
131 self.acCpus.append(c);
132 elif asArgs[iArg] == '--storage-ctrls':
133 iArg += 1;
134 if iArg >= len(asArgs):
135 raise base.InvalidOption('The "--storage-ctrls" takes a colon separated list of Storage controller types');
136 self.asStorageCtrls = asArgs[iArg].split(':');
137 elif asArgs[iArg] == '--disk-formats':
138 iArg += 1;
139 if iArg >= len(asArgs): raise base.InvalidOption('The "--disk-formats" takes a colon separated list of disk formats');
140 self.asDiskFormats = asArgs[iArg].split(':');
141 elif asArgs[iArg] == '--disk-dirs':
142 iArg += 1;
143 if iArg >= len(asArgs): raise base.InvalidOption('The "--disk-dirs" takes a colon separated list of directories');
144 self.asDirs = asArgs[iArg].split(':');
145 elif asArgs[iArg] == '--iscsi-targets':
146 iArg += 1;
147 if iArg >= len(asArgs):
148 raise base.InvalidOption('The "--iscsi-targets" takes a colon separated list of iscsi targets');
149 self.asIscsiTargets = asArgs[iArg].split(':');
150 elif asArgs[iArg] == '--tests':
151 iArg += 1;
152 if iArg >= len(asArgs): raise base.InvalidOption('The "--tests" takes a colon separated list of disk formats');
153 self.asTests = asArgs[iArg].split(':');
154 elif asArgs[iArg] == '--guest-fs':
155 iArg += 1;
156 if iArg >= len(asArgs):
157 raise base.InvalidOption('The "--guest-fs" takes a colon separated list of filesystem identifiers');
158 self.asGuestFs = asArgs[iArg].split(':');
159 elif asArgs[iArg] == '--test-vms':
160 iArg += 1;
161 if iArg >= len(asArgs): raise base.InvalidOption('The "--test-vms" takes colon separated list');
162 self.asTestVMs = asArgs[iArg].split(':');
163 for s in self.asTestVMs:
164 if s not in self.asTestVMsDef:
165 raise base.InvalidOption('The "--test-vms" value "%s" is not valid; valid values are: %s' \
166 % (s, ' '.join(self.asTestVMsDef)));
167 elif asArgs[iArg] == '--skip-vms':
168 iArg += 1;
169 if iArg >= len(asArgs): raise base.InvalidOption('The "--skip-vms" takes colon separated list');
170 self.asSkipVMs = asArgs[iArg].split(':');
171 for s in self.asSkipVMs:
172 if s not in self.asTestVMsDef:
173 reporter.log('warning: The "--test-vms" value "%s" does not specify any of our test VMs.' % (s));
174 else:
175 return vbox.TestDriver.parseOption(self, asArgs, iArg);
176 return iArg + 1;
177
178 def completeOptions(self):
179 # Remove skipped VMs from the test list.
180 for sVM in self.asSkipVMs:
181 try: self.asTestVMs.remove(sVM);
182 except: pass;
183
184 return vbox.TestDriver.completeOptions(self);
185
186 def getResourceSet(self):
187 # Construct the resource list the first time it's queried.
188 if self.asRsrcs is None:
189 self.asRsrcs = [];
190 if 'tst-debian' in self.asTestVMs:
191 self.asRsrcs.append('4.2/storage/debian.vdi');
192
193 return self.asRsrcs;
194
195 def actionConfig(self):
196 # Some stupid trickery to guess the location of the iso. ## fixme - testsuite unzip ++
197 sVBoxValidationKit_iso = os.path.abspath(os.path.join(os.path.dirname(__file__),
198 '../../VBoxValidationKitStorIo.iso'));
199 if not os.path.isfile(sVBoxValidationKit_iso):
200 sVBoxValidationKit_iso = os.path.abspath(os.path.join(os.path.dirname(__file__),
201 '../../VBoxValidationKitStorIo.iso'));
202 if not os.path.isfile(sVBoxValidationKit_iso):
203 sVBoxValidationKit_iso = '/mnt/ramdisk/vbox/svn/trunk/validationkit/VBoxValidationKitStorIo.iso';
204 if not os.path.isfile(sVBoxValidationKit_iso):
205 sVBoxValidationKit_iso = '/mnt/ramdisk/vbox/svn/trunk/testsuite/VBoxTestSuiteStorIo.iso';
206 if not os.path.isfile(sVBoxValidationKit_iso):
207 sCur = os.getcwd();
208 for i in range(0, 10):
209 sVBoxValidationKit_iso = os.path.join(sCur, 'validationkit/VBoxValidationKitStorIo.iso');
210 if os.path.isfile(sVBoxValidationKit_iso):
211 break;
212 sVBoxValidationKit_iso = os.path.join(sCur, 'testsuite/VBoxTestSuiteStorIo.iso');
213 if os.path.isfile(sVBoxValidationKit_iso):
214 break;
215 sCur = os.path.abspath(os.path.join(sCur, '..'));
216 if i is None: pass; # shut up pychecker/pylint.
217 if not os.path.isfile(sVBoxValidationKit_iso):
218 sVBoxValidationKit_iso = '/mnt/VirtualBox/VBoxValidationKitStorIo.iso';
219 if not os.path.isfile(sVBoxValidationKit_iso):
220 sVBoxValidationKit_iso = '/mnt/VirtualBox/VBoxTestSuiteStorIo.iso';
221
222
223
224 # Make sure vboxapi has been imported so we can use the constants.
225 if not self.importVBoxApi():
226 return False;
227
228 #
229 # Configure the VMs we're going to use.
230 #
231
232 # Linux VMs
233 if 'tst-debian' in self.asTestVMs:
234 oVM = self.createTestVM('tst-debian', 1, '4.2/storage/debian.vdi', sKind = 'Debian_64', fIoApic = True, \
235 eNic0AttachType = vboxcon.NetworkAttachmentType_NAT, \
236 eNic0Type = vboxcon.NetworkAdapterType_Am79C973, \
237 sDvdImage = sVBoxValidationKit_iso);
238 if oVM is None:
239 return False;
240
241 return True;
242
243 def actionExecute(self):
244 """
245 Execute the testcase.
246 """
247 fRc = self.test1();
248 return fRc;
249
250
251 #
252 # Test execution helpers.
253 #
254
255 def test1RunTestProgs(self, oSession, oTxsSession, fRc, sTestName, sGuestFs):
256 """
257 Runs all the test programs on the test machine.
258 """
259 _ = oSession;
260
261 reporter.testStart(sTestName);
262
263 sMkfsCmd = 'mkfs.' + sGuestFs;
264
265 # Prepare test disks, just create filesystem without partition
266 reporter.testStart('Preparation');
267 fRc = fRc and self.txsRunTest(oTxsSession, 'Create FS 1', 60000, \
268 '/sbin/' + sMkfsCmd,
269 (sMkfsCmd, '/dev/sdb'));
270
271 fRc = fRc and self.txsRunTest(oTxsSession, 'Create FS 2', 60000, \
272 '/sbin/' + sMkfsCmd,
273 (sMkfsCmd, '/dev/sdc'));
274
275 # Create test and scratch directory
276 fRc = fRc and self.txsRunTest(oTxsSession, 'Create /mnt/test', 10000, \
277 '/bin/mkdir',
278 ('mkdir', '/mnt/test'));
279
280 fRc = fRc and self.txsRunTest(oTxsSession, 'Create /mnt/scratch', 10000, \
281 '/bin/mkdir',
282 ('mkdir', '/mnt/scratch'));
283
284 # Mount test and scratch directory.
285 fRc = fRc and self.txsRunTest(oTxsSession, 'Mount /mnt/test', 10000, \
286 '/bin/mount',
287 ('mount', '/dev/sdb','/mnt/test'));
288
289 fRc = fRc and self.txsRunTest(oTxsSession, 'Mount /mnt/scratch', 10000, \
290 '/bin/mount',
291 ('mount', '/dev/sdc','/mnt/scratch'));
292
293 fRc = fRc and self.txsRunTest(oTxsSession, 'Copying xfstests', 10000, \
294 '/bin/cp',
295 ('cp', '-r','${CDROM}/${OS.ARCH}/xfstests', '/tmp'));
296
297 reporter.testDone();
298
299 # Run xfstests (this sh + cd crap is required because the cwd for the script must be in the root
300 # of the xfstests directory...)
301 reporter.testStart('xfstests');
302 if fRc and 'xfstests' in self.asTests:
303 fRc = self.txsRunTest(oTxsSession, 'xfstests', 3600000,
304 '/bin/sh',
305 ('sh', '-c', '(cd /tmp/xfstests && ./check -g auto)'),
306 ('TEST_DIR=/mnt/test', 'TEST_DEV=/dev/sdb', 'SCRATCH_MNT=/mnt/scratch', 'SCRATCH_DEV=/dev/sdc',
307 'FSTYP=' + sGuestFs));
308 reporter.testDone();
309 else:
310 reporter.testDone(fSkipped = True);
311
312 reporter.testDone(not fRc);
313 return fRc;
314
315 # pylint: disable=too-many-arguments
316
317 def test1OneCfg(self, sVmName, eStorageController, sDiskFormat, sDiskPath1, sDiskPath2, \
318 sGuestFs, cCpus, fHwVirt, fNestedPaging):
319 """
320 Runs the specified VM thru test #1.
321
322 Returns a success indicator on the general test execution. This is not
323 the actual test result.
324 """
325 oVM = self.getVmByName(sVmName);
326
327 # Reconfigure the VM
328 fRc = True;
329 oSession = self.openSession(oVM);
330 if oSession is not None:
331 # Attach HD
332 fRc = oSession.ensureControllerAttached(self.controllerTypeToName(eStorageController));
333 fRc = fRc and oSession.setStorageControllerType(eStorageController, self.controllerTypeToName(eStorageController));
334
335 if sDiskFormat == "iSCSI":
336 listNames = [];
337 listValues = [];
338 listValues = sDiskPath1.split('|');
339 listNames.append('TargetAddress');
340 listNames.append('TargetName');
341 listNames.append('LUN');
342
343 if self.fpApiVer >= 5.0:
344 oHd = oSession.oVBox.createMedium(sDiskFormat, sDiskPath1, vboxcon.AccessMode_ReadWrite, \
345 vboxcon.DeviceType_HardDisk);
346 else:
347 oHd = oSession.oVBox.createHardDisk(sDiskFormat, sDiskPath1);
348 oHd.type = vboxcon.MediumType_Normal;
349 oHd.setProperties(listNames, listValues);
350
351 # Attach it.
352 if fRc is True:
353 try:
354 if oSession.fpApiVer >= 4.0:
355 oSession.o.machine.attachDevice(self.controllerTypeToName(eStorageController),
356 1, 0, vboxcon.DeviceType_HardDisk, oHd);
357 else:
358 oSession.o.machine.attachDevice(self.controllerTypeToName(eStorageController),
359 1, 0, vboxcon.DeviceType_HardDisk, oHd.id);
360 except:
361 reporter.errorXcpt('attachDevice("%s",%s,%s,HardDisk,"%s") failed on "%s"' \
362 % (self.controllerTypeToName(eStorageController), 1, 0, oHd.id, oSession.sName) );
363 fRc = False;
364 else:
365 reporter.log('attached "%s" to %s' % (sDiskPath1, oSession.sName));
366 else:
367 fRc = fRc and oSession.createAndAttachHd(sDiskPath1, sDiskFormat, self.controllerTypeToName(eStorageController),
368 cb = 10*1024*1024*1024, iPort = 1, fImmutable = False);
369 fRc = fRc and oSession.createAndAttachHd(sDiskPath2, sDiskFormat, self.controllerTypeToName(eStorageController),
370 cb = 10*1024*1024*1024, iPort = 2, fImmutable = False);
371 fRc = fRc and oSession.enableVirtEx(fHwVirt);
372 fRc = fRc and oSession.enableNestedPaging(fNestedPaging);
373 fRc = fRc and oSession.setCpuCount(cCpus);
374 fRc = fRc and oSession.saveSettings();
375 fRc = oSession.close() and fRc and True; # pychecker hack.
376 oSession = None;
377 else:
378 fRc = False;
379
380 # Start up.
381 if fRc is True:
382 self.logVmInfo(oVM);
383 oSession, oTxsSession = self.startVmAndConnectToTxsViaTcp(sVmName, fCdWait = False, fNatForwardingForTxs = True);
384 if oSession is not None:
385 self.addTask(oTxsSession);
386
387 # Fudge factor - Allow the guest to finish starting up.
388 self.sleep(5);
389
390 fRc = self.test1RunTestProgs(oSession, oTxsSession, fRc, 'stress testing', sGuestFs);
391
392 # cleanup.
393 self.removeTask(oTxsSession);
394 self.terminateVmBySession(oSession)
395
396 # Remove disk
397 oSession = self.openSession(oVM);
398 if oSession is not None:
399 try:
400 oSession.o.machine.detachDevice(self.controllerTypeToName(eStorageController), 1, 0);
401 oSession.o.machine.detachDevice(self.controllerTypeToName(eStorageController), 2, 0);
402
403 # Remove storage controller if it is not an IDE controller.
404 if eStorageController not in (vboxcon.StorageControllerType_PIIX3, vboxcon.StorageControllerType_PIIX4,):
405 oSession.o.machine.removeStorageController(self.controllerTypeToName(eStorageController));
406
407 oSession.saveSettings();
408 oSession.oVBox.deleteHdByLocation(sDiskPath1);
409 oSession.oVBox.deleteHdByLocation(sDiskPath2);
410 oSession.saveSettings();
411 oSession.close();
412 oSession = None;
413 except:
414 reporter.errorXcpt('failed to detach/delete disks %s and %s from storage controller' % \
415 (sDiskPath1, sDiskPath2));
416 else:
417 fRc = False;
418 else:
419 fRc = False;
420 return fRc;
421
422 def test1OneVM(self, sVmName):
423 """
424 Runs one VM thru the various configurations.
425 """
426 reporter.testStart(sVmName);
427 fRc = True;
428 for sStorageCtrl in self.asStorageCtrls:
429 reporter.testStart(sStorageCtrl);
430
431 if sStorageCtrl == 'AHCI':
432 eStorageCtrl = vboxcon.StorageControllerType_IntelAhci;
433 elif sStorageCtrl == 'IDE':
434 eStorageCtrl = vboxcon.StorageControllerType_PIIX4;
435 elif sStorageCtrl == 'LsiLogicSAS':
436 eStorageCtrl = vboxcon.StorageControllerType_LsiLogicSas;
437 elif sStorageCtrl == 'LsiLogic':
438 eStorageCtrl = vboxcon.StorageControllerType_LsiLogic;
439 elif sStorageCtrl == 'BusLogic':
440 eStorageCtrl = vboxcon.StorageControllerType_BusLogic;
441 else:
442 eStorageCtrl = None;
443
444 for sDiskFormat in self.asDiskFormats:
445 reporter.testStart('%s' % (sDiskFormat,));
446
447 asPaths = self.asDirs;
448
449 for sDir in asPaths:
450 reporter.testStart('%s' % (sDir,));
451
452 sPathDisk1 = sDir + "/disk1.disk";
453 sPathDisk2 = sDir + "/disk2.disk";
454
455 for sGuestFs in self.asGuestFs:
456 reporter.testStart('%s' % (sGuestFs,));
457
458 for cCpus in self.acCpus:
459 if cCpus == 1: reporter.testStart('1 cpu');
460 else: reporter.testStart('%u cpus' % (cCpus,));
461
462 for sVirtMode in self.asVirtModes:
463 if sVirtMode == 'raw' and cCpus > 1:
464 continue;
465 hsVirtModeDesc = {};
466 hsVirtModeDesc['raw'] = 'Raw-mode';
467 hsVirtModeDesc['hwvirt'] = 'HwVirt';
468 hsVirtModeDesc['hwvirt-np'] = 'NestedPaging';
469 reporter.testStart(hsVirtModeDesc[sVirtMode]);
470
471 fHwVirt = sVirtMode != 'raw';
472 fNestedPaging = sVirtMode == 'hwvirt-np';
473 fRc = self.test1OneCfg(sVmName, eStorageCtrl, sDiskFormat, sPathDisk1, sPathDisk2, \
474 sGuestFs, cCpus, fHwVirt, fNestedPaging) and fRc and True;
475 reporter.testDone();
476 reporter.testDone();
477 reporter.testDone();
478 reporter.testDone();
479 reporter.testDone();
480 reporter.testDone();
481 reporter.testDone();
482 return fRc;
483
484 def test1(self):
485 """
486 Executes test #1.
487 """
488
489 # Loop thru the test VMs.
490 for sVM in self.asTestVMs:
491 # run test on the VM.
492 if not self.test1OneVM(sVM):
493 fRc = False;
494 else:
495 fRc = True;
496
497 return fRc;
498
499
500
501if __name__ == '__main__':
502 sys.exit(tdStorageStress().main(sys.argv));
503
Note: See TracBrowser for help on using the repository browser.

© 2024 Oracle Support Privacy / Do Not Sell My Info Terms of Use Trademark Policy Automated Access Etiquette