Add ThermalZone resource for RSD 2.1
Change-Id: Ia6cac7ab67029ddfb77bdc09daec50bba2928c77
This commit is contained in:
parent
a30a6b2320
commit
4e29ceb539
151
rsd_lib/resources/v2_1/chassis/thermal_zone.py
Normal file
151
rsd_lib/resources/v2_1/chassis/thermal_zone.py
Normal file
@ -0,0 +1,151 @@
|
|||||||
|
# Copyright 2018 Intel, Inc.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# 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 sushy.resources import base
|
||||||
|
|
||||||
|
from rsd_lib import utils as rsd_lib_utils
|
||||||
|
|
||||||
|
|
||||||
|
class StatusField(base.CompositeField):
|
||||||
|
state = base.Field('State')
|
||||||
|
health = base.Field('Health')
|
||||||
|
health_rollup = base.Field('HealthRollup')
|
||||||
|
|
||||||
|
|
||||||
|
class RackLocationField(base.CompositeField):
|
||||||
|
rack_units = base.Field('RackUnits')
|
||||||
|
"""Indicates the rack unit type"""
|
||||||
|
|
||||||
|
xlocation = base.Field('XLocation', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The horizontal location within uLocation, from left to right
|
||||||
|
(1.. MAXIMUM) 0 indicate not available
|
||||||
|
"""
|
||||||
|
|
||||||
|
ulocation = base.Field('ULocation', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The index of the top-most U of the component, from top to bottom
|
||||||
|
(1.. MAXIMUM) 0 indicate not available
|
||||||
|
"""
|
||||||
|
|
||||||
|
uheight = base.Field('UHeight', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The height of managed zone, e.g. 8 for 8U, 16 for 16U"""
|
||||||
|
|
||||||
|
|
||||||
|
class FansField(base.ListField):
|
||||||
|
name = base.Field('Name')
|
||||||
|
"""The Power Supply name"""
|
||||||
|
|
||||||
|
reading_rpm = base.Field('ReadingRPM', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""Fan RPM reading"""
|
||||||
|
|
||||||
|
status = StatusField('Status')
|
||||||
|
"""The Fan status"""
|
||||||
|
|
||||||
|
rack_location = RackLocationField('RackLocation')
|
||||||
|
"""The Fan physical location"""
|
||||||
|
|
||||||
|
|
||||||
|
class TemperaturesField(base.ListField):
|
||||||
|
name = base.Field('Name')
|
||||||
|
"""The Power Supply name"""
|
||||||
|
|
||||||
|
reading_celsius = base.Field(
|
||||||
|
'ReadingCelsius', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""Current value of the temperature sensor's reading"""
|
||||||
|
|
||||||
|
physical_context = base.Field('PhysicalContext')
|
||||||
|
"""Describes the area or device to which this temperature measurement
|
||||||
|
applies:
|
||||||
|
"Intake" - The intake point of the chassis
|
||||||
|
"Exhaust" - The exhaust point of the chassis
|
||||||
|
"Backplane" - A backplane within the chassis
|
||||||
|
"PowerSupply" - A power supply
|
||||||
|
"SystemBoard" - The system board (PCB)
|
||||||
|
"ComputeBay" - Within a compute bay
|
||||||
|
"PowerSupplyBay" - Within a power supply bay
|
||||||
|
"""
|
||||||
|
|
||||||
|
status = StatusField('Status')
|
||||||
|
"""The temperature sensors status"""
|
||||||
|
|
||||||
|
|
||||||
|
class ThermalZone(base.ResourceBase):
|
||||||
|
identity = base.Field('Id', required=True)
|
||||||
|
"""The ThermalZone identity string"""
|
||||||
|
|
||||||
|
name = base.Field('Name')
|
||||||
|
"""The ThermalZone name"""
|
||||||
|
|
||||||
|
description = base.Field('Description')
|
||||||
|
"""The ThermalZone description"""
|
||||||
|
|
||||||
|
status = StatusField('Status')
|
||||||
|
"""The ThermalZone status"""
|
||||||
|
|
||||||
|
rack_location = RackLocationField('RackLocation')
|
||||||
|
"""The ThermalZone physical location"""
|
||||||
|
|
||||||
|
presence = base.Field('Presence')
|
||||||
|
"""Indicates the aggregated Power Supply Unit presence information
|
||||||
|
Aggregated Power Supply Unit presence format: Length of string indicate
|
||||||
|
total slot of Power Supply Units in PowerZone.
|
||||||
|
|
||||||
|
For each byte the string:
|
||||||
|
"1" means present
|
||||||
|
"0" means not present
|
||||||
|
"""
|
||||||
|
|
||||||
|
desired_speed_pwm = base.Field(
|
||||||
|
'DesiredSpeedPWM', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The desired FAN speed in current ThermalZone present in PWM unit"""
|
||||||
|
|
||||||
|
desired_speed_rpm = base.Field(
|
||||||
|
'DesiredSpeedRPM', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The desired FAN speed in current ThermalZone present in RPM unit"""
|
||||||
|
|
||||||
|
max_fans_supported = base.Field(
|
||||||
|
'MaxFansSupported', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""Number of maximum fans that can be installed in a given Thermal Zone"""
|
||||||
|
|
||||||
|
number_of_fans_present = base.Field(
|
||||||
|
'NumberOfFansPresent', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""The existing number of fans in current ThermalZone"""
|
||||||
|
|
||||||
|
volumetric_airflow = base.Field(
|
||||||
|
'VolumetricAirflow', adapter=rsd_lib_utils.int_or_none)
|
||||||
|
"""Rack Level PTAS Telemetry - Volumetric airflow in current ThermalZone"""
|
||||||
|
|
||||||
|
fans = FansField('Fans')
|
||||||
|
"""Details of the fans associated with this thermal zone"""
|
||||||
|
|
||||||
|
temperatures = TemperaturesField('Temperatures')
|
||||||
|
"""Array of temperature sensors"""
|
||||||
|
|
||||||
|
|
||||||
|
class ThermalZoneCollection(base.ResourceCollectionBase):
|
||||||
|
@property
|
||||||
|
def _resource_type(self):
|
||||||
|
return ThermalZone
|
||||||
|
|
||||||
|
def __init__(self, connector, path, redfish_version=None):
|
||||||
|
"""A class representing a ThermalZone Collection
|
||||||
|
|
||||||
|
:param connector: A Connector instance
|
||||||
|
:param path: The canonical path to the power zone collection resource
|
||||||
|
:param redfish_version: The version of RedFish. Used to construct
|
||||||
|
the object according to schema of the given version.
|
||||||
|
"""
|
||||||
|
super(ThermalZoneCollection, self).__init__(connector,
|
||||||
|
path,
|
||||||
|
redfish_version)
|
66
rsd_lib/tests/unit/json_samples/v2_1/thermal_zone.json
Normal file
66
rsd_lib/tests/unit/json_samples/v2_1/thermal_zone.json
Normal file
@ -0,0 +1,66 @@
|
|||||||
|
{
|
||||||
|
"@odata.context": "/redfish/v1/$metadata#Chassis/Rack/ThermalZones/Members/$entity",
|
||||||
|
"@odata.type": "ThermalZone.v1_0_0.ThermalZone",
|
||||||
|
"@odata.id": "/redfish/v1/Chassis/Rack1/ThermalZones/1",
|
||||||
|
"Id": "1",
|
||||||
|
"Name": "thermal zone 1",
|
||||||
|
"Description": "thermal zone 1 description",
|
||||||
|
"RackLocation": {
|
||||||
|
"RackUnits": "OU",
|
||||||
|
"XLocation": 0,
|
||||||
|
"ULocation": 1,
|
||||||
|
"UHeight": 8
|
||||||
|
},
|
||||||
|
"Presence": "111100",
|
||||||
|
"DesiredSpeedPWM": 50,
|
||||||
|
"DesiredSpeedRPM": 3000,
|
||||||
|
"MaxFansSupported": 6,
|
||||||
|
"NumberOfFansPresent": 6,
|
||||||
|
"VolumetricAirflow": 80,
|
||||||
|
"Temperatures": [
|
||||||
|
{
|
||||||
|
"Name": "Inlet Temperature",
|
||||||
|
"Status": {
|
||||||
|
"State": "Enabled",
|
||||||
|
"Health": "OK",
|
||||||
|
"HealthRollup": null
|
||||||
|
},
|
||||||
|
"ReadingCelsius": 21,
|
||||||
|
"PhysicalContext": "Intake"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"Name": "Outlet Temperature",
|
||||||
|
"Status": {
|
||||||
|
"State": "Enabled",
|
||||||
|
"Health": "OK",
|
||||||
|
"HealthRollup": null
|
||||||
|
},
|
||||||
|
"ReadingCelsius": 35,
|
||||||
|
"PhysicalContext": "Exhaust"
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Status": {
|
||||||
|
"State": "Enabled",
|
||||||
|
"Health": "OK",
|
||||||
|
"HealthRollup": null
|
||||||
|
},
|
||||||
|
"Fans": [
|
||||||
|
{
|
||||||
|
"Name": "Fan 1",
|
||||||
|
"Status": {
|
||||||
|
"State": "Enabled",
|
||||||
|
"Health": "OK",
|
||||||
|
"HealthRollup": null
|
||||||
|
},
|
||||||
|
"ReadingRPM": 0,
|
||||||
|
"RackLocation": {
|
||||||
|
"RackUnits": "OU",
|
||||||
|
"XLocation": 0,
|
||||||
|
"ULocation": 1,
|
||||||
|
"UHeight": 8
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"Actions": {},
|
||||||
|
"Links": {}
|
||||||
|
}
|
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
"@odata.context": "/redfish/v1/$metadata#ThermalZoneCollection.ThermalZoneCollection",
|
||||||
|
"@odata.id": "/redfish/v1/Chassis/Rack1/ThermalZones",
|
||||||
|
"@odata.type": "#ThermalZoneCollection.ThermalZoneCollection",
|
||||||
|
"Name": "Thermal Zones Collection",
|
||||||
|
"Members@odata.count": 1,
|
||||||
|
"Members": [
|
||||||
|
{
|
||||||
|
"@odata.id": "/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1"
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -1,4 +1,4 @@
|
|||||||
# Copyright 2018 99cloud, Inc.
|
# Copyright 2018 Intel, Inc.
|
||||||
# All Rights Reserved.
|
# All Rights Reserved.
|
||||||
#
|
#
|
||||||
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
# Licensed under the Apache License, Version 2.0 (the "License"); you may
|
||||||
@ -39,7 +39,6 @@ class PowerZoneTestCase(testtools.TestCase):
|
|||||||
self.assertEqual('power zone 1', self.power_zone_inst.name)
|
self.assertEqual('power zone 1', self.power_zone_inst.name)
|
||||||
self.assertEqual('power zone 1 description',
|
self.assertEqual('power zone 1 description',
|
||||||
self.power_zone_inst.description)
|
self.power_zone_inst.description)
|
||||||
self.assertEqual('1', self.power_zone_inst.identity)
|
|
||||||
self.assertEqual('Enabled', self.power_zone_inst.status.state)
|
self.assertEqual('Enabled', self.power_zone_inst.status.state)
|
||||||
self.assertEqual('OK', self.power_zone_inst.status.health)
|
self.assertEqual('OK', self.power_zone_inst.status.health)
|
||||||
self.assertEqual('OK', self.power_zone_inst.status.health_rollup)
|
self.assertEqual('OK', self.power_zone_inst.status.health_rollup)
|
||||||
@ -106,16 +105,16 @@ class PowerZoneCollectionTestCase(testtools.TestCase):
|
|||||||
self.power_zone_col.members_identities)
|
self.power_zone_col.members_identities)
|
||||||
|
|
||||||
@mock.patch.object(power_zone, 'PowerZone', autospec=True)
|
@mock.patch.object(power_zone, 'PowerZone', autospec=True)
|
||||||
def test_get_member(self, mock_storage_subsystem):
|
def test_get_member(self, mock_power_zone):
|
||||||
self.power_zone_col.get_member(
|
self.power_zone_col.get_member(
|
||||||
'/redfish/v1/Chassis/Rack1/PowerZones/Power1')
|
'/redfish/v1/Chassis/Rack1/PowerZones/Power1')
|
||||||
mock_storage_subsystem.assert_called_once_with(
|
mock_power_zone.assert_called_once_with(
|
||||||
self.power_zone_col._conn,
|
self.power_zone_col._conn,
|
||||||
'/redfish/v1/Chassis/Rack1/PowerZones/Power1',
|
'/redfish/v1/Chassis/Rack1/PowerZones/Power1',
|
||||||
redfish_version=self.power_zone_col.redfish_version)
|
redfish_version=self.power_zone_col.redfish_version)
|
||||||
|
|
||||||
@mock.patch.object(power_zone, 'PowerZone', autospec=True)
|
@mock.patch.object(power_zone, 'PowerZone', autospec=True)
|
||||||
def test_get_members(self, mock_storage_subsystem):
|
def test_get_members(self, mock_power_zone):
|
||||||
members = self.power_zone_col.get_members()
|
members = self.power_zone_col.get_members()
|
||||||
calls = [
|
calls = [
|
||||||
mock.call(self.power_zone_col._conn,
|
mock.call(self.power_zone_col._conn,
|
||||||
@ -123,6 +122,6 @@ class PowerZoneCollectionTestCase(testtools.TestCase):
|
|||||||
redfish_version=self.power_zone_col.
|
redfish_version=self.power_zone_col.
|
||||||
redfish_version)
|
redfish_version)
|
||||||
]
|
]
|
||||||
mock_storage_subsystem.assert_has_calls(calls)
|
mock_power_zone.assert_has_calls(calls)
|
||||||
self.assertIsInstance(members, list)
|
self.assertIsInstance(members, list)
|
||||||
self.assertEqual(1, len(members))
|
self.assertEqual(1, len(members))
|
||||||
|
139
rsd_lib/tests/unit/resources/v2_1/chassis/test_thermal_zone.py
Normal file
139
rsd_lib/tests/unit/resources/v2_1/chassis/test_thermal_zone.py
Normal file
@ -0,0 +1,139 @@
|
|||||||
|
# Copyright 2018 Intel, Inc.
|
||||||
|
# All Rights Reserved.
|
||||||
|
#
|
||||||
|
# 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 json
|
||||||
|
import mock
|
||||||
|
import testtools
|
||||||
|
|
||||||
|
from rsd_lib.resources.v2_1.chassis import thermal_zone
|
||||||
|
|
||||||
|
|
||||||
|
class ThermalZoneTestCase(testtools.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ThermalZoneTestCase, self).setUp()
|
||||||
|
self.conn = mock.Mock()
|
||||||
|
with open('rsd_lib/tests/unit/json_samples/v2_1/'
|
||||||
|
'thermal_zone.json', 'r') as f:
|
||||||
|
self.conn.get.return_value.json.return_value = json.loads(f.read())
|
||||||
|
|
||||||
|
self.thermal_zone_inst = thermal_zone.ThermalZone(
|
||||||
|
self.conn, '/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1',
|
||||||
|
redfish_version='1.1.0')
|
||||||
|
|
||||||
|
def test__parse_attributes(self):
|
||||||
|
self.thermal_zone_inst._parse_attributes()
|
||||||
|
self.assertEqual('1', self.thermal_zone_inst.identity)
|
||||||
|
self.assertEqual('thermal zone 1', self.thermal_zone_inst.name)
|
||||||
|
self.assertEqual('thermal zone 1 description',
|
||||||
|
self.thermal_zone_inst.description)
|
||||||
|
self.assertEqual('Enabled', self.thermal_zone_inst.status.state)
|
||||||
|
self.assertEqual('OK', self.thermal_zone_inst.status.health)
|
||||||
|
self.assertEqual(None, self.thermal_zone_inst.status.health_rollup)
|
||||||
|
self.assertEqual('OU', self.thermal_zone_inst.rack_location.rack_units)
|
||||||
|
self.assertEqual(0, self.thermal_zone_inst.rack_location.xlocation)
|
||||||
|
self.assertEqual(1, self.thermal_zone_inst.rack_location.ulocation)
|
||||||
|
self.assertEqual(8, self.thermal_zone_inst.rack_location.uheight)
|
||||||
|
self.assertEqual('111100', self.thermal_zone_inst.presence)
|
||||||
|
self.assertEqual(50, self.thermal_zone_inst.desired_speed_pwm)
|
||||||
|
self.assertEqual(3000, self.thermal_zone_inst.desired_speed_rpm)
|
||||||
|
self.assertEqual(6, self.thermal_zone_inst.max_fans_supported)
|
||||||
|
self.assertEqual(6, self.thermal_zone_inst.number_of_fans_present)
|
||||||
|
self.assertEqual(80, self.thermal_zone_inst.volumetric_airflow)
|
||||||
|
self.assertEqual(
|
||||||
|
'Fan 1', self.thermal_zone_inst.fans[0].name)
|
||||||
|
self.assertEqual(
|
||||||
|
0, self.thermal_zone_inst.fans[0].reading_rpm)
|
||||||
|
self.assertEqual(
|
||||||
|
'Enabled', self.thermal_zone_inst.fans[0].status.state)
|
||||||
|
self.assertEqual(
|
||||||
|
'OK', self.thermal_zone_inst.fans[0].status.health)
|
||||||
|
self.assertEqual(
|
||||||
|
None, self.thermal_zone_inst.fans[0].status.health_rollup)
|
||||||
|
self.assertEqual(
|
||||||
|
'OU',
|
||||||
|
self.thermal_zone_inst.fans[0].rack_location.rack_units)
|
||||||
|
self.assertEqual(
|
||||||
|
0, self.thermal_zone_inst.fans[0].rack_location.xlocation)
|
||||||
|
self.assertEqual(
|
||||||
|
1, self.thermal_zone_inst.fans[0].rack_location.ulocation)
|
||||||
|
self.assertEqual(
|
||||||
|
8, self.thermal_zone_inst.fans[0].rack_location.uheight)
|
||||||
|
self.assertEqual(
|
||||||
|
'Inlet Temperature', self.thermal_zone_inst.temperatures[0].name)
|
||||||
|
self.assertEqual(
|
||||||
|
'Enabled', self.thermal_zone_inst.temperatures[0].status.state)
|
||||||
|
self.assertEqual(
|
||||||
|
'OK', self.thermal_zone_inst.temperatures[0].status.health)
|
||||||
|
self.assertEqual(
|
||||||
|
None, self.thermal_zone_inst.temperatures[0].status.health_rollup)
|
||||||
|
self.assertEqual(
|
||||||
|
21, self.thermal_zone_inst.temperatures[0].reading_celsius)
|
||||||
|
self.assertEqual(
|
||||||
|
'Intake', self.thermal_zone_inst.temperatures[0].physical_context)
|
||||||
|
self.assertEqual(
|
||||||
|
'Outlet Temperature', self.thermal_zone_inst.temperatures[1].name)
|
||||||
|
self.assertEqual(
|
||||||
|
'Enabled', self.thermal_zone_inst.temperatures[1].status.state)
|
||||||
|
self.assertEqual(
|
||||||
|
'OK', self.thermal_zone_inst.temperatures[1].status.health)
|
||||||
|
self.assertEqual(
|
||||||
|
None, self.thermal_zone_inst.temperatures[1].status.health_rollup)
|
||||||
|
self.assertEqual(
|
||||||
|
35, self.thermal_zone_inst.temperatures[1].reading_celsius)
|
||||||
|
self.assertEqual(
|
||||||
|
'Exhaust', self.thermal_zone_inst.temperatures[1].physical_context)
|
||||||
|
|
||||||
|
|
||||||
|
class ThermalZoneCollectionTestCase(testtools.TestCase):
|
||||||
|
|
||||||
|
def setUp(self):
|
||||||
|
super(ThermalZoneCollectionTestCase, self).setUp()
|
||||||
|
self.conn = mock.Mock()
|
||||||
|
with open('rsd_lib/tests/unit/json_samples/v2_1/'
|
||||||
|
'thermal_zone_collection.json', 'r') as f:
|
||||||
|
self.conn.get.return_value.json.return_value = json.loads(f.read())
|
||||||
|
self.thermal_zone_col = thermal_zone.\
|
||||||
|
ThermalZoneCollection(
|
||||||
|
self.conn, '/redfish/v1/Chassis/Rack1/ThermalZones',
|
||||||
|
redfish_version='1.1.0')
|
||||||
|
|
||||||
|
def test__parse_attributes(self):
|
||||||
|
self.thermal_zone_col._parse_attributes()
|
||||||
|
self.assertEqual('1.1.0', self.thermal_zone_col.redfish_version)
|
||||||
|
self.assertEqual(('/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1',),
|
||||||
|
self.thermal_zone_col.members_identities)
|
||||||
|
|
||||||
|
@mock.patch.object(thermal_zone, 'ThermalZone', autospec=True)
|
||||||
|
def test_get_member(self, mock_thermal_zone):
|
||||||
|
self.thermal_zone_col.get_member(
|
||||||
|
'/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1')
|
||||||
|
mock_thermal_zone.assert_called_once_with(
|
||||||
|
self.thermal_zone_col._conn,
|
||||||
|
'/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1',
|
||||||
|
redfish_version=self.thermal_zone_col.redfish_version)
|
||||||
|
|
||||||
|
@mock.patch.object(thermal_zone, 'ThermalZone', autospec=True)
|
||||||
|
def test_get_members(self, mock_thermal_zone):
|
||||||
|
members = self.thermal_zone_col.get_members()
|
||||||
|
calls = [
|
||||||
|
mock.call(self.thermal_zone_col._conn,
|
||||||
|
'/redfish/v1/Chassis/Rack1/ThermalZones/Thermal1',
|
||||||
|
redfish_version=self.thermal_zone_col.
|
||||||
|
redfish_version)
|
||||||
|
]
|
||||||
|
mock_thermal_zone.assert_has_calls(calls)
|
||||||
|
self.assertIsInstance(members, list)
|
||||||
|
self.assertEqual(1, len(members))
|
Loading…
x
Reference in New Issue
Block a user