blob: f577d9b253cacbb9a3a2c9e05d7001f5afe68fa0 [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)
Masayuki Igawa0670a2b2016-09-12 14:55:11 +090072 parser.add_argument('dir', nargs='?', default=os.getcwd(),
73 help="The path to the workspace directory. If you "
74 "omit this argument, the workspace directory is "
75 "your current directory")
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +010076 parser.add_argument('--config-dir', '-c', default=None)
Matthew Treinish054f45d2016-04-23 16:30:29 -040077 parser.add_argument('--show-global-config-dir', '-s',
78 action='store_true', dest='show_global_dir',
79 help="Print the global config dir location, "
80 "then exit")
step682980c14ec2016-02-23 14:53:52 -050081 parser.add_argument('--name', help="The workspace name", default=None)
82 parser.add_argument('--workspace-path', default=None,
83 help="The path to the workspace file, the default "
Masayuki Igawa0670a2b2016-09-12 14:55:11 +090084 "is ~/.tempest/workspace.yaml")
Matthew Treinishf610aca2015-06-30 15:32:34 -040085 return parser
86
87 def generate_testr_conf(self, local_path):
88 testr_conf_path = os.path.join(local_path, '.testr.conf')
89 top_level_path = os.path.dirname(os.path.dirname(__file__))
90 discover_path = os.path.join(top_level_path, 'test_discover')
91 testr_conf = TESTR_CONF % (top_level_path, discover_path)
92 with open(testr_conf_path, 'w+') as testr_conf_file:
93 testr_conf_file.write(testr_conf)
94
Matthew Treinish3c39bb62016-08-09 14:44:44 -040095 def get_configparser(self, conf_path):
Janonymous8254a3f2016-09-15 10:38:48 +053096 config_parse = moves.configparser.ConfigParser()
Matthew Treinishf610aca2015-06-30 15:32:34 -040097 config_parse.optionxform = str
Matthew Treinish3c39bb62016-08-09 14:44:44 -040098 # get any existing values if a config file already exists
99 if os.path.isfile(conf_path):
100 # use read() for Python 2 and 3 compatibility
101 config_parse.read(conf_path)
102 return config_parse
103
104 def update_local_conf(self, conf_path, lock_dir, log_dir):
105 config_parse = self.get_configparser(conf_path)
106 # Set local lock_dir in tempest conf
107 if not config_parse.has_section('oslo_concurrency'):
108 config_parse.add_section('oslo_concurrency')
109 config_parse.set('oslo_concurrency', 'lock_path', lock_dir)
110 # Set local log_dir in tempest conf
111 config_parse.set('DEFAULT', 'log_dir', log_dir)
112 # Set default log filename to tempest.log
113 config_parse.set('DEFAULT', 'log_file', 'tempest.log')
114
115 # write out a new file with the updated configurations
116 with open(conf_path, 'w+') as conf_file:
ghanshyam0e1dd842016-04-27 07:59:23 +0900117 config_parse.write(conf_file)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400118
119 def copy_config(self, etc_dir, config_dir):
Matthew Treinishd5cef952016-06-07 16:54:55 -0400120 if os.path.isdir(config_dir):
121 shutil.copytree(config_dir, etc_dir)
122 else:
123 LOG.warning("Global config dir %s can't be found" % config_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400124
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400125 def generate_sample_config(self, local_dir):
126 conf_generator = os.path.join(os.path.dirname(__file__),
127 'config-generator.tempest.conf')
128 output_file = os.path.join(local_dir, 'etc/tempest.conf.sample')
129 if os.path.isfile(conf_generator):
130 generator.main(['--config-file', conf_generator, '--output-file',
131 output_file])
Matthew Treinishd5cef952016-06-07 16:54:55 -0400132 else:
133 LOG.warning("Skipping sample config generation because global "
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400134 "config file %s can't be found" % conf_generator)
Matthew Treinishc8a39b42015-07-27 17:07:37 -0400135
Matthew Treinishf610aca2015-06-30 15:32:34 -0400136 def create_working_dir(self, local_dir, config_dir):
Jake Yip93464832016-06-18 00:57:40 +1000137 # make sure we are working with abspath however tempest init is called
138 local_dir = os.path.abspath(local_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400139 # Create local dir if missing
140 if not os.path.isdir(local_dir):
141 LOG.debug('Creating local working dir: %s' % local_dir)
142 os.mkdir(local_dir)
Marc Koderer090b5dc2015-11-04 10:35:48 +0100143 elif not os.listdir(local_dir) == []:
David Paterson0bf52d42015-04-13 21:55:58 -0400144 raise OSError("Directory you are trying to initialize already "
Marc Koderer090b5dc2015-11-04 10:35:48 +0100145 "exists and is not empty: %s" % local_dir)
David Paterson0bf52d42015-04-13 21:55:58 -0400146
Matthew Treinishf610aca2015-06-30 15:32:34 -0400147 lock_dir = os.path.join(local_dir, 'tempest_lock')
148 etc_dir = os.path.join(local_dir, 'etc')
149 config_path = os.path.join(etc_dir, 'tempest.conf')
150 log_dir = os.path.join(local_dir, 'logs')
151 testr_dir = os.path.join(local_dir, '.testrepository')
152 # Create lock dir
153 if not os.path.isdir(lock_dir):
154 LOG.debug('Creating lock dir: %s' % lock_dir)
155 os.mkdir(lock_dir)
156 # Create log dir
157 if not os.path.isdir(log_dir):
158 LOG.debug('Creating log dir: %s' % log_dir)
159 os.mkdir(log_dir)
160 # Create and copy local etc dir
161 self.copy_config(etc_dir, config_dir)
Matthew Treinishc8a39b42015-07-27 17:07:37 -0400162 # Generate the sample config file
Matthew Treinishbdef1c72016-06-21 18:06:49 -0400163 self.generate_sample_config(local_dir)
Matthew Treinishf610aca2015-06-30 15:32:34 -0400164 # Update local confs to reflect local paths
165 self.update_local_conf(config_path, lock_dir, log_dir)
166 # Generate a testr conf file
167 self.generate_testr_conf(local_dir)
168 # setup local testr working dir
169 if not os.path.isdir(testr_dir):
170 subprocess.call(['testr', 'init'], cwd=local_dir)
171
172 def take_action(self, parsed_args):
Cao Xuan Hoang36fe23c2016-08-25 16:11:14 +0700173 workspace_manager = workspace.WorkspaceManager(
174 parsed_args.workspace_path)
step682980c14ec2016-02-23 14:53:52 -0500175 name = parsed_args.name or parsed_args.dir.split(os.path.sep)[-1]
176 workspace_manager.register_new_workspace(
177 name, parsed_args.dir, init=True)
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +0100178 config_dir = parsed_args.config_dir or get_tempest_default_config_dir()
Matthew Treinish054f45d2016-04-23 16:30:29 -0400179 if parsed_args.show_global_dir:
180 print("Global config dir is located at: %s" % config_dir)
181 sys.exit(0)
Andrea Frittoli (andreaf)5a69e552015-07-31 18:40:17 +0100182 self.create_working_dir(parsed_args.dir, config_dir)