# -*- coding: utf-8 -*- # setup.py # Copyright (C) 2013 LEAP # # This program is free software: you can redistribute it and/or modify # it under the terms of the GNU General Public License as published by # the Free Software Foundation, either version 3 of the License, or # (at your option) any later version. # # This program is distributed in the hope that it will be useful, # but WITHOUT ANY WARRANTY; without even the implied warranty of # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the # GNU General Public License for more details. # # You should have received a copy of the GNU General Public License # along with this program. If not, see <http://www.gnu.org/licenses/>. """ setup file for leap.soledad.common """ import binascii import json from os import listdir from os.path import realpath, dirname, isdir, join, isfile, basename import re from distutils.command.build import build as _build from setuptools import setup from setuptools import find_packages from setuptools import Command from setuptools.command.develop import develop as _cmd_develop import versioneer from pkg import utils trove_classifiers = ( "Development Status :: 3 - Alpha", "Intended Audience :: Developers", "License :: OSI Approved :: " "GNU General Public License v3 or later (GPLv3+)", "Environment :: Console", "Operating System :: OS Independent", "Operating System :: POSIX", "Programming Language :: Python :: 2.6", "Programming Language :: Python :: 2.7", "Topic :: Database :: Front-Ends", "Topic :: Software Development :: Libraries :: Python Modules" ) DOWNLOAD_BASE = ('https://github.com/leapcode/bitmask_client/' 'archive/%s.tar.gz') _versions = versioneer.get_versions() VERSION = _versions['version'] VERSION_REVISION = _versions['full-revisionid'] DOWNLOAD_URL = "" # get the short version for the download url _version_short = re.findall('\d+\.\d+\.\d+', VERSION) if len(_version_short) > 0: VERSION_SHORT = _version_short[0] DOWNLOAD_URL = DOWNLOAD_BASE % VERSION_SHORT class freeze_debianver(Command): """ Freezes the version in a debian branch. To be used after merging the development branch onto the debian one. """ user_options = [] template = r""" # This file was generated by the `freeze_debianver` command in setup.py # Using 'versioneer.py' (0.16) from # revision-control system data, or from the parent directory name of an # unpacked source archive. Distribution tarballs contain a pre-generated copy # of this file. import json import sys version_json = ''' { "dirty": false, "error": null, "full-revisionid": "FULL_REVISIONID", "version": "VERSION_STRING" } ''' # END VERSION_JSON def get_versions(): return json.loads(version_json) """ def initialize_options(self): pass def finalize_options(self): pass def run(self): proceed = str(raw_input( "This will overwrite the file _version.py. Continue? [y/N] ")) if proceed != "y": print("He. You scared. Aborting.") return subst_template = self.template.replace( 'VERSION_STRING', VERSION_SHORT).replace( 'FULL_REVISIONID', VERSION_REVISION) versioneer_cfg = versioneer.get_config_from_root('.') with open(versioneer_cfg.versionfile_source, 'w') as f: f.write(subst_template) cmdclass = versioneer.get_cmdclass() # # Couch backend design docs file generation. # old_cmd_sdist = cmdclass["sdist"] def build_ddocs_py(basedir=None, with_src=True): """ Build `ddocs.py` file. For ease of development, couch backend design documents are stored as `.js` files in subdirectories of `src/leap/soledad/common/ddocs`. This function scans that directory for javascript files, builds the design documents structure, and encode those structures in the `ddocs.py` file. This function is used when installing in develop mode, building or generating source distributions (see the next classes and the `cmdclass` setuptools parameter. This funciton uses the following conventions to generate design documents: - Design documents are represented by directories in the form `<prefix>/<ddoc>`, there prefix is the `src/leap/soledad/common/ddocs` directory. - Design document directories might contain `views`, `lists` and `updates` subdirectories. - Views subdirectories must contain a `map.js` file and may contain a `reduce.js` file. - List and updates subdirectories may contain any number of javascript files (i.e. ending in `.js`) whose names will be mapped to the corresponding list or update function name. """ cur_pwd = dirname(realpath(__file__)) common_path = ('src', 'leap', 'soledad', 'common') dest_common_path = common_path if not with_src: dest_common_path = common_path[1:] prefix = join(cur_pwd, *common_path) dest_prefix = prefix if basedir is not None: # we're bulding a sdist dest_prefix = join(basedir, *dest_common_path) ddocs_prefix = join(prefix, 'ddocs') if not isdir(ddocs_prefix): print "No ddocs/ folder, bailing out..." return ddocs = {} # design docs are represented by subdirectories of `ddocs_prefix` for ddoc in [f for f in listdir(ddocs_prefix) if isdir(join(ddocs_prefix, f))]: ddocs[ddoc] = {'_id': '_design/%s' % ddoc} for t in ['views', 'lists', 'updates']: tdir = join(ddocs_prefix, ddoc, t) if isdir(tdir): ddocs[ddoc][t] = {} if t == 'views': # handle views (with map/reduce functions) for view in [f for f in listdir(tdir) if isdir(join(tdir, f))]: # look for map.js and reduce.js mapfile = join(tdir, view, 'map.js') reducefile = join(tdir, view, 'reduce.js') mapfun = None reducefun = None try: with open(mapfile) as f: mapfun = f.read() except IOError: pass try: with open(reducefile) as f: reducefun = f.read() except IOError: pass ddocs[ddoc]['views'][view] = {} if mapfun is not None: ddocs[ddoc]['views'][view]['map'] = mapfun if reducefun is not None: ddocs[ddoc]['views'][view]['reduce'] = reducefun else: # handle lists, updates, etc for fun in [f for f in listdir(tdir) if isfile(join(tdir, f))]: funfile = join(tdir, fun) funname = basename(funfile).replace('.js', '') try: with open(funfile) as f: ddocs[ddoc][t][funname] = f.read() except IOError: pass # write file containing design docs strings ddoc_filename = "ddocs.py" with open(join(dest_prefix, ddoc_filename), 'w') as f: for ddoc in ddocs: f.write( "%s = '%s'\n" % (ddoc, binascii.b2a_base64(json.dumps(ddocs[ddoc]))[:-1])) print "Wrote design docs in %s" % (dest_prefix + '/' + ddoc_filename,) class cmd_develop(_cmd_develop): def run(self): # versioneer: versions = versioneer.get_versions(verbose=True) self._versioneer_generated_versions = versions # unless we update this, the command will keep using the old version self.distribution.metadata.version = versions["version"] _cmd_develop.run(self) build_ddocs_py() class cmd_build(_build): def run(self): _build.run(self) build_ddocs_py(basedir=self.build_lib, with_src=False) cmdclass["freeze_debianver"] = freeze_debianver cmdclass["build"] = cmd_build cmdclass["develop"] = cmd_develop # XXX add ref to docs requirements = utils.parse_requirements() if utils.is_develop_mode(): print print ("[WARNING] Skipping leap-specific dependencies " "because development mode is detected.") print ("[WARNING] You can install " "the latest published versions with " "'pip install -r pkg/requirements-leap.pip'") print ("[WARNING] Or you can instead do 'python setup.py develop' " "from the parent folder of each one of them.") print else: requirements += utils.parse_requirements( reqfiles=["pkg/requirements-leap.pip"]) setup( name='leap.soledad.common', version=VERSION, cmdclass=cmdclass, url='https://leap.se/', download_url=DOWNLOAD_URL, license='GPLv3+', description='Synchronization of locally encrypted data among devices ' '(common files).', author='The LEAP Encryption Access Project', author_email='info@leap.se', maintainer='Kali Kaneko', maintainer_email='kali@leap.se', long_description=( "Soledad is the part of LEAP that allows application data to be " "securely shared among devices. It provides, to other parts of the " "LEAP project, an API for data storage and sync." ), classifiers=trove_classifiers, namespace_packages=["leap", "leap.soledad"], packages=find_packages('src', exclude=['*.tests', '*.tests.*']), package_dir={'': 'src'}, package_data={'': ["*.sql"]}, test_suite='leap.soledad.common.tests', install_requires=requirements, tests_require=utils.parse_requirements( reqfiles=['pkg/requirements-testing.pip']), )