Skip to content
Projects
Groups
Snippets
Help
This project
Loading...
Sign in / Register
Toggle navigation
N
news
Project
Overview
Details
Activity
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
Sartika Aritonang
news
Commits
b716502e
Commit
b716502e
authored
May 29, 2020
by
Sartika Aritonang
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Upload New File
parent
a63a7c8a
Show whitespace changes
Inline
Side-by-side
Showing
1 changed file
with
1339 additions
and
0 deletions
+1339
-0
database.py
stbi/Lib/site-packages/pip/_vendor/distlib/database.py
+1339
-0
No files found.
stbi/Lib/site-packages/pip/_vendor/distlib/database.py
0 → 100644
View file @
b716502e
# -*- coding: utf-8 -*-
#
# Copyright (C) 2012-2017 The Python Software Foundation.
# See LICENSE.txt and CONTRIBUTORS.txt.
#
"""PEP 376 implementation."""
from
__future__
import
unicode_literals
import
base64
import
codecs
import
contextlib
import
hashlib
import
logging
import
os
import
posixpath
import
sys
import
zipimport
from
.
import
DistlibException
,
resources
from
.compat
import
StringIO
from
.version
import
get_scheme
,
UnsupportedVersionError
from
.metadata
import
(
Metadata
,
METADATA_FILENAME
,
WHEEL_METADATA_FILENAME
,
LEGACY_METADATA_FILENAME
)
from
.util
import
(
parse_requirement
,
cached_property
,
parse_name_and_version
,
read_exports
,
write_exports
,
CSVReader
,
CSVWriter
)
__all__
=
[
'Distribution'
,
'BaseInstalledDistribution'
,
'InstalledDistribution'
,
'EggInfoDistribution'
,
'DistributionPath'
]
logger
=
logging
.
getLogger
(
__name__
)
EXPORTS_FILENAME
=
'pydist-exports.json'
COMMANDS_FILENAME
=
'pydist-commands.json'
DIST_FILES
=
(
'INSTALLER'
,
METADATA_FILENAME
,
'RECORD'
,
'REQUESTED'
,
'RESOURCES'
,
EXPORTS_FILENAME
,
'SHARED'
)
DISTINFO_EXT
=
'.dist-info'
class
_Cache
(
object
):
"""
A simple cache mapping names and .dist-info paths to distributions
"""
def
__init__
(
self
):
"""
Initialise an instance. There is normally one for each DistributionPath.
"""
self
.
name
=
{}
self
.
path
=
{}
self
.
generated
=
False
def
clear
(
self
):
"""
Clear the cache, setting it to its initial state.
"""
self
.
name
.
clear
()
self
.
path
.
clear
()
self
.
generated
=
False
def
add
(
self
,
dist
):
"""
Add a distribution to the cache.
:param dist: The distribution to add.
"""
if
dist
.
path
not
in
self
.
path
:
self
.
path
[
dist
.
path
]
=
dist
self
.
name
.
setdefault
(
dist
.
key
,
[])
.
append
(
dist
)
class
DistributionPath
(
object
):
"""
Represents a set of distributions installed on a path (typically sys.path).
"""
def
__init__
(
self
,
path
=
None
,
include_egg
=
False
):
"""
Create an instance from a path, optionally including legacy (distutils/
setuptools/distribute) distributions.
:param path: The path to use, as a list of directories. If not specified,
sys.path is used.
:param include_egg: If True, this instance will look for and return legacy
distributions as well as those based on PEP 376.
"""
if
path
is
None
:
path
=
sys
.
path
self
.
path
=
path
self
.
_include_dist
=
True
self
.
_include_egg
=
include_egg
self
.
_cache
=
_Cache
()
self
.
_cache_egg
=
_Cache
()
self
.
_cache_enabled
=
True
self
.
_scheme
=
get_scheme
(
'default'
)
def
_get_cache_enabled
(
self
):
return
self
.
_cache_enabled
def
_set_cache_enabled
(
self
,
value
):
self
.
_cache_enabled
=
value
cache_enabled
=
property
(
_get_cache_enabled
,
_set_cache_enabled
)
def
clear_cache
(
self
):
"""
Clears the internal cache.
"""
self
.
_cache
.
clear
()
self
.
_cache_egg
.
clear
()
def
_yield_distributions
(
self
):
"""
Yield .dist-info and/or .egg(-info) distributions.
"""
# We need to check if we've seen some resources already, because on
# some Linux systems (e.g. some Debian/Ubuntu variants) there are
# symlinks which alias other files in the environment.
seen
=
set
()
for
path
in
self
.
path
:
finder
=
resources
.
finder_for_path
(
path
)
if
finder
is
None
:
continue
r
=
finder
.
find
(
''
)
if
not
r
or
not
r
.
is_container
:
continue
rset
=
sorted
(
r
.
resources
)
for
entry
in
rset
:
r
=
finder
.
find
(
entry
)
if
not
r
or
r
.
path
in
seen
:
continue
if
self
.
_include_dist
and
entry
.
endswith
(
DISTINFO_EXT
):
possible_filenames
=
[
METADATA_FILENAME
,
WHEEL_METADATA_FILENAME
,
LEGACY_METADATA_FILENAME
]
for
metadata_filename
in
possible_filenames
:
metadata_path
=
posixpath
.
join
(
entry
,
metadata_filename
)
pydist
=
finder
.
find
(
metadata_path
)
if
pydist
:
break
else
:
continue
with
contextlib
.
closing
(
pydist
.
as_stream
())
as
stream
:
metadata
=
Metadata
(
fileobj
=
stream
,
scheme
=
'legacy'
)
logger
.
debug
(
'Found
%
s'
,
r
.
path
)
seen
.
add
(
r
.
path
)
yield
new_dist_class
(
r
.
path
,
metadata
=
metadata
,
env
=
self
)
elif
self
.
_include_egg
and
entry
.
endswith
((
'.egg-info'
,
'.egg'
)):
logger
.
debug
(
'Found
%
s'
,
r
.
path
)
seen
.
add
(
r
.
path
)
yield
old_dist_class
(
r
.
path
,
self
)
def
_generate_cache
(
self
):
"""
Scan the path for distributions and populate the cache with
those that are found.
"""
gen_dist
=
not
self
.
_cache
.
generated
gen_egg
=
self
.
_include_egg
and
not
self
.
_cache_egg
.
generated
if
gen_dist
or
gen_egg
:
for
dist
in
self
.
_yield_distributions
():
if
isinstance
(
dist
,
InstalledDistribution
):
self
.
_cache
.
add
(
dist
)
else
:
self
.
_cache_egg
.
add
(
dist
)
if
gen_dist
:
self
.
_cache
.
generated
=
True
if
gen_egg
:
self
.
_cache_egg
.
generated
=
True
@classmethod
def
distinfo_dirname
(
cls
,
name
,
version
):
"""
The *name* and *version* parameters are converted into their
filename-escaped form, i.e. any ``'-'`` characters are replaced
with ``'_'`` other than the one in ``'dist-info'`` and the one
separating the name from the version number.
:parameter name: is converted to a standard distribution name by replacing
any runs of non- alphanumeric characters with a single
``'-'``.
:type name: string
:parameter version: is converted to a standard version string. Spaces
become dots, and all other non-alphanumeric characters
(except dots) become dashes, with runs of multiple
dashes condensed to a single dash.
:type version: string
:returns: directory name
:rtype: string"""
name
=
name
.
replace
(
'-'
,
'_'
)
return
'-'
.
join
([
name
,
version
])
+
DISTINFO_EXT
def
get_distributions
(
self
):
"""
Provides an iterator that looks for distributions and returns
:class:`InstalledDistribution` or
:class:`EggInfoDistribution` instances for each one of them.
:rtype: iterator of :class:`InstalledDistribution` and
:class:`EggInfoDistribution` instances
"""
if
not
self
.
_cache_enabled
:
for
dist
in
self
.
_yield_distributions
():
yield
dist
else
:
self
.
_generate_cache
()
for
dist
in
self
.
_cache
.
path
.
values
():
yield
dist
if
self
.
_include_egg
:
for
dist
in
self
.
_cache_egg
.
path
.
values
():
yield
dist
def
get_distribution
(
self
,
name
):
"""
Looks for a named distribution on the path.
This function only returns the first result found, as no more than one
value is expected. If nothing is found, ``None`` is returned.
:rtype: :class:`InstalledDistribution`, :class:`EggInfoDistribution`
or ``None``
"""
result
=
None
name
=
name
.
lower
()
if
not
self
.
_cache_enabled
:
for
dist
in
self
.
_yield_distributions
():
if
dist
.
key
==
name
:
result
=
dist
break
else
:
self
.
_generate_cache
()
if
name
in
self
.
_cache
.
name
:
result
=
self
.
_cache
.
name
[
name
][
0
]
elif
self
.
_include_egg
and
name
in
self
.
_cache_egg
.
name
:
result
=
self
.
_cache_egg
.
name
[
name
][
0
]
return
result
def
provides_distribution
(
self
,
name
,
version
=
None
):
"""
Iterates over all distributions to find which distributions provide *name*.
If a *version* is provided, it will be used to filter the results.
This function only returns the first result found, since no more than
one values are expected. If the directory is not found, returns ``None``.
:parameter version: a version specifier that indicates the version
required, conforming to the format in ``PEP-345``
:type name: string
:type version: string
"""
matcher
=
None
if
version
is
not
None
:
try
:
matcher
=
self
.
_scheme
.
matcher
(
'
%
s (
%
s)'
%
(
name
,
version
))
except
ValueError
:
raise
DistlibException
(
'invalid name or version:
%
r,
%
r'
%
(
name
,
version
))
for
dist
in
self
.
get_distributions
():
# We hit a problem on Travis where enum34 was installed and doesn't
# have a provides attribute ...
if
not
hasattr
(
dist
,
'provides'
):
logger
.
debug
(
'No "provides":
%
s'
,
dist
)
else
:
provided
=
dist
.
provides
for
p
in
provided
:
p_name
,
p_ver
=
parse_name_and_version
(
p
)
if
matcher
is
None
:
if
p_name
==
name
:
yield
dist
break
else
:
if
p_name
==
name
and
matcher
.
match
(
p_ver
):
yield
dist
break
def
get_file_path
(
self
,
name
,
relative_path
):
"""
Return the path to a resource file.
"""
dist
=
self
.
get_distribution
(
name
)
if
dist
is
None
:
raise
LookupError
(
'no distribution named
%
r found'
%
name
)
return
dist
.
get_resource_path
(
relative_path
)
def
get_exported_entries
(
self
,
category
,
name
=
None
):
"""
Return all of the exported entries in a particular category.
:param category: The category to search for entries.
:param name: If specified, only entries with that name are returned.
"""
for
dist
in
self
.
get_distributions
():
r
=
dist
.
exports
if
category
in
r
:
d
=
r
[
category
]
if
name
is
not
None
:
if
name
in
d
:
yield
d
[
name
]
else
:
for
v
in
d
.
values
():
yield
v
class
Distribution
(
object
):
"""
A base class for distributions, whether installed or from indexes.
Either way, it must have some metadata, so that's all that's needed
for construction.
"""
build_time_dependency
=
False
"""
Set to True if it's known to be only a build-time dependency (i.e.
not needed after installation).
"""
requested
=
False
"""A boolean that indicates whether the ``REQUESTED`` metadata file is
present (in other words, whether the package was installed by user
request or it was installed as a dependency)."""
def
__init__
(
self
,
metadata
):
"""
Initialise an instance.
:param metadata: The instance of :class:`Metadata` describing this
distribution.
"""
self
.
metadata
=
metadata
self
.
name
=
metadata
.
name
self
.
key
=
self
.
name
.
lower
()
# for case-insensitive comparisons
self
.
version
=
metadata
.
version
self
.
locator
=
None
self
.
digest
=
None
self
.
extras
=
None
# additional features requested
self
.
context
=
None
# environment marker overrides
self
.
download_urls
=
set
()
self
.
digests
=
{}
@property
def
source_url
(
self
):
"""
The source archive download URL for this distribution.
"""
return
self
.
metadata
.
source_url
download_url
=
source_url
# Backward compatibility
@property
def
name_and_version
(
self
):
"""
A utility property which displays the name and version in parentheses.
"""
return
'
%
s (
%
s)'
%
(
self
.
name
,
self
.
version
)
@property
def
provides
(
self
):
"""
A set of distribution names and versions provided by this distribution.
:return: A set of "name (version)" strings.
"""
plist
=
self
.
metadata
.
provides
s
=
'
%
s (
%
s)'
%
(
self
.
name
,
self
.
version
)
if
s
not
in
plist
:
plist
.
append
(
s
)
return
plist
def
_get_requirements
(
self
,
req_attr
):
md
=
self
.
metadata
logger
.
debug
(
'Getting requirements from metadata
%
r'
,
md
.
todict
())
reqts
=
getattr
(
md
,
req_attr
)
return
set
(
md
.
get_requirements
(
reqts
,
extras
=
self
.
extras
,
env
=
self
.
context
))
@property
def
run_requires
(
self
):
return
self
.
_get_requirements
(
'run_requires'
)
@property
def
meta_requires
(
self
):
return
self
.
_get_requirements
(
'meta_requires'
)
@property
def
build_requires
(
self
):
return
self
.
_get_requirements
(
'build_requires'
)
@property
def
test_requires
(
self
):
return
self
.
_get_requirements
(
'test_requires'
)
@property
def
dev_requires
(
self
):
return
self
.
_get_requirements
(
'dev_requires'
)
def
matches_requirement
(
self
,
req
):
"""
Say if this instance matches (fulfills) a requirement.
:param req: The requirement to match.
:rtype req: str
:return: True if it matches, else False.
"""
# Requirement may contain extras - parse to lose those
# from what's passed to the matcher
r
=
parse_requirement
(
req
)
scheme
=
get_scheme
(
self
.
metadata
.
scheme
)
try
:
matcher
=
scheme
.
matcher
(
r
.
requirement
)
except
UnsupportedVersionError
:
# XXX compat-mode if cannot read the version
logger
.
warning
(
'could not read version
%
r - using name only'
,
req
)
name
=
req
.
split
()[
0
]
matcher
=
scheme
.
matcher
(
name
)
name
=
matcher
.
key
# case-insensitive
result
=
False
for
p
in
self
.
provides
:
p_name
,
p_ver
=
parse_name_and_version
(
p
)
if
p_name
!=
name
:
continue
try
:
result
=
matcher
.
match
(
p_ver
)
break
except
UnsupportedVersionError
:
pass
return
result
def
__repr__
(
self
):
"""
Return a textual representation of this instance,
"""
if
self
.
source_url
:
suffix
=
' [
%
s]'
%
self
.
source_url
else
:
suffix
=
''
return
'<Distribution
%
s (
%
s)
%
s>'
%
(
self
.
name
,
self
.
version
,
suffix
)
def
__eq__
(
self
,
other
):
"""
See if this distribution is the same as another.
:param other: The distribution to compare with. To be equal to one
another. distributions must have the same type, name,
version and source_url.
:return: True if it is the same, else False.
"""
if
type
(
other
)
is
not
type
(
self
):
result
=
False
else
:
result
=
(
self
.
name
==
other
.
name
and
self
.
version
==
other
.
version
and
self
.
source_url
==
other
.
source_url
)
return
result
def
__hash__
(
self
):
"""
Compute hash in a way which matches the equality test.
"""
return
hash
(
self
.
name
)
+
hash
(
self
.
version
)
+
hash
(
self
.
source_url
)
class
BaseInstalledDistribution
(
Distribution
):
"""
This is the base class for installed distributions (whether PEP 376 or
legacy).
"""
hasher
=
None
def
__init__
(
self
,
metadata
,
path
,
env
=
None
):
"""
Initialise an instance.
:param metadata: An instance of :class:`Metadata` which describes the
distribution. This will normally have been initialised
from a metadata file in the ``path``.
:param path: The path of the ``.dist-info`` or ``.egg-info``
directory for the distribution.
:param env: This is normally the :class:`DistributionPath`
instance where this distribution was found.
"""
super
(
BaseInstalledDistribution
,
self
)
.
__init__
(
metadata
)
self
.
path
=
path
self
.
dist_path
=
env
def
get_hash
(
self
,
data
,
hasher
=
None
):
"""
Get the hash of some data, using a particular hash algorithm, if
specified.
:param data: The data to be hashed.
:type data: bytes
:param hasher: The name of a hash implementation, supported by hashlib,
or ``None``. Examples of valid values are ``'sha1'``,
``'sha224'``, ``'sha384'``, '``sha256'``, ``'md5'`` and
``'sha512'``. If no hasher is specified, the ``hasher``
attribute of the :class:`InstalledDistribution` instance
is used. If the hasher is determined to be ``None``, MD5
is used as the hashing algorithm.
:returns: The hash of the data. If a hasher was explicitly specified,
the returned hash will be prefixed with the specified hasher
followed by '='.
:rtype: str
"""
if
hasher
is
None
:
hasher
=
self
.
hasher
if
hasher
is
None
:
hasher
=
hashlib
.
md5
prefix
=
''
else
:
hasher
=
getattr
(
hashlib
,
hasher
)
prefix
=
'
%
s='
%
self
.
hasher
digest
=
hasher
(
data
)
.
digest
()
digest
=
base64
.
urlsafe_b64encode
(
digest
)
.
rstrip
(
b
'='
)
.
decode
(
'ascii'
)
return
'
%
s
%
s'
%
(
prefix
,
digest
)
class
InstalledDistribution
(
BaseInstalledDistribution
):
"""
Created with the *path* of the ``.dist-info`` directory provided to the
constructor. It reads the metadata contained in ``pydist.json`` when it is
instantiated., or uses a passed in Metadata instance (useful for when
dry-run mode is being used).
"""
hasher
=
'sha256'
def
__init__
(
self
,
path
,
metadata
=
None
,
env
=
None
):
self
.
modules
=
[]
self
.
finder
=
finder
=
resources
.
finder_for_path
(
path
)
if
finder
is
None
:
raise
ValueError
(
'finder unavailable for
%
s'
%
path
)
if
env
and
env
.
_cache_enabled
and
path
in
env
.
_cache
.
path
:
metadata
=
env
.
_cache
.
path
[
path
]
.
metadata
elif
metadata
is
None
:
r
=
finder
.
find
(
METADATA_FILENAME
)
# Temporary - for Wheel 0.23 support
if
r
is
None
:
r
=
finder
.
find
(
WHEEL_METADATA_FILENAME
)
# Temporary - for legacy support
if
r
is
None
:
r
=
finder
.
find
(
'METADATA'
)
if
r
is
None
:
raise
ValueError
(
'no
%
s found in
%
s'
%
(
METADATA_FILENAME
,
path
))
with
contextlib
.
closing
(
r
.
as_stream
())
as
stream
:
metadata
=
Metadata
(
fileobj
=
stream
,
scheme
=
'legacy'
)
super
(
InstalledDistribution
,
self
)
.
__init__
(
metadata
,
path
,
env
)
if
env
and
env
.
_cache_enabled
:
env
.
_cache
.
add
(
self
)
r
=
finder
.
find
(
'REQUESTED'
)
self
.
requested
=
r
is
not
None
p
=
os
.
path
.
join
(
path
,
'top_level.txt'
)
if
os
.
path
.
exists
(
p
):
with
open
(
p
,
'rb'
)
as
f
:
data
=
f
.
read
()
.
decode
(
'utf-8'
)
self
.
modules
=
data
.
splitlines
()
def
__repr__
(
self
):
return
'<InstalledDistribution
%
r
%
s at
%
r>'
%
(
self
.
name
,
self
.
version
,
self
.
path
)
def
__str__
(
self
):
return
"
%
s
%
s"
%
(
self
.
name
,
self
.
version
)
def
_get_records
(
self
):
"""
Get the list of installed files for the distribution
:return: A list of tuples of path, hash and size. Note that hash and
size might be ``None`` for some entries. The path is exactly
as stored in the file (which is as in PEP 376).
"""
results
=
[]
r
=
self
.
get_distinfo_resource
(
'RECORD'
)
with
contextlib
.
closing
(
r
.
as_stream
())
as
stream
:
with
CSVReader
(
stream
=
stream
)
as
record_reader
:
# Base location is parent dir of .dist-info dir
#base_location = os.path.dirname(self.path)
#base_location = os.path.abspath(base_location)
for
row
in
record_reader
:
missing
=
[
None
for
i
in
range
(
len
(
row
),
3
)]
path
,
checksum
,
size
=
row
+
missing
#if not os.path.isabs(path):
# path = path.replace('/', os.sep)
# path = os.path.join(base_location, path)
results
.
append
((
path
,
checksum
,
size
))
return
results
@cached_property
def
exports
(
self
):
"""
Return the information exported by this distribution.
:return: A dictionary of exports, mapping an export category to a dict
of :class:`ExportEntry` instances describing the individual
export entries, and keyed by name.
"""
result
=
{}
r
=
self
.
get_distinfo_resource
(
EXPORTS_FILENAME
)
if
r
:
result
=
self
.
read_exports
()
return
result
def
read_exports
(
self
):
"""
Read exports data from a file in .ini format.
:return: A dictionary of exports, mapping an export category to a list
of :class:`ExportEntry` instances describing the individual
export entries.
"""
result
=
{}
r
=
self
.
get_distinfo_resource
(
EXPORTS_FILENAME
)
if
r
:
with
contextlib
.
closing
(
r
.
as_stream
())
as
stream
:
result
=
read_exports
(
stream
)
return
result
def
write_exports
(
self
,
exports
):
"""
Write a dictionary of exports to a file in .ini format.
:param exports: A dictionary of exports, mapping an export category to
a list of :class:`ExportEntry` instances describing the
individual export entries.
"""
rf
=
self
.
get_distinfo_file
(
EXPORTS_FILENAME
)
with
open
(
rf
,
'w'
)
as
f
:
write_exports
(
exports
,
f
)
def
get_resource_path
(
self
,
relative_path
):
"""
NOTE: This API may change in the future.
Return the absolute path to a resource file with the given relative
path.
:param relative_path: The path, relative to .dist-info, of the resource
of interest.
:return: The absolute path where the resource is to be found.
"""
r
=
self
.
get_distinfo_resource
(
'RESOURCES'
)
with
contextlib
.
closing
(
r
.
as_stream
())
as
stream
:
with
CSVReader
(
stream
=
stream
)
as
resources_reader
:
for
relative
,
destination
in
resources_reader
:
if
relative
==
relative_path
:
return
destination
raise
KeyError
(
'no resource file with relative path
%
r '
'is installed'
%
relative_path
)
def
list_installed_files
(
self
):
"""
Iterates over the ``RECORD`` entries and returns a tuple
``(path, hash, size)`` for each line.
:returns: iterator of (path, hash, size)
"""
for
result
in
self
.
_get_records
():
yield
result
def
write_installed_files
(
self
,
paths
,
prefix
,
dry_run
=
False
):
"""
Writes the ``RECORD`` file, using the ``paths`` iterable passed in. Any
existing ``RECORD`` file is silently overwritten.
prefix is used to determine when to write absolute paths.
"""
prefix
=
os
.
path
.
join
(
prefix
,
''
)
base
=
os
.
path
.
dirname
(
self
.
path
)
base_under_prefix
=
base
.
startswith
(
prefix
)
base
=
os
.
path
.
join
(
base
,
''
)
record_path
=
self
.
get_distinfo_file
(
'RECORD'
)
logger
.
info
(
'creating
%
s'
,
record_path
)
if
dry_run
:
return
None
with
CSVWriter
(
record_path
)
as
writer
:
for
path
in
paths
:
if
os
.
path
.
isdir
(
path
)
or
path
.
endswith
((
'.pyc'
,
'.pyo'
)):
# do not put size and hash, as in PEP-376
hash_value
=
size
=
''
else
:
size
=
'
%
d'
%
os
.
path
.
getsize
(
path
)
with
open
(
path
,
'rb'
)
as
fp
:
hash_value
=
self
.
get_hash
(
fp
.
read
())
if
path
.
startswith
(
base
)
or
(
base_under_prefix
and
path
.
startswith
(
prefix
)):
path
=
os
.
path
.
relpath
(
path
,
base
)
writer
.
writerow
((
path
,
hash_value
,
size
))
# add the RECORD file itself
if
record_path
.
startswith
(
base
):
record_path
=
os
.
path
.
relpath
(
record_path
,
base
)
writer
.
writerow
((
record_path
,
''
,
''
))
return
record_path
def
check_installed_files
(
self
):
"""
Checks that the hashes and sizes of the files in ``RECORD`` are
matched by the files themselves. Returns a (possibly empty) list of
mismatches. Each entry in the mismatch list will be a tuple consisting
of the path, 'exists', 'size' or 'hash' according to what didn't match
(existence is checked first, then size, then hash), the expected
value and the actual value.
"""
mismatches
=
[]
base
=
os
.
path
.
dirname
(
self
.
path
)
record_path
=
self
.
get_distinfo_file
(
'RECORD'
)
for
path
,
hash_value
,
size
in
self
.
list_installed_files
():
if
not
os
.
path
.
isabs
(
path
):
path
=
os
.
path
.
join
(
base
,
path
)
if
path
==
record_path
:
continue
if
not
os
.
path
.
exists
(
path
):
mismatches
.
append
((
path
,
'exists'
,
True
,
False
))
elif
os
.
path
.
isfile
(
path
):
actual_size
=
str
(
os
.
path
.
getsize
(
path
))
if
size
and
actual_size
!=
size
:
mismatches
.
append
((
path
,
'size'
,
size
,
actual_size
))
elif
hash_value
:
if
'='
in
hash_value
:
hasher
=
hash_value
.
split
(
'='
,
1
)[
0
]
else
:
hasher
=
None
with
open
(
path
,
'rb'
)
as
f
:
actual_hash
=
self
.
get_hash
(
f
.
read
(),
hasher
)
if
actual_hash
!=
hash_value
:
mismatches
.
append
((
path
,
'hash'
,
hash_value
,
actual_hash
))
return
mismatches
@cached_property
def
shared_locations
(
self
):
"""
A dictionary of shared locations whose keys are in the set 'prefix',
'purelib', 'platlib', 'scripts', 'headers', 'data' and 'namespace'.
The corresponding value is the absolute path of that category for
this distribution, and takes into account any paths selected by the
user at installation time (e.g. via command-line arguments). In the
case of the 'namespace' key, this would be a list of absolute paths
for the roots of namespace packages in this distribution.
The first time this property is accessed, the relevant information is
read from the SHARED file in the .dist-info directory.
"""
result
=
{}
shared_path
=
os
.
path
.
join
(
self
.
path
,
'SHARED'
)
if
os
.
path
.
isfile
(
shared_path
):
with
codecs
.
open
(
shared_path
,
'r'
,
encoding
=
'utf-8'
)
as
f
:
lines
=
f
.
read
()
.
splitlines
()
for
line
in
lines
:
key
,
value
=
line
.
split
(
'='
,
1
)
if
key
==
'namespace'
:
result
.
setdefault
(
key
,
[])
.
append
(
value
)
else
:
result
[
key
]
=
value
return
result
def
write_shared_locations
(
self
,
paths
,
dry_run
=
False
):
"""
Write shared location information to the SHARED file in .dist-info.
:param paths: A dictionary as described in the documentation for
:meth:`shared_locations`.
:param dry_run: If True, the action is logged but no file is actually
written.
:return: The path of the file written to.
"""
shared_path
=
os
.
path
.
join
(
self
.
path
,
'SHARED'
)
logger
.
info
(
'creating
%
s'
,
shared_path
)
if
dry_run
:
return
None
lines
=
[]
for
key
in
(
'prefix'
,
'lib'
,
'headers'
,
'scripts'
,
'data'
):
path
=
paths
[
key
]
if
os
.
path
.
isdir
(
paths
[
key
]):
lines
.
append
(
'
%
s=
%
s'
%
(
key
,
path
))
for
ns
in
paths
.
get
(
'namespace'
,
()):
lines
.
append
(
'namespace=
%
s'
%
ns
)
with
codecs
.
open
(
shared_path
,
'w'
,
encoding
=
'utf-8'
)
as
f
:
f
.
write
(
'
\n
'
.
join
(
lines
))
return
shared_path
def
get_distinfo_resource
(
self
,
path
):
if
path
not
in
DIST_FILES
:
raise
DistlibException
(
'invalid path for a dist-info file: '
'
%
r at
%
r'
%
(
path
,
self
.
path
))
finder
=
resources
.
finder_for_path
(
self
.
path
)
if
finder
is
None
:
raise
DistlibException
(
'Unable to get a finder for
%
s'
%
self
.
path
)
return
finder
.
find
(
path
)
def
get_distinfo_file
(
self
,
path
):
"""
Returns a path located under the ``.dist-info`` directory. Returns a
string representing the path.
:parameter path: a ``'/'``-separated path relative to the
``.dist-info`` directory or an absolute path;
If *path* is an absolute path and doesn't start
with the ``.dist-info`` directory path,
a :class:`DistlibException` is raised
:type path: str
:rtype: str
"""
# Check if it is an absolute path # XXX use relpath, add tests
if
path
.
find
(
os
.
sep
)
>=
0
:
# it's an absolute path?
distinfo_dirname
,
path
=
path
.
split
(
os
.
sep
)[
-
2
:]
if
distinfo_dirname
!=
self
.
path
.
split
(
os
.
sep
)[
-
1
]:
raise
DistlibException
(
'dist-info file
%
r does not belong to the
%
r
%
s '
'distribution'
%
(
path
,
self
.
name
,
self
.
version
))
# The file must be relative
if
path
not
in
DIST_FILES
:
raise
DistlibException
(
'invalid path for a dist-info file: '
'
%
r at
%
r'
%
(
path
,
self
.
path
))
return
os
.
path
.
join
(
self
.
path
,
path
)
def
list_distinfo_files
(
self
):
"""
Iterates over the ``RECORD`` entries and returns paths for each line if
the path is pointing to a file located in the ``.dist-info`` directory
or one of its subdirectories.
:returns: iterator of paths
"""
base
=
os
.
path
.
dirname
(
self
.
path
)
for
path
,
checksum
,
size
in
self
.
_get_records
():
# XXX add separator or use real relpath algo
if
not
os
.
path
.
isabs
(
path
):
path
=
os
.
path
.
join
(
base
,
path
)
if
path
.
startswith
(
self
.
path
):
yield
path
def
__eq__
(
self
,
other
):
return
(
isinstance
(
other
,
InstalledDistribution
)
and
self
.
path
==
other
.
path
)
# See http://docs.python.org/reference/datamodel#object.__hash__
__hash__
=
object
.
__hash__
class
EggInfoDistribution
(
BaseInstalledDistribution
):
"""Created with the *path* of the ``.egg-info`` directory or file provided
to the constructor. It reads the metadata contained in the file itself, or
if the given path happens to be a directory, the metadata is read from the
file ``PKG-INFO`` under that directory."""
requested
=
True
# as we have no way of knowing, assume it was
shared_locations
=
{}
def
__init__
(
self
,
path
,
env
=
None
):
def
set_name_and_version
(
s
,
n
,
v
):
s
.
name
=
n
s
.
key
=
n
.
lower
()
# for case-insensitive comparisons
s
.
version
=
v
self
.
path
=
path
self
.
dist_path
=
env
if
env
and
env
.
_cache_enabled
and
path
in
env
.
_cache_egg
.
path
:
metadata
=
env
.
_cache_egg
.
path
[
path
]
.
metadata
set_name_and_version
(
self
,
metadata
.
name
,
metadata
.
version
)
else
:
metadata
=
self
.
_get_metadata
(
path
)
# Need to be set before caching
set_name_and_version
(
self
,
metadata
.
name
,
metadata
.
version
)
if
env
and
env
.
_cache_enabled
:
env
.
_cache_egg
.
add
(
self
)
super
(
EggInfoDistribution
,
self
)
.
__init__
(
metadata
,
path
,
env
)
def
_get_metadata
(
self
,
path
):
requires
=
None
def
parse_requires_data
(
data
):
"""Create a list of dependencies from a requires.txt file.
*data*: the contents of a setuptools-produced requires.txt file.
"""
reqs
=
[]
lines
=
data
.
splitlines
()
for
line
in
lines
:
line
=
line
.
strip
()
if
line
.
startswith
(
'['
):
logger
.
warning
(
'Unexpected line: quitting requirement scan:
%
r'
,
line
)
break
r
=
parse_requirement
(
line
)
if
not
r
:
logger
.
warning
(
'Not recognised as a requirement:
%
r'
,
line
)
continue
if
r
.
extras
:
logger
.
warning
(
'extra requirements in requires.txt are '
'not supported'
)
if
not
r
.
constraints
:
reqs
.
append
(
r
.
name
)
else
:
cons
=
', '
.
join
(
'
%
s
%
s'
%
c
for
c
in
r
.
constraints
)
reqs
.
append
(
'
%
s (
%
s)'
%
(
r
.
name
,
cons
))
return
reqs
def
parse_requires_path
(
req_path
):
"""Create a list of dependencies from a requires.txt file.
*req_path*: the path to a setuptools-produced requires.txt file.
"""
reqs
=
[]
try
:
with
codecs
.
open
(
req_path
,
'r'
,
'utf-8'
)
as
fp
:
reqs
=
parse_requires_data
(
fp
.
read
())
except
IOError
:
pass
return
reqs
tl_path
=
tl_data
=
None
if
path
.
endswith
(
'.egg'
):
if
os
.
path
.
isdir
(
path
):
p
=
os
.
path
.
join
(
path
,
'EGG-INFO'
)
meta_path
=
os
.
path
.
join
(
p
,
'PKG-INFO'
)
metadata
=
Metadata
(
path
=
meta_path
,
scheme
=
'legacy'
)
req_path
=
os
.
path
.
join
(
p
,
'requires.txt'
)
tl_path
=
os
.
path
.
join
(
p
,
'top_level.txt'
)
requires
=
parse_requires_path
(
req_path
)
else
:
# FIXME handle the case where zipfile is not available
zipf
=
zipimport
.
zipimporter
(
path
)
fileobj
=
StringIO
(
zipf
.
get_data
(
'EGG-INFO/PKG-INFO'
)
.
decode
(
'utf8'
))
metadata
=
Metadata
(
fileobj
=
fileobj
,
scheme
=
'legacy'
)
try
:
data
=
zipf
.
get_data
(
'EGG-INFO/requires.txt'
)
tl_data
=
zipf
.
get_data
(
'EGG-INFO/top_level.txt'
)
.
decode
(
'utf-8'
)
requires
=
parse_requires_data
(
data
.
decode
(
'utf-8'
))
except
IOError
:
requires
=
None
elif
path
.
endswith
(
'.egg-info'
):
if
os
.
path
.
isdir
(
path
):
req_path
=
os
.
path
.
join
(
path
,
'requires.txt'
)
requires
=
parse_requires_path
(
req_path
)
path
=
os
.
path
.
join
(
path
,
'PKG-INFO'
)
tl_path
=
os
.
path
.
join
(
path
,
'top_level.txt'
)
metadata
=
Metadata
(
path
=
path
,
scheme
=
'legacy'
)
else
:
raise
DistlibException
(
'path must end with .egg-info or .egg, '
'got
%
r'
%
path
)
if
requires
:
metadata
.
add_requirements
(
requires
)
# look for top-level modules in top_level.txt, if present
if
tl_data
is
None
:
if
tl_path
is
not
None
and
os
.
path
.
exists
(
tl_path
):
with
open
(
tl_path
,
'rb'
)
as
f
:
tl_data
=
f
.
read
()
.
decode
(
'utf-8'
)
if
not
tl_data
:
tl_data
=
[]
else
:
tl_data
=
tl_data
.
splitlines
()
self
.
modules
=
tl_data
return
metadata
def
__repr__
(
self
):
return
'<EggInfoDistribution
%
r
%
s at
%
r>'
%
(
self
.
name
,
self
.
version
,
self
.
path
)
def
__str__
(
self
):
return
"
%
s
%
s"
%
(
self
.
name
,
self
.
version
)
def
check_installed_files
(
self
):
"""
Checks that the hashes and sizes of the files in ``RECORD`` are
matched by the files themselves. Returns a (possibly empty) list of
mismatches. Each entry in the mismatch list will be a tuple consisting
of the path, 'exists', 'size' or 'hash' according to what didn't match
(existence is checked first, then size, then hash), the expected
value and the actual value.
"""
mismatches
=
[]
record_path
=
os
.
path
.
join
(
self
.
path
,
'installed-files.txt'
)
if
os
.
path
.
exists
(
record_path
):
for
path
,
_
,
_
in
self
.
list_installed_files
():
if
path
==
record_path
:
continue
if
not
os
.
path
.
exists
(
path
):
mismatches
.
append
((
path
,
'exists'
,
True
,
False
))
return
mismatches
def
list_installed_files
(
self
):
"""
Iterates over the ``installed-files.txt`` entries and returns a tuple
``(path, hash, size)`` for each line.
:returns: a list of (path, hash, size)
"""
def
_md5
(
path
):
f
=
open
(
path
,
'rb'
)
try
:
content
=
f
.
read
()
finally
:
f
.
close
()
return
hashlib
.
md5
(
content
)
.
hexdigest
()
def
_size
(
path
):
return
os
.
stat
(
path
)
.
st_size
record_path
=
os
.
path
.
join
(
self
.
path
,
'installed-files.txt'
)
result
=
[]
if
os
.
path
.
exists
(
record_path
):
with
codecs
.
open
(
record_path
,
'r'
,
encoding
=
'utf-8'
)
as
f
:
for
line
in
f
:
line
=
line
.
strip
()
p
=
os
.
path
.
normpath
(
os
.
path
.
join
(
self
.
path
,
line
))
# "./" is present as a marker between installed files
# and installation metadata files
if
not
os
.
path
.
exists
(
p
):
logger
.
warning
(
'Non-existent file:
%
s'
,
p
)
if
p
.
endswith
((
'.pyc'
,
'.pyo'
)):
continue
#otherwise fall through and fail
if
not
os
.
path
.
isdir
(
p
):
result
.
append
((
p
,
_md5
(
p
),
_size
(
p
)))
result
.
append
((
record_path
,
None
,
None
))
return
result
def
list_distinfo_files
(
self
,
absolute
=
False
):
"""
Iterates over the ``installed-files.txt`` entries and returns paths for
each line if the path is pointing to a file located in the
``.egg-info`` directory or one of its subdirectories.
:parameter absolute: If *absolute* is ``True``, each returned path is
transformed into a local absolute path. Otherwise the
raw value from ``installed-files.txt`` is returned.
:type absolute: boolean
:returns: iterator of paths
"""
record_path
=
os
.
path
.
join
(
self
.
path
,
'installed-files.txt'
)
if
os
.
path
.
exists
(
record_path
):
skip
=
True
with
codecs
.
open
(
record_path
,
'r'
,
encoding
=
'utf-8'
)
as
f
:
for
line
in
f
:
line
=
line
.
strip
()
if
line
==
'./'
:
skip
=
False
continue
if
not
skip
:
p
=
os
.
path
.
normpath
(
os
.
path
.
join
(
self
.
path
,
line
))
if
p
.
startswith
(
self
.
path
):
if
absolute
:
yield
p
else
:
yield
line
def
__eq__
(
self
,
other
):
return
(
isinstance
(
other
,
EggInfoDistribution
)
and
self
.
path
==
other
.
path
)
# See http://docs.python.org/reference/datamodel#object.__hash__
__hash__
=
object
.
__hash__
new_dist_class
=
InstalledDistribution
old_dist_class
=
EggInfoDistribution
class
DependencyGraph
(
object
):
"""
Represents a dependency graph between distributions.
The dependency relationships are stored in an ``adjacency_list`` that maps
distributions to a list of ``(other, label)`` tuples where ``other``
is a distribution and the edge is labeled with ``label`` (i.e. the version
specifier, if such was provided). Also, for more efficient traversal, for
every distribution ``x``, a list of predecessors is kept in
``reverse_list[x]``. An edge from distribution ``a`` to
distribution ``b`` means that ``a`` depends on ``b``. If any missing
dependencies are found, they are stored in ``missing``, which is a
dictionary that maps distributions to a list of requirements that were not
provided by any other distributions.
"""
def
__init__
(
self
):
self
.
adjacency_list
=
{}
self
.
reverse_list
=
{}
self
.
missing
=
{}
def
add_distribution
(
self
,
distribution
):
"""Add the *distribution* to the graph.
:type distribution: :class:`distutils2.database.InstalledDistribution`
or :class:`distutils2.database.EggInfoDistribution`
"""
self
.
adjacency_list
[
distribution
]
=
[]
self
.
reverse_list
[
distribution
]
=
[]
#self.missing[distribution] = []
def
add_edge
(
self
,
x
,
y
,
label
=
None
):
"""Add an edge from distribution *x* to distribution *y* with the given
*label*.
:type x: :class:`distutils2.database.InstalledDistribution` or
:class:`distutils2.database.EggInfoDistribution`
:type y: :class:`distutils2.database.InstalledDistribution` or
:class:`distutils2.database.EggInfoDistribution`
:type label: ``str`` or ``None``
"""
self
.
adjacency_list
[
x
]
.
append
((
y
,
label
))
# multiple edges are allowed, so be careful
if
x
not
in
self
.
reverse_list
[
y
]:
self
.
reverse_list
[
y
]
.
append
(
x
)
def
add_missing
(
self
,
distribution
,
requirement
):
"""
Add a missing *requirement* for the given *distribution*.
:type distribution: :class:`distutils2.database.InstalledDistribution`
or :class:`distutils2.database.EggInfoDistribution`
:type requirement: ``str``
"""
logger
.
debug
(
'
%
s missing
%
r'
,
distribution
,
requirement
)
self
.
missing
.
setdefault
(
distribution
,
[])
.
append
(
requirement
)
def
_repr_dist
(
self
,
dist
):
return
'
%
s
%
s'
%
(
dist
.
name
,
dist
.
version
)
def
repr_node
(
self
,
dist
,
level
=
1
):
"""Prints only a subgraph"""
output
=
[
self
.
_repr_dist
(
dist
)]
for
other
,
label
in
self
.
adjacency_list
[
dist
]:
dist
=
self
.
_repr_dist
(
other
)
if
label
is
not
None
:
dist
=
'
%
s [
%
s]'
%
(
dist
,
label
)
output
.
append
(
' '
*
level
+
str
(
dist
))
suboutput
=
self
.
repr_node
(
other
,
level
+
1
)
subs
=
suboutput
.
split
(
'
\n
'
)
output
.
extend
(
subs
[
1
:])
return
'
\n
'
.
join
(
output
)
def
to_dot
(
self
,
f
,
skip_disconnected
=
True
):
"""Writes a DOT output for the graph to the provided file *f*.
If *skip_disconnected* is set to ``True``, then all distributions
that are not dependent on any other distribution are skipped.
:type f: has to support ``file``-like operations
:type skip_disconnected: ``bool``
"""
disconnected
=
[]
f
.
write
(
"digraph dependencies {
\n
"
)
for
dist
,
adjs
in
self
.
adjacency_list
.
items
():
if
len
(
adjs
)
==
0
and
not
skip_disconnected
:
disconnected
.
append
(
dist
)
for
other
,
label
in
adjs
:
if
not
label
is
None
:
f
.
write
(
'"
%
s" -> "
%
s" [label="
%
s"]
\n
'
%
(
dist
.
name
,
other
.
name
,
label
))
else
:
f
.
write
(
'"
%
s" -> "
%
s"
\n
'
%
(
dist
.
name
,
other
.
name
))
if
not
skip_disconnected
and
len
(
disconnected
)
>
0
:
f
.
write
(
'subgraph disconnected {
\n
'
)
f
.
write
(
'label = "Disconnected"
\n
'
)
f
.
write
(
'bgcolor = red
\n
'
)
for
dist
in
disconnected
:
f
.
write
(
'"
%
s"'
%
dist
.
name
)
f
.
write
(
'
\n
'
)
f
.
write
(
'}
\n
'
)
f
.
write
(
'}
\n
'
)
def
topological_sort
(
self
):
"""
Perform a topological sort of the graph.
:return: A tuple, the first element of which is a topologically sorted
list of distributions, and the second element of which is a
list of distributions that cannot be sorted because they have
circular dependencies and so form a cycle.
"""
result
=
[]
# Make a shallow copy of the adjacency list
alist
=
{}
for
k
,
v
in
self
.
adjacency_list
.
items
():
alist
[
k
]
=
v
[:]
while
True
:
# See what we can remove in this run
to_remove
=
[]
for
k
,
v
in
list
(
alist
.
items
())[:]:
if
not
v
:
to_remove
.
append
(
k
)
del
alist
[
k
]
if
not
to_remove
:
# What's left in alist (if anything) is a cycle.
break
# Remove from the adjacency list of others
for
k
,
v
in
alist
.
items
():
alist
[
k
]
=
[(
d
,
r
)
for
d
,
r
in
v
if
d
not
in
to_remove
]
logger
.
debug
(
'Moving to result:
%
s'
,
[
'
%
s (
%
s)'
%
(
d
.
name
,
d
.
version
)
for
d
in
to_remove
])
result
.
extend
(
to_remove
)
return
result
,
list
(
alist
.
keys
())
def
__repr__
(
self
):
"""Representation of the graph"""
output
=
[]
for
dist
,
adjs
in
self
.
adjacency_list
.
items
():
output
.
append
(
self
.
repr_node
(
dist
))
return
'
\n
'
.
join
(
output
)
def
make_graph
(
dists
,
scheme
=
'default'
):
"""Makes a dependency graph from the given distributions.
:parameter dists: a list of distributions
:type dists: list of :class:`distutils2.database.InstalledDistribution` and
:class:`distutils2.database.EggInfoDistribution` instances
:rtype: a :class:`DependencyGraph` instance
"""
scheme
=
get_scheme
(
scheme
)
graph
=
DependencyGraph
()
provided
=
{}
# maps names to lists of (version, dist) tuples
# first, build the graph and find out what's provided
for
dist
in
dists
:
graph
.
add_distribution
(
dist
)
for
p
in
dist
.
provides
:
name
,
version
=
parse_name_and_version
(
p
)
logger
.
debug
(
'Add to provided:
%
s,
%
s,
%
s'
,
name
,
version
,
dist
)
provided
.
setdefault
(
name
,
[])
.
append
((
version
,
dist
))
# now make the edges
for
dist
in
dists
:
requires
=
(
dist
.
run_requires
|
dist
.
meta_requires
|
dist
.
build_requires
|
dist
.
dev_requires
)
for
req
in
requires
:
try
:
matcher
=
scheme
.
matcher
(
req
)
except
UnsupportedVersionError
:
# XXX compat-mode if cannot read the version
logger
.
warning
(
'could not read version
%
r - using name only'
,
req
)
name
=
req
.
split
()[
0
]
matcher
=
scheme
.
matcher
(
name
)
name
=
matcher
.
key
# case-insensitive
matched
=
False
if
name
in
provided
:
for
version
,
provider
in
provided
[
name
]:
try
:
match
=
matcher
.
match
(
version
)
except
UnsupportedVersionError
:
match
=
False
if
match
:
graph
.
add_edge
(
dist
,
provider
,
req
)
matched
=
True
break
if
not
matched
:
graph
.
add_missing
(
dist
,
req
)
return
graph
def
get_dependent_dists
(
dists
,
dist
):
"""Recursively generate a list of distributions from *dists* that are
dependent on *dist*.
:param dists: a list of distributions
:param dist: a distribution, member of *dists* for which we are interested
"""
if
dist
not
in
dists
:
raise
DistlibException
(
'given distribution
%
r is not a member '
'of the list'
%
dist
.
name
)
graph
=
make_graph
(
dists
)
dep
=
[
dist
]
# dependent distributions
todo
=
graph
.
reverse_list
[
dist
]
# list of nodes we should inspect
while
todo
:
d
=
todo
.
pop
()
dep
.
append
(
d
)
for
succ
in
graph
.
reverse_list
[
d
]:
if
succ
not
in
dep
:
todo
.
append
(
succ
)
dep
.
pop
(
0
)
# remove dist from dep, was there to prevent infinite loops
return
dep
def
get_required_dists
(
dists
,
dist
):
"""Recursively generate a list of distributions from *dists* that are
required by *dist*.
:param dists: a list of distributions
:param dist: a distribution, member of *dists* for which we are interested
"""
if
dist
not
in
dists
:
raise
DistlibException
(
'given distribution
%
r is not a member '
'of the list'
%
dist
.
name
)
graph
=
make_graph
(
dists
)
req
=
[]
# required distributions
todo
=
graph
.
adjacency_list
[
dist
]
# list of nodes we should inspect
while
todo
:
d
=
todo
.
pop
()[
0
]
req
.
append
(
d
)
for
pred
in
graph
.
adjacency_list
[
d
]:
if
pred
not
in
req
:
todo
.
append
(
pred
)
return
req
def
make_dist
(
name
,
version
,
**
kwargs
):
"""
A convenience method for making a dist given just a name and version.
"""
summary
=
kwargs
.
pop
(
'summary'
,
'Placeholder for summary'
)
md
=
Metadata
(
**
kwargs
)
md
.
name
=
name
md
.
version
=
version
md
.
summary
=
summary
or
'Placeholder for summary'
return
Distribution
(
md
)
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment