init

Dependencies:   mbed

Committer:
Nathan Yonkee
Date:
Fri Mar 02 07:16:49 2018 -0700
Revision:
10:46a4cf51ee38
Parent:
9:d58e77ebd769
remove mbed-os

Who changed what in which revision?

UserRevisionLine numberNew contents of line
Nathan Yonkee 9:d58e77ebd769 1 #! /usr/bin/env python2
Nathan Yonkee 9:d58e77ebd769 2 """
Nathan Yonkee 9:d58e77ebd769 3 mbed SDK
Nathan Yonkee 9:d58e77ebd769 4 Copyright (c) 2011-2013 ARM Limited
Nathan Yonkee 9:d58e77ebd769 5
Nathan Yonkee 9:d58e77ebd769 6 Licensed under the Apache License, Version 2.0 (the "License");
Nathan Yonkee 9:d58e77ebd769 7 you may not use this file except in compliance with the License.
Nathan Yonkee 9:d58e77ebd769 8 You may obtain a copy of the License at
Nathan Yonkee 9:d58e77ebd769 9
Nathan Yonkee 9:d58e77ebd769 10 http://www.apache.org/licenses/LICENSE-2.0
Nathan Yonkee 9:d58e77ebd769 11
Nathan Yonkee 9:d58e77ebd769 12 Unless required by applicable law or agreed to in writing, software
Nathan Yonkee 9:d58e77ebd769 13 distributed under the License is distributed on an "AS IS" BASIS,
Nathan Yonkee 9:d58e77ebd769 14 WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
Nathan Yonkee 9:d58e77ebd769 15 See the License for the specific language governing permissions and
Nathan Yonkee 9:d58e77ebd769 16 limitations under the License.
Nathan Yonkee 9:d58e77ebd769 17
Nathan Yonkee 9:d58e77ebd769 18
Nathan Yonkee 9:d58e77ebd769 19 TEST BUILD & RUN
Nathan Yonkee 9:d58e77ebd769 20 """
Nathan Yonkee 9:d58e77ebd769 21 from __future__ import print_function, division, absolute_import
Nathan Yonkee 9:d58e77ebd769 22 import sys
Nathan Yonkee 9:d58e77ebd769 23 import os
Nathan Yonkee 9:d58e77ebd769 24 import json
Nathan Yonkee 9:d58e77ebd769 25 import fnmatch
Nathan Yonkee 9:d58e77ebd769 26
Nathan Yonkee 9:d58e77ebd769 27 ROOT = os.path.abspath(os.path.join(os.path.dirname(__file__), ".."))
Nathan Yonkee 9:d58e77ebd769 28 sys.path.insert(0, ROOT)
Nathan Yonkee 9:d58e77ebd769 29
Nathan Yonkee 9:d58e77ebd769 30 from tools.config import ConfigException
Nathan Yonkee 9:d58e77ebd769 31 from tools.test_api import test_path_to_name, find_tests, get_test_config, print_tests, build_tests, test_spec_from_test_builds
Nathan Yonkee 9:d58e77ebd769 32 import tools.test_configs as TestConfig
Nathan Yonkee 9:d58e77ebd769 33 from tools.options import get_default_options_parser, extract_profile, extract_mcus
Nathan Yonkee 9:d58e77ebd769 34 from tools.build_api import build_project, build_library
Nathan Yonkee 9:d58e77ebd769 35 from tools.build_api import print_build_memory_usage
Nathan Yonkee 9:d58e77ebd769 36 from tools.build_api import merge_build_data
Nathan Yonkee 9:d58e77ebd769 37 from tools.targets import TARGET_MAP
Nathan Yonkee 9:d58e77ebd769 38 from tools.utils import mkdir, ToolException, NotSupportedException, args_error
Nathan Yonkee 9:d58e77ebd769 39 from tools.test_exporters import ReportExporter, ResultExporterType
Nathan Yonkee 9:d58e77ebd769 40 from tools.utils import argparse_filestring_type, argparse_lowercase_type, argparse_many
Nathan Yonkee 9:d58e77ebd769 41 from tools.utils import argparse_dir_not_parent
Nathan Yonkee 9:d58e77ebd769 42 from tools.toolchains import mbedToolchain, TOOLCHAIN_PATHS, TOOLCHAIN_CLASSES
Nathan Yonkee 9:d58e77ebd769 43 from tools.settings import CLI_COLOR_MAP
Nathan Yonkee 9:d58e77ebd769 44
Nathan Yonkee 9:d58e77ebd769 45 if __name__ == '__main__':
Nathan Yonkee 9:d58e77ebd769 46 try:
Nathan Yonkee 9:d58e77ebd769 47 # Parse Options
Nathan Yonkee 9:d58e77ebd769 48 parser = get_default_options_parser(add_app_config=True)
Nathan Yonkee 9:d58e77ebd769 49
Nathan Yonkee 9:d58e77ebd769 50 parser.add_argument("-D",
Nathan Yonkee 9:d58e77ebd769 51 action="append",
Nathan Yonkee 9:d58e77ebd769 52 dest="macros",
Nathan Yonkee 9:d58e77ebd769 53 help="Add a macro definition")
Nathan Yonkee 9:d58e77ebd769 54
Nathan Yonkee 9:d58e77ebd769 55 parser.add_argument("-j", "--jobs",
Nathan Yonkee 9:d58e77ebd769 56 type=int,
Nathan Yonkee 9:d58e77ebd769 57 dest="jobs",
Nathan Yonkee 9:d58e77ebd769 58 default=0,
Nathan Yonkee 9:d58e77ebd769 59 help="Number of concurrent jobs. Default: 0/auto (based on host machine's number of CPUs)")
Nathan Yonkee 9:d58e77ebd769 60
Nathan Yonkee 9:d58e77ebd769 61 parser.add_argument("--source", dest="source_dir",
Nathan Yonkee 9:d58e77ebd769 62 type=argparse_filestring_type,
Nathan Yonkee 9:d58e77ebd769 63 default=None, help="The source (input) directory (for sources other than tests). Defaults to current directory.", action="append")
Nathan Yonkee 9:d58e77ebd769 64
Nathan Yonkee 9:d58e77ebd769 65 parser.add_argument("--build", dest="build_dir", type=argparse_dir_not_parent(ROOT),
Nathan Yonkee 9:d58e77ebd769 66 default=None, help="The build (output) directory")
Nathan Yonkee 9:d58e77ebd769 67
Nathan Yonkee 9:d58e77ebd769 68 parser.add_argument("-l", "--list", action="store_true", dest="list",
Nathan Yonkee 9:d58e77ebd769 69 default=False, help="List (recursively) available tests in order and exit")
Nathan Yonkee 9:d58e77ebd769 70
Nathan Yonkee 9:d58e77ebd769 71 parser.add_argument("-p", "--paths", dest="paths",
Nathan Yonkee 9:d58e77ebd769 72 type=argparse_many(argparse_filestring_type),
Nathan Yonkee 9:d58e77ebd769 73 default=None, help="Limit the tests to those within the specified comma separated list of paths")
Nathan Yonkee 9:d58e77ebd769 74
Nathan Yonkee 9:d58e77ebd769 75 format_choices = ["list", "json"]
Nathan Yonkee 9:d58e77ebd769 76 format_default_choice = "list"
Nathan Yonkee 9:d58e77ebd769 77 format_help = "Change the format in which tests are listed. Choices include: %s. Default: %s" % (", ".join(format_choices), format_default_choice)
Nathan Yonkee 9:d58e77ebd769 78 parser.add_argument("-f", "--format", dest="format",
Nathan Yonkee 9:d58e77ebd769 79 type=argparse_lowercase_type(format_choices, "format"),
Nathan Yonkee 9:d58e77ebd769 80 default=format_default_choice, help=format_help)
Nathan Yonkee 9:d58e77ebd769 81
Nathan Yonkee 9:d58e77ebd769 82 parser.add_argument("--continue-on-build-fail", action="store_true", dest="continue_on_build_fail",
Nathan Yonkee 9:d58e77ebd769 83 default=None, help="Continue trying to build all tests if a build failure occurs")
Nathan Yonkee 9:d58e77ebd769 84
Nathan Yonkee 9:d58e77ebd769 85 #TODO validate the names instead of just passing through str
Nathan Yonkee 9:d58e77ebd769 86 parser.add_argument("-n", "--names", dest="names", type=argparse_many(str),
Nathan Yonkee 9:d58e77ebd769 87 default=None, help="Limit the tests to a comma separated list of names")
Nathan Yonkee 9:d58e77ebd769 88
Nathan Yonkee 9:d58e77ebd769 89 parser.add_argument("--test-config", dest="test_config", type=str,
Nathan Yonkee 9:d58e77ebd769 90 default=None, help="Test config for a module")
Nathan Yonkee 9:d58e77ebd769 91
Nathan Yonkee 9:d58e77ebd769 92 parser.add_argument("--test-spec", dest="test_spec",
Nathan Yonkee 9:d58e77ebd769 93 default=None, help="Destination path for a test spec file that can be used by the Greentea automated test tool")
Nathan Yonkee 9:d58e77ebd769 94
Nathan Yonkee 9:d58e77ebd769 95 parser.add_argument("--build-report-junit", dest="build_report_junit",
Nathan Yonkee 9:d58e77ebd769 96 default=None, help="Destination path for a build report in the JUnit xml format")
Nathan Yonkee 9:d58e77ebd769 97 parser.add_argument("--build-data",
Nathan Yonkee 9:d58e77ebd769 98 dest="build_data",
Nathan Yonkee 9:d58e77ebd769 99 default=None,
Nathan Yonkee 9:d58e77ebd769 100 help="Dump build_data to this file")
Nathan Yonkee 9:d58e77ebd769 101
Nathan Yonkee 9:d58e77ebd769 102 parser.add_argument("-v", "--verbose",
Nathan Yonkee 9:d58e77ebd769 103 action="store_true",
Nathan Yonkee 9:d58e77ebd769 104 dest="verbose",
Nathan Yonkee 9:d58e77ebd769 105 default=False,
Nathan Yonkee 9:d58e77ebd769 106 help="Verbose diagnostic output")
Nathan Yonkee 9:d58e77ebd769 107
Nathan Yonkee 9:d58e77ebd769 108 parser.add_argument("--stats-depth",
Nathan Yonkee 9:d58e77ebd769 109 type=int,
Nathan Yonkee 9:d58e77ebd769 110 dest="stats_depth",
Nathan Yonkee 9:d58e77ebd769 111 default=2,
Nathan Yonkee 9:d58e77ebd769 112 help="Depth level for static memory report")
Nathan Yonkee 9:d58e77ebd769 113
Nathan Yonkee 9:d58e77ebd769 114 options = parser.parse_args()
Nathan Yonkee 9:d58e77ebd769 115
Nathan Yonkee 9:d58e77ebd769 116 # Filter tests by path if specified
Nathan Yonkee 9:d58e77ebd769 117 if options.paths:
Nathan Yonkee 9:d58e77ebd769 118 all_paths = options.paths
Nathan Yonkee 9:d58e77ebd769 119 else:
Nathan Yonkee 9:d58e77ebd769 120 all_paths = ["."]
Nathan Yonkee 9:d58e77ebd769 121
Nathan Yonkee 9:d58e77ebd769 122 all_tests = {}
Nathan Yonkee 9:d58e77ebd769 123 tests = {}
Nathan Yonkee 9:d58e77ebd769 124
Nathan Yonkee 9:d58e77ebd769 125 # Target
Nathan Yonkee 9:d58e77ebd769 126 if options.mcu is None :
Nathan Yonkee 9:d58e77ebd769 127 args_error(parser, "argument -m/--mcu is required")
Nathan Yonkee 9:d58e77ebd769 128 mcu = extract_mcus(parser, options)[0]
Nathan Yonkee 9:d58e77ebd769 129
Nathan Yonkee 9:d58e77ebd769 130 # Toolchain
Nathan Yonkee 9:d58e77ebd769 131 if options.tool is None:
Nathan Yonkee 9:d58e77ebd769 132 args_error(parser, "argument -t/--tool is required")
Nathan Yonkee 9:d58e77ebd769 133 toolchain = options.tool[0]
Nathan Yonkee 9:d58e77ebd769 134
Nathan Yonkee 9:d58e77ebd769 135 if not TOOLCHAIN_CLASSES[toolchain].check_executable():
Nathan Yonkee 9:d58e77ebd769 136 search_path = TOOLCHAIN_PATHS[toolchain] or "No path set"
Nathan Yonkee 9:d58e77ebd769 137 args_error(parser, "Could not find executable for %s.\n"
Nathan Yonkee 9:d58e77ebd769 138 "Currently set search path: %s"
Nathan Yonkee 9:d58e77ebd769 139 % (toolchain, search_path))
Nathan Yonkee 9:d58e77ebd769 140
Nathan Yonkee 9:d58e77ebd769 141 # Assign config file. Precedence: test_config>app_config
Nathan Yonkee 9:d58e77ebd769 142 # TODO: merge configs if both given
Nathan Yonkee 9:d58e77ebd769 143 if options.test_config:
Nathan Yonkee 9:d58e77ebd769 144 config = get_test_config(options.test_config, mcu)
Nathan Yonkee 9:d58e77ebd769 145 if not config:
Nathan Yonkee 9:d58e77ebd769 146 args_error(parser, "argument --test-config contains invalid path or identifier")
Nathan Yonkee 9:d58e77ebd769 147 elif not options.app_config:
Nathan Yonkee 9:d58e77ebd769 148 config = TestConfig.get_default_config(options.source_dir or ['.'], mcu)
Nathan Yonkee 9:d58e77ebd769 149 else:
Nathan Yonkee 9:d58e77ebd769 150 config = options.app_config
Nathan Yonkee 9:d58e77ebd769 151
Nathan Yonkee 9:d58e77ebd769 152 # Find all tests in the relevant paths
Nathan Yonkee 9:d58e77ebd769 153 for path in all_paths:
Nathan Yonkee 9:d58e77ebd769 154 all_tests.update(find_tests(path, mcu, toolchain,
Nathan Yonkee 9:d58e77ebd769 155 app_config=config))
Nathan Yonkee 9:d58e77ebd769 156
Nathan Yonkee 9:d58e77ebd769 157 # Filter tests by name if specified
Nathan Yonkee 9:d58e77ebd769 158 if options.names:
Nathan Yonkee 9:d58e77ebd769 159 all_names = options.names
Nathan Yonkee 9:d58e77ebd769 160 all_names = [x.lower() for x in all_names]
Nathan Yonkee 9:d58e77ebd769 161
Nathan Yonkee 9:d58e77ebd769 162 for name in all_names:
Nathan Yonkee 9:d58e77ebd769 163 if any(fnmatch.fnmatch(testname, name) for testname in all_tests):
Nathan Yonkee 9:d58e77ebd769 164 for testname, test in all_tests.items():
Nathan Yonkee 9:d58e77ebd769 165 if fnmatch.fnmatch(testname, name):
Nathan Yonkee 9:d58e77ebd769 166 tests[testname] = test
Nathan Yonkee 9:d58e77ebd769 167 else:
Nathan Yonkee 9:d58e77ebd769 168 print("[Warning] Test with name '%s' was not found in the "
Nathan Yonkee 9:d58e77ebd769 169 "available tests" % (name))
Nathan Yonkee 9:d58e77ebd769 170 else:
Nathan Yonkee 9:d58e77ebd769 171 tests = all_tests
Nathan Yonkee 9:d58e77ebd769 172
Nathan Yonkee 9:d58e77ebd769 173 if options.color:
Nathan Yonkee 9:d58e77ebd769 174 # This import happens late to prevent initializing colorization when we don't need it
Nathan Yonkee 9:d58e77ebd769 175 import colorize
Nathan Yonkee 9:d58e77ebd769 176 if options.verbose:
Nathan Yonkee 9:d58e77ebd769 177 notify = mbedToolchain.print_notify_verbose
Nathan Yonkee 9:d58e77ebd769 178 else:
Nathan Yonkee 9:d58e77ebd769 179 notify = mbedToolchain.print_notify
Nathan Yonkee 9:d58e77ebd769 180 notify = colorize.print_in_color_notifier(CLI_COLOR_MAP, notify)
Nathan Yonkee 9:d58e77ebd769 181 else:
Nathan Yonkee 9:d58e77ebd769 182 notify = None
Nathan Yonkee 9:d58e77ebd769 183
Nathan Yonkee 9:d58e77ebd769 184 if options.list:
Nathan Yonkee 9:d58e77ebd769 185 # Print available tests in order and exit
Nathan Yonkee 9:d58e77ebd769 186 print_tests(tests, options.format)
Nathan Yonkee 9:d58e77ebd769 187 sys.exit(0)
Nathan Yonkee 9:d58e77ebd769 188 else:
Nathan Yonkee 9:d58e77ebd769 189 # Build all tests
Nathan Yonkee 9:d58e77ebd769 190 if not options.build_dir:
Nathan Yonkee 9:d58e77ebd769 191 args_error(parser, "argument --build is required")
Nathan Yonkee 9:d58e77ebd769 192
Nathan Yonkee 9:d58e77ebd769 193 base_source_paths = options.source_dir
Nathan Yonkee 9:d58e77ebd769 194
Nathan Yonkee 9:d58e77ebd769 195 # Default base source path is the current directory
Nathan Yonkee 9:d58e77ebd769 196 if not base_source_paths:
Nathan Yonkee 9:d58e77ebd769 197 base_source_paths = ['.']
Nathan Yonkee 9:d58e77ebd769 198
Nathan Yonkee 9:d58e77ebd769 199 build_report = {}
Nathan Yonkee 9:d58e77ebd769 200 build_properties = {}
Nathan Yonkee 9:d58e77ebd769 201
Nathan Yonkee 9:d58e77ebd769 202 library_build_success = False
Nathan Yonkee 9:d58e77ebd769 203 profile = extract_profile(parser, options, toolchain)
Nathan Yonkee 9:d58e77ebd769 204 try:
Nathan Yonkee 9:d58e77ebd769 205 # Build sources
Nathan Yonkee 9:d58e77ebd769 206 build_library(base_source_paths, options.build_dir, mcu,
Nathan Yonkee 9:d58e77ebd769 207 toolchain, jobs=options.jobs,
Nathan Yonkee 9:d58e77ebd769 208 clean=options.clean, report=build_report,
Nathan Yonkee 9:d58e77ebd769 209 properties=build_properties, name="mbed-build",
Nathan Yonkee 9:d58e77ebd769 210 macros=options.macros, verbose=options.verbose,
Nathan Yonkee 9:d58e77ebd769 211 notify=notify, archive=False,
Nathan Yonkee 9:d58e77ebd769 212 app_config=config,
Nathan Yonkee 9:d58e77ebd769 213 build_profile=profile)
Nathan Yonkee 9:d58e77ebd769 214
Nathan Yonkee 9:d58e77ebd769 215 library_build_success = True
Nathan Yonkee 9:d58e77ebd769 216 except ToolException as e:
Nathan Yonkee 9:d58e77ebd769 217 # ToolException output is handled by the build log
Nathan Yonkee 9:d58e77ebd769 218 pass
Nathan Yonkee 9:d58e77ebd769 219 except NotSupportedException as e:
Nathan Yonkee 9:d58e77ebd769 220 # NotSupportedException is handled by the build log
Nathan Yonkee 9:d58e77ebd769 221 pass
Nathan Yonkee 9:d58e77ebd769 222 except Exception as e:
Nathan Yonkee 9:d58e77ebd769 223 # Some other exception occurred, print the error message
Nathan Yonkee 9:d58e77ebd769 224 print(e)
Nathan Yonkee 9:d58e77ebd769 225
Nathan Yonkee 9:d58e77ebd769 226 if not library_build_success:
Nathan Yonkee 9:d58e77ebd769 227 print("Failed to build library")
Nathan Yonkee 9:d58e77ebd769 228 else:
Nathan Yonkee 9:d58e77ebd769 229 # Build all the tests
Nathan Yonkee 9:d58e77ebd769 230
Nathan Yonkee 9:d58e77ebd769 231 test_build_success, test_build = build_tests(tests, [options.build_dir], options.build_dir, mcu, toolchain,
Nathan Yonkee 9:d58e77ebd769 232 clean=options.clean,
Nathan Yonkee 9:d58e77ebd769 233 report=build_report,
Nathan Yonkee 9:d58e77ebd769 234 properties=build_properties,
Nathan Yonkee 9:d58e77ebd769 235 macros=options.macros,
Nathan Yonkee 9:d58e77ebd769 236 verbose=options.verbose,
Nathan Yonkee 9:d58e77ebd769 237 notify=notify,
Nathan Yonkee 9:d58e77ebd769 238 jobs=options.jobs,
Nathan Yonkee 9:d58e77ebd769 239 continue_on_build_fail=options.continue_on_build_fail,
Nathan Yonkee 9:d58e77ebd769 240 app_config=config,
Nathan Yonkee 9:d58e77ebd769 241 build_profile=profile,
Nathan Yonkee 9:d58e77ebd769 242 stats_depth=options.stats_depth)
Nathan Yonkee 9:d58e77ebd769 243
Nathan Yonkee 9:d58e77ebd769 244 # If a path to a test spec is provided, write it to a file
Nathan Yonkee 9:d58e77ebd769 245 if options.test_spec:
Nathan Yonkee 9:d58e77ebd769 246 test_spec_data = test_spec_from_test_builds(test_build)
Nathan Yonkee 9:d58e77ebd769 247
Nathan Yonkee 9:d58e77ebd769 248 # Create the target dir for the test spec if necessary
Nathan Yonkee 9:d58e77ebd769 249 # mkdir will not create the dir if it already exists
Nathan Yonkee 9:d58e77ebd769 250 test_spec_dir = os.path.dirname(options.test_spec)
Nathan Yonkee 9:d58e77ebd769 251 if test_spec_dir:
Nathan Yonkee 9:d58e77ebd769 252 mkdir(test_spec_dir)
Nathan Yonkee 9:d58e77ebd769 253
Nathan Yonkee 9:d58e77ebd769 254 try:
Nathan Yonkee 9:d58e77ebd769 255 with open(options.test_spec, 'w') as f:
Nathan Yonkee 9:d58e77ebd769 256 f.write(json.dumps(test_spec_data, indent=2))
Nathan Yonkee 9:d58e77ebd769 257 except IOError as e:
Nathan Yonkee 9:d58e77ebd769 258 print("[ERROR] Error writing test spec to file")
Nathan Yonkee 9:d58e77ebd769 259 print(e)
Nathan Yonkee 9:d58e77ebd769 260
Nathan Yonkee 9:d58e77ebd769 261 # If a path to a JUnit build report spec is provided, write it to a file
Nathan Yonkee 9:d58e77ebd769 262 if options.build_report_junit:
Nathan Yonkee 9:d58e77ebd769 263 report_exporter = ReportExporter(ResultExporterType.JUNIT, package="build")
Nathan Yonkee 9:d58e77ebd769 264 report_exporter.report_to_file(build_report, options.build_report_junit, test_suite_properties=build_properties)
Nathan Yonkee 9:d58e77ebd769 265
Nathan Yonkee 9:d58e77ebd769 266 # Print memory map summary on screen
Nathan Yonkee 9:d58e77ebd769 267 if build_report:
Nathan Yonkee 9:d58e77ebd769 268 print
Nathan Yonkee 9:d58e77ebd769 269 print(print_build_memory_usage(build_report))
Nathan Yonkee 9:d58e77ebd769 270
Nathan Yonkee 9:d58e77ebd769 271 print_report_exporter = ReportExporter(ResultExporterType.PRINT, package="build")
Nathan Yonkee 9:d58e77ebd769 272 status = print_report_exporter.report(build_report)
Nathan Yonkee 9:d58e77ebd769 273 if options.build_data:
Nathan Yonkee 9:d58e77ebd769 274 merge_build_data(options.build_data, build_report, "test")
Nathan Yonkee 9:d58e77ebd769 275
Nathan Yonkee 9:d58e77ebd769 276 if status:
Nathan Yonkee 9:d58e77ebd769 277 sys.exit(0)
Nathan Yonkee 9:d58e77ebd769 278 else:
Nathan Yonkee 9:d58e77ebd769 279 sys.exit(1)
Nathan Yonkee 9:d58e77ebd769 280
Nathan Yonkee 9:d58e77ebd769 281 except KeyboardInterrupt as e:
Nathan Yonkee 9:d58e77ebd769 282 print("\n[CTRL+c] exit")
Nathan Yonkee 9:d58e77ebd769 283 except ConfigException as e:
Nathan Yonkee 9:d58e77ebd769 284 # Catching ConfigException here to prevent a traceback
Nathan Yonkee 9:d58e77ebd769 285 print("[ERROR] %s" % str(e))
Nathan Yonkee 9:d58e77ebd769 286 except Exception as e:
Nathan Yonkee 9:d58e77ebd769 287 import traceback
Nathan Yonkee 9:d58e77ebd769 288 traceback.print_exc(file=sys.stdout)
Nathan Yonkee 9:d58e77ebd769 289 print("[ERROR] %s" % str(e))
Nathan Yonkee 9:d58e77ebd769 290 sys.exit(1)