]> git.proxmox.com Git - ceph.git/blame - ceph/src/pybind/mgr/cephadm/tests/fixtures.py
import ceph quincy 17.2.4
[ceph.git] / ceph / src / pybind / mgr / cephadm / tests / fixtures.py
CommitLineData
9f95a23c 1import fnmatch
20effc67
TL
2import asyncio
3import sys
4from tempfile import NamedTemporaryFile
f6b5b4d7
TL
5from contextlib import contextmanager
6
7from ceph.deployment.service_spec import PlacementSpec, ServiceSpec
adb31ebb 8from ceph.utils import datetime_to_str, datetime_now
20effc67 9from cephadm.serve import CephadmServe, cephadmNoImage
f6b5b4d7 10
9f95a23c 11try:
20effc67 12 from typing import Any, Iterator, List, Callable, Dict
9f95a23c
TL
13except ImportError:
14 pass
9f95a23c
TL
15
16from cephadm import CephadmOrchestrator
20effc67 17from orchestrator import raise_if_exception, OrchResult, HostSpec, DaemonDescriptionStatus
9f95a23c
TL
18from tests import mock
19
20
20effc67
TL
21def async_side_effect(result):
22 async def side_effect(*args, **kwargs):
23 return result
24 return side_effect
25
26
9f95a23c
TL
27def get_ceph_option(_, key):
28 return __file__
29
30
33c7a0ef
TL
31def get_module_option_ex(_, module, key, default=None):
32 if module == 'prometheus':
33 if key == 'server_port':
34 return 9283
35 return None
36
37
9f95a23c 38def _run_cephadm(ret):
20effc67 39 async def foo(s, host, entity, cmd, e, **kwargs):
b3b6e05e
TL
40 if cmd == 'gather-facts':
41 return '{}', '', 0
f91f0fd5 42 return [ret], '', 0
9f95a23c
TL
43 return foo
44
45
46def match_glob(val, pat):
47 ok = fnmatch.fnmatchcase(val, pat)
48 if not ok:
49 assert pat in val
50
51
20effc67
TL
52class MockEventLoopThread:
53 def get_result(self, coro):
54 if sys.version_info >= (3, 7):
55 return asyncio.run(coro)
56
57 loop = asyncio.new_event_loop()
58 asyncio.set_event_loop(loop)
59 try:
60 return loop.run_until_complete(coro)
61 finally:
62 loop.close()
63 asyncio.set_event_loop(None)
64
65
66def receive_agent_metadata(m: CephadmOrchestrator, host: str, ops: List[str] = None) -> None:
67 to_update: Dict[str, Callable[[str, Any], None]] = {
68 'ls': m._process_ls_output,
69 'gather-facts': m.cache.update_host_facts,
70 'list-networks': m.cache.update_host_networks,
71 }
72 if ops:
73 for op in ops:
74 out = m.wait_async(CephadmServe(m)._run_cephadm_json(host, cephadmNoImage, op, []))
75 to_update[op](host, out)
76 m.cache.last_daemon_update[host] = datetime_now()
77 m.cache.last_facts_update[host] = datetime_now()
78 m.cache.last_network_update[host] = datetime_now()
79 m.cache.metadata_up_to_date[host] = True
80
81
82def receive_agent_metadata_all_hosts(m: CephadmOrchestrator) -> None:
83 for host in m.cache.get_hosts():
84 receive_agent_metadata(m, host)
85
86
f6b5b4d7
TL
87@contextmanager
88def with_cephadm_module(module_options=None, store=None):
89 """
90 :param module_options: Set opts as if they were set before module.__init__ is called
91 :param store: Set the store before module.__init__ is called
92 """
9f95a23c 93 with mock.patch("cephadm.module.CephadmOrchestrator.get_ceph_option", get_ceph_option),\
f6b5b4d7 94 mock.patch("cephadm.services.osd.RemoveUtil._run_mon_cmd"), \
33c7a0ef 95 mock.patch('cephadm.module.CephadmOrchestrator.get_module_option_ex', get_module_option_ex),\
f6b5b4d7 96 mock.patch("cephadm.module.CephadmOrchestrator.get_osdmap"), \
20effc67
TL
97 mock.patch("cephadm.module.CephadmOrchestrator.remote"), \
98 mock.patch("cephadm.agent.CephadmAgentHelpers._request_agent_acks"), \
99 mock.patch("cephadm.agent.CephadmAgentHelpers._apply_agent", return_value=False), \
100 mock.patch("cephadm.agent.CephadmAgentHelpers._agent_down", return_value=False), \
33c7a0ef 101 mock.patch('cephadm.agent.CherryPyThread.run'), \
2a845540
TL
102 mock.patch('cephadm.offline_watcher.OfflineHostWatcher.run'), \
103 mock.patch('cephadm.tuned_profiles.TunedProfileUtils._remove_stray_tuned_profiles'):
9f95a23c 104
f91f0fd5 105 m = CephadmOrchestrator.__new__(CephadmOrchestrator)
f6b5b4d7
TL
106 if module_options is not None:
107 for k, v in module_options.items():
108 m._ceph_set_module_option('cephadm', k, v)
109 if store is None:
110 store = {}
111 if '_ceph_get/mon_map' not in store:
112 m.mock_store_set('_ceph_get', 'mon_map', {
adb31ebb 113 'modified': datetime_to_str(datetime_now()),
f6b5b4d7
TL
114 'fsid': 'foobar',
115 })
a4b75251
TL
116 if '_ceph_get/mgr_map' not in store:
117 m.mock_store_set('_ceph_get', 'mgr_map', {
118 'services': {
119 'dashboard': 'http://[::1]:8080',
120 'prometheus': 'http://[::1]:8081'
121 },
122 'modules': ['dashboard', 'prometheus'],
123 })
f6b5b4d7
TL
124 for k, v in store.items():
125 m._ceph_set_store(k, v)
126
9f95a23c
TL
127 m.__init__('cephadm', 0, 0)
128 m._cluster_fsid = "fsid"
20effc67
TL
129
130 m.event_loop = MockEventLoopThread()
131 m.tkey = NamedTemporaryFile(prefix='test-cephadm-identity-')
132
9f95a23c
TL
133 yield m
134
135
136def wait(m, c):
f67539c2
TL
137 # type: (CephadmOrchestrator, OrchResult) -> Any
138 return raise_if_exception(c)
f6b5b4d7
TL
139
140
141@contextmanager
20effc67 142def with_host(m: CephadmOrchestrator, name, addr='1::4', refresh_hosts=True, rm_with_force=True):
b3b6e05e
TL
143 with mock.patch("cephadm.utils.resolve_ip", return_value=addr):
144 wait(m, m.add_host(HostSpec(hostname=name)))
145 if refresh_hosts:
146 CephadmServe(m)._refresh_hosts_and_daemons()
20effc67 147 receive_agent_metadata(m, name)
b3b6e05e 148 yield
20effc67 149 wait(m, m.remove_host(name, force=rm_with_force))
f6b5b4d7
TL
150
151
f67539c2
TL
152def assert_rm_service(cephadm: CephadmOrchestrator, srv_name):
153 mon_or_mgr = cephadm.spec_store[srv_name].spec.service_type in ('mon', 'mgr')
154 if mon_or_mgr:
155 assert 'Unable' in wait(cephadm, cephadm.remove_service(srv_name))
156 return
f6b5b4d7 157 assert wait(cephadm, cephadm.remove_service(srv_name)) == f'Removed service {srv_name}'
f67539c2
TL
158 assert cephadm.spec_store[srv_name].deleted is not None
159 CephadmServe(cephadm)._check_daemons()
f91f0fd5 160 CephadmServe(cephadm)._apply_all_services()
f67539c2
TL
161 assert cephadm.spec_store[srv_name].deleted
162 unmanaged = cephadm.spec_store[srv_name].spec.unmanaged
163 CephadmServe(cephadm)._purge_deleted_services()
164 if not unmanaged: # cause then we're not deleting daemons
165 assert srv_name not in cephadm.spec_store, f'{cephadm.spec_store[srv_name]!r}'
f6b5b4d7
TL
166
167
168@contextmanager
20effc67 169def with_service(cephadm_module: CephadmOrchestrator, spec: ServiceSpec, meth=None, host: str = '', status_running=False) -> Iterator[List[str]]:
f67539c2 170 if spec.placement.is_empty() and host:
f6b5b4d7 171 spec.placement = PlacementSpec(hosts=[host], count=1)
f67539c2
TL
172 if meth is not None:
173 c = meth(cephadm_module, spec)
174 assert wait(cephadm_module, c) == f'Scheduled {spec.service_name()} update...'
175 else:
176 c = cephadm_module.apply([spec])
177 assert wait(cephadm_module, c) == [f'Scheduled {spec.service_name()} update...']
178
f6b5b4d7
TL
179 specs = [d.spec for d in wait(cephadm_module, cephadm_module.describe_service())]
180 assert spec in specs
181
f91f0fd5 182 CephadmServe(cephadm_module)._apply_all_services()
f6b5b4d7 183
20effc67
TL
184 if status_running:
185 make_daemons_running(cephadm_module, spec.service_name())
186
f6b5b4d7 187 dds = wait(cephadm_module, cephadm_module.list_daemons())
f91f0fd5 188 own_dds = [dd for dd in dds if dd.service_name() == spec.service_name()]
20effc67 189 if host and spec.service_type != 'osd':
f67539c2 190 assert own_dds
f6b5b4d7 191
f91f0fd5 192 yield [dd.name() for dd in own_dds]
f6b5b4d7 193
f91f0fd5 194 assert_rm_service(cephadm_module, spec.service_name())
f67539c2
TL
195
196
20effc67
TL
197def make_daemons_running(cephadm_module, service_name):
198 own_dds = cephadm_module.cache.get_daemons_by_service(service_name)
199 for dd in own_dds:
200 dd.status = DaemonDescriptionStatus.running # We're changing the reference