Skip to content

Commit 77a1342

Browse files
committed
Packaging implementation
1 parent 47a99fa commit 77a1342

File tree

93 files changed

+7355
-95
lines changed

Some content is hidden

Large Commits have some content hidden by default. Use the searchbox below for content that may be hidden.

93 files changed

+7355
-95
lines changed

conftest.py

+85-1
Original file line numberDiff line numberDiff line change
@@ -155,10 +155,94 @@ def _loaded_test_contract_fixtures(populus_source_root, project_dir, request):
155155
shutil.copy(src_path, dst_path)
156156

157157

158+
EXAMPLE_PACKAGES_BASE_PATH = './tests/example-packages'
159+
160+
161+
@pytest.fixture()
162+
def _loaded_installed_dependencies(populus_source_root, project_dir, request):
163+
from populus.utils.dependencies import (
164+
get_installed_packages_dir,
165+
)
166+
from populus.utils.filesystem import (
167+
find_solidity_source_files,
168+
)
169+
from populus.utils.packaging import (
170+
load_release_lockfile,
171+
extract_package_metadata,
172+
)
173+
from populus.utils.ipfs import (
174+
generate_file_hash,
175+
)
176+
from populus.packages.installation import (
177+
write_installed_packages,
178+
)
179+
180+
packages_to_load_from_fn = getattr(request.function, '_populus_packages_to_load', [])
181+
packages_to_load_from_module = getattr(request.module, '_populus_packages_to_load', [])
182+
183+
packages_to_load = itertools.chain(
184+
packages_to_load_from_fn,
185+
packages_to_load_from_module,
186+
)
187+
188+
def load_example_package_data(example_package_name):
189+
example_package_dir = os.path.join(
190+
populus_source_root,
191+
EXAMPLE_PACKAGES_BASE_PATH,
192+
example_package_name,
193+
)
194+
195+
if not os.path.exists(example_package_dir):
196+
raise ValueError(
197+
"Unable to load example package '{0}".format(example_package_name)
198+
)
199+
200+
release_lockfile_path = os.path.join(example_package_dir, '1.0.0.json')
201+
release_lockfile_uri = generate_file_hash(release_lockfile_path)
202+
release_lockfile = load_release_lockfile(release_lockfile_path)
203+
source_file_paths = find_solidity_source_files(example_package_dir)
204+
source_tree = {
205+
os.path.relpath(source_file_path, example_package_dir): open(source_file_path).read()
206+
for source_file_path
207+
in source_file_paths
208+
}
209+
package_meta = extract_package_metadata(
210+
[
211+
example_package_name,
212+
"{0}==1.0.0".format(example_package_name),
213+
release_lockfile_uri,
214+
],
215+
release_lockfile,
216+
)
217+
package_dependencies = tuple(
218+
load_example_package_data(dependency_name)
219+
for dependency_name
220+
in release_lockfile.get('build_dependencies', {}).keys()
221+
)
222+
223+
package_data = {
224+
'meta': package_meta,
225+
'lockfile': release_lockfile,
226+
'source_tree': source_tree,
227+
'dependencies': package_dependencies,
228+
}
229+
return package_data
230+
231+
installed_packages_dir = get_installed_packages_dir(project_dir)
232+
233+
package_data_to_install = tuple(
234+
load_example_package_data(item)
235+
for item
236+
in packages_to_load
237+
)
238+
write_installed_packages(installed_packages_dir, package_data_to_install)
239+
240+
158241
@pytest.fixture()
159242
def project(project_dir,
160243
_loaded_contract_fixtures,
161-
_loaded_test_contract_fixtures):
244+
_loaded_test_contract_fixtures,
245+
_loaded_installed_dependencies):
162246
return Project()
163247

164248

docs/index.rst

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ Contents
1919
project
2020
config
2121
chain
22+
packaging
2223
release
2324
modules
2425
API Documentation <populus>

docs/packaging.quickstart.rst

