2016-09-14 08:51:49 +00:00
|
|
|
#!/usr/bin/env python
|
2015-05-05 17:09:44 +00:00
|
|
|
|
2016-06-09 10:38:50 +00:00
|
|
|
# Copyright (C) 2016 g10 Code GmbH
|
2015-05-05 17:09:44 +00:00
|
|
|
# Copyright (C) 2004 Igor Belyi <belyi@users.sourceforge.net>
|
|
|
|
# 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
|
|
|
|
import os, os.path, sys
|
2016-07-11 14:38:37 +00:00
|
|
|
import glob
|
2015-05-05 17:09:44 +00:00
|
|
|
import subprocess
|
|
|
|
|
2016-10-28 20:45:49 +00:00
|
|
|
# Out-of-tree build of the gpg bindings.
|
python: Release the GIL during calls into GPGME.
* lang/python/helpers.c (pyme_raise_callback_exception): Re-acquire
the Global Interpreter Lock.
(pyPassphraseCb, pyme_set_passphrase_cb, pyProgressCb,
pyme_set_progress_cb, pyStatusCb, pyme_set_status_cb,
_pyme_interact_cb, pyDataReadCb, pyDataWriteCb, pyDataSeekCb,
pyDataReleaseCb, pyme_data_new_from_cbs, _pyme_assuan_data_cb,
_pyme_assuan_inquire_cb, _pyme_assuan_status_cb): Likewise.
* lang/python/setup.py.in: Make 'gpgme-config' emit the correct
cflags, and SWIG generate code to release the GIL before calling us.
Co-authored-by: Kai Michaelis <kai@gnupg.org>
Signed-off-by: Justus Winter <justus@g10code.com>
2016-09-16 10:32:24 +00:00
|
|
|
gpg_error_config = ["gpg-error-config"]
|
|
|
|
gpgme_config_flags = ["--thread=pthread"]
|
|
|
|
gpgme_config = ["gpgme-config"] + gpgme_config_flags
|
2016-07-11 14:38:37 +00:00
|
|
|
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
|
|
|
|
|
|
|
if os.path.exists("../../src/gpgme-config"):
|
|
|
|
# In-tree build.
|
|
|
|
in_tree = True
|
python: Release the GIL during calls into GPGME.
* lang/python/helpers.c (pyme_raise_callback_exception): Re-acquire
the Global Interpreter Lock.
(pyPassphraseCb, pyme_set_passphrase_cb, pyProgressCb,
pyme_set_progress_cb, pyStatusCb, pyme_set_status_cb,
_pyme_interact_cb, pyDataReadCb, pyDataWriteCb, pyDataSeekCb,
pyDataReleaseCb, pyme_data_new_from_cbs, _pyme_assuan_data_cb,
_pyme_assuan_inquire_cb, _pyme_assuan_status_cb): Likewise.
* lang/python/setup.py.in: Make 'gpgme-config' emit the correct
cflags, and SWIG generate code to release the GIL before calling us.
Co-authored-by: Kai Michaelis <kai@gnupg.org>
Signed-off-by: Justus Winter <justus@g10code.com>
2016-09-16 10:32:24 +00:00
|
|
|
gpgme_config = ["../../src/gpgme-config"] + gpgme_config_flags
|
2016-07-11 14:38:37 +00:00
|
|
|
gpgme_h = "../../src/gpgme.h"
|
|
|
|
library_dirs = ["../../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
|
|
|
|
2016-09-12 14:18:31 +00:00
|
|
|
if hasattr(subprocess, "DEVNULL"):
|
|
|
|
devnull = subprocess.DEVNULL
|
|
|
|
else:
|
|
|
|
devnull = open(os.devnull, "w")
|
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
try:
|
python: Release the GIL during calls into GPGME.
* lang/python/helpers.c (pyme_raise_callback_exception): Re-acquire
the Global Interpreter Lock.
(pyPassphraseCb, pyme_set_passphrase_cb, pyProgressCb,
pyme_set_progress_cb, pyStatusCb, pyme_set_status_cb,
_pyme_interact_cb, pyDataReadCb, pyDataWriteCb, pyDataSeekCb,
pyDataReleaseCb, pyme_data_new_from_cbs, _pyme_assuan_data_cb,
_pyme_assuan_inquire_cb, _pyme_assuan_status_cb): Likewise.
* lang/python/setup.py.in: Make 'gpgme-config' emit the correct
cflags, and SWIG generate code to release the GIL before calling us.
Co-authored-by: Kai Michaelis <kai@gnupg.org>
Signed-off-by: Justus Winter <justus@g10code.com>
2016-09-16 10:32:24 +00:00
|
|
|
subprocess.check_call(gpg_error_config + ['--version'],
|
2016-09-12 14:18:31 +00:00
|
|
|
stdout=devnull)
|
2016-07-11 14:38:37 +00:00
|
|
|
except:
|
|
|
|
sys.exit("Could not find gpg-error-config. " +
|
|
|
|
"Please install the libgpg-error development package.")
|
|
|
|
|
|
|
|
try:
|
python: Release the GIL during calls into GPGME.
* lang/python/helpers.c (pyme_raise_callback_exception): Re-acquire
the Global Interpreter Lock.
(pyPassphraseCb, pyme_set_passphrase_cb, pyProgressCb,
pyme_set_progress_cb, pyStatusCb, pyme_set_status_cb,
_pyme_interact_cb, pyDataReadCb, pyDataWriteCb, pyDataSeekCb,
pyDataReleaseCb, pyme_data_new_from_cbs, _pyme_assuan_data_cb,
_pyme_assuan_inquire_cb, _pyme_assuan_status_cb): Likewise.
* lang/python/setup.py.in: Make 'gpgme-config' emit the correct
cflags, and SWIG generate code to release the GIL before calling us.
Co-authored-by: Kai Michaelis <kai@gnupg.org>
Signed-off-by: Justus Winter <justus@g10code.com>
2016-09-16 10:32:24 +00:00
|
|
|
subprocess.check_call(gpgme_config + ['--version'],
|
2016-09-12 14:18:31 +00:00
|
|
|
stdout=devnull)
|
2016-07-11 14:38:37 +00:00
|
|
|
except:
|
|
|
|
sys.exit("Could not find gpgme-config. " +
|
|
|
|
"Please install the libgpgme development package.")
|
|
|
|
|
|
|
|
def getconfig(what, config=gpgme_config):
|
python: Release the GIL during calls into GPGME.
* lang/python/helpers.c (pyme_raise_callback_exception): Re-acquire
the Global Interpreter Lock.
(pyPassphraseCb, pyme_set_passphrase_cb, pyProgressCb,
pyme_set_progress_cb, pyStatusCb, pyme_set_status_cb,
_pyme_interact_cb, pyDataReadCb, pyDataWriteCb, pyDataSeekCb,
pyDataReleaseCb, pyme_data_new_from_cbs, _pyme_assuan_data_cb,
_pyme_assuan_inquire_cb, _pyme_assuan_status_cb): Likewise.
* lang/python/setup.py.in: Make 'gpgme-config' emit the correct
cflags, and SWIG generate code to release the GIL before calling us.
Co-authored-by: Kai Michaelis <kai@gnupg.org>
Signed-off-by: Justus Winter <justus@g10code.com>
2016-09-16 10:32:24 +00:00
|
|
|
confdata = subprocess.Popen(config + ["--%s" % what],
|
2016-05-11 09:42:00 +00:00
|
|
|
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 != '']
|
|
|
|
|
2016-07-11 14:38:37 +00:00
|
|
|
version = version_raw = getconfig("version")[0]
|
|
|
|
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:
|
|
|
|
gpgme_h = os.path.join(getconfig("prefix")[0], "include", "gpgme.h")
|
|
|
|
|
|
|
|
gpg_error_prefix = getconfig("prefix", config=gpg_error_config)[0]
|
|
|
|
gpg_error_h = os.path.join(gpg_error_prefix, "include", "gpg-error.h")
|
|
|
|
if not os.path.exists(gpg_error_h):
|
|
|
|
gpg_error_h = \
|
|
|
|
glob.glob(os.path.join(gpg_error_prefix, "include",
|
|
|
|
"*", "gpg-error.h"))[0]
|
|
|
|
|
2016-10-28 20:45:49 +00:00
|
|
|
print("Building python gpg module using {} and {}.".format(gpgme_h, gpg_error_h))
|
2016-07-11 14:38:37 +00:00
|
|
|
|
|
|
|
# Cleanup gpgme.h from deprecated functions and typedefs.
|
2016-09-12 14:19:07 +00:00
|
|
|
subprocess.check_call([sys.executable, "gpgme-h-clean.py", gpgme_h],
|
2016-07-11 14:38:37 +00:00
|
|
|
stdout=open("gpgme.h", "w"))
|
2016-09-12 14:19:07 +00:00
|
|
|
subprocess.check_call([sys.executable, "gpgme-h-clean.py", gpg_error_h],
|
2016-07-11 14:38:37 +00:00
|
|
|
stdout=open("errors.i", "w"))
|
|
|
|
|
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():
|
|
|
|
extra_swig_opts.append("-D{0}={1}".format(k, v))
|
|
|
|
define_macros.append((k, str(v)))
|
|
|
|
|
2015-05-05 17:09:44 +00:00
|
|
|
for item in getconfig('cflags'):
|
|
|
|
if item.startswith("-I"):
|
|
|
|
include_dirs.append(item[2:])
|
|
|
|
elif item.startswith("-D"):
|
|
|
|
defitem = item[2:].split("=", 1)
|
|
|
|
if len(defitem)==2:
|
|
|
|
define_macros.append((defitem[0], defitem[1]))
|
|
|
|
else:
|
|
|
|
define_macros.append((defitem[0], None))
|
|
|
|
|
|
|
|
# Adjust include and library locations in case of win32
|
|
|
|
uname_s = os.popen("uname -s").read()
|
|
|
|
if uname_s.startswith("MINGW32"):
|
2016-05-10 11:30:30 +00:00
|
|
|
mnts = [x.split()[0:3:2] for x in os.popen("mount").read().split("\n") if x]
|
2015-05-05 17:09:44 +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
|
|
|
|
for item in [x[2:] for x in libs if x.startswith("-L")]:
|
|
|
|
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
|
|
|
|
|
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.
|
|
|
|
# Bug: http://bugs.python.org/issue1016626
|
|
|
|
# Workaround:
|
|
|
|
# http://stackoverflow.com/questions/12491328/python-distutils-not-include-the-swig-generated-module
|
|
|
|
from distutils.command.build import build
|
|
|
|
class BuildExtFirstHack(build):
|
|
|
|
def run(self):
|
|
|
|
self.run_command('build_ext')
|
|
|
|
build.run(self)
|
|
|
|
|
2016-12-20 17:00:36 +00:00
|
|
|
py3 = [] if sys.version_info.major < 3 else ['-py3']
|
2016-10-28 20:45:49 +00:00
|
|
|
swige = Extension("gpg._gpgme", ["gpgme.i", "helpers.c"],
|
2016-12-20 17:00:36 +00:00
|
|
|
swig_opts = ['-threads', '-builtin',
|
|
|
|
'-outdir', 'gpg'] + py3 + extra_swig_opts,
|
2015-05-05 17:09:44 +00:00
|
|
|
include_dirs = include_dirs,
|
|
|
|
define_macros = define_macros,
|
|
|
|
library_dirs = library_dirs,
|
|
|
|
extra_link_args = libs)
|
|
|
|
|
2016-10-28 20:45:49 +00:00
|
|
|
setup(name="gpg",
|
2016-07-11 14:38:37 +00:00
|
|
|
cmdclass={'build': BuildExtFirstHack},
|
2016-06-09 10:38:50 +00:00
|
|
|
version="@VERSION@",
|
|
|
|
description='Python bindings for GPGME GnuPG cryptography library',
|
2016-07-12 14:20:35 +00:00
|
|
|
# XXX add a long description
|
|
|
|
#long_description=long_description,
|
2016-06-09 10:38:50 +00:00
|
|
|
author='The GnuPG hackers',
|
|
|
|
author_email='gnupg-devel@gnupg.org',
|
|
|
|
url='https://www.gnupg.org',
|
2015-05-05 17:09:44 +00:00
|
|
|
ext_modules=[swige],
|
2016-10-28 20:45:49 +00:00
|
|
|
packages = ['gpg', 'gpg.constants', 'gpg.constants.data',
|
|
|
|
'gpg.constants.keylist', 'gpg.constants.sig'],
|
2016-07-12 14:20:35 +00:00
|
|
|
license="LGPL2.1+ (the library), GPL2+ (tests and examples)",
|
|
|
|
classifiers=[
|
|
|
|
'Development Status :: 4 - Beta',
|
|
|
|
'Intended Audience :: Developers',
|
|
|
|
'License :: OSI Approved :: GNU Lesser General Public License v2 or later (LGPLv2+)',
|
2016-09-19 10:49:14 +00:00
|
|
|
'Programming Language :: Python :: 2',
|
2016-09-14 12:36:29 +00:00
|
|
|
'Programming Language :: Python :: 2.7',
|
2016-07-12 14:20:35 +00:00
|
|
|
'Programming Language :: Python :: 3',
|
|
|
|
'Programming Language :: Python :: 3.4',
|
|
|
|
'Programming Language :: Python :: 3.5',
|
|
|
|
'Programming Language :: Python :: 3.6',
|
|
|
|
'Operating System :: POSIX',
|
|
|
|
'Operating System :: Microsoft :: Windows',
|
|
|
|
'Topic :: Communications :: Email',
|
|
|
|
'Topic :: Security :: Cryptography',
|
|
|
|
],
|
2015-05-05 17:09:44 +00:00
|
|
|
)
|