summaryrefslogtreecommitdiffstats
path: root/qa/tasks/rbd_pwl_cache_recovery.py
blob: e13c1f66447663c90e3b870953f685da770224af (plain)
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
"""
persistent write log cache recovery task
"""
import contextlib
import logging
import random
import json
import time

from teuthology import misc as teuthology
from teuthology import contextutil

DEFAULT_NUM_ITERATIONS = 20
IO_PATTERNS = ("full-seq", "rand")
IO_SIZES = ('4K', '16K', '128K', '1024K')

log = logging.getLogger(__name__)

@contextlib.contextmanager
def thrashes_rbd_bench_on_persistent_cache(ctx, config):
    """
    thrashes rbd bench on persistent write log cache.
    It can test recovery feature of persistent write log cache.
    """
    log.info("thrashes rbd bench on persistent write log cache")

    client, client_config = list(config.items())[0]
    (remote,) = ctx.cluster.only(client).remotes.keys()
    client_config = client_config if client_config is not None else dict()
    image_name = client_config.get('image_name', 'testimage')
    num_iterations = client_config.get('num_iterations', DEFAULT_NUM_ITERATIONS)

    for i in range(num_iterations):
        log.info("start rbd bench")
        # rbd bench could not specify the run time so set a large enough test size.
        remote.run(
            args=[
                'rbd', 'bench',
                '--io-type', 'write',
                '--io-pattern', random.choice(IO_PATTERNS),
                '--io-size', random.choice(IO_SIZES),
                '--io-total', '100G',
                image_name,
                ],
            wait=False,
        )
        # Wait a few seconds for the rbd bench process to run
        # and complete the pwl cache initialization
        time.sleep(10)
        log.info("dump cache state when rbd bench running.")
        remote.sh(['rbd', 'status', image_name, '--format=json'])
        log.info("sleep...")
        time.sleep(random.randint(10, 60))
        log.info("rbd bench crash.")
        remote.run(
            args=[
                'killall', '-9', 'rbd',
                ],
            check_status=False,
        )
        log.info("wait for watch timeout.")
        time.sleep(40)
        log.info("check cache state after crash.")
        out = remote.sh(['rbd', 'status', image_name, '--format=json'])
        rbd_status = json.loads(out)
        assert len(rbd_status['watchers']) == 0
        assert rbd_status['persistent_cache']['present'] == True
        assert rbd_status['persistent_cache']['empty'] == False
        assert rbd_status['persistent_cache']['clean'] == False
        log.info("check dirty cache file.")
        remote.run(
            args=[
                'test', '-e', rbd_status['persistent_cache']['path'],
                ]
        )
    try:
        yield
    finally:
        log.info("cleanup")

@contextlib.contextmanager
def task(ctx, config):
    """
    This is task for testing persistent write log cache recovery.
    """
    assert isinstance(config, dict), \
            "task rbd_pwl_cache_recovery only supports a dictionary for configuration"

    managers = []
    config = teuthology.replace_all_with_clients(ctx.cluster, config)
    managers.append(
        lambda: thrashes_rbd_bench_on_persistent_cache(ctx=ctx, config=config)
        )

    with contextutil.nested(*managers):
        yield