mirror of
https://github.com/apple/swift.git
synced 2025-12-14 20:36:38 +01:00
* Implemented a presets module which includes a more robust and easy to understand parser. Moved the swift-sdks migration code to a new migration module and added testing for both the presets and migration modules. Also converted build-script to use the new presets parser. * Switched the expansion algorithm to resolve mixins in-place (little known feature) and also changed the parser to skip all non-preset sections. Tests are included for these two behaviors. * Re-worked the presets error hierarchy to have more descriptive and information packed exception classes. Also re-worked the PresetParser to catch duplicate preset declarations and duplicate options in a single preset. There's some special shim-code to handle the disconnect between the Python 2 ConfigParser module and the Python 3 update which adds DuplicateOptionError.
91 lines
2.0 KiB
Python
91 lines
2.0 KiB
Python
# This source file is part of the Swift.org open source project
|
|
#
|
|
# Copyright (c) 2014 - 2017 Apple Inc. and the Swift project authors
|
|
# Licensed under Apache License v2.0 with Runtime Library Exception
|
|
#
|
|
# See https://swift.org/LICENSE.txt for license information
|
|
# See https://swift.org/CONTRIBUTORS.txt for the list of Swift project authors
|
|
|
|
|
|
try:
|
|
from StringIO import StringIO
|
|
except ImportError:
|
|
from io import StringIO
|
|
|
|
import os
|
|
import sys
|
|
import unittest
|
|
from contextlib import contextmanager
|
|
|
|
|
|
__all__ = [
|
|
'add_metaclass',
|
|
'redirect_stderr',
|
|
'redirect_stdout',
|
|
'TestCase',
|
|
|
|
'UTILS_PATH',
|
|
]
|
|
|
|
|
|
UTILS_PATH = os.path.abspath(os.path.join(
|
|
os.path.dirname(__file__),
|
|
os.pardir,
|
|
os.pardir,
|
|
))
|
|
|
|
|
|
# -----------------------------------------------------------------------------
|
|
|
|
def add_metaclass(metacls):
|
|
def wrapper(cls):
|
|
body = vars(cls).copy()
|
|
|
|
body.pop('__dict__', None)
|
|
body.pop('__weakref__', None)
|
|
|
|
return metacls(cls.__name__, cls.__bases__, body)
|
|
|
|
return wrapper
|
|
|
|
|
|
@contextmanager
|
|
def redirect_stderr(stream=None):
|
|
stream = stream or StringIO()
|
|
old_stderr, sys.stderr = sys.stderr, stream
|
|
try:
|
|
yield stream
|
|
finally:
|
|
sys.stderr = old_stderr
|
|
|
|
|
|
@contextmanager
|
|
def redirect_stdout(stream=None):
|
|
stream = stream or StringIO()
|
|
old_stdout, sys.stdout = sys.stdout, stream
|
|
try:
|
|
yield stream
|
|
finally:
|
|
sys.stdout = old_stdout
|
|
|
|
|
|
# -----------------------------------------------------------------------------
|
|
|
|
class TestCase(unittest.TestCase):
|
|
|
|
@contextmanager
|
|
def quietOutput(self):
|
|
with open(os.devnull, 'w') as devnull:
|
|
with redirect_stderr(devnull), redirect_stdout(devnull):
|
|
yield
|
|
|
|
@contextmanager
|
|
def assertNotRaises(self, exception=BaseException):
|
|
assert issubclass(exception, BaseException)
|
|
|
|
try:
|
|
yield
|
|
except exception as e:
|
|
message = '{} raised: {}'.format(exception.__name__, str(e))
|
|
raise self.failureException(message)
|