1 # This Source Code Form is subject to the terms of the Mozilla Public
2 # License, v. 2.0. If a copy of the MPL was not distributed with this
3 # file, You can obtain one at http://mozilla.org/MPL/2.0/.
5 from __future__
import absolute_import
, print_function
, unicode_literals
13 from collections
import Iterable
16 base_dir
= os
.path
.abspath(os
.path
.dirname(__file__
))
17 sys
.path
.insert(0, os
.path
.join(base_dir
, 'python', 'mozbuild'))
18 sys
.path
.insert(0, os
.path
.join(base_dir
, 'third_party', 'python', 'six'))
19 from mozbuild
.configure
import (
23 from mozbuild
.pythonutil
import iter_modules_in_path
24 from mozbuild
.backend
.configenvironment
import PartialConfigEnvironment
25 from mozbuild
.util
import (
28 import mozpack
.path
as mozpath
35 sandbox
= ConfigureSandbox(config
, os
.environ
, argv
)
37 clobber_file
= 'CLOBBER'
38 if not os
.path
.exists(clobber_file
):
39 # Simply touch the file.
40 with
open(clobber_file
, 'a'):
43 if os
.environ
.get('MOZ_CONFIGURE_TRACE'):
44 sandbox
._logger
.setLevel(TRACE
)
46 sandbox
.run(os
.path
.join(os
.path
.dirname(__file__
), 'moz.configure'))
51 return config_status(config
)
54 def check_unicode(obj
):
55 '''Recursively check that all strings in the object are unicode strings.'''
56 if isinstance(obj
, dict):
58 for k
, v
in six
.iteritems(obj
):
59 if not check_unicode(k
):
60 print("%s key is not unicode." % k
, file=sys
.stderr
)
62 elif not check_unicode(v
):
63 print("%s value is not unicode." % k
, file=sys
.stderr
)
66 if isinstance(obj
, bytes
):
68 if isinstance(obj
, six
.text_type
):
70 if isinstance(obj
, Iterable
):
71 return all(check_unicode(o
) for o
in obj
)
75 def config_status(config
):
76 # Sanitize config data to feed config.status
77 # Ideally, all the backend and frontend code would handle the booleans, but
78 # there are so many things involved, that it's easier to keep config.status
80 def sanitized_bools(v
):
88 sanitized_config
['substs'] = {
89 k
: sanitized_bools(v
) for k
, v
in six
.iteritems(config
)
90 if k
not in ('DEFINES', 'non_global_defines', 'TOPSRCDIR', 'TOPOBJDIR',
93 sanitized_config
['defines'] = {
94 k
: sanitized_bools(v
) for k
, v
in six
.iteritems(config
['DEFINES'])
96 sanitized_config
['non_global_defines'] = config
['non_global_defines']
97 sanitized_config
['topsrcdir'] = config
['TOPSRCDIR']
98 sanitized_config
['topobjdir'] = config
['TOPOBJDIR']
99 sanitized_config
['mozconfig'] = config
.get('MOZCONFIG')
101 if not check_unicode(sanitized_config
):
102 print("Configuration should be all unicode.", file=sys
.stderr
)
103 print("Please file a bug for the above.", file=sys
.stderr
)
106 # Create config.status. Eventually, we'll want to just do the work it does
107 # here, when we're able to skip configure tests/use cached results/not rely
109 logging
.getLogger('moz.configure').info('Creating config.status')
110 with codecs
.open('config.status', 'w', 'utf-8') as fh
:
111 fh
.write(textwrap
.dedent('''\
114 from __future__ import unicode_literals
115 ''') % {'python': config
['PYTHON']})
116 for k
, v
in six
.iteritems(sanitized_config
):
117 fh
.write('%s = %s\n' % (k
, indented_repr(v
)))
118 fh
.write("__all__ = ['topobjdir', 'topsrcdir', 'defines', "
119 "'non_global_defines', 'substs', 'mozconfig']")
121 if config
.get('MOZ_BUILD_APP') != 'js' or config
.get('JS_STANDALONE'):
122 fh
.write(textwrap
.dedent('''
123 if __name__ == '__main__':
124 from mozbuild.util import patch_main
126 from mozbuild.config_status import config_status
127 args = dict([(name, globals()[name]) for name in __all__])
128 config_status(**args)
131 partial_config
= PartialConfigEnvironment(config
['TOPOBJDIR'])
132 partial_config
.write_vars(sanitized_config
)
134 # Write out a file so the build backend knows to re-run configure when
135 # relevant Python changes.
136 with
open('config_status_deps.in', 'w') as fh
:
137 for f
in itertools
.chain(config
['CONFIG_STATUS_DEPS'],
138 iter_modules_in_path(config
['TOPOBJDIR'],
139 config
['TOPSRCDIR'])):
140 fh
.write('%s\n' % mozpath
.normpath(f
))
142 # Other things than us are going to run this file, so we need to give it
143 # executable permissions.
144 os
.chmod('config.status', 0o755)
145 if config
.get('MOZ_BUILD_APP') != 'js' or config
.get('JS_STANDALONE'):
146 from mozbuild
.config_status
import config_status
148 # Some values in sanitized_config also have more complex types, such as
149 # EnumString, which using when calling config_status would currently
150 # break the build, as well as making it inconsistent with re-running
151 # config.status, for which they are normalized to plain strings via
152 # indented_repr. Likewise for non-dict non-string iterables being
153 # converted to lists.
155 if isinstance(obj
, dict):
158 for k
, v
in six
.iteritems(obj
)
160 if isinstance(obj
, six
.text_type
):
161 return six
.text_type(obj
)
162 if isinstance(obj
, Iterable
):
163 return [normalize(o
) for o
in obj
]
165 return config_status(args
=[], **normalize(sanitized_config
))
169 if __name__
== '__main__':
170 sys
.exit(main(sys
.argv
))