+131
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,131 @@
1+
Packaging Quickstart
2+
====================
3+
4+
.. contents:: :local:
5+
6+
Introduction
7+
------------
8+
9+
Populus can be used as a package manager to interact with any ERC190 smart
10+
contract packages.
11+
12+
13+
Project Manifest
14+
----------------
15+
16+
In order to take advantage of the packaging features you will first need to
17+
create a package manifest for your project. This can either be done manually
18+
or using the command line helper ``$ populus package init`` which will present
19+
an interactive prompt for creating the ``ethpm.json`` file.
20+
21+
.. code-block:: bash
22+
23+
$ populus package init
24+
Writing new ethpm.json file.
25+
Package Name: fancy-greeter
26+
Author(s) [[]]: Piper Merriam <[email protected]>
27+
Version [1.0.0]:
28+
License [MIT]:
29+
Description []: A fancy greeter contract
30+
Keywords [[]]: greeter, greetings
31+
Links [{}]:
32+
Wrote package manifest: ethpm.json
33+
34+
35+
Installing Packages
36+
-------------------
37+
38+
Packages can be installed using the ``$populus package install`` command.
39+
Packages may be specified in the following formats.
40+
41+
* ``populus package install .``:
42+
43+
To install all of the declared dependencies found within the project's package manifest.
44+
45+
* ``populus package install some-package-name``
46+
47+
To install a named package ``some-package-name`` sourced from a package index.
48+
49+
* ``populus package install ipfs://QmUwVUMVtkVctrLDeL12SoeCPUacELBU8nAxRtHUzvtjND``
50+
51+
To install directly from a release lockfile via IPFS
52+
53+
* ``populus package install /path/to/release-lockfile.json``
54+
55+
To install directly from a release lockfile on the local filesystem.
56+
57+
58+
Populus also supports installing packages under aliased names. This can be
59+
used to allow multiple versions of the same package to be installed in tandem.
60+
61+
* ``populus package install some-alias:some-package-name``
62+
63+
To install a named package ``some-package-name`` under the name
64+
``some-alias`` sourced from a package index.
65+
66+
* ``populus package install some-alias@ipfs://QmUwVUMVtkVctrLDeL12SoeCPUacELBU8nAxRtHUzvtjND``
67+
68+
To install directly from a release lockfile via IPFS using the name ``some-alias``.
69+
70+
* ``populus package install some-alias@/path/to/release-lockfile.json``
71+
72+
To install directly from a release lockfile on the local filesystem using
73+
the name ``some-alias``
74+
75+
76+
Packages are installed in the ``./installed_packages`` directory in the root
77+
project directory under their aliased name, or their package name if no alias
78+
is used.
79+
80+
When a package is installed it is automatically saved to the project
81+
dependencies within the package manifest. This can be disabled by passing in
82+
the ``--no-save`` flag during installation.
83+
84+
85+
Using Contracts from Installed Packages
86+
---------------------------------------
87+
88+
Importing a contract from an installed package is done by prefixing the source
89+
path with the name of the installed package, or the alias name if an alias was
90+
used.
91+
92+
Lets use the common *owned* pattern for an example. Suppose we have the
93+
``owned`` package installed in our project. We know that this package has a
94+
single solidity source file that contains the ``owned`` contract located at
95+
``./contracts/owned.sol``.
96+
97+
To import a contract from this file into local solidity source files you would
98+
simply prefix the import path with the package name.
99+
100+
.. code-block:: solidity
101+
102+
pragma solidity ^0.4.0;
103+
104+
import "owned/contracts/owned.sol";
105+
106+
contract MyContract is owned {
107+
...
108+
}
109+
110+
.. note::
111+
112+
If you install a package which either has source files which do not compile
113+
with the solidity compiler version you are using, or which have a ``pragma
114+
solidity`` statement which is incompatable with your version of solidity
115+
then compilation will fail.
116+
117+
118+
Library Linking
119+
---------------
120+
121+
If you have a package installed which contains a library contract with a deployed instance of that library, populus will automatically find and link against that existing deployed library. One of the default contract backends that populus uses will check all installed packages
122+
123+
124+
125+
Building and Publishing Releases
126+
--------------------------------
127+
128+
Populus can be used to build and publish packages to The Ethereum Package
129+
Registry or any registry which implements a compatable API.
130+
131+
To build a release use the ``$ populus package build`` command.

docs/packaging.rst

+15
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,15 @@
1+
Packaging
2+
=========
3+
4+
.. contents:: :local:
5+
6+
.. warning:: The packaging functionality is highly experimental. All APIs are subject to change without notice.
7+
8+
9+
Contents
10+
--------
11+
12+
.. toctree::
13+
:maxdepth: 1
14+
15+
packaging.quickstart

0 commit comments

Comments
 (0)