geoseeq 0.2.2__py3-none-any.whl → 0.2.3__py3-none-any.whl
Sign up to get free protection for your applications and to get access to all the features.
- geoseeq/cli/download.py +38 -8
- geoseeq/cli/main.py +1 -1
- geoseeq/cli/progress_bar.py +28 -0
- geoseeq/cli/upload/upload_reads.py +1 -30
- geoseeq/result/file_download.py +67 -63
- geoseeq/result/utils.py +0 -16
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/METADATA +1 -1
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/RECORD +12 -11
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/LICENSE +0 -0
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/WHEEL +0 -0
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/entry_points.txt +0 -0
- {geoseeq-0.2.2.dist-info → geoseeq-0.2.3.dist-info}/top_level.txt +0 -0
geoseeq/cli/download.py
CHANGED
@@ -5,7 +5,7 @@ from os.path import dirname, join
|
|
5
5
|
|
6
6
|
import click
|
7
7
|
import pandas as pd
|
8
|
-
|
8
|
+
from multiprocessing import Pool
|
9
9
|
from .shared_params import (
|
10
10
|
handle_project_id,
|
11
11
|
project_id_arg,
|
@@ -14,13 +14,14 @@ from .shared_params import (
|
|
14
14
|
use_common_state,
|
15
15
|
flatten_list_of_els_and_files
|
16
16
|
)
|
17
|
-
from geoseeq.result.
|
17
|
+
from geoseeq.result.file_download import download_url
|
18
18
|
from geoseeq.utils import download_ftp
|
19
19
|
from geoseeq.blob_constructors import (
|
20
20
|
sample_result_file_from_uuid,
|
21
21
|
project_result_file_from_uuid,
|
22
22
|
)
|
23
23
|
from geoseeq.knex import GeoseeqNotFoundError
|
24
|
+
from .progress_bar import PBarManager
|
24
25
|
from .utils import convert_size
|
25
26
|
|
26
27
|
logger = logging.getLogger('geoseeq_api')
|
@@ -83,8 +84,16 @@ def cli_download_metadata(state, sample_ids):
|
|
83
84
|
click.echo("Metadata successfully downloaded for samples.", err=True)
|
84
85
|
|
85
86
|
|
87
|
+
def _download_one_file(args):
|
88
|
+
url, file_path, pbar = args
|
89
|
+
return download_url(url, filename=file_path, progress_tracker=pbar)
|
90
|
+
|
91
|
+
|
92
|
+
cores_option = click.option('--cores', default=1, help='Number of downloads to run in parallel')
|
93
|
+
|
86
94
|
@cli_download.command("files")
|
87
95
|
@use_common_state
|
96
|
+
@cores_option
|
88
97
|
@click.option("--target-dir", default=".")
|
89
98
|
@click.option('--yes/--confirm', default=False, help='Skip confirmation prompts')
|
90
99
|
@click.option("--download/--urls-only", default=True, help="Download files or just print urls")
|
@@ -98,6 +107,7 @@ def cli_download_metadata(state, sample_ids):
|
|
98
107
|
@sample_ids_arg
|
99
108
|
def cli_download_files(
|
100
109
|
state,
|
110
|
+
cores,
|
101
111
|
sample_name_includes,
|
102
112
|
target_dir,
|
103
113
|
yes,
|
@@ -186,23 +196,32 @@ def cli_download_files(
|
|
186
196
|
if not yes:
|
187
197
|
click.confirm('Do you want to download these files?', abort=True)
|
188
198
|
|
199
|
+
download_args = []
|
200
|
+
pbars = PBarManager()
|
189
201
|
for fname, url in response["links"].items():
|
190
202
|
click.echo(f"Downloading file {fname}")
|
191
203
|
file_path = join(target_dir, fname)
|
192
204
|
makedirs(dirname(file_path), exist_ok=True)
|
193
|
-
|
194
|
-
|
195
|
-
|
196
|
-
|
205
|
+
pbar = pbars.get_new_bar(file_path)
|
206
|
+
download_args.append((url, file_path, pbar))
|
207
|
+
if cores == 1:
|
208
|
+
download_url(url, filename=file_path, progress_tracker=pbar)
|
209
|
+
|
210
|
+
if cores > 1:
|
211
|
+
with Pool(cores) as p:
|
212
|
+
for _ in p.imap_unordered(_download_one_file, download_args):
|
213
|
+
pass
|
197
214
|
|
198
215
|
|
199
216
|
@cli_download.command("ids")
|
200
217
|
@use_common_state
|
218
|
+
@cores_option
|
201
219
|
@click.option("--target-dir", default=".")
|
202
220
|
@click.option('--yes/--confirm', default=False, help='Skip confirmation prompts')
|
203
221
|
@click.option("--download/--urls-only", default=True, help="Download files or just print urls")
|
222
|
+
@click.option('--head', default=None, type=int, help='Download the first N bytes of each file')
|
204
223
|
@click.argument("ids", nargs=-1)
|
205
|
-
def cli_download_ids(state, target_dir, yes, download, ids):
|
224
|
+
def cli_download_ids(state, cores, target_dir, yes, download, head, ids):
|
206
225
|
"""Download a files from GeoSeeq based on their UUID or GeoSeeq Resource Number (GRN).
|
207
226
|
|
208
227
|
This command downloads files directly based on their ID. This is used for "manual"
|
@@ -228,6 +247,7 @@ def cli_download_ids(state, target_dir, yes, download, ids):
|
|
228
247
|
---
|
229
248
|
"""
|
230
249
|
result_file_ids = flatten_list_of_els_and_files(ids)
|
250
|
+
cores = max(cores, len(result_file_ids)) # don't use more cores than files
|
231
251
|
knex = state.get_knex()
|
232
252
|
result_files = []
|
233
253
|
for result_id in result_file_ids:
|
@@ -249,8 +269,18 @@ def cli_download_ids(state, target_dir, yes, download, ids):
|
|
249
269
|
if not yes:
|
250
270
|
click.confirm('Do you want to download these files?', abort=True)
|
251
271
|
|
272
|
+
download_args = []
|
273
|
+
pbars = PBarManager()
|
252
274
|
for result_file in result_files:
|
253
275
|
click.echo(f"Downloading file {result_file.get_referenced_filename()}")
|
254
276
|
file_path = join(target_dir, result_file.get_referenced_filename())
|
255
277
|
makedirs(dirname(file_path), exist_ok=True)
|
256
|
-
|
278
|
+
pbar = pbars.get_new_bar(file_path)
|
279
|
+
download_args.append((result_file, file_path, pbar))
|
280
|
+
if cores == 1:
|
281
|
+
result_file.download(file_path, progress_tracker=pbar, head=head)
|
282
|
+
|
283
|
+
if cores > 1:
|
284
|
+
with Pool(cores) as p:
|
285
|
+
for _ in p.imap_unordered(_download_one_file, download_args):
|
286
|
+
pass
|
geoseeq/cli/main.py
CHANGED
@@ -0,0 +1,28 @@
|
|
1
|
+
from tqdm import tqdm
|
2
|
+
from os.path import basename
|
3
|
+
|
4
|
+
class TQBar:
|
5
|
+
|
6
|
+
def __init__(self, pos, desc) -> None:
|
7
|
+
self.n_bars = 0
|
8
|
+
self.pos = pos
|
9
|
+
self.desc = desc
|
10
|
+
self.bar = None
|
11
|
+
|
12
|
+
def set_num_chunks(self, n_chunks):
|
13
|
+
self.n_bars = n_chunks
|
14
|
+
self.bar = tqdm(total=n_chunks, position=self.pos, desc=self.desc, leave=False)
|
15
|
+
|
16
|
+
def update(self, chunk_num):
|
17
|
+
self.bar.update(chunk_num)
|
18
|
+
|
19
|
+
|
20
|
+
class PBarManager:
|
21
|
+
|
22
|
+
def __init__(self):
|
23
|
+
self.n_bars = 0
|
24
|
+
self.pbars = []
|
25
|
+
|
26
|
+
def get_new_bar(self, filepath):
|
27
|
+
self.n_bars += 1
|
28
|
+
return TQBar(self.n_bars, basename(filepath))
|
@@ -1,5 +1,4 @@
|
|
1
1
|
import logging
|
2
|
-
from tqdm import tqdm
|
3
2
|
import click
|
4
3
|
import requests
|
5
4
|
from os.path import basename
|
@@ -20,39 +19,11 @@ from geoseeq.cli.shared_params import (
|
|
20
19
|
)
|
21
20
|
|
22
21
|
from geoseeq.constants import FASTQ_MODULE_NAMES
|
23
|
-
|
24
|
-
|
22
|
+
from geoseeq.cli.progress_bar import PBarManager
|
25
23
|
|
26
24
|
logger = logging.getLogger('geoseeq_api')
|
27
25
|
|
28
26
|
|
29
|
-
class TQBar:
|
30
|
-
|
31
|
-
def __init__(self, pos, desc) -> None:
|
32
|
-
self.n_bars = 0
|
33
|
-
self.pos = pos
|
34
|
-
self.desc = desc
|
35
|
-
self.bar = None
|
36
|
-
|
37
|
-
def set_num_chunks(self, n_chunks):
|
38
|
-
self.n_bars = n_chunks
|
39
|
-
self.bar = tqdm(total=n_chunks, position=self.pos, desc=self.desc, leave=False)
|
40
|
-
|
41
|
-
def update(self, chunk_num):
|
42
|
-
self.bar.update(chunk_num)
|
43
|
-
|
44
|
-
|
45
|
-
class PBarManager:
|
46
|
-
|
47
|
-
def __init__(self):
|
48
|
-
self.n_bars = 0
|
49
|
-
self.pbars = []
|
50
|
-
|
51
|
-
def get_new_bar(self, filepath):
|
52
|
-
self.n_bars += 1
|
53
|
-
return TQBar(self.n_bars, basename(filepath))
|
54
|
-
|
55
|
-
|
56
27
|
def _make_in_process_logger(log_level):
|
57
28
|
logger = logging.getLogger('geoseeq_api')
|
58
29
|
logger.setLevel(log_level)
|
geoseeq/result/file_download.py
CHANGED
@@ -1,38 +1,88 @@
|
|
1
1
|
|
2
2
|
import urllib.request
|
3
3
|
import logging
|
4
|
+
import requests
|
4
5
|
from os.path import basename, getsize, join
|
5
6
|
from pathlib import Path
|
6
7
|
from tempfile import NamedTemporaryFile
|
7
8
|
|
8
9
|
from geoseeq.utils import download_ftp
|
9
|
-
|
10
|
-
from .utils import _download_head
|
10
|
+
from geoseeq.constants import FIVE_MB
|
11
11
|
|
12
12
|
logger = logging.getLogger("geoseeq_api") # Same name as calling module
|
13
13
|
|
14
14
|
|
15
|
+
def _download_head(url, filename, head=None, progress_tracker=None):
|
16
|
+
headers = None
|
17
|
+
if head and head > 0:
|
18
|
+
headers = {"Range": f"bytes=0-{head}"}
|
19
|
+
response = requests.get(url, stream=True, headers=headers)
|
20
|
+
total_size_in_bytes = int(response.headers.get('content-length', 0))
|
21
|
+
if progress_tracker: progress_tracker.set_num_chunks(total_size_in_bytes)
|
22
|
+
block_size = FIVE_MB
|
23
|
+
with open(filename, 'wb') as file:
|
24
|
+
for data in response.iter_content(block_size):
|
25
|
+
if progress_tracker: progress_tracker.update(len(data))
|
26
|
+
file.write(data)
|
27
|
+
return filename
|
28
|
+
|
29
|
+
|
30
|
+
def _download_generic(url, filename, head=None):
|
31
|
+
urllib.request.urlretrieve(url, filename)
|
32
|
+
return filename
|
33
|
+
|
34
|
+
|
35
|
+
def guess_download_kind(url):
|
36
|
+
if 'azure' in url:
|
37
|
+
return 'azure'
|
38
|
+
elif 's3' in url:
|
39
|
+
return 's3'
|
40
|
+
elif 'ftp' in url:
|
41
|
+
return 'ftp'
|
42
|
+
else:
|
43
|
+
return 'generic'
|
44
|
+
|
45
|
+
|
46
|
+
def download_url(url, kind='guess', filename=None, head=None, progress_tracker=None):
|
47
|
+
"""Return a local filepath to the downloaded file. Download the file."""
|
48
|
+
if kind == 'guess':
|
49
|
+
kind = guess_download_kind(url)
|
50
|
+
logger.info(f"Guessed download kind: {kind} for {url}")
|
51
|
+
logger.info(f"Downloading {kind} file to {filename}")
|
52
|
+
if kind == 'generic':
|
53
|
+
return _download_generic(url, filename, head=head)
|
54
|
+
elif kind == 's3':
|
55
|
+
return _download_head(url, filename, head=head, progress_tracker=progress_tracker)
|
56
|
+
elif kind == 'azure':
|
57
|
+
return _download_head(url, filename, head=head)
|
58
|
+
elif kind == 'ftp':
|
59
|
+
return download_ftp(url, filename, head=head)
|
60
|
+
else:
|
61
|
+
raise ValueError(f"Unknown download kind: {kind}")
|
62
|
+
|
63
|
+
|
64
|
+
|
15
65
|
class ResultFileDownload:
|
16
66
|
"""Abstract class that handles download methods for result files."""
|
17
67
|
|
18
68
|
def get_download_url(self):
|
19
69
|
"""Return a URL that can be used to download the file for this result."""
|
20
70
|
blob_type = self.stored_data.get("__type__", "").lower()
|
21
|
-
if blob_type not in ["s3", "sra"]:
|
22
|
-
raise
|
23
|
-
if
|
71
|
+
if blob_type not in ["s3", "sra", "ftp", "azure"]:
|
72
|
+
raise ValueError(f'Unknown URL type: "{blob_type}"')
|
73
|
+
key = 'url' if 'url' in self.stored_data else 'uri'
|
74
|
+
if blob_type in ["s3", "azure"]:
|
24
75
|
try:
|
25
76
|
url = self.stored_data["presigned_url"]
|
26
77
|
except KeyError:
|
27
|
-
url = self.stored_data[
|
78
|
+
url = self.stored_data[key]
|
28
79
|
if url.startswith("s3://"):
|
29
80
|
url = self.stored_data["endpoint_url"] + "/" + url[5:]
|
30
81
|
return url
|
31
|
-
|
32
|
-
|
33
|
-
return url
|
82
|
+
else:
|
83
|
+
return self.stored_data[key]
|
34
84
|
|
35
|
-
def
|
85
|
+
def download(self, filename=None, cache=True, head=None, progress_tracker=None):
|
36
86
|
"""Return a local filepath to the file this result points to."""
|
37
87
|
if not filename:
|
38
88
|
self._temp_filename = True
|
@@ -42,57 +92,11 @@ class ResultFileDownload:
|
|
42
92
|
blob_type = self.stored_data.get("__type__", "").lower()
|
43
93
|
if cache and self._cached_filename:
|
44
94
|
return self._cached_filename
|
45
|
-
|
46
|
-
|
47
|
-
|
48
|
-
|
49
|
-
|
50
|
-
return self._download_ftp(filename, cache)
|
51
|
-
elif blob_type == "azure":
|
52
|
-
return self._download_azure(filename, cache, head=head)
|
53
|
-
else:
|
54
|
-
raise TypeError("Cannot fetch a file for a BLOB type result field.")
|
55
|
-
|
56
|
-
def _download_s3(self, filename, cache, head=None):
|
57
|
-
logger.info(f"Downloading S3 file to {filename}")
|
58
|
-
try:
|
59
|
-
url = self.stored_data["presigned_url"]
|
60
|
-
except KeyError:
|
61
|
-
key = 'uri' if 'uri' in self.stored_data else 'url'
|
62
|
-
url = self.stored_data[key]
|
63
|
-
if url.startswith("s3://"):
|
64
|
-
url = self.stored_data["endpoint_url"] + "/" + url[5:]
|
65
|
-
_download_head(url, filename, head=head)
|
66
|
-
if cache:
|
67
|
-
self._cached_filename = filename
|
68
|
-
return filename
|
69
|
-
|
70
|
-
def _download_azure(self, filename, cache, head=None):
|
71
|
-
logger.info(f"Downloading Azure file to {filename}")
|
72
|
-
try:
|
73
|
-
url = self.stored_data["presigned_url"]
|
74
|
-
except KeyError:
|
75
|
-
key = 'uri' if 'uri' in self.stored_data else 'url'
|
76
|
-
url = self.stored_data[key]
|
77
|
-
_download_head(url, filename, head=head)
|
78
|
-
if cache:
|
79
|
-
self._cached_filename = filename
|
80
|
-
return filename
|
81
|
-
|
82
|
-
def _download_sra(self, filename, cache):
|
83
|
-
return self._download_generic_url(filename, cache)
|
84
|
-
|
85
|
-
def _download_ftp(self, filename, cache, head=None):
|
86
|
-
logger.info(f"Downloading FTP file to {filename}")
|
87
|
-
key = 'url' if 'url' in self.stored_data else 'uri'
|
88
|
-
download_ftp(self.stored_data[key], filename, head=head)
|
89
|
-
return filename
|
90
|
-
|
91
|
-
def _download_generic_url(self, filename, cache):
|
92
|
-
logger.info(f"Downloading generic URL file to {filename}")
|
93
|
-
key = 'url' if 'url' in self.stored_data else 'uri'
|
94
|
-
url = self.stored_data[key]
|
95
|
-
urllib.request.urlretrieve(url, filename)
|
95
|
+
url = self.get_download_url()
|
96
|
+
filepath = download_url(
|
97
|
+
url, blob_type, filename,
|
98
|
+
head=head, progress_tracker=progress_tracker
|
99
|
+
)
|
96
100
|
if cache:
|
97
|
-
self._cached_filename =
|
98
|
-
return
|
101
|
+
self._cached_filename = filepath
|
102
|
+
return filepath
|
geoseeq/result/utils.py
CHANGED
@@ -15,22 +15,6 @@ from geoseeq.utils import download_ftp, md5_checksum
|
|
15
15
|
|
16
16
|
logger = logging.getLogger("geoseeq_api") # Same name as calling module
|
17
17
|
logger.addHandler(logging.NullHandler()) # No output unless configured by calling program
|
18
|
-
|
19
|
-
|
20
|
-
def _download_head(url, filename, head=None):
|
21
|
-
if head and head > 0:
|
22
|
-
opener = urllib.request.build_opener()
|
23
|
-
if head:
|
24
|
-
opener.addheaders = [('Range', f'bytes=0-{head}')]
|
25
|
-
urllib.request.install_opener(opener)
|
26
|
-
try:
|
27
|
-
urllib.request.urlretrieve(url, filename) # can throw 416 error if head is too large
|
28
|
-
except urllib.error.HTTPError as e:
|
29
|
-
if e.code == 416:
|
30
|
-
logger.warning(f"HEAD request failed, trying again without HEAD.")
|
31
|
-
_download_head(url, filename, head=None)
|
32
|
-
else:
|
33
|
-
raise e
|
34
18
|
|
35
19
|
|
36
20
|
def diff_dicts(blob1, blob2):
|
@@ -18,10 +18,11 @@ geoseeq/cli/constants.py,sha256=Do5AUf9lMO9_P8KpFJ3XwwFBAWsxSjZ6sx9_QEGyC_c,176
|
|
18
18
|
geoseeq/cli/copy.py,sha256=ocJ-T6vZBTPELUSBA65KndzAugrBWMq-3ux0dbffacE,1997
|
19
19
|
geoseeq/cli/create.py,sha256=nMixc5DgLdHS8JhJE8Jt1LAvDprQdwojaOQ2bvvI-QU,6686
|
20
20
|
geoseeq/cli/delete.py,sha256=uvdmBqxBTquwLQAdP-fNpSfOd2hjImh8hMe-AnBd12Q,919
|
21
|
-
geoseeq/cli/download.py,sha256=
|
21
|
+
geoseeq/cli/download.py,sha256=T4jB9lZCaksoZFGigk7sJX1_QyFR6Zt9MfNUWTrUjB8,10930
|
22
22
|
geoseeq/cli/fastq_utils.py,sha256=-bmeQLaiMBm57zWOF0R5OlWTU0_3sh1JBC1RYw2BOFM,3083
|
23
23
|
geoseeq/cli/list.py,sha256=BvqwUJeOGvEVy7O6H6xYR2raYoyvR0yjvY2iklOCWCk,1712
|
24
|
-
geoseeq/cli/main.py,sha256=
|
24
|
+
geoseeq/cli/main.py,sha256=mgNakRaRrA_jTqU_OF7LkGR9vUKBI-Yom4pGK_IFK4g,1231
|
25
|
+
geoseeq/cli/progress_bar.py,sha256=LrhMTbTXHaRSX99NJQzRncreDXl6edov3Zo-Y3-d5_c,644
|
25
26
|
geoseeq/cli/user.py,sha256=fARv_5vx-QYT765Hsjgwr6J5ddA_viCPQdv9iUoVX6w,925
|
26
27
|
geoseeq/cli/utils.py,sha256=f0wX5-19uDrYZ8d-aBLplqfbGOkr5BH8x3AMSAyPCes,2873
|
27
28
|
geoseeq/cli/view.py,sha256=YmGcN4uGFuXWd_OA6Pi0KYvT-z4AD3nUwaOLewwJhBw,645
|
@@ -33,7 +34,7 @@ geoseeq/cli/shared_params/obj_getters.py,sha256=wmT37kwwcguy_JxqE7G9UH5mwcSZ0byw
|
|
33
34
|
geoseeq/cli/shared_params/opts_and_args.py,sha256=kxqqsbXoGg58GE1VtIySb2TFw8KlRodsvh4xXWEVBRw,1352
|
34
35
|
geoseeq/cli/upload/__init__.py,sha256=QqBh0WXD9BaEDcKsKpb-WgWwQ0WzZdQUgYXYwyowW_c,340
|
35
36
|
geoseeq/cli/upload/upload.py,sha256=JC5AH9CNR7dKryZzkxSf7DX9ZqPer7-fpvahNMtlVQQ,6153
|
36
|
-
geoseeq/cli/upload/upload_reads.py,sha256=
|
37
|
+
geoseeq/cli/upload/upload_reads.py,sha256=2y1rN1zLDpEj0rwKIDLjVnNA6BeXxM6ZDYn6lNtIsbM,6834
|
37
38
|
geoseeq/contrib/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
38
39
|
geoseeq/contrib/ncbi/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
39
40
|
geoseeq/contrib/ncbi/api.py,sha256=WQeLoGA_-Zha-QeSO8_i7HpvXyD8UkV0qc5okm11KiA,1056
|
@@ -42,11 +43,11 @@ geoseeq/contrib/ncbi/cli.py,sha256=j9zEcaZPTryK3a4xluRxigcJKDhRpRxbp3KZSx-Bfhk,2
|
|
42
43
|
geoseeq/contrib/ncbi/setup_logging.py,sha256=Tp1bY1U0f-o739aHpvVYriG2qdd1lFvCYBXZeXQgt-w,175
|
43
44
|
geoseeq/result/__init__.py,sha256=IFHIyRV8ZzuKIfwfze1SXgcKwNMcSgMAknLHMkwjXIU,356
|
44
45
|
geoseeq/result/bioinfo.py,sha256=QQtbyogrdro9avJSN0713sxLVnVeA24mFw3hWtKDKyw,1782
|
45
|
-
geoseeq/result/file_download.py,sha256=
|
46
|
+
geoseeq/result/file_download.py,sha256=XoWht-Al0HkvQxkDnU7IKqtisNBTrseDIV9IPwuDcuY,3617
|
46
47
|
geoseeq/result/file_upload.py,sha256=6HhsxeTsMHMrXjRdhL5k8bAMIxx7fBYQ6oesFKOcMqs,6424
|
47
48
|
geoseeq/result/result_file.py,sha256=YMCAivUXgHUNEhH6yTGCHJh9gXHd8OLjjbnccdvW7iw,7138
|
48
49
|
geoseeq/result/result_folder.py,sha256=LhCD75837cip3Sjv4thl-Kqv3oyLJI-NVRPwqThktvM,7660
|
49
|
-
geoseeq/result/utils.py,sha256=
|
50
|
+
geoseeq/result/utils.py,sha256=C-CxGzB3WddlnRiqFSkrY78I_m0yFgNqsTBRzGU-y8Q,2772
|
50
51
|
geoseeq/vc/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
51
52
|
geoseeq/vc/checksum.py,sha256=y8rh1asUZNbE_NLiFO0-9hImLNiTOc2YXQBRKORWK7k,710
|
52
53
|
geoseeq/vc/cli.py,sha256=VlAvBoXckENKAvREZ5nKXl2F6Z6Glh_mKBDiSfaT_lQ,2853
|
@@ -59,9 +60,9 @@ geoseeq/vc/vc_stub.py,sha256=BNFr7CZtpvLPe0MOqgB0zYL037vRCSwd2A_Fcy9UTKA,3160
|
|
59
60
|
tests/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
|
60
61
|
tests/test_api_client.py,sha256=JafItPjtL8QnhZs7Vcl5bg4E6N5rLDYMqbrKaWwGjWs,14444
|
61
62
|
tests/test_work_orders.py,sha256=CB6QbGGMNeEePZiOVioiSQP2dQsyMnsy4L31AQMRs3A,498
|
62
|
-
geoseeq-0.2.
|
63
|
-
geoseeq-0.2.
|
64
|
-
geoseeq-0.2.
|
65
|
-
geoseeq-0.2.
|
66
|
-
geoseeq-0.2.
|
67
|
-
geoseeq-0.2.
|
63
|
+
geoseeq-0.2.3.dist-info/LICENSE,sha256=IuhIl1XCxXLPLJT_coN1CNqQU4Khlq7x4IdW7ioOJD8,1067
|
64
|
+
geoseeq-0.2.3.dist-info/METADATA,sha256=f5sqGnpDcExJyNFt6DoEJ7BXjp32M125L91Y3kojrow,454
|
65
|
+
geoseeq-0.2.3.dist-info/WHEEL,sha256=5sUXSg9e4bi7lTLOHcm6QEYwO5TIF1TNbTSVFVjcJcc,92
|
66
|
+
geoseeq-0.2.3.dist-info/entry_points.txt,sha256=yF-6KDM8zXib4Al0qn49TX-qM7PUkWUIcYtsgt36rjM,45
|
67
|
+
geoseeq-0.2.3.dist-info/top_level.txt,sha256=zZk7mmeaqAYqFJG8nq2DTgSQPbflRjJwkDIhNURPDEU,14
|
68
|
+
geoseeq-0.2.3.dist-info/RECORD,,
|
File without changes
|
File without changes
|
File without changes
|
File without changes
|