8ddbf276bc0e7138d0565b37cdcc1d0bb6b3cb3c
[gpgme.git] / lang / python / setup.py.in
1 #!/usr/bin/env python
2
3 # Copyright (C) 2016 g10 Code GmbH
4 # Copyright (C) 2004 Igor Belyi <belyi@users.sourceforge.net>
5 # Copyright (C) 2002 John Goerzen <jgoerzen@complete.org>
6 #
7 #    This library is free software; you can redistribute it and/or
8 #    modify it under the terms of the GNU Lesser General Public
9 #    License as published by the Free Software Foundation; either
10 #    version 2.1 of the License, or (at your option) any later version.
11 #
12 #    This library is distributed in the hope that it will be useful,
13 #    but WITHOUT ANY WARRANTY; without even the implied warranty of
14 #    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU
15 #    Lesser General Public License for more details.
16 #
17 #    You should have received a copy of the GNU Lesser General Public
18 #    License along with this library; if not, write to the Free Software
19 #    Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307  USA
20
21 from distutils.core import setup, Extension
22 import os, os.path, sys
23 import glob
24 import subprocess
25
26 # Out-of-tree build of the gpg bindings.
27 gpg_error_config = ["gpg-error-config"]
28 gpgme_config_flags = ["--thread=pthread"]
29 gpgme_config = ["gpgme-config"] + gpgme_config_flags
30 gpgme_h = ""
31 include_dirs = [os.getcwd()]
32 library_dirs = []
33 in_tree = False
34 extra_swig_opts = []
35 extra_macros = dict()
36
37 abs_top_builddir = os.environ.get("abs_top_builddir")
38 if abs_top_builddir:
39     # In-tree build.
40     in_tree = True
41     gpgme_config = [os.path.join(abs_top_builddir, "src/gpgme-config")] + gpgme_config_flags
42     gpgme_h = os.path.join(abs_top_builddir, "src/gpgme.h")
43     library_dirs = [os.path.join(abs_top_builddir, "src/.libs")] # XXX uses libtool internals
44     extra_macros.update(
45         HAVE_CONFIG_H=1,
46         HAVE_DATA_H=1,
47         IN_TREE_BUILD=1,
48     )
49
50 if hasattr(subprocess, "DEVNULL"):
51     devnull = subprocess.DEVNULL
52 else:
53     devnull = open(os.devnull, "w")
54
55 try:
56     subprocess.check_call(gpg_error_config + ['--version'],
57                           stdout=devnull)
58 except:
59     sys.exit("Could not find gpg-error-config.  " +
60              "Please install the libgpg-error development package.")
61
62 try:
63     subprocess.check_call(gpgme_config + ['--version'],
64                           stdout=devnull)
65 except:
66     sys.exit("Could not find gpgme-config.  " +
67              "Please install the libgpgme development package.")
68
69 def getconfig(what, config=gpgme_config):
70     confdata = subprocess.Popen(config + ["--%s" % what],
71                                 stdout=subprocess.PIPE).communicate()[0]
72     return [x for x in confdata.decode('utf-8').split() if x != '']
73
74 version = version_raw = getconfig("version")[0]
75 if '-' in version:
76     version = version.split('-')[0]
77 major, minor, patch = map(int, version.split('.'))
78
79 if not (major > 1 or (major == 1 and minor >= 7)):
80     sys.exit('Need at least GPGME version 1.7, found {}.'.format(version_raw))
81
82 if not gpgme_h:
83     gpgme_h = os.path.join(getconfig("prefix")[0], "include", "gpgme.h")
84
85 gpg_error_prefix = getconfig("prefix", config=gpg_error_config)[0]
86 gpg_error_h = os.path.join(gpg_error_prefix, "include", "gpg-error.h")
87 if not os.path.exists(gpg_error_h):
88     gpg_error_h = \
89         glob.glob(os.path.join(gpg_error_prefix, "include",
90                                "*", "gpg-error.h"))[0]
91
92 print("Building python gpg module using {} and {}.".format(gpgme_h, gpg_error_h))
93
94 # Cleanup gpgme.h from deprecated functions and typedefs.
95 subprocess.check_call([sys.executable, "gpgme-h-clean.py", gpgme_h],
96                       stdout=open("gpgme.h", "w"))
97 subprocess.check_call([sys.executable, "gpgme-h-clean.py", gpg_error_h],
98                       stdout=open("errors.i", "w"))
99
100 define_macros = []
101 libs = getconfig('libs')
102
103 # Define extra_macros for both the SWIG and C code
104 for k, v in extra_macros.items():
105     extra_swig_opts.append("-D{0}={1}".format(k, v))
106     define_macros.append((k, str(v)))
107
108 for item in getconfig('cflags'):
109     if item.startswith("-I"):
110         include_dirs.append(item[2:])
111     elif item.startswith("-D"):
112         defitem = item[2:].split("=", 1)
113         if len(defitem)==2:
114             define_macros.append((defitem[0], defitem[1]))
115         else:
116             define_macros.append((defitem[0], None))
117
118 # Adjust include and library locations in case of win32
119 uname_s = os.popen("uname -s").read()
120 if uname_s.startswith("MINGW32"):
121    mnts = [x.split()[0:3:2] for x in os.popen("mount").read().split("\n") if x]
122    tmplist = sorted([(len(x[1]), x[1], x[0]) for x in mnts])
123    tmplist.reverse()
124    extra_dirs = []
125    for item in include_dirs:
126        for ln, mnt, tgt in tmplist:
127            if item.startswith(mnt):
128                item = os.path.normpath(item[ln:])
129                while item[0] == os.path.sep:
130                    item = item[1:]
131                extra_dirs.append(os.path.join(tgt, item))
132                break
133    include_dirs += extra_dirs
134    for item in [x[2:] for x in libs if x.startswith("-L")]:
135        for ln, mnt, tgt in tmplist:
136            if item.startswith(mnt):
137                item = os.path.normpath(item[ln:])
138                while item[0] == os.path.sep:
139                    item = item[1:]
140                library_dirs.append(os.path.join(tgt, item))
141                break
142
143 # We build an Extension using SWIG, which generates a Python module.
144 # By default, the 'build_py' step is run before 'build_ext', and
145 # therefore the generated Python module is not copied into the build
146 # directory.
147 # Bug: http://bugs.python.org/issue1016626
148 # Workaround:
149 # http://stackoverflow.com/questions/12491328/python-distutils-not-include-the-swig-generated-module
150 from distutils.command.build import build
151 class BuildExtFirstHack(build):
152     def run(self):
153         self.run_command('build_ext')
154         build.run(self)
155
156 py3 = [] if sys.version_info.major < 3 else ['-py3']
157 swige = Extension("gpg._gpgme", ["gpgme.i", "helpers.c"],
158                   swig_opts = ['-threads',
159                                '-outdir', 'gpg'] + py3 + extra_swig_opts,
160                   include_dirs = include_dirs,
161                   define_macros = define_macros,
162                   library_dirs = library_dirs,
163                   extra_link_args = libs)
164
165 setup(name="gpg",
166       cmdclass={'build': BuildExtFirstHack},
167       version="@VERSION@",
168       description='Python bindings for GPGME GnuPG cryptography library',
169       # XXX add a long description
170       #long_description=long_description,
171       author='The GnuPG hackers',
172       author_email='gnupg-devel@gnupg.org',
173       url='https://www.gnupg.org',
174       ext_modules=[swige],
175       packages = ['gpg', 'gpg.constants', 'gpg.constants.data',
176                   'gpg.constants.keylist', 'gpg.constants.sig',
177                   'gpg.constants.tofu'],
178       license="LGPL2.1+ (the library), GPL2+ (tests and examples)",
179       classifiers=[
180           'Development Status :: 4 - Beta',
181           'Intended Audience :: Developers',
182           'License :: OSI Approved :: GNU Lesser General Public License v2 or later (LGPLv2+)',
183           'Programming Language :: Python :: 2',
184           'Programming Language :: Python :: 2.7',
185           'Programming Language :: Python :: 3',
186           'Programming Language :: Python :: 3.4',
187           'Programming Language :: Python :: 3.5',
188           'Programming Language :: Python :: 3.6',
189           'Operating System :: POSIX',
190           'Operating System :: Microsoft :: Windows',
191           'Topic :: Communications :: Email',
192           'Topic :: Security :: Cryptography',
193       ],
194 )