2016-09-14 08:51:49 +00:00
|
|
|
#!/usr/bin/env python
|
2015-05-05 17:09:44 +00:00
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
# Copyright (C) 2016-2018 g10 Code GmbH
|
2017-04-03 13:44:14 +00:00
|
|
|
# Copyright (C) 2004,2008 Igor Belyi <belyi@users.sourceforge.net>
|
2015-05-05 17:09:44 +00:00
|
|
|
# Copyright (C) 2002 John Goerzen <jgoerzen@complete.org>
|
|
|
|
#
|
|
|
|
# This library is free software; you can redistribute it and/or
|
|
|
|
# modify it under the terms of the GNU Lesser General Public
|
|
|
|
# License as published by the Free Software Foundation; either
|
|
|
|
# version 2.1 of the License, or (at your option) any later version.
|
|
|
|
#
|
|
|
|
# This library 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
|
|
|
|
# Lesser General Public License for more details.
|
|
|
|
#
|
|
|
|
# You should have received a copy of the GNU Lesser General Public
|
|
|
|
# License along with this library; if not, write to the Free Software
|
|
|
|
# Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA
|
|
|
|
|
|
|
|
from distutils.core import setup, Extension
|
2018-08-18 14:21:47 +00:00
|
|
|
from distutils.command.build import build
|
2018-08-18 08:19:16 +00:00
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
import glob
|
2018-08-18 08:19:16 +00:00
|
|
|
import os
|
|
|
|
import os.path
|
2017-04-03 13:44:14 +00:00
|
|
|
import re
|
2017-04-01 23:29:52 +00:00
|
|
|
import shutil
|
2015-05-05 17:09:44 +00:00
|
|
|
import subprocess
|
2018-08-18 08:19:16 +00:00
|
|
|
import sys
|
2015-05-05 17:09:44 +00:00
|
|
|
|
2016-10-28 20:45:49 +00:00
|
|
|
# Out-of-tree build of the gpg bindings.
|
2018-11-30 20:45:50 +00:00
|
|
|
gpg_error_config = ['gpg-error-config']
|
|
|
|
gpgme_config_flags = ['--thread=pthread']
|
|
|
|
gpgme_config = ['gpgme-config'] + gpgme_config_flags
|
|
|
|
gpgme_h = ''
|
2016-09-26 09:35:40 +00:00
|
|
|
include_dirs = [os.getcwd()]
|
2016-07-11 14:38:37 +00:00
|
|
|
library_dirs = []
|
2016-08-02 16:45:10 +00:00
|
|
|
in_tree = False
|
2016-07-11 14:38:37 +00:00
|
|
|
extra_swig_opts = []
|
2016-08-02 16:45:10 +00:00
|
|
|
extra_macros = dict()
|
2016-07-11 14:38:37 +00:00
|
|
|
|
2018-11-30 20:45:50 +00:00
|
|
|
top_builddir = os.environ.get('top_builddir')
|
2017-04-05 16:47:08 +00:00
|
|
|
if top_builddir:
|
2016-07-11 14:38:37 +00:00
|
|
|
# In-tree build.
|
|
|
|
in_tree = True
|
2018-11-30 20:45:50 +00:00
|
|
|
gpgme_config = [os.path.join(top_builddir, 'src/gpgme-config')
|
2018-08-18 08:19:16 +00:00
|
|
|
] + gpgme_config_flags
|
2018-11-30 20:45:50 +00:00
|
|
|
gpgme_h = os.path.join(top_builddir, 'src/gpgme.h')
|
2018-08-18 08:19:16 +00:00
|
|
|
library_dirs = [os.path.join(top_builddir,
|
2018-11-30 20:45:50 +00:00
|
|
|
'src/.libs')] # XXX uses libtool internals
|
2016-08-02 16:45:10 +00:00
|
|
|
extra_macros.update(
|
2016-09-26 09:35:40 +00:00
|
|
|
HAVE_CONFIG_H=1,
|
2016-08-02 16:45:10 +00:00
|
|
|
HAVE_DATA_H=1,
|
|
|
|
IN_TREE_BUILD=1,
|
|
|
|
)
|
2016-07-11 14:38:37 +00:00
|
|
|
|
2018-11-30 20:45:50 +00:00
|
|
|
if hasattr(subprocess, 'DEVNULL'):
|
2016-09-12 14:18:31 +00:00
|
|
|
devnull = subprocess.DEVNULL
|
|
|
|
else:
|
2018-11-30 20:45:50 +00:00
|
|
|
devnull = open(os.devnull, 'w')
|
2016-09-12 14:18:31 +00:00
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
try:
|
2018-08-18 08:19:16 +00:00
|
|
|
subprocess.check_call(gpgme_config + ['--version'], stdout=devnull)
|
2016-07-11 14:38:37 +00:00
|
|
|
except:
|
2018-11-30 20:45:50 +00:00
|
|
|
sys.exit('Could not find gpgme-config. ' +
|
|
|
|
'Please install the libgpgme development package.')
|
2016-07-11 14:38:37 +00:00
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
def getconfig(what, config=gpgme_config):
|
2018-08-18 08:19:16 +00:00
|
|
|
confdata = subprocess.Popen(
|
2018-11-30 20:45:50 +00:00
|
|
|
config + ['--%s' % what], stdout=subprocess.PIPE).communicate()[0]
|
2015-05-05 17:09:44 +00:00
|
|
|
return [x for x in confdata.decode('utf-8').split() if x != '']
|
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
|
2018-11-30 20:45:50 +00:00
|
|
|
version = version_raw = getconfig('version')[0]
|
2016-07-11 14:38:37 +00:00
|
|
|
if '-' in version:
|
|
|
|
version = version.split('-')[0]
|
|
|
|
major, minor, patch = map(int, version.split('.'))
|
|
|
|
|
2016-11-09 12:56:00 +00:00
|
|
|
if not (major > 1 or (major == 1 and minor >= 7)):
|
|
|
|
sys.exit('Need at least GPGME version 1.7, found {}.'.format(version_raw))
|
2016-07-11 14:38:37 +00:00
|
|
|
|
|
|
|
if not gpgme_h:
|
2018-11-30 20:45:50 +00:00
|
|
|
gpgme_h = os.path.join(getconfig('prefix')[0], 'include', 'gpgme.h')
|
2016-07-11 14:38:37 +00:00
|
|
|
|
2015-05-05 17:09:44 +00:00
|
|
|
define_macros = []
|
|
|
|
libs = getconfig('libs')
|
2016-05-11 09:42:00 +00:00
|
|
|
|
2016-08-02 16:45:10 +00:00
|
|
|
# Define extra_macros for both the SWIG and C code
|
|
|
|
for k, v in extra_macros.items():
|
2018-11-30 20:45:50 +00:00
|
|
|
extra_swig_opts.append('-D{0}={1}'.format(k, v))
|
2016-08-02 16:45:10 +00:00
|
|
|
define_macros.append((k, str(v)))
|
|
|
|
|
2015-05-05 17:09:44 +00:00
|
|
|
for item in getconfig('cflags'):
|
2018-11-30 20:45:50 +00:00
|
|
|
if item.startswith('-I'):
|
2015-05-05 17:09:44 +00:00
|
|
|
include_dirs.append(item[2:])
|
2018-11-30 20:45:50 +00:00
|
|
|
elif item.startswith('-D'):
|
|
|
|
defitem = item[2:].split('=', 1)
|
2018-08-18 08:19:16 +00:00
|
|
|
if len(defitem) == 2:
|
2015-05-05 17:09:44 +00:00
|
|
|
define_macros.append((defitem[0], defitem[1]))
|
|
|
|
else:
|
|
|
|
define_macros.append((defitem[0], None))
|
|
|
|
|
|
|
|
# Adjust include and library locations in case of win32
|
2018-11-30 20:45:50 +00:00
|
|
|
uname_s = os.popen('uname -s').read()
|
|
|
|
if uname_s.startswith('MINGW32'):
|
2018-08-18 08:19:16 +00:00
|
|
|
mnts = [
|
2018-11-30 20:45:50 +00:00
|
|
|
x.split()[0:3:2] for x in os.popen('mount').read().split('\n') if x
|
2018-08-18 08:19:16 +00:00
|
|
|
]
|
|
|
|
tmplist = sorted([(len(x[1]), x[1], x[0]) for x in mnts])
|
|
|
|
tmplist.reverse()
|
|
|
|
extra_dirs = []
|
|
|
|
for item in include_dirs:
|
|
|
|
for ln, mnt, tgt in tmplist:
|
|
|
|
if item.startswith(mnt):
|
|
|
|
item = os.path.normpath(item[ln:])
|
|
|
|
while item[0] == os.path.sep:
|
|
|
|
item = item[1:]
|
|
|
|
extra_dirs.append(os.path.join(tgt, item))
|
|
|
|
break
|
|
|
|
include_dirs += extra_dirs
|
2018-11-30 20:45:50 +00:00
|
|
|
for item in [x[2:] for x in libs if x.startswith('-L')]:
|
2018-08-18 08:19:16 +00:00
|
|
|
for ln, mnt, tgt in tmplist:
|
|
|
|
if item.startswith(mnt):
|
|
|
|
item = os.path.normpath(item[ln:])
|
|
|
|
while item[0] == os.path.sep:
|
|
|
|
item = item[1:]
|
|
|
|
library_dirs.append(os.path.join(tgt, item))
|
|
|
|
break
|
|
|
|
|
2015-05-05 17:09:44 +00:00
|
|
|
|
2017-04-03 13:44:14 +00:00
|
|
|
def in_srcdir(name):
|
2018-11-30 20:45:50 +00:00
|
|
|
return os.path.join(os.environ.get('srcdir', ''), name)
|
2018-08-18 08:19:16 +00:00
|
|
|
|
|
|
|
|
2017-04-03 13:44:14 +00:00
|
|
|
def up_to_date(source, target):
|
2018-08-18 08:19:16 +00:00
|
|
|
return (os.path.exists(target) and
|
|
|
|
os.path.getmtime(source) <= os.path.getmtime(target))
|
|
|
|
|
2017-04-03 13:44:14 +00:00
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
# We build an Extension using SWIG, which generates a Python module.
|
|
|
|
# By default, the 'build_py' step is run before 'build_ext', and
|
|
|
|
# therefore the generated Python module is not copied into the build
|
|
|
|
# directory.
|
2018-08-18 08:19:16 +00:00
|
|
|
# Bugs: https://bugs.python.org/issue1016626
|
|
|
|
# https://bugs.python.org/issue2624
|
2016-07-11 14:38:37 +00:00
|
|
|
# Workaround:
|
2018-08-18 08:19:16 +00:00
|
|
|
# https://stackoverflow.com/questions/12491328/python-distutils-not-include-the-swig-generated-module
|
2018-08-18 14:21:47 +00:00
|
|
|
#
|
|
|
|
# To install to multiple Python installations or to alternate ones run the
|
|
|
|
# following three commands (yes, run the build one twice):
|
|
|
|
#
|
|
|
|
# /path/to/pythonX.Y setup.py build
|
|
|
|
# /path/to/pythonX.Y setup.py build
|
|
|
|
# /path/to/pythonX.Y setup.py install
|
|
|
|
#
|
|
|
|
# It is highly likely that this will need to be run as root or with sudo (or
|
|
|
|
# sudo -H). It may or may not work with venv. and outside a virtualenv
|
2018-08-18 08:19:16 +00:00
|
|
|
|
|
|
|
class BuildExtFirstHack(build):
|
2017-04-08 13:34:32 +00:00
|
|
|
def _read_header(self, header, cflags):
|
2018-11-30 20:45:50 +00:00
|
|
|
tmp_include = self._in_build_base('include1.h')
|
2017-04-08 13:34:32 +00:00
|
|
|
with open(tmp_include, 'w') as f:
|
2018-11-30 20:45:50 +00:00
|
|
|
f.write('#include <%s>' % header)
|
2018-08-18 08:19:16 +00:00
|
|
|
return subprocess.check_output(
|
|
|
|
os.environ.get('CPP', 'cc -E').split() + cflags +
|
|
|
|
[tmp_include]).decode('utf-8')
|
2017-04-08 13:34:32 +00:00
|
|
|
|
|
|
|
def _write_if_unchanged(self, target, content):
|
|
|
|
if os.path.exists(target):
|
|
|
|
with open(target) as f:
|
|
|
|
if f.read() == content:
|
|
|
|
return
|
|
|
|
|
2018-11-30 20:45:50 +00:00
|
|
|
with open(target, 'w') as sink:
|
2017-04-08 13:34:32 +00:00
|
|
|
sink.write(content)
|
|
|
|
|
2017-04-03 13:44:14 +00:00
|
|
|
def _generate_gpgme_h(self, source_name, sink_name):
|
2018-11-30 20:45:50 +00:00
|
|
|
print('Using gpgme.h from {}'.format(source_name))
|
2018-02-20 16:34:14 +00:00
|
|
|
shutil.copy2(source_name, sink_name)
|
2017-04-03 13:44:14 +00:00
|
|
|
|
2017-04-08 13:34:32 +00:00
|
|
|
def _generate_errors_i(self):
|
|
|
|
|
|
|
|
try:
|
2018-08-18 08:19:16 +00:00
|
|
|
subprocess.check_call(
|
|
|
|
gpg_error_config + ['--version'], stdout=devnull)
|
2017-04-08 13:34:32 +00:00
|
|
|
except:
|
2018-11-30 20:45:50 +00:00
|
|
|
sys.exit('Could not find gpg-error-config. ' +
|
|
|
|
'Please install the libgpg-error development package.')
|
2017-04-08 13:34:32 +00:00
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
gpg_error_content = self._read_header(
|
2018-11-30 20:45:50 +00:00
|
|
|
'gpg-error.h', getconfig('cflags', config=gpg_error_config))
|
2017-04-03 13:44:14 +00:00
|
|
|
|
|
|
|
filter_re = re.compile(r'GPG_ERR_[^ ]* =')
|
|
|
|
rewrite_re = re.compile(r' *(.*) = .*')
|
|
|
|
|
2017-04-08 13:34:32 +00:00
|
|
|
errors_i_content = ''
|
|
|
|
for line in gpg_error_content.splitlines():
|
|
|
|
if not filter_re.search(line):
|
|
|
|
continue
|
2018-08-18 08:19:16 +00:00
|
|
|
errors_i_content += rewrite_re.sub(
|
|
|
|
r'%constant long \1 = \1;' + '\n', line.strip())
|
2017-04-08 13:34:32 +00:00
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
self._write_if_unchanged(
|
2018-11-30 20:45:50 +00:00
|
|
|
self._in_build_base('errors.i'), errors_i_content)
|
2017-04-03 13:44:14 +00:00
|
|
|
|
2017-04-07 14:31:47 +00:00
|
|
|
def _in_build_base(self, name):
|
|
|
|
return os.path.join(self.build_base, name)
|
|
|
|
|
2017-04-01 23:29:52 +00:00
|
|
|
def _generate(self):
|
|
|
|
# Cleanup gpgme.h from deprecated functions and typedefs.
|
|
|
|
if not os.path.exists(self.build_base):
|
|
|
|
os.makedirs(self.build_base)
|
|
|
|
|
2018-11-30 20:45:50 +00:00
|
|
|
self._generate_gpgme_h(gpgme_h, self._in_build_base('gpgme.h'))
|
2017-04-08 13:34:32 +00:00
|
|
|
self._generate_errors_i()
|
2017-04-03 13:44:14 +00:00
|
|
|
|
2018-10-18 04:10:19 +00:00
|
|
|
# Copy due to https://bugs.python.org/issue2624
|
2017-04-01 23:29:52 +00:00
|
|
|
# Avoid creating in srcdir
|
2017-04-07 14:31:47 +00:00
|
|
|
for source, target in ((in_srcdir(n), self._in_build_base(n))
|
2018-08-18 08:19:16 +00:00
|
|
|
for n in ('gpgme.i', 'helpers.c', 'private.h',
|
|
|
|
'helpers.h')):
|
2017-04-03 13:44:14 +00:00
|
|
|
if not up_to_date(source, target):
|
|
|
|
shutil.copy2(source, target)
|
2017-04-01 23:29:52 +00:00
|
|
|
|
|
|
|
# Append generated files via build_base
|
2018-11-30 20:45:50 +00:00
|
|
|
if not os.path.exists(os.path.join(self.build_lib, 'gpg')):
|
|
|
|
os.makedirs(os.path.join(self.build_lib, 'gpg'))
|
|
|
|
shutil.copy2('version.py', os.path.join(self.build_lib, 'gpg'))
|
2017-04-05 16:47:08 +00:00
|
|
|
|
|
|
|
def run(self):
|
|
|
|
self._generate()
|
2017-04-01 23:29:52 +00:00
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
swig_sources.extend((self._in_build_base('gpgme.i'),
|
|
|
|
self._in_build_base('helpers.c')))
|
|
|
|
swig_opts.extend([
|
|
|
|
'-I' + self.build_base, '-outdir',
|
|
|
|
os.path.join(self.build_lib, 'gpg')
|
|
|
|
])
|
2017-05-18 09:42:13 +00:00
|
|
|
include_dirs.insert(0, self.build_base)
|
2017-04-01 23:29:52 +00:00
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
self.run_command('build_ext')
|
|
|
|
build.run(self)
|
|
|
|
|
2018-08-18 08:19:16 +00:00
|
|
|
|
2016-12-20 17:00:36 +00:00
|
|
|
py3 = [] if sys.version_info.major < 3 else ['-py3']
|
2017-04-07 14:31:47 +00:00
|
|
|
swig_sources = []
|
2017-04-01 23:29:52 +00:00
|
|
|
swig_opts = ['-threads'] + py3 + extra_swig_opts
|
2018-08-18 08:19:16 +00:00
|
|
|
swige = Extension(
|
2018-11-30 20:45:50 +00:00
|
|
|
'gpg._gpgme',
|
2018-08-18 08:19:16 +00:00
|
|
|
sources=swig_sources,
|
|
|
|
swig_opts=swig_opts,
|
|
|
|
include_dirs=include_dirs,
|
|
|
|
define_macros=define_macros,
|
|
|
|
library_dirs=library_dirs,
|
|
|
|
extra_link_args=libs)
|
|
|
|
|
|
|
|
setup(
|
2018-11-30 20:45:50 +00:00
|
|
|
name='gpg',
|
2018-08-18 08:19:16 +00:00
|
|
|
cmdclass={'build': BuildExtFirstHack},
|
2018-11-30 20:45:50 +00:00
|
|
|
version='@VERSION@',
|
|
|
|
# Note: description appears as Summary in egg-info file.
|
|
|
|
description='Python bindings to the GPGME API of the GnuPG cryptography library.',
|
|
|
|
# Note: long-description appears as Description in egg-info file.
|
|
|
|
long_description='''Dynamically generated bindings to the C API of the GNU Privacy Guard.
|
|
|
|
|
|
|
|
The GPG Made Easy (GPGME) library provides a high-level API in C to all the
|
|
|
|
component software and libraries in the GnuPG Project, including GPG itself
|
|
|
|
(the GnuPG OpenPGP implementation), libgcrypt, libgpg-error, libassuan and
|
|
|
|
more.
|
|
|
|
|
|
|
|
The official CPython bindings to GPGME are generated during the compiling
|
|
|
|
process of GPGME itself and built for the specific C header and include files
|
|
|
|
produced when GPGME is compiled using SWIG. This provides access to over two
|
|
|
|
thousand functions, methods and values via both the lower level dynamically
|
|
|
|
generated bindings and a more intuitively pythonic higher level layer.
|
|
|
|
|
|
|
|
While the lower level, dynamically generated bindings provide access to
|
|
|
|
everything which GPGME itself provides; the higher level layer is easier to use
|
|
|
|
by Python developers, provides access to the vast majority of functionality
|
|
|
|
developers would want from GnuPG and is extensively documented.
|
|
|
|
|
|
|
|
GPGME and these bindings is available here:
|
|
|
|
|
|
|
|
https://gnupg.org/software/gpgme/index.html
|
|
|
|
''',
|
2018-08-18 08:19:16 +00:00
|
|
|
author='The GnuPG hackers',
|
|
|
|
author_email='gnupg-devel@gnupg.org',
|
|
|
|
url='https://www.gnupg.org',
|
|
|
|
ext_modules=[swige],
|
|
|
|
packages=[
|
|
|
|
'gpg', 'gpg.constants', 'gpg.constants.data', 'gpg.constants.keylist',
|
|
|
|
'gpg.constants.sig', 'gpg.constants.tofu'
|
|
|
|
],
|
2018-11-30 20:45:50 +00:00
|
|
|
license='LGPL2.1+ (the library), GPL2+ (tests and examples)',
|
2018-08-18 08:19:16 +00:00
|
|
|
classifiers=[
|
|
|
|
'Development Status :: 4 - Beta',
|
|
|
|
'Intended Audience :: Developers',
|
|
|
|
'License :: OSI Approved :: GNU Lesser General Public License v2 or later (LGPLv2+)',
|
|
|
|
'Programming Language :: Python :: 2',
|
|
|
|
'Programming Language :: Python :: 2.7',
|
|
|
|
'Programming Language :: Python :: 3',
|
|
|
|
'Programming Language :: Python :: 3.4',
|
|
|
|
'Programming Language :: Python :: 3.5',
|
|
|
|
'Programming Language :: Python :: 3.6',
|
|
|
|
'Programming Language :: Python :: 3.7',
|
|
|
|
'Operating System :: POSIX',
|
|
|
|
'Operating System :: Microsoft :: Windows',
|
|
|
|
'Topic :: Communications :: Email',
|
|
|
|
'Topic :: Security :: Cryptography',
|
|
|
|
],
|
2015-05-05 17:09:44 +00:00
|
|
|
)
|