171 lines
7.7 KiB
Python
Executable File
171 lines
7.7 KiB
Python
Executable File
#!/usr/bin/env python
|
|
# Copyright The Lightning AI team.
|
|
#
|
|
# Licensed under the Apache License, Version 2.0 (the "License");
|
|
# you may not use this file except in compliance with the License.
|
|
# You may obtain a copy of the License at
|
|
#
|
|
# http://www.apache.org/licenses/LICENSE-2.0
|
|
#
|
|
# Unless required by applicable law or agreed to in writing, software
|
|
# distributed under the License is distributed on an "AS IS" BASIS,
|
|
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
# See the License for the specific language governing permissions and
|
|
# limitations under the License.
|
|
"""This is the main and only one setup entry point for installing each package as stand-alone as well as joint
|
|
installation for all packages.
|
|
|
|
There are considered three main scenarios for installing this project:
|
|
|
|
1. Using PyPI registry when you can install `pytorch-lightning`, `lightning-app`, etc. or `lightning` for all.
|
|
|
|
2. Installation from source code after cloning repository.
|
|
In such case we recommend to use command `pip install .` or `pip install -e .` for development version
|
|
(development ver. do not copy python files to your pip file system, just create links, so you can edit here)
|
|
In case you want to install just one package you need to export env. variable before calling `pip`
|
|
|
|
- for `pytorch-lightning` use `export PACKAGE_NAME=pytorch ; pip install .`
|
|
- for `lightning-fabric` use `export PACKAGE_NAME=fabric ; pip install .`
|
|
- for `lightning-app` use `export PACKAGE_NAME=app ; pip install .`
|
|
|
|
3. Building packages as sdist or binary wheel and installing or publish to PyPI afterwords you use command
|
|
`python setup.py sdist` or `python setup.py bdist_wheel` accordingly.
|
|
In case you want to build just a particular package you want to set an environment variable:
|
|
`PACKAGE_NAME=lightning|pytorch|app|fabric python setup.py sdist|bdist_wheel`
|
|
|
|
4. Automated releasing with GitHub action is natural extension of 3) is composed of three consecutive steps:
|
|
a) determine which packages shall be released based on version increment in `__version__.py` and eventually
|
|
compared against PyPI registry
|
|
b) with a parameterization build desired packages in to standard `dist/` folder
|
|
c) validate packages and publish to PyPI
|
|
"""
|
|
import contextlib
|
|
import glob
|
|
import logging
|
|
import os
|
|
import tempfile
|
|
from importlib.util import module_from_spec, spec_from_file_location
|
|
from types import ModuleType
|
|
from typing import Generator, Mapping, Optional
|
|
|
|
import setuptools
|
|
import setuptools.command.egg_info
|
|
|
|
_PACKAGE_NAME = os.environ.get("PACKAGE_NAME")
|
|
_PACKAGE_MAPPING = {
|
|
"lightning": "lightning",
|
|
"pytorch": "pytorch_lightning",
|
|
"app": "lightning_app",
|
|
"fabric": "lightning_fabric",
|
|
}
|
|
# https://packaging.python.org/guides/single-sourcing-package-version/
|
|
# http://blog.ionelmc.ro/2014/05/25/python-packaging/
|
|
_PATH_ROOT = os.path.dirname(__file__)
|
|
_PATH_SRC = os.path.join(_PATH_ROOT, "src")
|
|
_PATH_REQUIRE = os.path.join(_PATH_ROOT, "requirements")
|
|
_FREEZE_REQUIREMENTS = os.environ.get("FREEZE_REQUIREMENTS", "0").lower() in ("1", "true")
|
|
|
|
|
|
def _load_py_module(name: str, location: str) -> ModuleType:
|
|
spec = spec_from_file_location(name, location)
|
|
assert spec, f"Failed to load module {name} from {location}"
|
|
py = module_from_spec(spec)
|
|
assert spec.loader, f"ModuleSpec.loader is None for {name} from {location}"
|
|
spec.loader.exec_module(py)
|
|
return py
|
|
|
|
|
|
def _named_temporary_file(directory: Optional[str] = None) -> str:
|
|
# `tempfile.NamedTemporaryFile` has issues in Windows
|
|
# https://github.com/deepchem/deepchem/issues/707#issuecomment-556002823
|
|
if directory is None:
|
|
directory = tempfile.gettempdir()
|
|
return os.path.join(directory, os.urandom(24).hex())
|
|
|
|
|
|
@contextlib.contextmanager
|
|
def _set_manifest_path(manifest_dir: str, aggregate: bool = False, mapping: Mapping = _PACKAGE_MAPPING) -> Generator:
|
|
if aggregate:
|
|
# aggregate all MANIFEST.in contents into a single temporary file
|
|
manifest_path = _named_temporary_file(manifest_dir)
|
|
lines = []
|
|
# load manifest and aggregated all manifests
|
|
for pkg in mapping.values():
|
|
pkg_manifest = os.path.join(_PATH_SRC, pkg, "MANIFEST.in")
|
|
if os.path.isfile(pkg_manifest):
|
|
with open(pkg_manifest) as fh:
|
|
lines.extend(fh.readlines())
|
|
# convert lightning_foo to lightning/foo
|
|
for new, old in mapping.items():
|
|
if old == "lightning":
|
|
continue # avoid `lightning` -> `lightning/lightning`
|
|
lines = [ln.replace(old, f"lightning/{new}") for ln in lines]
|
|
lines = sorted(set(filter(lambda ln: not ln.strip().startswith("#"), lines)))
|
|
logging.debug(f"aggregated manifest consists of: {lines}")
|
|
with open(manifest_path, mode="w") as fp:
|
|
fp.writelines(lines)
|
|
else:
|
|
manifest_path = os.path.join(manifest_dir, "MANIFEST.in")
|
|
assert os.path.exists(manifest_path)
|
|
# avoid error: setup script specifies an absolute path
|
|
manifest_path = os.path.relpath(manifest_path, _PATH_ROOT)
|
|
logging.info("Set manifest path to", manifest_path)
|
|
setuptools.command.egg_info.manifest_maker.template = manifest_path
|
|
yield
|
|
# cleanup
|
|
setuptools.command.egg_info.manifest_maker.template = "MANIFEST.in"
|
|
if aggregate:
|
|
os.remove(manifest_path)
|
|
|
|
|
|
if __name__ == "__main__":
|
|
assistant = _load_py_module(name="assistant", location=os.path.join(_PATH_ROOT, ".actions", "assistant.py"))
|
|
|
|
if os.path.isdir(_PATH_SRC):
|
|
# copy the version information to all packages
|
|
assistant.distribute_version(_PATH_SRC)
|
|
print(f"Requested package: '{_PACKAGE_NAME}'") # requires `-v` to appear
|
|
|
|
local_pkgs = [
|
|
os.path.basename(p)
|
|
for p in glob.glob(os.path.join(_PATH_SRC, "*"))
|
|
if os.path.isdir(p) and not p.endswith(".egg-info")
|
|
]
|
|
print(f"Local package candidates: {local_pkgs}")
|
|
is_source_install = len(local_pkgs) > 2
|
|
print(f"Installing from source: {is_source_install}")
|
|
if is_source_install:
|
|
if _PACKAGE_NAME is not None and _PACKAGE_NAME not in _PACKAGE_MAPPING:
|
|
raise ValueError(
|
|
f"Unexpected package name: {_PACKAGE_NAME}. Possible choices are: {list(_PACKAGE_MAPPING)}"
|
|
)
|
|
package_to_install = _PACKAGE_MAPPING.get(_PACKAGE_NAME, "lightning")
|
|
if package_to_install == "lightning":
|
|
# merge all requirements files
|
|
assistant._load_aggregate_requirements(_PATH_REQUIRE, _FREEZE_REQUIREMENTS)
|
|
else:
|
|
# replace imports and copy the code
|
|
assistant.create_mirror_package(_PATH_SRC, _PACKAGE_MAPPING)
|
|
else:
|
|
assert len(local_pkgs) > 0
|
|
# PL as a package is distributed together with Fabric, so in such case there are more than one candidate
|
|
package_to_install = "pytorch_lightning" if "pytorch_lightning" in local_pkgs else local_pkgs[0]
|
|
print(f"Installing package: {package_to_install}")
|
|
|
|
# going to install with `setuptools.setup`
|
|
pkg_path = os.path.join(_PATH_SRC, package_to_install)
|
|
pkg_setup = os.path.join(pkg_path, "__setup__.py")
|
|
if not os.path.exists(pkg_setup):
|
|
raise RuntimeError(f"Something's wrong, no package was installed. Package name: {_PACKAGE_NAME}")
|
|
setup_module = _load_py_module(name=f"{package_to_install}_setup", location=pkg_setup)
|
|
setup_args = setup_module._setup_args()
|
|
is_main_pkg = package_to_install == "lightning"
|
|
print(f"Installing as the main package: {is_main_pkg}")
|
|
if is_source_install:
|
|
# we are installing from source, set the correct manifest path
|
|
with _set_manifest_path(pkg_path, aggregate=is_main_pkg):
|
|
setuptools.setup(**setup_args)
|
|
else:
|
|
setuptools.setup(**setup_args)
|
|
print("Finished setup configuration.")
|