forked from MouseLand/suite2p
-
Notifications
You must be signed in to change notification settings - Fork 0
/
conftest.py
100 lines (92 loc) · 3.66 KB
/
conftest.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
import pytest
import suite2p
import os
import zipfile
import tempfile
import shutil
from tqdm import tqdm
from pathlib import Path
from urllib.request import urlopen
from tenacity import retry
@pytest.fixture()
def data_dir():
"""
Initializes data input directory for tests. Downloads test_inputs and test_outputs if not present.
"""
data_path = Path('data/')
data_path.mkdir(exist_ok=True)
download_cached_inputs(data_path)
cached_outputs = data_path.joinpath('test_outputs')
cached_outputs_url = 'https://www.suite2p.org/static/test_data/test_outputs.zip'
if not os.path.exists(cached_outputs):
extract_zip(data_path.joinpath('test_outputs.zip'), cached_outputs_url, data_path)
return data_path
@pytest.fixture()
def test_ops(tmpdir, data_dir):
"""Initializes ops to be used for tests. Also, uses tmpdir fixture to create a unique temporary dir for each test."""
return initialize_ops(tmpdir, data_dir)
def download_cached_inputs(data_path):
""" Downloads test_input data if not present on machine. This function was created so it can also be used by scripts/generate_test_data.py."""
cached_inputs = data_path.joinpath('test_inputs')
cached_inputs_url = 'https://www.suite2p.org/static/test_data/test_inputs.zip'
if not os.path.exists(cached_inputs):
extract_zip(data_path.joinpath('test_inputs.zip'), cached_inputs_url, data_path)
def initialize_ops(tmpdir, data_dir):
"""Initializes ops. Used for both the test_ops function above and for generate_test_data script. This function was made to accomodate creation of ops for both pytest and non-pytest settings."""
ops = suite2p.default_ops()
ops.update(
{
'use_builtin_classifier': True,
'data_path': [Path(data_dir).joinpath('test_inputs')],
'save_path0': str(tmpdir),
'norm_frames': False,
'denoise': False,
'soma_crop': False
}
)
return ops
def extract_zip(cached_file, url, data_path):
download_url_to_file(url, cached_file)
with zipfile.ZipFile(cached_file,"r") as zip_ref:
zip_ref.extractall(data_path)
@retry
def download_url_to_file(url, dst, progress=True):
r"""Download object at the given URL to a local path.
Thanks to torch, slightly modified
Args:
url (string): URL of the object to download
dst (string): Full path where object will be saved, e.g. `/tmp/temporary_file`
progress (bool, optional): whether or not to display a progress bar to stderr
Default: True
"""
file_size = None
import ssl
ssl._create_default_https_context = ssl._create_unverified_context
u = urlopen(url)
meta = u.info()
if hasattr(meta, 'getheaders'):
content_length = meta.getheaders("Content-Length")
else:
content_length = meta.get_all("Content-Length")
if content_length is not None and len(content_length) > 0:
file_size = int(content_length[0])
# We deliberately save it in a temp file and move it after
dst = os.path.expanduser(dst)
dst_dir = os.path.dirname(dst)
f = tempfile.NamedTemporaryFile(delete=False, dir=dst_dir)
print(f"\nDownloading: {url}")
try:
with tqdm(total=file_size, disable=not progress,
unit='B', unit_scale=True, unit_divisor=1024) as pbar:
while True:
buffer = u.read(8192)
if len(buffer) == 0:
break
f.write(buffer)
pbar.update(len(buffer))
f.close()
shutil.move(f.name, dst)
finally:
f.close()
if os.path.exists(f.name):
os.remove(f.name)