forked from wamdam/backy2
-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathsmoketest.py
executable file
·168 lines (143 loc) · 5.38 KB
/
smoketest.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
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
#!/usr/bin/env python3
import os
import shutil
import random
import json
import hashlib
import math
from backy2.scripts.backy import hints_from_rbd_diff
from backy2.logging import init_logging
from backy2.utils import backy_from_config
from backy2.config import Config as _Config
from functools import partial
import logging
kB = 1024
MB = kB * 1024
GB = MB * 1024
HASH_FUNCTION = hashlib.sha512
def patch(path, filename, offset, data=None):
""" write data into a file at offset """
filename = os.path.join(path, filename)
if not os.path.exists(filename):
open(filename, 'wb')
with open(filename, 'r+b') as f:
f.seek(offset)
f.write(data)
def same(f1, f2):
""" returns False if files differ, True if they are the same """
d1 = open(f1, 'rb').read()
d2 = open(f1, 'rb').read()
return d1 == d2
class TestPath():
path = '_smoketest'
def __enter__(self):
os.mkdir(self.path)
return self.path
def __exit__(self, type, value, traceback):
shutil.rmtree(self.path)
with TestPath() as testpath:
from_version = None
init_logging(testpath+'/backy.log', logging.INFO)
version_uids = []
old_size = 0
initdb = True
for i in range(100):
print('Run {}'.format(i+1))
hints = []
if old_size and random.randint(0, 10) == 0: # every 10th time or so do not apply any changes.
size = old_size
else:
size = 32*4*kB + random.randint(-4*kB, 4*kB)
if size > old_size:
hints.append({'offset': old_size, 'length': size-old_size, 'exists': 'true'})
if size < old_size:
last_block_offset = math.floor(size/(4*kB))*4*kB
hints.append({'offset': last_block_offset, 'length': size-last_block_offset, 'exists': 'true'})
old_size = size
for j in range(random.randint(0, 10)): # up to 10 changes
if random.randint(0, 1):
patch_size = random.randint(0, 4*kB)
data = os.urandom(patch_size)
#exists = True
exists = "true"
else:
patch_size = random.randint(0, 4*4*kB) # we want full blocks sometimes
data = b'\0' * patch_size
#exists = False
exists = "false"
offset = random.randint(0, size-1-patch_size)
print(' Applied change at {}:{}, exists {}'.format(offset, patch_size, exists))
patch(testpath, 'data', offset, data)
hints.append({'offset': offset, 'length': patch_size, 'exists': exists})
# truncate?
open(os.path.join(testpath, 'data'), 'r+b').truncate(size)
print(' Applied {} changes, size is {}.'.format(len(hints), size))
open(os.path.join(testpath, 'hints'), 'w').write(json.dumps(hints))
# create backy
config = """
[DEFAULTS]
logfile: /var/log/backy.log
block_size: 4096
hash_function: sha512
lock_dir: /tmp
process_name: backy2
encryption_key: decafbaddecafbaddecafbaddecafbaddecafbaddecafbaddecafbaddecafbad
encryption_version: 1
deduplication: 0
[MetaBackend]
type: backy2.meta_backends.sql
engine: sqlite:///{testpath}/backy.sqlite
[DataBackend]
type: backy2.data_backends.file
path: {testpath}
simultaneous_writes: 5
bandwidth_read: 100000
bandwidth_write: 100000
[NBD]
cachedir: /tmp
[io_file]
simultaneous_reads: 5
[io_rbd]
ceph_conffile: /etc/ceph/ceph.conf
simultaneous_reads: 10
""".format(testpath=testpath)
Config = partial(_Config, cfg=config)
backy = backy_from_config(Config)(initdb=initdb)
initdb = False
version_uid = backy.backup(
'data-backup',
'snapshot-name',
'file://'+os.path.join(testpath, 'data'),
from_version and hints_from_rbd_diff(open(os.path.join(testpath, 'hints')).read()) or None,
from_version
)
backy.close()
version_uids.append(version_uid)
try:
backy = backy_from_config(Config)()
assert backy.scrub(version_uid) == True
backy.close()
print(' Scrub successful')
backy = backy_from_config(Config)()
assert backy.scrub(version_uid, 'file://'+os.path.join(testpath, 'data')) == True
backy.close()
print(' Deep scrub successful')
backy = backy_from_config(Config)()
backy.restore(version_uid, 'file://'+os.path.join(testpath, 'restore'), sparse=False, force=False)
backy.close()
assert same(os.path.join(testpath, 'data'), os.path.join(testpath, 'restore')) == True
os.unlink(os.path.join(testpath, 'restore'))
print(' Restore successful')
except AssertionError:
import pdb; pdb.set_trace()
from_version = version_uid
# delete old versions
if len(version_uids) > 10:
backy = backy_from_config(Config)()
backy.rm(version_uids.pop(0))
backy.close()
if (i%7) == 0:
backy = backy_from_config(Config)()
backy.cleanup_fast(dt=0)
backy.close()
#import pdb; pdb.set_trace()