forked from drone-plugins/drone-devpi
-
Notifications
You must be signed in to change notification settings - Fork 1
/
run_devpi.py
149 lines (122 loc) · 4.79 KB
/
run_devpi.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
#!/usr/bin/env python
"""
Everything needed to publish Python modules to a DevPi server.
Recommended reading: http://doc.devpi.net/latest/quickstart-releaseprocess.html
"""
import sys
import urllib.parse
import drone
import subprocess
# devpi uses a 'clientdir' arg to determine where to store state. We make
# this overridable below to facilitate the integration test process.
DEFAULT_CLIENTDIR = '/tmp/devpi-clientdir'
def die_on_error(f):
"""
Since this plugin wraps the devpi client's CLI, we do a lot of the same
error handling. We don't capture stdout/stderr, so they get emitted.
This decorator just finishes the job and kills the build (by exiting).
"""
def wrapped(*args, **kwargs):
result = f(*args, **kwargs)
if result.returncode == 1:
sys.exit(1)
return wrapped
@die_on_error
def select_server(server, clientdir=DEFAULT_CLIENTDIR):
"""
Before the devpi CLI can do much of anything, it has to be pointed at the
root of a devpi server.
:param str server: Absolute URI to the root of a devpi server (not an
index within the devpi server).
:param str clientdir: Path to a directory for the devpi CLI to store state.
:rtype: subprocess.CompletedProcess
"""
return subprocess.run(['devpi', 'use', '--clientdir', clientdir, server])
@die_on_error
def login(username, password, clientdir=DEFAULT_CLIENTDIR):
"""
Uploading packages to a devpi server usually requires an authenticated
account with write permissions.
:param str username: The devpi username we'll be uploading as.
:param str password: The devpi user's password.
:param str clientdir: Path to a directory for the devpi CLI to store state.
:rtype: subprocess.CompletedProcess
"""
return subprocess.run([
'devpi', 'login', '--clientdir', clientdir,
username, '--password', password])
@die_on_error
def select_index(index, clientdir=DEFAULT_CLIENTDIR):
"""
Before we can upload a package to an index, we must select it since there's
no one-shot select + upload command.
:param str index: The index to upload to. For example, ``root/devpitest``.
This gets appended to whatever ``server`` value gets passed into
:py:func:`select_server`.
:param str clientdir: Path to a directory for the devpi CLI to store state.
:rtype: subprocess.CompletedProcess
"""
return subprocess.run(['devpi', 'use', '--clientdir', clientdir, index])
@die_on_error
def create_index(index, clientdir=DEFAULT_CLIENTDIR):
"""
Creates an index on the devpi server.
:param str index: The index to create. For example, ``root/devpitest``.
This gets appended to whatever ``server`` value gets passed into
:py:func:`select_server`.
:param str clientdir: Path to a directory for the devpi CLI to store state.
:rtype: subprocess.CompletedProcess
"""
return subprocess.run([
'devpi', 'index', '--clientdir', clientdir, '-c', index])
@die_on_error
def upload_package(path, clientdir=DEFAULT_CLIENTDIR):
"""
Upload the package residing at ``path`` to the currently selected devpi
server + index.
:param str path: An absolute or relative path to the directory containing
the package you'd like to upload.
:param str clientdir: Path to a directory for the devpi CLI to store state.
:rtype: subprocess.CompletedProcess
"""
cmd = subprocess.Popen([
'devpi', 'upload', '--clientdir', clientdir,
'--from-dir', '--no-vcs'], cwd=path)
cmd.wait()
return cmd
def check_vargs(vargs):
"""
Check over the args passed in to make sure that we have everything we
need to get the upload done. Exit with code 1 if the input is bad.
:param dict vargs: Contents of the 'vargs' JSON array in the
the plugin input.
"""
server_uri = vargs.get('server', '')
parsed = urllib.parse.urlsplit(server_uri)
if not all([parsed.scheme, parsed.netloc]):
print(
"You must specify the full, absolute URI to your devpi server "
"(including protocol).")
sys.exit(1)
index = vargs.get('index')
if not index:
print("You must specify an index on your devpi server to upload to.")
sys.exit(1)
username = vargs.get('username')
if not username:
print("You must specify a username to upload packages as.")
sys.exit(1)
password = vargs.get('password')
if password is None:
print("You must specify a password.")
sys.exit(1)
def main():
payload = drone.plugin.get_input()
vargs = payload['vargs']
check_vargs(vargs)
select_server(vargs['server'])
login(vargs['username'], vargs['password'])
select_index(vargs['index'])
upload_package(payload['workspace']['path'])
if __name__ == "__main__":
main()