373e17c2a2ce65a9e17613c0b8973947ca3569ad
[gpgme.git] / lang / python / setup.py.in
1 #!/usr/bin/env python3
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 pyme3 bindings.
27 gpg_error_config = "gpg-error-config"
28 gpgme_config = "gpgme-config"
29 gpgme_h = ""
30 library_dirs = []
31 extra_swig_opts = []
32
33 if os.path.exists("../../src/gpgme-config"):
34     # In-tree build.
35     in_tree = True
36     gpgme_config = "../../src/gpgme-config"
37     gpgme_h = "../../src/gpgme.h"
38     library_dirs = ["../../src/.libs"] # XXX uses libtool internals
39     extra_swig_opts = ["-DHAVE_DATA_H=1"]
40
41 try:
42     subprocess.check_call([gpg_error_config, '--version'],
43                           stdout=subprocess.DEVNULL)
44 except:
45     sys.exit("Could not find gpg-error-config.  " +
46              "Please install the libgpg-error development package.")
47
48 try:
49     subprocess.check_call([gpgme_config, '--version'],
50                           stdout=subprocess.DEVNULL)
51 except:
52     sys.exit("Could not find gpgme-config.  " +
53              "Please install the libgpgme development package.")
54
55 def getconfig(what, config=gpgme_config):
56     confdata = subprocess.Popen([config, "--%s" % what],
57                                 stdout=subprocess.PIPE).communicate()[0]
58     return [x for x in confdata.decode('utf-8').split() if x != '']
59
60 version = version_raw = getconfig("version")[0]
61 if '-' in version:
62     version = version.split('-')[0]
63 major, minor, patch = map(int, version.split('.'))
64
65 if not (major > 1 or (major == 1 and minor >= 6)):
66     sys.exit('Need at least GPGME version 1.6, found {}.'.format(version_raw))
67
68 if not gpgme_h:
69     gpgme_h = os.path.join(getconfig("prefix")[0], "include", "gpgme.h")
70
71 gpg_error_prefix = getconfig("prefix", config=gpg_error_config)[0]
72 gpg_error_h = os.path.join(gpg_error_prefix, "include", "gpg-error.h")
73 if not os.path.exists(gpg_error_h):
74     gpg_error_h = \
75         glob.glob(os.path.join(gpg_error_prefix, "include",
76                                "*", "gpg-error.h"))[0]
77
78 print("Building pyme3 using {} and {}.".format(gpgme_h, gpg_error_h))
79
80 # Cleanup gpgme.h from deprecated functions and typedefs.
81 subprocess.check_call(["python3", "gpgme-h-clean.py", gpgme_h],
82                       stdout=open("gpgme.h", "w"))
83 subprocess.check_call(["python3", "gpgme-h-clean.py", gpg_error_h],
84                       stdout=open("errors.i", "w"))
85
86 include_dirs = [os.getcwd()]
87 define_macros = []
88 libs = getconfig('libs')
89
90 for item in getconfig('cflags'):
91     if item.startswith("-I"):
92         include_dirs.append(item[2:])
93     elif item.startswith("-D"):
94         defitem = item[2:].split("=", 1)
95         if len(defitem)==2:
96             define_macros.append((defitem[0], defitem[1]))
97         else:
98             define_macros.append((defitem[0], None))
99
100 # Adjust include and library locations in case of win32
101 uname_s = os.popen("uname -s").read()
102 if uname_s.startswith("MINGW32"):
103    mnts = [x.split()[0:3:2] for x in os.popen("mount").read().split("\n") if x]
104    tmplist = sorted([(len(x[1]), x[1], x[0]) for x in mnts])
105    tmplist.reverse()
106    extra_dirs = []
107    for item in include_dirs:
108        for ln, mnt, tgt in tmplist:
109            if item.startswith(mnt):
110                item = os.path.normpath(item[ln:])
111                while item[0] == os.path.sep:
112                    item = item[1:]
113                extra_dirs.append(os.path.join(tgt, item))
114                break
115    include_dirs += extra_dirs
116    for item in [x[2:] for x in libs if x.startswith("-L")]:
117        for ln, mnt, tgt in tmplist:
118            if item.startswith(mnt):
119                item = os.path.normpath(item[ln:])
120                while item[0] == os.path.sep:
121                    item = item[1:]
122                library_dirs.append(os.path.join(tgt, item))
123                break
124
125 # We build an Extension using SWIG, which generates a Python module.
126 # By default, the 'build_py' step is run before 'build_ext', and
127 # therefore the generated Python module is not copied into the build
128 # directory.
129 # Bug: http://bugs.python.org/issue1016626
130 # Workaround:
131 # http://stackoverflow.com/questions/12491328/python-distutils-not-include-the-swig-generated-module
132 from distutils.command.build import build
133 class BuildExtFirstHack(build):
134     def run(self):
135         self.run_command('build_ext')
136         build.run(self)
137
138 swige = Extension("pyme._pygpgme", ["gpgme.i", "helpers.c"],
139                   swig_opts = ['-py3', '-builtin',
140                                '-outdir', 'pyme'] + extra_swig_opts,
141                   include_dirs = include_dirs,
142                   define_macros = define_macros,
143                   library_dirs = library_dirs,
144                   extra_link_args = libs)
145
146 setup(name="pyme3",
147       cmdclass={'build': BuildExtFirstHack},
148       version="@VERSION@",
149       description='Python bindings for GPGME GnuPG cryptography library',
150       # XXX add a long description
151       #long_description=long_description,
152       author='The GnuPG hackers',
153       author_email='gnupg-devel@gnupg.org',
154       url='https://www.gnupg.org',
155       ext_modules=[swige],
156       packages = ['pyme', 'pyme.constants', 'pyme.constants.data',
157                   'pyme.constants.keylist', 'pyme.constants.sig'],
158       license="LGPL2.1+ (the library), GPL2+ (tests and examples)",
159       classifiers=[
160           'Development Status :: 4 - Beta',
161           'Intended Audience :: Developers',
162           'License :: OSI Approved :: GNU Lesser General Public License v2 or later (LGPLv2+)',
163           'Programming Language :: Python :: 3',
164           'Programming Language :: Python :: 3 :: Only',
165           'Programming Language :: Python :: 3.4',
166           'Programming Language :: Python :: 3.5',
167           'Programming Language :: Python :: 3.6',
168           'Operating System :: POSIX',
169           'Operating System :: Microsoft :: Windows',
170           'Topic :: Communications :: Email',
171           'Topic :: Security :: Cryptography',
172       ],
173 )