blob: eeca0631c9e13b657dfa24ab791c686495ad4127 [file] [log] [blame]
Matthew Treinishf610aca2015-06-30 15:32:34 -04001# Copyright 2015 Hewlett-Packard Development Company, L.P.
2#
3# Licensed under the Apache License, Version 2.0 (the "License"); you may
4# not use this file except in compliance with the License. You may obtain
5# a copy of the License at
6#
7# http://www.apache.org/licenses/LICENSE-2.0
8#
9# Unless required by applicable law or agreed to in writing, software
10# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
11# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
12# License for the specific language governing permissions and limitations
13# under the License.
14
15import os
16import shutil
17import subprocess
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010018import sys
Matthew Treinishf610aca2015-06-30 15:32:34 -040019
20from cliff import command
Matthew Treinishbdef1c72016-06-21 18:06:49 -040021from oslo_config import generator
Matthew Treinishf610aca2015-06-30 15:32:34 -040022from oslo_log import log as logging
23from six import moves
24
Cao Xuan Hoang36fe23c2016-08-25 16:11:14 +070025from tempest.cmd import workspace
step682980c14ec2016-02-23 14:53:52 -050026
Matthew Treinishf610aca2015-06-30 15:32:34 -040027LOG = logging.getLogger(__name__)
28
29TESTR_CONF = """[DEFAULT]
30test_command=OS_STDOUT_CAPTURE=${OS_STDOUT_CAPTURE:-1} \\
31 OS_STDERR_CAPTURE=${OS_STDERR_CAPTURE:-1} \\
32 OS_TEST_TIMEOUT=${OS_TEST_TIMEOUT:-500} \\
33 ${PYTHON:-python} -m subunit.run discover -t %s %s $LISTOPT $IDOPTION
34test_id_option=--load-list $IDFILE
35test_list_option=--list
36group_regex=([^\.]*\.)*
37"""
38
39
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010040def get_tempest_default_config_dir():
Ken'ichi Ohmichi2e2ee192015-11-19 09:48:27 +000041 """Get default config directory of tempest
42
Matthew Treinish3fe57b32016-06-21 14:39:00 -040043 There are 3 dirs that get tried in priority order. First is /etc/tempest,
44 if that doesn't exist it looks for a tempest dir in the XDG_CONFIG_HOME
45 dir (defaulting to ~/.config/tempest) and last it tries for a
46 ~/.tempest/etc directory. If none of these exist a ~/.tempest/etc
47 directory will be created.
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010048
49 :return: default config dir
50 """
Ken'ichi Ohmichi0ce16522016-04-23 12:52:05 -070051 global_conf_dir = '/etc/tempest'
Matthew Treinish3fe57b32016-06-21 14:39:00 -040052 xdg_config = os.environ.get('XDG_CONFIG_HOME',
53 os.path.expanduser('~/.config'))
54 user_xdg_global_path = os.path.join(xdg_config, 'tempest')
55 user_global_path = os.path.join(os.path.expanduser('~'), '.tempest/etc')
56 if os.path.isdir(global_conf_dir):
Ken'ichi Ohmichi0ce16522016-04-23 12:52:05 -070057 return global_conf_dir
Matthew Treinish3fe57b32016-06-21 14:39:00 -040058 elif os.path.isdir(user_xdg_global_path):
59 return user_xdg_global_path
60 elif os.path.isdir(user_global_path):
61 return user_global_path
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010062 else:
Matthew Treinish3fe57b32016-06-21 14:39:00 -040063 os.makedirs(user_global_path)
64 return user_global_path
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010065
66
Matthew Treinishf610aca2015-06-30 15:32:34 -040067class TempestInit(command.Command):
68 """Setup a local working environment for running tempest"""
69
70 def get_parser(self, prog_name):
71 parser = super(TempestInit, self).get_parser(prog_name)
72 parser.add_argument('dir', nargs='?', default=os.getcwd())
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010073 parser.add_argument('--config-dir', '-c', default=None)
Matthew Treinish054f45d2016-04-23 16:30:29 -040074 parser.add_argument('--show-global-config-dir', '-s',
75 action='store_true', dest='show_global_dir',
76 help="Print the global config dir location, "
77 "then exit")
step682980c14ec2016-02-23 14:53:52 -050078 parser.add_argument('--name', help="The workspace name", default=None)
79 parser.add_argument('--workspace-path', default=None,
80 help="The path to the workspace file, the default "
81 "is ~/.tempest/workspace")
Matthew Treinishf610aca2015-06-30 15:32:34 -040082 return parser
83
84 def generate_testr_conf(self, local_path):
85 testr_conf_path = os.path.join(local_path, '.testr.conf')
86 top_level_path = os.path.dirname(os.path.dirname(__file__))
87 discover_path = os.path.join(top_level_path, 'test_discover')
88 testr_conf = TESTR_CONF % (top_level_path, discover_path)
89 with open(testr_conf_path, 'w+') as testr_conf_file:
90 testr_conf_file.write(testr_conf)
91
Matthew Treinish3c39bb62016-08-09 14:44:44 -040092 def get_configparser(self, conf_path):
Matthew Treinishf610aca2015-06-30 15:32:34 -040093 config_parse = moves.configparser.SafeConfigParser()
94 config_parse.optionxform = str
Matthew Treinish3c39bb62016-08-09 14:44:44 -040095 # get any existing values if a config file already exists
96 if os.path.isfile(conf_path):
97 # use read() for Python 2 and 3 compatibility
98 config_parse.read(conf_path)
99 return config_parse
100
101 def update_local_conf(self, conf_path, lock_dir, log_dir):
102 config_parse = self.get_configparser(conf_path)
103 # Set local lock_dir in tempest conf
104 if not config_parse.has_section('oslo_concurrency'):
105 config_parse.add_section('oslo_concurrency')
106 config_parse.set('oslo_concurrency', 'lock_path', lock_dir)
107 # Set local log_dir in tempest conf
108 config_parse.set('DEFAULT', 'log_dir', log_dir)
109 # Set default log filename to tempest.log
110 config_parse.set('DEFAULT', 'log_file', 'tempest.log')
111
112 # write out a new file with the updated configurations
113 with open(conf_path, 'w+') as conf_file:
ghanshyam0e1dd842016-04-27 07:59:23 +0900114 config_parse.write(conf_file)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400115
116 def copy_config(self, etc_dir, config_dir):
Matthew Treinishd5cef952016-06-07 16:54:55 -0400117 if os.path.isdir(config_dir):
118 shutil.copytree(config_dir, etc_dir)
119 else:
120 LOG.warning("Global config dir %s can't be found" % config_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400121
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400122 def generate_sample_config(self, local_dir):
123 conf_generator = os.path.join(os.path.dirname(__file__),
124 'config-generator.tempest.conf')
125 output_file = os.path.join(local_dir, 'etc/tempest.conf.sample')
126 if os.path.isfile(conf_generator):
127 generator.main(['--config-file', conf_generator, '--output-file',
128 output_file])
Matthew Treinishd5cef952016-06-07 16:54:55 -0400129 else:
130 LOG.warning("Skipping sample config generation because global "
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400131 "config file %s can't be found" % conf_generator)
Matthew Treinishc8a39b42015-07-27 17:07:37 -0400132
Matthew Treinishf610aca2015-06-30 15:32:34 -0400133 def create_working_dir(self, local_dir, config_dir):
Jake Yip93464832016-06-18 00:57:40 +1000134 # make sure we are working with abspath however tempest init is called
135 local_dir = os.path.abspath(local_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400136 # Create local dir if missing
137 if not os.path.isdir(local_dir):
138 LOG.debug('Creating local working dir: %s' % local_dir)
139 os.mkdir(local_dir)
Marc Koderer090b5dc2015-11-04 10:35:48 +0100140 elif not os.listdir(local_dir) == []:
David Paterson0bf52d42015-04-13 21:55:58 -0400141 raise OSError("Directory you are trying to initialize already "
Marc Koderer090b5dc2015-11-04 10:35:48 +0100142 "exists and is not empty: %s" % local_dir)
David Paterson0bf52d42015-04-13 21:55:58 -0400143
Matthew Treinishf610aca2015-06-30 15:32:34 -0400144 lock_dir = os.path.join(local_dir, 'tempest_lock')
145 etc_dir = os.path.join(local_dir, 'etc')
146 config_path = os.path.join(etc_dir, 'tempest.conf')
147 log_dir = os.path.join(local_dir, 'logs')
148 testr_dir = os.path.join(local_dir, '.testrepository')
149 # Create lock dir
150 if not os.path.isdir(lock_dir):
151 LOG.debug('Creating lock dir: %s' % lock_dir)
152 os.mkdir(lock_dir)
153 # Create log dir
154 if not os.path.isdir(log_dir):
155 LOG.debug('Creating log dir: %s' % log_dir)
156 os.mkdir(log_dir)
157 # Create and copy local etc dir
158 self.copy_config(etc_dir, config_dir)
Matthew Treinishc8a39b42015-07-27 17:07:37 -0400159 # Generate the sample config file
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400160 self.generate_sample_config(local_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400161 # Update local confs to reflect local paths
162 self.update_local_conf(config_path, lock_dir, log_dir)
163 # Generate a testr conf file
164 self.generate_testr_conf(local_dir)
165 # setup local testr working dir
166 if not os.path.isdir(testr_dir):
167 subprocess.call(['testr', 'init'], cwd=local_dir)
168
169 def take_action(self, parsed_args):
Cao Xuan Hoang36fe23c2016-08-25 16:11:14 +0700170 workspace_manager = workspace.WorkspaceManager(
171 parsed_args.workspace_path)
step682980c14ec2016-02-23 14:53:52 -0500172 name = parsed_args.name or parsed_args.dir.split(os.path.sep)[-1]
173 workspace_manager.register_new_workspace(
174 name, parsed_args.dir, init=True)
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +0100175 config_dir = parsed_args.config_dir or get_tempest_default_config_dir()
Matthew Treinish054f45d2016-04-23 16:30:29 -0400176 if parsed_args.show_global_dir:
177 print("Global config dir is located at: %s" % config_dir)
178 sys.exit(0)
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +0100179 self.create_working_dir(parsed_args.dir, config_dir)