Add Gnocchi metric scenarios
This patch is based on [1] and adds following scenarios: GnocchiMetric.list_metric GnocchiMetric.create_metric GnocchiMetric.create_delete_metric [1] https://review.openstack.org/#/c/453861/ Change-Id: Ia391b89fb7c57dbf30d4ed1756ddca184b1e592f Signed-off-by: Juha Kosonen <juha.kosonen@nokia.com>
This commit is contained in:
parent
f24365c646
commit
367b14102f
@ -16,6 +16,16 @@ Changelog
|
|||||||
.. Release notes for existing releases are MUTABLE! If there is something that
|
.. Release notes for existing releases are MUTABLE! If there is something that
|
||||||
was missed or can be improved, feel free to change it!
|
was missed or can be improved, feel free to change it!
|
||||||
|
|
||||||
|
[Unreleased]
|
||||||
|
------------
|
||||||
|
|
||||||
|
Added
|
||||||
|
~~~~~
|
||||||
|
|
||||||
|
* [scenario plugin] GnocchiMetric.list_metric
|
||||||
|
* [scenario plugin] GnocchiMetric.create_metric
|
||||||
|
* [scenario plugin] GnocchiMetric.create_delete_metric
|
||||||
|
|
||||||
[1.0.0] - 2018-03-28
|
[1.0.0] - 2018-03-28
|
||||||
--------------------
|
--------------------
|
||||||
Start a fork of `rally/plugins/openstack module of original OpenStack Rally
|
Start a fork of `rally/plugins/openstack module of original OpenStack Rally
|
||||||
|
@ -156,3 +156,51 @@
|
|||||||
sla:
|
sla:
|
||||||
failure_rate:
|
failure_rate:
|
||||||
max: 0
|
max: 0
|
||||||
|
|
||||||
|
GnocchiMetric.list_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
limit: 10000
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
||||||
|
|
||||||
|
GnocchiMetric.create_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
archive_policy_name: "low"
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
||||||
|
|
||||||
|
GnocchiMetric.create_delete_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
archive_policy_name: "low"
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
||||||
|
@ -876,6 +876,29 @@ class GnocchiResourceType(GnocchiMixin):
|
|||||||
pass
|
pass
|
||||||
|
|
||||||
|
|
||||||
|
@base.resource("gnocchi", "metric", order=next(_gnocchi_order),
|
||||||
|
tenant_resource=True)
|
||||||
|
class GnocchiMetric(GnocchiMixin):
|
||||||
|
|
||||||
|
def id(self):
|
||||||
|
return self.raw_resource["id"]
|
||||||
|
|
||||||
|
def list(self):
|
||||||
|
result = []
|
||||||
|
marker = None
|
||||||
|
while True:
|
||||||
|
metrics = self._manager().list(marker=marker)
|
||||||
|
if not metrics:
|
||||||
|
break
|
||||||
|
result.extend(metrics)
|
||||||
|
marker = metrics[-1]["id"]
|
||||||
|
if self.tenant_uuid:
|
||||||
|
result = [r for r in result
|
||||||
|
if r["creator"].partition(":")[2] == self.tenant_uuid]
|
||||||
|
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
# WATCHER
|
# WATCHER
|
||||||
|
|
||||||
_watcher_order = get_order(1500)
|
_watcher_order = get_order(1500)
|
||||||
|
73
rally_openstack/scenarios/gnocchi/metric.py
Normal file
73
rally_openstack/scenarios/gnocchi/metric.py
Normal file
@ -0,0 +1,73 @@
|
|||||||
|
# Copyright 2017 Red Hat, Inc. <http://www.redhat.com>
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
from rally import consts
|
||||||
|
from rally.task import validation
|
||||||
|
|
||||||
|
from rally_openstack import scenario
|
||||||
|
from rally_openstack.scenarios.gnocchi import utils as gnocchiutils
|
||||||
|
|
||||||
|
"""Scenarios for Gnocchi metric."""
|
||||||
|
|
||||||
|
|
||||||
|
@validation.add("required_services", services=[consts.Service.GNOCCHI])
|
||||||
|
@validation.add("required_platform", platform="openstack", users=True)
|
||||||
|
@scenario.configure(name="GnocchiMetric.list_metric")
|
||||||
|
class ListMetric(gnocchiutils.GnocchiBase):
|
||||||
|
|
||||||
|
def run(self, limit=None):
|
||||||
|
"""List metrics.
|
||||||
|
|
||||||
|
:param limit: Maximum number of metrics to list
|
||||||
|
"""
|
||||||
|
self.gnocchi.list_metric(limit=limit)
|
||||||
|
|
||||||
|
|
||||||
|
@validation.add("required_services", services=[consts.Service.GNOCCHI])
|
||||||
|
@validation.add("required_platform", platform="openstack", users=True)
|
||||||
|
@scenario.configure(context={"cleanup@openstack": ["gnocchi.metric"]},
|
||||||
|
name="GnocchiMetric.create_metric")
|
||||||
|
class CreateMetric(gnocchiutils.GnocchiBase):
|
||||||
|
|
||||||
|
def run(self, archive_policy_name="low", resource_id=None, unit=None):
|
||||||
|
"""Create metric.
|
||||||
|
|
||||||
|
:param archive_policy_name: Archive policy name
|
||||||
|
:param resource_id: The resource ID to attach the metric to
|
||||||
|
:param unit: The unit of the metric
|
||||||
|
"""
|
||||||
|
name = self.generate_random_name()
|
||||||
|
self.gnocchi.create_metric(name,
|
||||||
|
archive_policy_name=archive_policy_name,
|
||||||
|
resource_id=resource_id, unit=unit)
|
||||||
|
|
||||||
|
|
||||||
|
@validation.add("required_services", services=[consts.Service.GNOCCHI])
|
||||||
|
@validation.add("required_platform", platform="openstack", users=True)
|
||||||
|
@scenario.configure(context={"cleanup@openstack": ["gnocchi.metric"]},
|
||||||
|
name="GnocchiMetric.create_delete_metric")
|
||||||
|
class CreateDeleteMetric(gnocchiutils.GnocchiBase):
|
||||||
|
|
||||||
|
def run(self, archive_policy_name="low", resource_id=None, unit=None):
|
||||||
|
"""Create metric and then delete it.
|
||||||
|
|
||||||
|
:param archive_policy_name: Archive policy name
|
||||||
|
:param resource_id: The resource ID to attach the metric to
|
||||||
|
:param unit: The unit of the metric
|
||||||
|
"""
|
||||||
|
name = self.generate_random_name()
|
||||||
|
metric = self.gnocchi.create_metric(
|
||||||
|
name, archive_policy_name=archive_policy_name,
|
||||||
|
resource_id=resource_id, unit=unit)
|
||||||
|
self.gnocchi.delete_metric(metric["id"])
|
@ -128,9 +128,26 @@ class GnocchiService(service.Service):
|
|||||||
return self._clients.gnocchi().metric.delete(metric_id)
|
return self._clients.gnocchi().metric.delete(metric_id)
|
||||||
|
|
||||||
@atomic.action_timer("gnocchi.list_metric")
|
@atomic.action_timer("gnocchi.list_metric")
|
||||||
def list_metric(self):
|
def list_metric(self, limit=None):
|
||||||
"""List metrics."""
|
"""List metrics."""
|
||||||
return self._clients.gnocchi().metric.list()
|
metrics = []
|
||||||
|
marker = None
|
||||||
|
limit_val = limit
|
||||||
|
while True:
|
||||||
|
page = self._clients.gnocchi().metric.list(limit=limit_val,
|
||||||
|
marker=marker)
|
||||||
|
if not page:
|
||||||
|
break
|
||||||
|
metrics.extend(page)
|
||||||
|
marker = page[-1]["id"]
|
||||||
|
if limit_val is not None:
|
||||||
|
cnt = len(metrics)
|
||||||
|
if cnt < limit:
|
||||||
|
limit_val = limit - cnt
|
||||||
|
else:
|
||||||
|
break
|
||||||
|
|
||||||
|
return metrics
|
||||||
|
|
||||||
@atomic.action_timer("gnocchi.create_resource")
|
@atomic.action_timer("gnocchi.create_resource")
|
||||||
def create_resource(self, resource_type="generic"):
|
def create_resource(self, resource_type="generic"):
|
||||||
|
25
samples/tasks/scenarios/gnocchi/create-delete-metric.json
Normal file
25
samples/tasks/scenarios/gnocchi/create-delete-metric.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"GnocchiMetric.create_delete_metric": [
|
||||||
|
{
|
||||||
|
"args": {
|
||||||
|
"archive_policy_name": "low"
|
||||||
|
},
|
||||||
|
"runner": {
|
||||||
|
"type": "constant",
|
||||||
|
"times": 10,
|
||||||
|
"concurrency": 2
|
||||||
|
},
|
||||||
|
"context": {
|
||||||
|
"users": {
|
||||||
|
"tenants": 2,
|
||||||
|
"users_per_tenant": 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sla": {
|
||||||
|
"failure_rate": {
|
||||||
|
"max": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
16
samples/tasks/scenarios/gnocchi/create-delete-metric.yaml
Normal file
16
samples/tasks/scenarios/gnocchi/create-delete-metric.yaml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
---
|
||||||
|
GnocchiMetric.create_delete_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
archive_policy_name: "low"
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
25
samples/tasks/scenarios/gnocchi/create-metric.json
Normal file
25
samples/tasks/scenarios/gnocchi/create-metric.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"GnocchiMetric.create_metric": [
|
||||||
|
{
|
||||||
|
"args": {
|
||||||
|
"archive_policy_name": "low"
|
||||||
|
},
|
||||||
|
"runner": {
|
||||||
|
"type": "constant",
|
||||||
|
"times": 10,
|
||||||
|
"concurrency": 2
|
||||||
|
},
|
||||||
|
"context": {
|
||||||
|
"users": {
|
||||||
|
"tenants": 2,
|
||||||
|
"users_per_tenant": 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sla": {
|
||||||
|
"failure_rate": {
|
||||||
|
"max": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
16
samples/tasks/scenarios/gnocchi/create-metric.yaml
Normal file
16
samples/tasks/scenarios/gnocchi/create-metric.yaml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
---
|
||||||
|
GnocchiMetric.create_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
archive_policy_name: "low"
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
25
samples/tasks/scenarios/gnocchi/list-metric.json
Normal file
25
samples/tasks/scenarios/gnocchi/list-metric.json
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
{
|
||||||
|
"GnocchiMetric.list_metric": [
|
||||||
|
{
|
||||||
|
"args": {
|
||||||
|
"limit": 10000
|
||||||
|
},
|
||||||
|
"runner": {
|
||||||
|
"type": "constant",
|
||||||
|
"times": 10,
|
||||||
|
"concurrency": 2
|
||||||
|
},
|
||||||
|
"context": {
|
||||||
|
"users": {
|
||||||
|
"tenants": 2,
|
||||||
|
"users_per_tenant": 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"sla": {
|
||||||
|
"failure_rate": {
|
||||||
|
"max": 0
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
16
samples/tasks/scenarios/gnocchi/list-metric.yaml
Normal file
16
samples/tasks/scenarios/gnocchi/list-metric.yaml
Normal file
@ -0,0 +1,16 @@
|
|||||||
|
---
|
||||||
|
GnocchiMetric.list_metric:
|
||||||
|
-
|
||||||
|
args:
|
||||||
|
limit: 10000
|
||||||
|
runner:
|
||||||
|
type: "constant"
|
||||||
|
times: 10
|
||||||
|
concurrency: 2
|
||||||
|
context:
|
||||||
|
users:
|
||||||
|
tenants: 2
|
||||||
|
users_per_tenant: 3
|
||||||
|
sla:
|
||||||
|
failure_rate:
|
||||||
|
max: 0
|
@ -320,12 +320,23 @@ class Gnocchi(ResourceManager):
|
|||||||
def list_archive_policy_rules(self):
|
def list_archive_policy_rules(self):
|
||||||
return self.client.archive_policy_rule.list()
|
return self.client.archive_policy_rule.list()
|
||||||
|
|
||||||
def list_archive_policy(self):
|
def list_archive_policys(self):
|
||||||
return self.client.archive_policy.list()
|
return self.client.archive_policy.list()
|
||||||
|
|
||||||
def list_resource_type(self):
|
def list_resource_types(self):
|
||||||
return self.client.resource_type.list()
|
return self.client.resource_type.list()
|
||||||
|
|
||||||
|
def list_metrics(self):
|
||||||
|
result = []
|
||||||
|
marker = None
|
||||||
|
while True:
|
||||||
|
metrics = self.client.metric.list(marker=marker)
|
||||||
|
if not metrics:
|
||||||
|
break
|
||||||
|
result.extend(metrics)
|
||||||
|
marker = metrics[-1]["id"]
|
||||||
|
return result
|
||||||
|
|
||||||
|
|
||||||
class Ironic(ResourceManager):
|
class Ironic(ResourceManager):
|
||||||
|
|
||||||
|
@ -1118,3 +1118,29 @@ class GnocchiMixinTestCase(test.TestCase):
|
|||||||
gnocchi = self.get_gnocchi()
|
gnocchi = self.get_gnocchi()
|
||||||
gnocchi.raw_resource = {"name": "test_name"}
|
gnocchi.raw_resource = {"name": "test_name"}
|
||||||
self.assertEqual("test_name", gnocchi.name())
|
self.assertEqual("test_name", gnocchi.name())
|
||||||
|
|
||||||
|
|
||||||
|
class GnocchiMetricTestCase(test.TestCase):
|
||||||
|
|
||||||
|
def get_gnocchi(self):
|
||||||
|
gnocchi = resources.GnocchiMetric()
|
||||||
|
gnocchi._service = "gnocchi"
|
||||||
|
return gnocchi
|
||||||
|
|
||||||
|
def test_id(self):
|
||||||
|
gnocchi = self.get_gnocchi()
|
||||||
|
gnocchi.raw_resource = {"id": "test_id"}
|
||||||
|
self.assertEqual("test_id", gnocchi.id())
|
||||||
|
|
||||||
|
def test_list(self):
|
||||||
|
gnocchi = self.get_gnocchi()
|
||||||
|
gnocchi._manager = mock.MagicMock()
|
||||||
|
metrics = [mock.MagicMock(), mock.MagicMock(), mock.MagicMock(),
|
||||||
|
mock.MagicMock()]
|
||||||
|
gnocchi._manager.return_value.list.side_effect = (
|
||||||
|
metrics[:2], metrics[2:4], [])
|
||||||
|
self.assertEqual(metrics, gnocchi.list())
|
||||||
|
self.assertEqual(
|
||||||
|
[mock.call(marker=None), mock.call(marker=metrics[1]["id"]),
|
||||||
|
mock.call(marker=metrics[3]["id"])],
|
||||||
|
gnocchi._manager.return_value.list.call_args_list)
|
||||||
|
66
tests/unit/scenarios/gnocchi/test_metric.py
Normal file
66
tests/unit/scenarios/gnocchi/test_metric.py
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
# Copyright 2017 Red Hat, Inc. <http://www.redhat.com>
|
||||||
|
#
|
||||||
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
|
# not use this file except in compliance with the License. You may obtain
|
||||||
|
# a copy of the License at
|
||||||
|
#
|
||||||
|
# http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
#
|
||||||
|
# Unless required by applicable law or agreed to in writing, software
|
||||||
|
# distributed under the License is distributed on an "AS IS" BASIS, WITHOUT
|
||||||
|
# WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
|
||||||
|
# License for the specific language governing permissions and limitations
|
||||||
|
# under the License.
|
||||||
|
|
||||||
|
import mock
|
||||||
|
|
||||||
|
from rally_openstack.scenarios.gnocchi import metric
|
||||||
|
from tests.unit import test
|
||||||
|
|
||||||
|
|
||||||
|
class GnocchiMetricTestCase(test.ScenarioTestCase):
|
||||||
|
|
||||||
|
def get_test_context(self):
|
||||||
|
context = super(GnocchiMetricTestCase, self).get_test_context()
|
||||||
|
context.update({
|
||||||
|
"admin": {
|
||||||
|
"user_id": "fake",
|
||||||
|
"credential": mock.MagicMock()
|
||||||
|
},
|
||||||
|
"user": {
|
||||||
|
"user_id": "fake",
|
||||||
|
"credential": mock.MagicMock()
|
||||||
|
},
|
||||||
|
"tenant": {"id": "fake"}
|
||||||
|
})
|
||||||
|
return context
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(GnocchiMetricTestCase, self).setUp()
|
||||||
|
patch = mock.patch(
|
||||||
|
"rally_openstack.services.gnocchi.metric.GnocchiService")
|
||||||
|
self.addCleanup(patch.stop)
|
||||||
|
self.mock_metric = patch.start()
|
||||||
|
|
||||||
|
def test_list_metric(self):
|
||||||
|
metric_service = self.mock_metric.return_value
|
||||||
|
scenario = metric.ListMetric(self.context)
|
||||||
|
scenario.run(limit=42)
|
||||||
|
metric_service.list_metric.assert_called_once_with(limit=42)
|
||||||
|
|
||||||
|
def test_create_metric(self):
|
||||||
|
metric_service = self.mock_metric.return_value
|
||||||
|
scenario = metric.CreateMetric(self.context)
|
||||||
|
scenario.generate_random_name = mock.MagicMock(return_value="name")
|
||||||
|
scenario.run(archive_policy_name="foo", resource_id="123", unit="u")
|
||||||
|
metric_service.create_metric.assert_called_once_with(
|
||||||
|
"name", archive_policy_name="foo", resource_id="123", unit="u")
|
||||||
|
|
||||||
|
def test_create_delete_metric(self):
|
||||||
|
metric_service = self.mock_metric.return_value
|
||||||
|
scenario = metric.CreateDeleteMetric(self.context)
|
||||||
|
scenario.generate_random_name = mock.MagicMock(return_value="name")
|
||||||
|
scenario.run(archive_policy_name="bar", resource_id="123", unit="v")
|
||||||
|
metric_service.create_metric.assert_called_once_with(
|
||||||
|
"name", archive_policy_name="bar", resource_id="123", unit="v")
|
||||||
|
self.assertEqual(1, metric_service.delete_metric.call_count)
|
@ -129,17 +129,17 @@ class GnocchiServiceTestCase(test.TestCase):
|
|||||||
"gnocchi.get_measures")
|
"gnocchi.get_measures")
|
||||||
|
|
||||||
def test__create_metric(self):
|
def test__create_metric(self):
|
||||||
metric = {"name": "fake_name"}
|
param = {"name": "fake_name"}
|
||||||
metric["archive_policy_name"] = "fake_archive_policy"
|
param["archive_policy_name"] = "fake_archive_policy"
|
||||||
metric["unit"] = "fake_unit"
|
param["unit"] = "fake_unit"
|
||||||
metric["resource_id"] = "fake_resource_id"
|
param["resource_id"] = "fake_resource_id"
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.service.create_metric(
|
self.service.create_metric(
|
||||||
name="fake_name",
|
name="fake_name",
|
||||||
archive_policy_name="fake_archive_policy",
|
archive_policy_name="fake_archive_policy",
|
||||||
unit="fake_unit",
|
unit="fake_unit",
|
||||||
resource_id="fake_resource_id"),
|
resource_id="fake_resource_id"),
|
||||||
self.service._clients.gnocchi().metric.create(metric)
|
self.service._clients.gnocchi().metric.create(param)
|
||||||
)
|
)
|
||||||
self._test_atomic_action_timer(self.atomic_actions(),
|
self._test_atomic_action_timer(self.atomic_actions(),
|
||||||
"gnocchi.create_metric")
|
"gnocchi.create_metric")
|
||||||
@ -152,10 +152,9 @@ class GnocchiServiceTestCase(test.TestCase):
|
|||||||
"gnocchi.delete_metric")
|
"gnocchi.delete_metric")
|
||||||
|
|
||||||
def test__list_metric(self):
|
def test__list_metric(self):
|
||||||
|
self.service.list_metric(limit=0)
|
||||||
self.assertEqual(
|
self.assertEqual(
|
||||||
self.service.list_metric(),
|
1, self.service._clients.gnocchi().metric.list.call_count)
|
||||||
self.service._clients.gnocchi().metric.list.return_value
|
|
||||||
)
|
|
||||||
self._test_atomic_action_timer(self.atomic_actions(),
|
self._test_atomic_action_timer(self.atomic_actions(),
|
||||||
"gnocchi.list_metric")
|
"gnocchi.list_metric")
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user