2 # -*- coding: utf-8 -*-
4 # Copyright 2010 Per Øyvind Karlsen <proyvind@moondrake.org>
5 # Copyright 2015 Neal Gompa <ngompa13@gmail.com>
7 # This program is free software. It may be redistributed and/or modified under
8 # the terms of the LGPL version 2.1 (or later).
10 # RPM python dependency generator, using .egg-info/.egg-link/.dist-info data
13 from __future__ import print_function
14 from getopt import getopt
15 from os.path import basename, dirname, isdir, sep
16 from sys import argv, stdin, version
17 from distutils.sysconfig import get_python_lib
18 from warnings import warn
22 argv[1:], 'hPRrCEMmLl:',
23 ['help', 'provides', 'requires', 'recommends', 'conflicts', 'extras', 'majorver-provides', 'majorver-only', 'legacy-provides' , 'legacy'])
30 Provides_PyMajorVer_Variant = False
31 PyMajorVer_Deps = False
32 legacy_Provides = False
36 if o in ('-h', '--help'):
37 print('-h, --help\tPrint help')
38 print('-P, --provides\tPrint Provides')
39 print('-R, --requires\tPrint Requires')
40 print('-r, --recommends\tPrint Recommends')
41 print('-C, --conflicts\tPrint Conflicts')
42 print('-E, --extras\tPrint Extras ')
43 print('-M, --majorver-provides\tPrint extra Provides with Python major version only')
44 print('-m, --majorver-only\tPrint Provides/Requires with Python major version only')
45 print('-L, --legacy-provides\tPrint extra legacy pythonegg Provides')
46 print('-l, --legacy\tPrint legacy pythonegg Provides/Requires instead')
48 elif o in ('-P', '--provides'):
50 elif o in ('-R', '--requires'):
52 elif o in ('-r', '--recommends'):
54 elif o in ('-C', '--conflicts'):
56 elif o in ('-E', '--extras'):
58 elif o in ('-M', '--majorver-provides'):
59 Provides_PyMajorVer_Variant = True
60 elif o in ('-m', '--majorver-only'):
61 PyMajorVer_Deps = True
62 elif o in ('-L', '--legacy-provides'):
63 legacy_Provides = True
64 elif o in ('-l', '--legacy'):
75 files = stdin.readlines()
81 # add dependency based on path, versioned if within versioned python directory
82 if py_abi and (lower.endswith('.py') or lower.endswith('.pyc') or lower.endswith('.pyo')):
83 if name not in py_deps:
85 purelib = get_python_lib(standard_lib=1, plat_specific=0).split(version[:3])[0]
86 platlib = get_python_lib(standard_lib=1, plat_specific=1).split(version[:3])[0]
87 for lib in (purelib, platlib):
89 spec = ('==', f.split(lib)[1].split(sep)[0])
90 if spec not in py_deps[name]:
91 py_deps[name].append(spec)
93 # XXX: hack to workaround RPM internal dependency generator not passing directories
94 lower_dir = dirname(lower)
95 if lower_dir.endswith('.egg') or \
96 lower_dir.endswith('.egg-info') or \
97 lower_dir.endswith('.dist-info'):
100 # Determine provide, requires, conflicts & recommends based on egg/dist metadata
101 if lower.endswith('.egg') or \
102 lower.endswith('.egg-info') or \
103 lower.endswith('.dist-info'):
104 # This import is very slow, so only do it if needed
105 from pkg_resources import Distribution, FileMetadata, PathMetadata
106 dist_name = basename(f)
108 path_item = dirname(f)
109 metadata = PathMetadata(path_item, f)
112 metadata = FileMetadata(f)
113 dist = Distribution.from_location(path_item, dist_name, metadata)
114 # Check if py_version is defined in the metadata file/directory name
115 if not dist.py_version:
116 # Try to parse the Python version from the path the metadata
117 # resides at (e.g. /usr/lib/pythonX.Y/site-packages/...)
119 res = re.search(r"/python(?P<pyver>\d+\.\d)/", path_item)
121 dist.py_version = res.group('pyver')
123 warn("Version for {!r} has not been found".format(dist), RuntimeWarning)
125 if Provides_PyMajorVer_Variant or PyMajorVer_Deps or legacy_Provides or legacy:
126 # Get the Python major version
127 pyver_major = dist.py_version.split('.')[0]
129 # If egg/dist metadata says package name is python, we provide python(abi)
130 if dist.key == 'python':
132 if name not in py_deps:
134 py_deps[name].append(('==', dist.py_version))
135 if not legacy or not PyMajorVer_Deps:
136 name = 'python{}dist({})'.format(dist.py_version, dist.key)
137 if name not in py_deps:
139 if Provides_PyMajorVer_Variant or PyMajorVer_Deps:
140 pymajor_name = 'python{}dist({})'.format(pyver_major, dist.key)
141 if pymajor_name not in py_deps:
142 py_deps[pymajor_name] = []
143 if legacy or legacy_Provides:
144 legacy_name = 'pythonegg({})({})'.format(pyver_major, dist.key)
145 if legacy_name not in py_deps:
146 py_deps[legacy_name] = []
148 spec = ('==', dist.version)
149 if spec not in py_deps[name]:
151 py_deps[name].append(spec)
152 if Provides_PyMajorVer_Variant:
153 py_deps[pymajor_name].append(spec)
154 if legacy or legacy_Provides:
155 py_deps[legacy_name].append(spec)
156 if Requires or (Recommends and dist.extras):
158 # If egg/dist metadata says package name is python, we don't add dependency on python(abi)
159 if dist.key == 'python':
163 elif py_abi and dist.py_version:
164 if name not in py_deps:
166 spec = ('==', dist.py_version)
167 if spec not in py_deps[name]:
168 py_deps[name].append(spec)
169 deps = dist.requires()
171 depsextras = dist.requires(extras=dist.extras)
173 for dep in reversed(depsextras):
175 depsextras.remove(dep)
177 # add requires/recommends based on egg/dist metadata
180 name = 'pythonegg({})({})'.format(pyver_major, dep.key)
183 name = 'python{}dist({})'.format(pyver_major, dep.key)
185 name = 'python{}dist({})'.format(dist.py_version, dep.key)
186 for spec in dep.specs:
188 if name not in py_deps:
190 if spec not in py_deps[name]:
191 py_deps[name].append(spec)
194 # Unused, for automatic sub-package generation based on 'extras' from egg/dist metadata
195 # TODO: implement in rpm later, or...?
197 deps = dist.requires()
201 print('%%package\textras-{}'.format(extra))
202 print('Summary:\t{} extra for {} python package'.format(extra, dist.key))
203 print('Group:\t\tDevelopment/Python')
204 depsextras = dist.requires(extras=[extra])
205 for dep in reversed(depsextras):
207 depsextras.remove(dep)
210 for spec in dep.specs:
212 print('Conflicts:\t{} {} {}'.format(dep.key, '==', spec[1]))
214 print('Requires:\t{} {} {}'.format(dep.key, spec[0], spec[1]))
215 print('%%description\t{}'.format(extra))
216 print('{} extra for {} python package'.format(extra, dist.key))
217 print('%%files\t\textras-{}\n'.format(extra))
219 # Should we really add conflicts for extras?
220 # Creating a meta package per extra with recommends on, which has
221 # the requires/conflicts in stead might be a better solution...
222 for dep in dist.requires(extras=dist.extras):
224 for spec in dep.specs:
226 if name not in py_deps:
228 spec = ('==', spec[1])
229 if spec not in py_deps[name]:
230 py_deps[name].append(spec)
231 names = list(py_deps.keys())
235 # Print out versioned provides, requires, recommends, conflicts
236 for spec in py_deps[name]:
237 print('{} {} {}'.format(name, spec[0], spec[1]))
239 # Print out unversioned provides, requires, recommends, conflicts