# -*- 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 . """ setup file for leap.soledad.common """ import re from setuptools import setup from setuptools import find_packages import versioneer versioneer.versionfile_source = 'src/leap/soledad/common/_version.py' versioneer.versionfile_build = 'leap/soledad/common/_version.py' versioneer.tag_prefix = '' # tags are like 1.2.0 versioneer.parentdir_prefix = 'leap.soledad.common-' 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/soledad/' 'archive/%s.tar.gz') _versions = versioneer.get_versions() VERSION = _versions['version'] VERSION_FULL = _versions['full'] 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 cmdclass = versioneer.get_cmdclass() from setuptools import Command 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 = [] 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 template = r""" # This file was generated by the `freeze_debianver` command in setup.py # Using 'versioneer.py' (0.7+) 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. version_version = '{version}' version_full = '{version_full}' """ templatefun = r""" def get_versions(default={}, verbose=False): return {'version': version_version, 'full': version_full} """ subst_template = template.format( version=VERSION_SHORT, version_full=VERSION_FULL) + templatefun with open(versioneer.versionfile_source, 'w') as f: f.write(subst_template) # # Couch backend design docs file generation. # from os import listdir from os.path import realpath, dirname, isdir, join, isfile, basename import json import binascii 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 `/`, 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') 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,) from setuptools.command.develop import develop as _cmd_develop 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() # versioneer powered old_cmd_sdist = cmdclass["sdist"] class cmd_sdist(old_cmd_sdist): """ Generate 'src/leap/soledad/common/ddocs.py' which contains couch design documents scripts. """ def run(self): old_cmd_sdist.run(self) def make_release_tree(self, base_dir, files): old_cmd_sdist.make_release_tree(self, base_dir, files) build_ddocs_py(basedir=base_dir) # versioneer powered old_cmd_build = cmdclass["build"] class cmd_build(old_cmd_build): def run(self): old_cmd_build.run(self) build_ddocs_py(basedir=self.build_lib, with_src=False) cmdclass["freeze_debianver"] = freeze_debianver cmdclass["build"] = cmd_build cmdclass["sdist"] = cmd_sdist cmdclass["develop"] = cmd_develop # XXX add ref to docs 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=['leap.soledad.common.tests']), package_dir={'': 'src'}, test_suite='leap.soledad.common.tests.load_tests', install_requires=utils.parse_requirements(), tests_require=utils.parse_requirements( reqfiles=['pkg/requirements-testing.pip']), extras_require={ 'couchdb': ['couchdb'], }, )