| 
									
										
											  
											
												prefer Python 3 to Python 2 for running Spack
The Python landscape is going to be changing in 2020, and Python 2 will
be end of life. Spack should *prefer* Python 3 to Python 2 by default,
but we still need to run on systems that only have Python2 available.
This is trickier than it sounds, as on some systems, the `python` command
is `python2`; on others it's `python3`, and RHEL8 doesn't even have the
`python` command.  Instead, it makes you choose `python3` or
`python2`. You can thus no longer make a simple shebang to handle all the
cases.
This commit makes the `spack` script bilingual.  It is still valid
Python, but its shebang is `#!/bin/sh`, and it has a tiny bit of shell
code at the beginning to pick the right python and execute itself with
what it finds.
This has a lot of advantages.  I think this will help ensure that Spack
works well in Python3 -- there are cases where we've missed things
because Python2 is still the default `python` on most systems.  Also,
with this change, you do not lose the ability to execute the `spack`
script directly with a python interpreter.  This is useful for forcing
your own version of python, running coverage tools, and running profiling
tools.  i.e., these will not break with this change:
```console
$ python2 $(which spack) <args>
$ coverage run $(which spack) <args>
$ pyinstrument $(which spack) <args>
```
These would not work if we split `spack` into a python file and a shell
script (see #11783).  So, this gives us the best of both worlds.  We get
to control our interpreter *and* remain a mostly pure python executable.
											
										 
											2019-09-28 20:59:02 -07:00
										 |  |  | #!/bin/sh | 
					
						
							|  |  |  | # -*- python -*- | 
					
						
							| 
									
										
										
										
											2014-01-08 10:21:02 +01:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2022-01-12 11:21:41 -08:00
										 |  |  | # Copyright 2013-2022 Lawrence Livermore National Security, LLC and other | 
					
						
							| 
									
										
										
										
											2018-10-07 13:52:23 -07:00
										 |  |  | # Spack Project Developers. See the top-level COPYRIGHT file for details. | 
					
						
							| 
									
										
										
										
											2014-01-08 10:21:02 +01:00
										 |  |  | # | 
					
						
							| 
									
										
										
										
											2018-10-07 13:52:23 -07:00
										 |  |  | # SPDX-License-Identifier: (Apache-2.0 OR MIT) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												prefer Python 3 to Python 2 for running Spack
The Python landscape is going to be changing in 2020, and Python 2 will
be end of life. Spack should *prefer* Python 3 to Python 2 by default,
but we still need to run on systems that only have Python2 available.
This is trickier than it sounds, as on some systems, the `python` command
is `python2`; on others it's `python3`, and RHEL8 doesn't even have the
`python` command.  Instead, it makes you choose `python3` or
`python2`. You can thus no longer make a simple shebang to handle all the
cases.
This commit makes the `spack` script bilingual.  It is still valid
Python, but its shebang is `#!/bin/sh`, and it has a tiny bit of shell
code at the beginning to pick the right python and execute itself with
what it finds.
This has a lot of advantages.  I think this will help ensure that Spack
works well in Python3 -- there are cases where we've missed things
because Python2 is still the default `python` on most systems.  Also,
with this change, you do not lose the ability to execute the `spack`
script directly with a python interpreter.  This is useful for forcing
your own version of python, running coverage tools, and running profiling
tools.  i.e., these will not break with this change:
```console
$ python2 $(which spack) <args>
$ coverage run $(which spack) <args>
$ pyinstrument $(which spack) <args>
```
These would not work if we split `spack` into a python file and a shell
script (see #11783).  So, this gives us the best of both worlds.  We get
to control our interpreter *and* remain a mostly pure python executable.
											
										 
											2019-09-28 20:59:02 -07:00
										 |  |  | # This file is bilingual. The following shell code finds our preferred python. | 
					
						
							|  |  |  | # Following line is a shell no-op, and starts a multi-line Python comment. | 
					
						
							|  |  |  | # See https://stackoverflow.com/a/47886254 | 
					
						
							|  |  |  | """:" | 
					
						
							| 
									
										
										
										
											2021-02-12 13:52:44 -05:00
										 |  |  | # prefer SPACK_PYTHON environment variable, python3, python, then python2 | 
					
						
							| 
									
										
										
										
											2021-05-22 15:35:07 -07:00
										 |  |  | SPACK_PREFERRED_PYTHONS="python3 python python2 /usr/libexec/platform-python" | 
					
						
							|  |  |  | for cmd in "${SPACK_PYTHON:-}" ${SPACK_PREFERRED_PYTHONS}; do | 
					
						
							| 
									
										
										
										
											2021-02-12 13:52:44 -05:00
										 |  |  |     if command -v > /dev/null "$cmd"; then | 
					
						
							|  |  |  |         export SPACK_PYTHON="$(command -v "$cmd")" | 
					
						
							|  |  |  |         exec "${SPACK_PYTHON}" "$0" "$@" | 
					
						
							|  |  |  |     fi | 
					
						
							| 
									
										
											  
											
												prefer Python 3 to Python 2 for running Spack
The Python landscape is going to be changing in 2020, and Python 2 will
be end of life. Spack should *prefer* Python 3 to Python 2 by default,
but we still need to run on systems that only have Python2 available.
This is trickier than it sounds, as on some systems, the `python` command
is `python2`; on others it's `python3`, and RHEL8 doesn't even have the
`python` command.  Instead, it makes you choose `python3` or
`python2`. You can thus no longer make a simple shebang to handle all the
cases.
This commit makes the `spack` script bilingual.  It is still valid
Python, but its shebang is `#!/bin/sh`, and it has a tiny bit of shell
code at the beginning to pick the right python and execute itself with
what it finds.
This has a lot of advantages.  I think this will help ensure that Spack
works well in Python3 -- there are cases where we've missed things
because Python2 is still the default `python` on most systems.  Also,
with this change, you do not lose the ability to execute the `spack`
script directly with a python interpreter.  This is useful for forcing
your own version of python, running coverage tools, and running profiling
tools.  i.e., these will not break with this change:
```console
$ python2 $(which spack) <args>
$ coverage run $(which spack) <args>
$ pyinstrument $(which spack) <args>
```
These would not work if we split `spack` into a python file and a shell
script (see #11783).  So, this gives us the best of both worlds.  We get
to control our interpreter *and* remain a mostly pure python executable.
											
										 
											2019-09-28 20:59:02 -07:00
										 |  |  | done | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | echo "==> Error: spack could not find a python interpreter!" >&2 | 
					
						
							|  |  |  | exit 1 | 
					
						
							|  |  |  | ":""" | 
					
						
							|  |  |  | # Line above is a shell no-op, and ends a python multi-line comment. | 
					
						
							|  |  |  | # The code above runs this file with our preferred python interpreter. | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-03-07 14:25:48 -08:00
										 |  |  | from __future__ import print_function | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | import os | 
					
						
							| 
									
										
										
										
											2021-08-17 17:52:51 +02:00
										 |  |  | import os.path | 
					
						
							| 
									
										
										
										
											2013-06-02 13:54:46 -07:00
										 |  |  | import sys | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  | min_python3 = (3, 5) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-23 18:06:17 +01:00
										 |  |  | if sys.version_info[:2] < (2, 7) or ( | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  |     sys.version_info[:2] >= (3, 0) and sys.version_info[:2] < min_python3 | 
					
						
							|  |  |  | ): | 
					
						
							| 
									
										
										
										
											2014-12-19 11:09:37 -08:00
										 |  |  |     v_info = sys.version_info[:3] | 
					
						
							| 
									
										
										
										
											2021-11-23 18:06:17 +01:00
										 |  |  |     msg = "Spack requires Python 2.7 or %d.%d or higher " % min_python3 | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  |     msg += "You are running spack with Python %d.%d.%d." % v_info | 
					
						
							|  |  |  |     sys.exit(msg) | 
					
						
							| 
									
										
										
										
											2013-02-13 17:50:44 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Find spack's location and its prefix. | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | spack_file = os.path.realpath(os.path.expanduser(__file__)) | 
					
						
							|  |  |  | spack_prefix = os.path.dirname(os.path.dirname(spack_file)) | 
					
						
							| 
									
										
										
										
											2013-02-13 17:50:44 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Allow spack libs to be imported in our scripts | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | spack_lib_path = os.path.join(spack_prefix, "lib", "spack") | 
					
						
							|  |  |  | sys.path.insert(0, spack_lib_path) | 
					
						
							| 
									
										
										
										
											2016-10-12 18:25:18 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Add external libs | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | spack_external_libs = os.path.join(spack_lib_path, "external") | 
					
						
							| 
									
										
										
										
											2018-01-16 07:00:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
											  
											
												add mypy to style checks; rename `spack flake8` to `spack style` (#20384)
I lost my mind a bit after getting the completion stuff working and
decided to get Mypy working for spack as well. This adds a 
`.mypy.ini` that checks all of the spack and llnl modules, though
not yet packages, and fixes all of the identified missing types and
type issues for the spack library.
In addition to these changes, this includes:
* rename `spack flake8` to `spack style`
Aliases flake8 to style, and just runs flake8 as before, but with
a warning.  The style command runs both `flake8` and `mypy`,
in sequence. Added --no-<tool> options to turn off one or the
other, they are on by default.  Fixed two issues caught by the tools.
* stub typing module for python2.x
We don't support typing in Spack for python 2.x. To allow 2.x to
support `import typing` and `from typing import ...` without a
try/except dance to support old versions, this adds a stub module
*just* for python 2.x.  Doing it this way means we can only reliably
use all type hints in python3.7+, and mypi.ini has been updated to
reflect that.
* add non-default black check to spack style
This is a first step to requiring black.  It doesn't enforce it by
default, but it will check it if requested.  Currently enforcing the
line length of 79 since that's what flake8 requires, but it's a bit odd
for a black formatted project to be quite that narrow.  All settings are
in the style command since spack has no pyproject.toml and I don't
want to add one until more discussion happens. Also re-format
`style.py` since it no longer passed the black style check
with the new length.
* use style check in github action
Update the style and docs action to use `spack style`, adding in mypy
and black to the action even if it isn't running black right now.
											
										 
											2020-12-22 21:39:10 -08:00
										 |  |  | if sys.version_info[:2] <= (2, 7): | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  |     sys.path.insert(0, os.path.join(spack_external_libs, "py2")) | 
					
						
							| 
									
										
										
										
											2018-01-16 07:00:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | sys.path.insert(0, spack_external_libs) | 
					
						
							| 
									
										
										
										
											2013-02-13 17:50:44 -08:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2018-11-07 01:06:18 +01:00
										 |  |  | # Here we delete ruamel.yaml in case it has been already imported from site | 
					
						
							|  |  |  | # (see #9206 for a broader description of the issue). | 
					
						
							|  |  |  | # | 
					
						
							|  |  |  | # Briefly: ruamel.yaml produces a .pth file when installed with pip that | 
					
						
							| 
									
										
										
										
											2019-05-18 21:00:46 -05:00
										 |  |  | # makes the site installed package the preferred one, even though sys.path | 
					
						
							| 
									
										
										
										
											2018-11-07 01:06:18 +01:00
										 |  |  | # is modified to point to another version of ruamel.yaml. | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  | if "ruamel.yaml" in sys.modules: | 
					
						
							|  |  |  |     del sys.modules["ruamel.yaml"] | 
					
						
							| 
									
										
										
										
											2018-11-07 01:06:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-03-20 08:01:23 -07:00
										 |  |  | if "ruamel" in sys.modules: | 
					
						
							|  |  |  |     del sys.modules["ruamel"] | 
					
						
							| 
									
										
										
										
											2018-11-07 01:06:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-08-17 17:52:51 +02:00
										 |  |  | # The following code is here to avoid failures when updating | 
					
						
							|  |  |  | # the develop version, due to spurious argparse.pyc files remaining | 
					
						
							|  |  |  | # in the libs/spack/external directory, see: | 
					
						
							|  |  |  | # https://github.com/spack/spack/pull/25376 | 
					
						
							|  |  |  | # TODO: Remove in v0.18.0 or later | 
					
						
							|  |  |  | try: | 
					
						
							|  |  |  |     import argparse | 
					
						
							|  |  |  | except ImportError: | 
					
						
							|  |  |  |     argparse_pyc = os.path.join(spack_external_libs, 'argparse.pyc') | 
					
						
							|  |  |  |     if not os.path.exists(argparse_pyc): | 
					
						
							|  |  |  |         raise | 
					
						
							|  |  |  |     try: | 
					
						
							|  |  |  |         os.remove(argparse_pyc) | 
					
						
							|  |  |  |         import argparse  # noqa | 
					
						
							|  |  |  |     except Exception: | 
					
						
							|  |  |  |         msg = ('The file\n\n\t{0}\n\nis corrupted and cannot be deleted by Spack. ' | 
					
						
							|  |  |  |                'Either delete it manually or ask some administrator to ' | 
					
						
							|  |  |  |                'delete it for you.') | 
					
						
							|  |  |  |         print(msg.format(argparse_pyc)) | 
					
						
							|  |  |  |         sys.exit(1) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-05-08 13:18:29 -07:00
										 |  |  | import spack.main  # noqa | 
					
						
							| 
									
										
										
										
											2020-11-12 12:26:23 -08:00
										 |  |  | 
 | 
					
						
							|  |  |  | # Once we've set up the system path, run the spack main method | 
					
						
							|  |  |  | if __name__ == "__main__": | 
					
						
							|  |  |  |     sys.exit(spack.main.main()) |