1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231 |
- #!/usr/bin/env python
- # -*- coding: utf-8 -*-
- '''
- The setup script for salt
- '''
- # pylint: disable=file-perms,ungrouped-imports,wrong-import-order,wrong-import-position,repr-flag-used-in-string
- # pylint: disable=3rd-party-local-module-not-gated,resource-leakage
- # pylint: disable=C0111,E1101,E1103,F0401,W0611,W0201,W0232,R0201,R0902,R0903
- # For Python 2.5. A no-op on 2.6 and above.
- from __future__ import absolute_import, print_function, with_statement
- import os
- import sys
- import glob
- import time
- import operator
- import platform
- try:
- from urllib2 import urlopen
- except ImportError:
- from urllib.request import urlopen # pylint: disable=no-name-in-module
- from datetime import datetime
- # pylint: disable=E0611
- import distutils.dist
- from distutils import log
- from distutils.cmd import Command
- from distutils.errors import DistutilsArgError
- from distutils.command.build import build
- from distutils.command.clean import clean
- from distutils.command.sdist import sdist
- from distutils.command.install_lib import install_lib
- from distutils.version import LooseVersion # pylint: disable=blacklisted-module
- from ctypes.util import find_library
- # pylint: enable=E0611
- try:
- import zmq
- HAS_ZMQ = True
- except ImportError:
- HAS_ZMQ = False
- try:
- DATE = datetime.utcfromtimestamp(int(os.environ['SOURCE_DATE_EPOCH']))
- except (KeyError, ValueError):
- DATE = datetime.utcnow()
- # Change to salt source's directory prior to running any command
- try:
- SETUP_DIRNAME = os.path.dirname(__file__)
- except NameError:
- # We're most likely being frozen and __file__ triggered this NameError
- # Let's work around that
- SETUP_DIRNAME = os.path.dirname(sys.argv[0])
- if SETUP_DIRNAME != '':
- os.chdir(SETUP_DIRNAME)
- SETUP_DIRNAME = os.path.abspath(SETUP_DIRNAME)
- BOOTSTRAP_SCRIPT_DISTRIBUTED_VERSION = os.environ.get(
- # The user can provide a different bootstrap-script version.
- # ATTENTION: A tag for that version MUST exist
- 'BOOTSTRAP_SCRIPT_VERSION',
- # If no bootstrap-script version was provided from the environment, let's
- # provide the one we define.
- 'v2014.06.21'
- )
- # Store a reference to the executing platform
- IS_WINDOWS_PLATFORM = sys.platform.startswith('win')
- if IS_WINDOWS_PLATFORM:
- IS_SMARTOS_PLATFORM = False
- else:
- # os.uname() not available on Windows.
- IS_SMARTOS_PLATFORM = os.uname()[0] == 'SunOS' and os.uname()[3].startswith('joyent_')
- # Store a reference whether if we're running under Python 3 and above
- IS_PY3 = sys.version_info > (3,)
- # Use setuptools only if the user opts-in by setting the USE_SETUPTOOLS env var
- # Or if setuptools was previously imported (which is the case when using
- # 'distribute')
- # This ensures consistent behavior but allows for advanced usage with
- # virtualenv, buildout, and others.
- WITH_SETUPTOOLS = False
- if 'USE_SETUPTOOLS' in os.environ or 'setuptools' in sys.modules:
- try:
- from setuptools import setup
- from setuptools.command.develop import develop
- from setuptools.command.install import install
- from setuptools.command.sdist import sdist
- from setuptools.command.egg_info import egg_info
- WITH_SETUPTOOLS = True
- except ImportError:
- WITH_SETUPTOOLS = False
- if WITH_SETUPTOOLS is False:
- import warnings
- # pylint: disable=E0611
- from distutils.command.install import install
- from distutils.core import setup
- # pylint: enable=E0611
- warnings.filterwarnings(
- 'ignore',
- 'Unknown distribution option: \'(extras_require|tests_require|install_requires|zip_safe)\'',
- UserWarning,
- 'distutils.dist'
- )
- try:
- # Add the esky bdist target if the module is available
- # may require additional modules depending on platform
- from esky import bdist_esky
- # bbfreeze chosen for its tight integration with distutils
- import bbfreeze
- HAS_ESKY = True
- except ImportError:
- HAS_ESKY = False
- SALT_VERSION = os.path.join(os.path.abspath(SETUP_DIRNAME), 'salt', 'version.py')
- SALT_VERSION_HARDCODED = os.path.join(os.path.abspath(SETUP_DIRNAME), 'salt', '_version.py')
- SALT_SYSPATHS_HARDCODED = os.path.join(os.path.abspath(SETUP_DIRNAME), 'salt', '_syspaths.py')
- SALT_REQS = os.path.join(os.path.abspath(SETUP_DIRNAME), 'requirements', 'base.txt')
- SALT_ZEROMQ_REQS = os.path.join(os.path.abspath(SETUP_DIRNAME), 'requirements', 'zeromq.txt')
- SALT_WINDOWS_REQS = os.path.join(os.path.abspath(SETUP_DIRNAME), 'pkg', 'windows', 'req.txt')
- SALT_LONG_DESCRIPTION_FILE = os.path.join(os.path.abspath(SETUP_DIRNAME), 'README.rst')
- # Salt SSH Packaging Detection
- PACKAGED_FOR_SALT_SSH_FILE = os.path.join(os.path.abspath(SETUP_DIRNAME), '.salt-ssh-package')
- PACKAGED_FOR_SALT_SSH = os.path.isfile(PACKAGED_FOR_SALT_SSH_FILE)
- # pylint: disable=W0122
- exec(compile(open(SALT_VERSION).read(), SALT_VERSION, 'exec'))
- # pylint: enable=W0122
- # ----- Helper Functions -------------------------------------------------------------------------------------------->
- def _parse_op(op):
- '''
- >>> _parse_op('>')
- 'gt'
- >>> _parse_op('>=')
- 'ge'
- >>> _parse_op('=>')
- 'ge'
- >>> _parse_op('=> ')
- 'ge'
- >>> _parse_op('<')
- 'lt'
- >>> _parse_op('<=')
- 'le'
- >>> _parse_op('==')
- 'eq'
- >>> _parse_op(' <= ')
- 'le'
- '''
- op = op.strip()
- if '>' in op:
- if '=' in op:
- return 'ge'
- else:
- return 'gt'
- elif '<' in op:
- if '=' in op:
- return 'le'
- else:
- return 'lt'
- elif '!' in op:
- return 'ne'
- else:
- return 'eq'
- def _parse_ver(ver):
- '''
- >>> _parse_ver("'3.4' # pyzmq 17.1.0 stopped building wheels for python3.4")
- '3.4'
- >>> _parse_ver('"3.4"')
- '3.4'
- >>> _parse_ver('"2.6.17"')
- '2.6.17'
- '''
- if '#' in ver:
- ver, _ = ver.split('#', 1)
- ver = ver.strip()
- return ver.strip('\'').strip('"')
- def _check_ver(pyver, op, wanted):
- '''
- >>> _check_ver('2.7.15', 'gt', '2.7')
- True
- >>> _check_ver('2.7.15', 'gt', '2.7.15')
- False
- >>> _check_ver('2.7.15', 'ge', '2.7.15')
- True
- >>> _check_ver('2.7.15', 'eq', '2.7.15')
- True
- '''
- pyver = distutils.version.LooseVersion(pyver)
- wanted = distutils.version.LooseVersion(wanted)
- if IS_PY3:
- if not isinstance(pyver, str):
- pyver = str(pyver)
- if not isinstance(wanted, str):
- wanted = str(wanted)
- return getattr(operator, '__{}__'.format(op))(pyver, wanted)
- def _parse_requirements_file(requirements_file):
- parsed_requirements = []
- with open(requirements_file) as rfh:
- for line in rfh.readlines():
- line = line.strip()
- if not line or line.startswith(('#', '-r')):
- continue
- if IS_WINDOWS_PLATFORM:
- if 'libcloud' in line:
- continue
- if IS_PY3 and 'futures' in line.lower():
- # Python 3 already has futures, installing it will only break
- # the current python installation whenever futures is imported
- continue
- try:
- pkg, pyverspec = line.rsplit(';', 1)
- except ValueError:
- pkg, pyverspec = line, ''
- pyverspec = pyverspec.strip()
- if pyverspec:
- _, op, ver = pyverspec.split(' ', 2)
- if not _check_ver(platform.python_version(), _parse_op(op), _parse_ver(ver)):
- continue
- parsed_requirements.append(pkg)
- return parsed_requirements
- # <---- Helper Functions ---------------------------------------------------------------------------------------------
- # ----- Custom Distutils/Setuptools Commands ------------------------------------------------------------------------>
- class WriteSaltVersion(Command):
- description = 'Write salt\'s hardcoded version file'
- user_options = []
- def initialize_options(self):
- '''
- Abstract method that is required to be overwritten
- '''
- def finalize_options(self):
- '''
- Abstract method that is required to be overwritten
- '''
- def run(self):
- if not os.path.exists(SALT_VERSION_HARDCODED) or self.distribution.with_salt_version:
- # Write the version file
- if getattr(self.distribution, 'salt_version_hardcoded_path', None) is None:
- print('This command is not meant to be called on it\'s own')
- exit(1)
- if not self.distribution.with_salt_version:
- salt_version = __saltstack_version__ # pylint: disable=undefined-variable
- else:
- from salt.version import SaltStackVersion
- salt_version = SaltStackVersion.parse(self.distribution.with_salt_version)
- # pylint: disable=E0602
- open(self.distribution.salt_version_hardcoded_path, 'w').write(
- INSTALL_VERSION_TEMPLATE.format(
- date=DATE,
- full_version_info=salt_version.full_info
- )
- )
- # pylint: enable=E0602
- class GenerateSaltSyspaths(Command):
- description = 'Generate salt\'s hardcoded syspaths file'
- def initialize_options(self):
- pass
- def finalize_options(self):
- pass
- def run(self):
- # Write the syspaths file
- if getattr(self.distribution, 'salt_syspaths_hardcoded_path', None) is None:
- print('This command is not meant to be called on it\'s own')
- exit(1)
- # Write the system paths file
- open(self.distribution.salt_syspaths_hardcoded_path, 'w').write(
- INSTALL_SYSPATHS_TEMPLATE.format(
- date=DATE,
- root_dir=self.distribution.salt_root_dir,
- share_dir=self.distribution.salt_share_dir,
- config_dir=self.distribution.salt_config_dir,
- cache_dir=self.distribution.salt_cache_dir,
- sock_dir=self.distribution.salt_sock_dir,
- srv_root_dir=self.distribution.salt_srv_root_dir,
- base_file_roots_dir=self.distribution.salt_base_file_roots_dir,
- base_pillar_roots_dir=self.distribution.salt_base_pillar_roots_dir,
- base_master_roots_dir=self.distribution.salt_base_master_roots_dir,
- base_thorium_roots_dir=self.distribution.salt_base_thorium_roots_dir,
- logs_dir=self.distribution.salt_logs_dir,
- pidfile_dir=self.distribution.salt_pidfile_dir,
- spm_parent_path=self.distribution.salt_spm_parent_dir,
- spm_formula_path=self.distribution.salt_spm_formula_dir,
- spm_pillar_path=self.distribution.salt_spm_pillar_dir,
- spm_reactor_path=self.distribution.salt_spm_reactor_dir,
- home_dir=self.distribution.salt_home_dir,
- )
- )
- class WriteSaltSshPackagingFile(Command):
- description = 'Write salt\'s ssh packaging file'
- user_options = []
- def initialize_options(self):
- '''
- Abstract method that is required to be overwritten
- '''
- def finalize_options(self):
- '''
- Abstract method that is required to be overwritten
- '''
- def run(self):
- if not os.path.exists(PACKAGED_FOR_SALT_SSH_FILE):
- # Write the salt-ssh packaging file
- if getattr(self.distribution, 'salt_ssh_packaging_file', None) is None:
- print('This command is not meant to be called on it\'s own')
- exit(1)
- # pylint: disable=E0602
- open(self.distribution.salt_ssh_packaging_file, 'w').write('Packaged for Salt-SSH\n')
- # pylint: enable=E0602
- if WITH_SETUPTOOLS:
- class Develop(develop):
- user_options = develop.user_options + [
- ('write-salt-version', None,
- 'Generate Salt\'s _version.py file which allows proper version '
- 'reporting. This defaults to False on develop/editable setups. '
- 'If WRITE_SALT_VERSION is found in the environment this flag is '
- 'switched to True.'),
- ('generate-salt-syspaths', None,
- 'Generate Salt\'s _syspaths.py file which allows tweaking some '
- 'common paths that salt uses. This defaults to False on '
- 'develop/editable setups. If GENERATE_SALT_SYSPATHS is found in '
- 'the environment this flag is switched to True.'),
- ('mimic-salt-install', None,
- 'Mimmic the install command when running the develop command. '
- 'This will generate salt\'s _version.py and _syspaths.py files. '
- 'Generate Salt\'s _syspaths.py file which allows tweaking some '
- 'This defaults to False on develop/editable setups. '
- 'If MIMIC_INSTALL is found in the environment this flag is '
- 'switched to True.')
- ]
- boolean_options = develop.boolean_options + [
- 'write-salt-version',
- 'generate-salt-syspaths',
- 'mimic-salt-install'
- ]
- def initialize_options(self):
- develop.initialize_options(self)
- self.write_salt_version = False
- self.generate_salt_syspaths = False
- self.mimic_salt_install = False
- def finalize_options(self):
- develop.finalize_options(self)
- if 'WRITE_SALT_VERSION' in os.environ:
- self.write_salt_version = True
- if 'GENERATE_SALT_SYSPATHS' in os.environ:
- self.generate_salt_syspaths = True
- if 'MIMIC_SALT_INSTALL' in os.environ:
- self.mimic_salt_install = True
- if self.mimic_salt_install:
- self.write_salt_version = True
- self.generate_salt_syspaths = True
- def run(self):
- if IS_WINDOWS_PLATFORM:
- # Download the required DLLs
- self.distribution.salt_download_windows_dlls = True
- self.run_command('download-windows-dlls')
- self.distribution.salt_download_windows_dlls = None
- if self.write_salt_version is True:
- self.distribution.running_salt_install = True
- self.distribution.salt_version_hardcoded_path = SALT_VERSION_HARDCODED
- self.run_command('write_salt_version')
- if self.generate_salt_syspaths:
- self.distribution.salt_syspaths_hardcoded_path = SALT_SYSPATHS_HARDCODED
- self.run_command('generate_salt_syspaths')
- # Resume normal execution
- develop.run(self)
- class DownloadWindowsDlls(Command):
- description = 'Download required DLL\'s for windows'
- def initialize_options(self):
- pass
- def finalize_options(self):
- pass
- def run(self):
- if getattr(self.distribution, 'salt_download_windows_dlls', None) is None:
- print('This command is not meant to be called on it\'s own')
- exit(1)
- import pip
- # pip has moved many things to `_internal` starting with pip 10
- if LooseVersion(pip.__version__) < LooseVersion('10.0'):
- from pip.utils.logging import indent_log # pylint: disable=no-name-in-module
- else:
- from pip._internal.utils.logging import indent_log # pylint: disable=no-name-in-module
- platform_bits, _ = platform.architecture()
- url = 'https://repo.saltstack.com/windows/dependencies/{bits}/{fname}.dll'
- dest = os.path.join(os.path.dirname(sys.executable), '{fname}.dll')
- with indent_log():
- for fname in ('libeay32', 'ssleay32', 'msvcr120'):
- # See if the library is already on the system
- if find_library(fname):
- continue
- furl = url.format(bits=platform_bits[:2], fname=fname)
- fdest = dest.format(fname=fname)
- if not os.path.exists(fdest):
- log.info('Downloading {0}.dll to {1} from {2}'.format(fname, fdest, furl))
- try:
- import requests
- from contextlib import closing
- with closing(requests.get(furl, stream=True)) as req:
- if req.status_code == 200:
- with open(fdest, 'wb') as wfh:
- for chunk in req.iter_content(chunk_size=4096):
- if chunk: # filter out keep-alive new chunks
- wfh.write(chunk)
- wfh.flush()
- else:
- log.error(
- 'Failed to download {0}.dll to {1} from {2}'.format(
- fname, fdest, furl
- )
- )
- except ImportError:
- req = urlopen(furl)
- if req.getcode() == 200:
- with open(fdest, 'wb') as wfh:
- if IS_PY3:
- while True:
- chunk = req.read(4096)
- if len(chunk) == 0:
- break
- wfh.write(chunk)
- wfh.flush()
- else:
- while True:
- for chunk in req.read(4096):
- if not chunk:
- break
- wfh.write(chunk)
- wfh.flush()
- else:
- log.error(
- 'Failed to download {0}.dll to {1} from {2}'.format(
- fname, fdest, furl
- )
- )
- class Sdist(sdist):
- def make_release_tree(self, base_dir, files):
- if self.distribution.ssh_packaging:
- self.distribution.salt_ssh_packaging_file = PACKAGED_FOR_SALT_SSH_FILE
- self.run_command('write_salt_ssh_packaging_file')
- self.filelist.files.append(os.path.basename(PACKAGED_FOR_SALT_SSH_FILE))
- if not IS_PY3 and not isinstance(base_dir, str):
- # Work around some bad code in distutils which logs unicode paths
- # against a str format string.
- base_dir = base_dir.encode('utf-8')
- sdist.make_release_tree(self, base_dir, files)
- # Let's generate salt/_version.py to include in the sdist tarball
- self.distribution.running_salt_sdist = True
- self.distribution.salt_version_hardcoded_path = os.path.join(
- base_dir, 'salt', '_version.py'
- )
- self.run_command('write_salt_version')
- def make_distribution(self):
- sdist.make_distribution(self)
- if self.distribution.ssh_packaging:
- os.unlink(PACKAGED_FOR_SALT_SSH_FILE)
- class CloudSdist(Sdist): # pylint: disable=too-many-ancestors
- user_options = Sdist.user_options + [
- ('download-bootstrap-script', None,
- 'Download the latest stable bootstrap-salt.sh script. This '
- 'can also be triggered by having `DOWNLOAD_BOOTSTRAP_SCRIPT=1` as an '
- 'environment variable.')
- ]
- boolean_options = Sdist.boolean_options + [
- 'download-bootstrap-script'
- ]
- def initialize_options(self):
- Sdist.initialize_options(self)
- self.skip_bootstrap_download = True
- self.download_bootstrap_script = False
- def finalize_options(self):
- Sdist.finalize_options(self)
- if 'SKIP_BOOTSTRAP_DOWNLOAD' in os.environ:
- log('Please stop using \'SKIP_BOOTSTRAP_DOWNLOAD\' and use ' # pylint: disable=not-callable
- '\'DOWNLOAD_BOOTSTRAP_SCRIPT\' instead')
- if 'DOWNLOAD_BOOTSTRAP_SCRIPT' in os.environ:
- download_bootstrap_script = os.environ.get(
- 'DOWNLOAD_BOOTSTRAP_SCRIPT', '0'
- )
- self.download_bootstrap_script = download_bootstrap_script == '1'
- def run(self):
- if self.download_bootstrap_script is True:
- # Let's update the bootstrap-script to the version defined to be
- # distributed. See BOOTSTRAP_SCRIPT_DISTRIBUTED_VERSION above.
- url = (
- 'https://github.com/saltstack/salt-bootstrap/raw/{0}'
- '/bootstrap-salt.sh'.format(
- BOOTSTRAP_SCRIPT_DISTRIBUTED_VERSION
- )
- )
- deploy_path = os.path.join(
- SETUP_DIRNAME,
- 'salt',
- 'cloud',
- 'deploy',
- 'bootstrap-salt.sh'
- )
- log.info(
- 'Updating bootstrap-salt.sh.'
- '\n\tSource: {0}'
- '\n\tDestination: {1}'.format(
- url,
- deploy_path
- )
- )
- try:
- import requests
- req = requests.get(url)
- if req.status_code == 200:
- script_contents = req.text.encode(req.encoding)
- else:
- log.error(
- 'Failed to update the bootstrap-salt.sh script. HTTP '
- 'Error code: {0}'.format(
- req.status_code
- )
- )
- except ImportError:
- req = urlopen(url)
- if req.getcode() == 200:
- script_contents = req.read()
- else:
- log.error(
- 'Failed to update the bootstrap-salt.sh script. HTTP '
- 'Error code: {0}'.format(
- req.getcode()
- )
- )
- try:
- with open(deploy_path, 'w') as fp_:
- fp_.write(script_contents)
- except (OSError, IOError) as err:
- log.error(
- 'Failed to write the updated script: {0}'.format(err)
- )
- # Let's the rest of the build command
- Sdist.run(self)
- def write_manifest(self):
- # We only need to ship the scripts which are supposed to be installed
- dist_scripts = self.distribution.scripts
- for script in self.filelist.files[:]:
- if not script.startswith('scripts/'):
- continue
- if script not in dist_scripts:
- self.filelist.files.remove(script)
- return Sdist.write_manifest(self)
- class TestCommand(Command):
- description = 'Run tests'
- user_options = [
- ('runtests-opts=', 'R', 'Command line options to pass to runtests.py')
- ]
- def initialize_options(self):
- self.runtests_opts = None
- def finalize_options(self):
- '''
- Abstract method that is required to be overwritten
- '''
- def run(self):
- from subprocess import Popen
- self.run_command('build')
- build_cmd = self.get_finalized_command('build_ext')
- runner = os.path.abspath('tests/runtests.py')
- test_cmd = sys.executable + ' {0}'.format(runner)
- if self.runtests_opts:
- test_cmd += ' {0}'.format(self.runtests_opts)
- print('running test')
- test_process = Popen(
- test_cmd, shell=True,
- stdout=sys.stdout, stderr=sys.stderr,
- cwd=build_cmd.build_lib
- )
- test_process.communicate()
- sys.exit(test_process.returncode)
- class Clean(clean):
- def run(self):
- clean.run(self)
- # Let's clean compiled *.py[c,o]
- for subdir in ('salt', 'tests', 'doc'):
- root = os.path.join(os.path.dirname(__file__), subdir)
- for dirname, _, _ in os.walk(root):
- for to_remove_filename in glob.glob('{0}/*.py[oc]'.format(dirname)):
- os.remove(to_remove_filename)
- INSTALL_VERSION_TEMPLATE = '''\
- # This file was auto-generated by salt's setup
- from salt.version import SaltStackVersion
- __saltstack_version__ = SaltStackVersion{full_version_info!r}
- '''
- INSTALL_SYSPATHS_TEMPLATE = '''\
- # This file was auto-generated by salt's setup on \
- {date:%A, %d %B %Y @ %H:%m:%S UTC}.
- ROOT_DIR = {root_dir!r}
- SHARE_DIR = {share_dir!r}
- CONFIG_DIR = {config_dir!r}
- CACHE_DIR = {cache_dir!r}
- SOCK_DIR = {sock_dir!r}
- SRV_ROOT_DIR= {srv_root_dir!r}
- BASE_FILE_ROOTS_DIR = {base_file_roots_dir!r}
- BASE_PILLAR_ROOTS_DIR = {base_pillar_roots_dir!r}
- BASE_MASTER_ROOTS_DIR = {base_master_roots_dir!r}
- BASE_THORIUM_ROOTS_DIR = {base_thorium_roots_dir!r}
- LOGS_DIR = {logs_dir!r}
- PIDFILE_DIR = {pidfile_dir!r}
- SPM_PARENT_PATH = {spm_parent_path!r}
- SPM_FORMULA_PATH = {spm_formula_path!r}
- SPM_PILLAR_PATH = {spm_pillar_path!r}
- SPM_REACTOR_PATH = {spm_reactor_path!r}
- HOME_DIR = {home_dir!r}
- '''
- class Build(build):
- def run(self):
- # Run build.run function
- build.run(self)
- if getattr(self.distribution, 'with_salt_version', False):
- # Write the hardcoded salt version module salt/_version.py
- self.distribution.salt_version_hardcoded_path = os.path.join(
- self.build_lib, 'salt', '_version.py'
- )
- self.run_command('write_salt_version')
- if getattr(self.distribution, 'running_salt_install', False):
- # If our install attribute is present and set to True, we'll go
- # ahead and write our install time python modules.
- # Write the hardcoded salt version module salt/_version.py
- self.run_command('write_salt_version')
- # Write the system paths file
- self.distribution.salt_syspaths_hardcoded_path = os.path.join(
- self.build_lib, 'salt', '_syspaths.py'
- )
- self.run_command('generate_salt_syspaths')
- class Install(install):
- def initialize_options(self):
- install.initialize_options(self)
- def finalize_options(self):
- install.finalize_options(self)
- def run(self):
- # Let's set the running_salt_install attribute so we can add
- # _version.py in the build command
- self.distribution.running_salt_install = True
- self.distribution.salt_version_hardcoded_path = os.path.join(
- self.build_lib, 'salt', '_version.py'
- )
- if IS_WINDOWS_PLATFORM:
- # Download the required DLLs
- self.distribution.salt_download_windows_dlls = True
- self.run_command('download-windows-dlls')
- self.distribution.salt_download_windows_dlls = None
- # Run install.run
- install.run(self)
- class InstallLib(install_lib):
- def run(self):
- executables = [
- 'salt/templates/git/ssh-id-wrapper',
- 'salt/templates/lxc/salt_tarball',
- ]
- install_lib.run(self)
- # input and outputs match 1-1
- inp = self.get_inputs()
- out = self.get_outputs()
- chmod = []
- for idx, inputfile in enumerate(inp):
- for executable in executables:
- if inputfile.endswith(executable):
- chmod.append(idx)
- for idx in chmod:
- filename = out[idx]
- os.chmod(filename, 0o755)
- # <---- Custom Distutils/Setuptools Commands -------------------------------------------------------------------------
- # ----- Custom Distribution Class ----------------------------------------------------------------------------------->
- # We use this to override the package name in case --ssh-packaging is passed to
- # setup.py or the special .salt-ssh-package is found
- class SaltDistribution(distutils.dist.Distribution):
- '''
- Just so it's completely clear
- Under windows, the following scripts should be installed:
- * salt-call
- * salt-cp
- * salt-minion
- * salt-unity
- * salt-proxy
- When packaged for salt-ssh, the following scripts should be installed:
- * salt-call
- * salt-run
- * salt-ssh
- * salt-cloud
- Under windows, the following scripts should be omitted from the salt-ssh package:
- * salt-cloud
- * salt-run
- Under *nix, all scripts should be installed
- '''
- global_options = distutils.dist.Distribution.global_options + [
- ('ssh-packaging', None, 'Run in SSH packaging mode'),
- ('salt-transport=', None, 'The transport to prepare salt for. Currently, the only choice '
- 'is \'zeromq\'. This may be expanded in the future. Defaults to '
- '\'zeromq\'', 'zeromq')] + [
- ('with-salt-version=', None, 'Set a fixed version for Salt instead calculating it'),
- # Salt's Paths Configuration Settings
- ('salt-root-dir=', None,
- 'Salt\'s pre-configured root directory'),
- ('salt-share-dir=', None,
- 'Salt\'s pre-configured share directory'),
- ('salt-config-dir=', None,
- 'Salt\'s pre-configured configuration directory'),
- ('salt-cache-dir=', None,
- 'Salt\'s pre-configured cache directory'),
- ('salt-sock-dir=', None,
- 'Salt\'s pre-configured socket directory'),
- ('salt-srv-root-dir=', None,
- 'Salt\'s pre-configured service directory'),
- ('salt-base-file-roots-dir=', None,
- 'Salt\'s pre-configured file roots directory'),
- ('salt-base-pillar-roots-dir=', None,
- 'Salt\'s pre-configured pillar roots directory'),
- ('salt-base-master-roots-dir=', None,
- 'Salt\'s pre-configured master roots directory'),
- ('salt-logs-dir=', None,
- 'Salt\'s pre-configured logs directory'),
- ('salt-pidfile-dir=', None,
- 'Salt\'s pre-configured pidfiles directory'),
- ('salt-spm-formula-dir=', None,
- 'Salt\'s pre-configured SPM formulas directory'),
- ('salt-spm-pillar-dir=', None,
- 'Salt\'s pre-configured SPM pillar directory'),
- ('salt-spm-reactor-dir=', None,
- 'Salt\'s pre-configured SPM reactor directory'),
- ('salt-home-dir=', None,
- 'Salt\'s pre-configured user home directory'),
- ]
- def __init__(self, attrs=None):
- distutils.dist.Distribution.__init__(self, attrs)
- self.ssh_packaging = PACKAGED_FOR_SALT_SSH
- self.salt_transport = None
- # Salt Paths Configuration Settings
- self.salt_root_dir = None
- self.salt_share_dir = None
- self.salt_config_dir = None
- self.salt_cache_dir = None
- self.salt_sock_dir = None
- self.salt_srv_root_dir = None
- self.salt_base_file_roots_dir = None
- self.salt_base_thorium_roots_dir = None
- self.salt_base_pillar_roots_dir = None
- self.salt_base_master_roots_dir = None
- self.salt_logs_dir = None
- self.salt_pidfile_dir = None
- self.salt_spm_parent_dir = None
- self.salt_spm_formula_dir = None
- self.salt_spm_pillar_dir = None
- self.salt_spm_reactor_dir = None
- self.salt_home_dir = None
- # Salt version
- self.with_salt_version = None
- self.name = 'salt-ssh' if PACKAGED_FOR_SALT_SSH else 'salt'
- self.salt_version = __version__ # pylint: disable=undefined-variable
- self.description = 'Portable, distributed, remote execution and configuration management system'
- kwargs = {}
- if IS_PY3:
- kwargs['encoding'] = 'utf-8'
- with open(SALT_LONG_DESCRIPTION_FILE, **kwargs) as f:
- self.long_description = f.read()
- self.long_description_content_type = 'text/x-rst'
- self.author = 'Thomas S Hatch'
- self.author_email = 'thatch45@gmail.com'
- self.url = 'http://saltstack.org'
- self.cmdclass.update({'test': TestCommand,
- 'clean': Clean,
- 'build': Build,
- 'sdist': Sdist,
- 'install': Install,
- 'write_salt_version': WriteSaltVersion,
- 'generate_salt_syspaths': GenerateSaltSyspaths,
- 'write_salt_ssh_packaging_file': WriteSaltSshPackagingFile})
- if not IS_WINDOWS_PLATFORM:
- self.cmdclass.update({'sdist': CloudSdist,
- 'install_lib': InstallLib})
- if IS_WINDOWS_PLATFORM:
- self.cmdclass.update({'download-windows-dlls': DownloadWindowsDlls})
- if WITH_SETUPTOOLS:
- self.cmdclass.update({'develop': Develop})
- self.license = 'Apache Software License 2.0'
- self.packages = self.discover_packages()
- self.zip_safe = False
- if HAS_ESKY:
- self.setup_esky()
- self.update_metadata()
- def update_metadata(self):
- for attrname in dir(self):
- if attrname.startswith('__'):
- continue
- attrvalue = getattr(self, attrname, None)
- if attrvalue == 0:
- continue
- if attrname == 'salt_version':
- attrname = 'version'
- if hasattr(self.metadata, 'set_{0}'.format(attrname)):
- getattr(self.metadata, 'set_{0}'.format(attrname))(attrvalue)
- elif hasattr(self.metadata, attrname):
- try:
- setattr(self.metadata, attrname, attrvalue)
- except AttributeError:
- pass
- def discover_packages(self):
- modules = []
- for root, _, files in os.walk(os.path.join(SETUP_DIRNAME, 'salt')):
- if '__init__.py' not in files:
- continue
- modules.append(os.path.relpath(root, SETUP_DIRNAME).replace(os.sep, '.'))
- return modules
- # ----- Static Data -------------------------------------------------------------------------------------------->
- @property
- def _property_classifiers(self):
- return ['Programming Language :: Python',
- 'Programming Language :: Cython',
- 'Programming Language :: Python :: 2.6',
- 'Programming Language :: Python :: 2.7',
- 'Development Status :: 5 - Production/Stable',
- 'Environment :: Console',
- 'Intended Audience :: Developers',
- 'Intended Audience :: Information Technology',
- 'Intended Audience :: System Administrators',
- 'License :: OSI Approved :: Apache Software License',
- 'Operating System :: POSIX :: Linux',
- 'Topic :: System :: Clustering',
- 'Topic :: System :: Distributed Computing']
- @property
- def _property_dependency_links(self):
- return ['https://github.com/saltstack/salt-testing/tarball/develop#egg=SaltTesting']
- @property
- def _property_tests_require(self):
- return ['SaltTesting']
- # <---- Static Data ----------------------------------------------------------------------------------------------
- # ----- Dynamic Data -------------------------------------------------------------------------------------------->
- @property
- def _property_package_data(self):
- package_data = {'salt.templates': ['rh_ip/*.jinja',
- 'debian_ip/*.jinja',
- 'virt/*.jinja',
- 'git/*',
- 'lxc/*',
- ]}
- if not IS_WINDOWS_PLATFORM:
- package_data['salt.cloud'] = ['deploy/*.sh']
- if not self.ssh_packaging and not PACKAGED_FOR_SALT_SSH:
- package_data['salt.daemons.flo'] = ['*.flo']
- return package_data
- @property
- def _property_data_files(self):
- # Data files common to all scenarios
- data_files = [
- ('share/man/man1', ['doc/man/salt-call.1', 'doc/man/salt-run.1']),
- ('share/man/man7', ['doc/man/salt.7'])
- ]
- if self.ssh_packaging or PACKAGED_FOR_SALT_SSH:
- data_files[0][1].append('doc/man/salt-ssh.1')
- if IS_WINDOWS_PLATFORM:
- return data_files
- data_files[0][1].append('doc/man/salt-cloud.1')
- return data_files
- if IS_WINDOWS_PLATFORM:
- data_files[0][1].extend(['doc/man/salt-api.1',
- 'doc/man/salt-cp.1',
- 'doc/man/salt-key.1',
- 'doc/man/salt-master.1',
- 'doc/man/salt-minion.1',
- 'doc/man/salt-proxy.1',
- 'doc/man/salt-unity.1'])
- return data_files
- # *nix, so, we need all man pages
- data_files[0][1].extend(['doc/man/salt-api.1',
- 'doc/man/salt-cloud.1',
- 'doc/man/salt-cp.1',
- 'doc/man/salt-key.1',
- 'doc/man/salt-master.1',
- 'doc/man/salt-minion.1',
- 'doc/man/salt-proxy.1',
- 'doc/man/spm.1',
- 'doc/man/salt.1',
- 'doc/man/salt-ssh.1',
- 'doc/man/salt-syndic.1',
- 'doc/man/salt-unity.1'])
- return data_files
- @property
- def _property_install_requires(self):
- install_requires = _parse_requirements_file(SALT_REQS)
- if self.salt_transport == 'zeromq':
- install_requires += _parse_requirements_file(SALT_ZEROMQ_REQS)
- if IS_WINDOWS_PLATFORM:
- install_requires = _parse_requirements_file(SALT_WINDOWS_REQS)
- return install_requires
- @property
- def _property_scripts(self):
- # Scripts common to all scenarios
- scripts = ['scripts/salt-call', 'scripts/salt-run']
- if self.ssh_packaging or PACKAGED_FOR_SALT_SSH:
- scripts.append('scripts/salt-ssh')
- if IS_WINDOWS_PLATFORM:
- return scripts
- scripts.extend(['scripts/salt-cloud', 'scripts/spm'])
- return scripts
- if IS_WINDOWS_PLATFORM:
- scripts.extend(['scripts/salt',
- 'scripts/salt-api',
- 'scripts/salt-cp',
- 'scripts/salt-key',
- 'scripts/salt-master',
- 'scripts/salt-minion',
- 'scripts/salt-proxy',
- 'scripts/salt-unity'])
- return scripts
- # *nix, so, we need all scripts
- scripts.extend(['scripts/salt',
- 'scripts/salt-api',
- 'scripts/salt-cloud',
- 'scripts/salt-cp',
- 'scripts/salt-key',
- 'scripts/salt-master',
- 'scripts/salt-minion',
- 'scripts/salt-support',
- 'scripts/salt-ssh',
- 'scripts/salt-syndic',
- 'scripts/salt-unity',
- 'scripts/salt-proxy',
- 'scripts/spm'])
- return scripts
- @property
- def _property_entry_points(self):
- # console scripts common to all scenarios
- scripts = ['salt-call = salt.scripts:salt_call',
- 'salt-run = salt.scripts:salt_run']
- if self.ssh_packaging or PACKAGED_FOR_SALT_SSH:
- scripts.append('salt-ssh = salt.scripts:salt_ssh')
- if IS_WINDOWS_PLATFORM:
- return {'console_scripts': scripts}
- scripts.append('salt-cloud = salt.scripts:salt_cloud')
- return {'console_scripts': scripts}
- if IS_WINDOWS_PLATFORM:
- scripts.extend(['salt = salt.scripts:salt_main',
- 'salt-api = salt.scripts:salt_api',
- 'salt-cp = salt.scripts:salt_cp',
- 'salt-key = salt.scripts:salt_key',
- 'salt-master = salt.scripts:salt_master',
- 'salt-minion = salt.scripts:salt_minion',
- 'salt-unity = salt.scripts:salt_unity',
- 'spm = salt.scripts:salt_spm'])
- return {'console_scripts': scripts}
- # *nix, so, we need all scripts
- scripts.extend(['salt = salt.scripts:salt_main',
- 'salt-api = salt.scripts:salt_api',
- 'salt-cloud = salt.scripts:salt_cloud',
- 'salt-cp = salt.scripts:salt_cp',
- 'salt-key = salt.scripts:salt_key',
- 'salt-master = salt.scripts:salt_master',
- 'salt-minion = salt.scripts:salt_minion',
- 'salt-support = salt.scripts:salt_support',
- 'salt-ssh = salt.scripts:salt_ssh',
- 'salt-syndic = salt.scripts:salt_syndic',
- 'salt-unity = salt.scripts:salt_unity',
- 'spm = salt.scripts:salt_spm'])
- return {'console_scripts': scripts}
- # <---- Dynamic Data ---------------------------------------------------------------------------------------------
- # ----- Esky Setup ---------------------------------------------------------------------------------------------->
- def setup_esky(self):
- opt_dict = self.get_option_dict('bdist_esky')
- opt_dict['freezer_module'] = ('setup script', 'bbfreeze')
- opt_dict['freezer_options'] = ('setup script', {'includes': self.get_esky_freezer_includes()})
- @property
- def _property_freezer_options(self):
- return {'includes': self.get_esky_freezer_includes()}
- def get_esky_freezer_includes(self):
- # Sometimes the auto module traversal doesn't find everything, so we
- # explicitly add it. The auto dependency tracking especially does not work for
- # imports occurring in salt.modules, as they are loaded at salt runtime.
- # Specifying includes that don't exist doesn't appear to cause a freezing
- # error.
- freezer_includes = [
- 'zmq.core.*',
- 'zmq.utils.*',
- 'ast',
- 'csv',
- 'difflib',
- 'distutils',
- 'distutils.version',
- 'numbers',
- 'json',
- 'M2Crypto',
- 'Cookie',
- 'asyncore',
- 'fileinput',
- 'sqlite3',
- 'email',
- 'email.mime.*',
- 'requests',
- 'sqlite3',
- ]
- if HAS_ZMQ and hasattr(zmq, 'pyzmq_version_info'):
- if HAS_ZMQ and zmq.pyzmq_version_info() >= (0, 14):
- # We're freezing, and when freezing ZMQ needs to be installed, so this
- # works fine
- if 'zmq.core.*' in freezer_includes:
- # For PyZMQ >= 0.14, freezing does not need 'zmq.core.*'
- freezer_includes.remove('zmq.core.*')
- if IS_WINDOWS_PLATFORM:
- freezer_includes.extend([
- 'imp',
- 'win32api',
- 'win32file',
- 'win32con',
- 'win32com',
- 'win32net',
- 'win32netcon',
- 'win32gui',
- 'win32security',
- 'ntsecuritycon',
- 'pywintypes',
- 'pythoncom',
- '_winreg',
- 'wmi',
- 'site',
- 'psutil',
- 'pytz',
- ])
- elif IS_SMARTOS_PLATFORM:
- # we have them as requirements in pkg/smartos/esky/requirements.txt
- # all these should be safe to force include
- freezer_includes.extend([
- 'cherrypy',
- 'python-dateutil',
- 'pyghmi',
- 'croniter',
- 'mako',
- 'gnupg',
- ])
- elif sys.platform.startswith('linux'):
- freezer_includes.append('spwd')
- try:
- import yum # pylint: disable=unused-variable
- freezer_includes.append('yum')
- except ImportError:
- pass
- elif sys.platform.startswith('sunos'):
- # (The sledgehammer approach)
- # Just try to include everything
- # (This may be a better way to generate freezer_includes generally)
- try:
- from bbfreeze.modulegraph.modulegraph import ModuleGraph
- mgraph = ModuleGraph(sys.path[:])
- for arg in glob.glob('salt/modules/*.py'):
- mgraph.run_script(arg)
- for mod in mgraph.flatten():
- if type(mod).__name__ != 'Script' and mod.filename:
- freezer_includes.append(str(os.path.basename(mod.identifier)))
- except ImportError:
- pass
- return freezer_includes
- # <---- Esky Setup -----------------------------------------------------------------------------------------------
- # ----- Overridden Methods -------------------------------------------------------------------------------------->
- def parse_command_line(self):
- args = distutils.dist.Distribution.parse_command_line(self)
- if not self.ssh_packaging and PACKAGED_FOR_SALT_SSH:
- self.ssh_packaging = 1
- if self.ssh_packaging:
- self.metadata.name = 'salt-ssh'
- self.salt_transport = 'ssh'
- elif self.salt_transport is None:
- self.salt_transport = 'zeromq'
- if self.salt_transport not in ('zeromq', 'both', 'ssh', 'none'):
- raise DistutilsArgError(
- 'The value of --salt-transport needs be \'zeromq\', '
- '\'both\', \'ssh\', or \'none\' not \'{0}\''.format(
- self.salt_transport
- )
- )
- # Setup our property functions after class initialization and
- # after parsing the command line since most are set to None
- # ATTENTION: This should be the last step before returning the args or
- # some of the requirements won't be correctly set
- for funcname in dir(self):
- if not funcname.startswith('_property_'):
- continue
- property_name = funcname.split('_property_', 1)[-1]
- setattr(self, property_name, getattr(self, funcname))
- return args
- # <---- Overridden Methods ---------------------------------------------------------------------------------------
- # <---- Custom Distribution Class ------------------------------------------------------------------------------------
- if __name__ == '__main__':
- setup(distclass=SaltDistribution)
|