mirror of
https://github.com/ansible/awx.git
synced 2026-02-27 07:56:06 -03:30
Merge pull request #1088 from chrismeyersfsu/mongoectomy_cleanup
postgres fact cleanup
This commit is contained in:
@@ -12,7 +12,7 @@ from django.db import transaction
|
|||||||
from django.utils.timezone import now
|
from django.utils.timezone import now
|
||||||
|
|
||||||
# AWX
|
# AWX
|
||||||
from awx.fact.models.fact import * # noqa
|
from awx.main.models.fact import Fact
|
||||||
from awx.api.license import feature_enabled
|
from awx.api.license import feature_enabled
|
||||||
|
|
||||||
OLDER_THAN = 'older_than'
|
OLDER_THAN = 'older_than'
|
||||||
@@ -31,7 +31,7 @@ class CleanupFacts(object):
|
|||||||
# pivot -= granularity
|
# pivot -= granularity
|
||||||
# group by host
|
# group by host
|
||||||
def cleanup(self, older_than_abs, granularity, module=None):
|
def cleanup(self, older_than_abs, granularity, module=None):
|
||||||
fact_oldest = FactVersion.objects.all().order_by('timestamp').first()
|
fact_oldest = Fact.objects.all().order_by('timestamp').first()
|
||||||
if not fact_oldest:
|
if not fact_oldest:
|
||||||
return 0
|
return 0
|
||||||
|
|
||||||
@@ -44,7 +44,10 @@ class CleanupFacts(object):
|
|||||||
# Special case, granularity=0x where x is d, w, or y
|
# Special case, granularity=0x where x is d, w, or y
|
||||||
# The intent is to delete all facts < older_than_abs
|
# The intent is to delete all facts < older_than_abs
|
||||||
if granularity == relativedelta():
|
if granularity == relativedelta():
|
||||||
return FactVersion.objects.filter(**kv).order_by('-timestamp').delete()
|
qs = Fact.objects.filter(**kv)
|
||||||
|
count = qs.count()
|
||||||
|
qs.delete()
|
||||||
|
return count
|
||||||
|
|
||||||
total = 0
|
total = 0
|
||||||
|
|
||||||
@@ -61,18 +64,17 @@ class CleanupFacts(object):
|
|||||||
kv['module'] = module
|
kv['module'] = module
|
||||||
|
|
||||||
|
|
||||||
fact_version_objs = FactVersion.objects.filter(**kv).order_by('-timestamp').limit(1)
|
fact_version_obj = Fact.objects.filter(**kv).order_by('-timestamp').first()
|
||||||
if fact_version_objs:
|
if fact_version_obj:
|
||||||
fact_version_obj = fact_version_objs[0]
|
|
||||||
kv = {
|
kv = {
|
||||||
'timestamp__lt': fact_version_obj.timestamp,
|
'timestamp__lt': fact_version_obj.timestamp,
|
||||||
'timestamp__gt': date_pivot_next
|
'timestamp__gt': date_pivot_next
|
||||||
}
|
}
|
||||||
if module:
|
if module:
|
||||||
kv['module'] = module
|
kv['module'] = module
|
||||||
count = FactVersion.objects.filter(**kv).delete()
|
qs = Fact.objects.filter(**kv)
|
||||||
# FIXME: These two deletes should be a transaction
|
count = qs.count()
|
||||||
count = Fact.objects.filter(**kv).delete()
|
qs.delete()
|
||||||
total += count
|
total += count
|
||||||
|
|
||||||
date_pivot = date_pivot_next
|
date_pivot = date_pivot_next
|
||||||
|
|||||||
182
awx/main/tests/functional/commands/test_cleanup_facts.py
Normal file
182
awx/main/tests/functional/commands/test_cleanup_facts.py
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
# Copyright (c) 2016 Ansible, Inc.
|
||||||
|
# All Rights Reserved
|
||||||
|
|
||||||
|
# Python
|
||||||
|
import pytest
|
||||||
|
from dateutil.relativedelta import relativedelta
|
||||||
|
from datetime import timedelta
|
||||||
|
|
||||||
|
# Django
|
||||||
|
from django.utils import timezone
|
||||||
|
from django.core.management.base import CommandError
|
||||||
|
|
||||||
|
# AWX
|
||||||
|
from awx.main.management.commands.cleanup_facts import CleanupFacts, Command
|
||||||
|
from awx.main.models.fact import Fact
|
||||||
|
from awx.main.models.inventory import Host
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_cleanup_granularity(fact_scans, hosts):
|
||||||
|
epoch = timezone.now()
|
||||||
|
hosts(5)
|
||||||
|
fact_scans(10, timestamp_epoch=epoch)
|
||||||
|
fact_newest = Fact.objects.all().order_by('-timestamp').first()
|
||||||
|
timestamp_future = fact_newest.timestamp + timedelta(days=365)
|
||||||
|
granularity = relativedelta(days=2)
|
||||||
|
|
||||||
|
cleanup_facts = CleanupFacts()
|
||||||
|
deleted_count = cleanup_facts.cleanup(timestamp_future, granularity)
|
||||||
|
assert 60 == deleted_count
|
||||||
|
|
||||||
|
'''
|
||||||
|
Delete half of the scans
|
||||||
|
'''
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_cleanup_older_than(fact_scans, hosts):
|
||||||
|
epoch = timezone.now()
|
||||||
|
hosts(5)
|
||||||
|
fact_scans(28, timestamp_epoch=epoch)
|
||||||
|
qs = Fact.objects.all().order_by('-timestamp')
|
||||||
|
fact_middle = qs[qs.count() / 2]
|
||||||
|
granularity = relativedelta()
|
||||||
|
|
||||||
|
cleanup_facts = CleanupFacts()
|
||||||
|
deleted_count = cleanup_facts.cleanup(fact_middle.timestamp, granularity)
|
||||||
|
assert 210 == deleted_count
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_cleanup_older_than_granularity_module(fact_scans, hosts):
|
||||||
|
epoch = timezone.now()
|
||||||
|
hosts(5)
|
||||||
|
fact_scans(10, timestamp_epoch=epoch)
|
||||||
|
fact_newest = Fact.objects.all().order_by('-timestamp').first()
|
||||||
|
timestamp_future = fact_newest.timestamp + timedelta(days=365)
|
||||||
|
granularity = relativedelta(days=2)
|
||||||
|
|
||||||
|
cleanup_facts = CleanupFacts()
|
||||||
|
deleted_count = cleanup_facts.cleanup(timestamp_future, granularity, module='ansible')
|
||||||
|
assert 20 == deleted_count
|
||||||
|
|
||||||
|
|
||||||
|
'''
|
||||||
|
Reduce the granularity of half of the facts scans, by half.
|
||||||
|
'''
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_cleanup_logic(fact_scans, hosts):
|
||||||
|
epoch = timezone.now()
|
||||||
|
hosts = hosts(5)
|
||||||
|
fact_scans(60, timestamp_epoch=epoch)
|
||||||
|
timestamp_middle = epoch + timedelta(days=30)
|
||||||
|
granularity = relativedelta(days=2)
|
||||||
|
module = 'ansible'
|
||||||
|
|
||||||
|
cleanup_facts = CleanupFacts()
|
||||||
|
deleted_count = cleanup_facts.cleanup(timestamp_middle, granularity, module=module)
|
||||||
|
|
||||||
|
|
||||||
|
host_ids = Host.objects.all().values_list('id', flat=True)
|
||||||
|
host_facts = {}
|
||||||
|
for host_id in host_ids:
|
||||||
|
facts = Fact.objects.filter(host__id=host_id, module=module, timestamp__lt=timestamp_middle).order_by('-timestamp')
|
||||||
|
host_facts[host_id] = facts
|
||||||
|
|
||||||
|
for host_id, facts in host_facts.iteritems():
|
||||||
|
assert 15 == len(facts)
|
||||||
|
|
||||||
|
timestamp_pivot = timestamp_middle
|
||||||
|
for fact in facts:
|
||||||
|
timestamp_pivot -= granularity
|
||||||
|
assert fact.timestamp == timestamp_pivot
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_parameters_ok(mocker):
|
||||||
|
run = mocker.patch('awx.main.management.commands.cleanup_facts.CleanupFacts.run')
|
||||||
|
kv = {
|
||||||
|
'older_than': '1d',
|
||||||
|
'granularity': '1d',
|
||||||
|
'module': None,
|
||||||
|
}
|
||||||
|
cmd = Command()
|
||||||
|
cmd.handle(None, **kv)
|
||||||
|
run.assert_called_once_with(relativedelta(days=1), relativedelta(days=1), module=None)
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_string_time_to_timestamp_ok():
|
||||||
|
kvs = [
|
||||||
|
{
|
||||||
|
'time': '2w',
|
||||||
|
'timestamp': relativedelta(weeks=2),
|
||||||
|
'msg': '2 weeks',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': '23d',
|
||||||
|
'timestamp': relativedelta(days=23),
|
||||||
|
'msg': '23 days',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': '11m',
|
||||||
|
'timestamp': relativedelta(months=11),
|
||||||
|
'msg': '11 months',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': '14y',
|
||||||
|
'timestamp': relativedelta(years=14),
|
||||||
|
'msg': '14 years',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
for kv in kvs:
|
||||||
|
cmd = Command()
|
||||||
|
res = cmd.string_time_to_timestamp(kv['time'])
|
||||||
|
assert kv['timestamp'] == res
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_string_time_to_timestamp_invalid():
|
||||||
|
kvs = [
|
||||||
|
{
|
||||||
|
'time': '2weeks',
|
||||||
|
'msg': 'weeks instead of w',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': '2days',
|
||||||
|
'msg': 'days instead of d',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': '23',
|
||||||
|
'msg': 'no unit specified',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': None,
|
||||||
|
'msg': 'no value specified',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'time': 'zigzag',
|
||||||
|
'msg': 'random string specified',
|
||||||
|
},
|
||||||
|
]
|
||||||
|
for kv in kvs:
|
||||||
|
cmd = Command()
|
||||||
|
res = cmd.string_time_to_timestamp(kv['time'])
|
||||||
|
assert res is None
|
||||||
|
|
||||||
|
@pytest.mark.django_db
|
||||||
|
def test_parameters_fail(mocker):
|
||||||
|
# Mock run() just in case, but it should never get called because an error should be thrown
|
||||||
|
mocker.patch('awx.main.management.commands.cleanup_facts.CleanupFacts.run')
|
||||||
|
kvs = [
|
||||||
|
{
|
||||||
|
'older_than': '1week',
|
||||||
|
'granularity': '1d',
|
||||||
|
'msg': '--older_than invalid value "1week"',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
'older_than': '1d',
|
||||||
|
'granularity': '1year',
|
||||||
|
'msg': '--granularity invalid value "1year"',
|
||||||
|
}
|
||||||
|
]
|
||||||
|
for kv in kvs:
|
||||||
|
cmd = Command()
|
||||||
|
with pytest.raises(CommandError) as err:
|
||||||
|
cmd.handle(None, older_than=kv['older_than'], granularity=kv['granularity'])
|
||||||
|
assert kv['msg'] in err.value
|
||||||
|
|
||||||
@@ -1,238 +0,0 @@
|
|||||||
# Copyright (c) 2015 Ansible, Inc.
|
|
||||||
# All Rights Reserved
|
|
||||||
|
|
||||||
# Python
|
|
||||||
from datetime import datetime
|
|
||||||
from dateutil.relativedelta import relativedelta
|
|
||||||
import mock
|
|
||||||
|
|
||||||
#Django
|
|
||||||
from django.core.management.base import CommandError
|
|
||||||
|
|
||||||
# AWX
|
|
||||||
from awx.main.tests.base import BaseTest
|
|
||||||
from awx.fact.tests.base import MongoDBRequired, FactScanBuilder, TEST_FACT_PACKAGES, TEST_FACT_ANSIBLE, TEST_FACT_SERVICES
|
|
||||||
from command_base import BaseCommandMixin
|
|
||||||
from awx.main.management.commands.cleanup_facts import Command, CleanupFacts
|
|
||||||
from awx.fact.models.fact import * # noqa
|
|
||||||
|
|
||||||
__all__ = ['CommandTest','CleanupFactsUnitTest', 'CleanupFactsCommandFunctionalTest']
|
|
||||||
|
|
||||||
class CleanupFactsCommandFunctionalTest(BaseCommandMixin, BaseTest, MongoDBRequired):
|
|
||||||
def setUp(self):
|
|
||||||
super(CleanupFactsCommandFunctionalTest, self).setUp()
|
|
||||||
self.create_test_license_file()
|
|
||||||
self.builder = FactScanBuilder()
|
|
||||||
self.builder.add_fact('ansible', TEST_FACT_ANSIBLE)
|
|
||||||
|
|
||||||
def test_invoke_zero_ok(self):
|
|
||||||
self.builder.set_epoch(datetime(year=2015, day=2, month=1, microsecond=0))
|
|
||||||
self.builder.build(scan_count=20, host_count=10)
|
|
||||||
|
|
||||||
result, stdout, stderr = self.run_command('cleanup_facts', granularity='2y', older_than='1d')
|
|
||||||
self.assertEqual(stdout, 'Deleted %s facts.\n' % ((200 / 2)))
|
|
||||||
|
|
||||||
def test_invoke_zero_deleted(self):
|
|
||||||
result, stdout, stderr = self.run_command('cleanup_facts', granularity='1w',older_than='5d')
|
|
||||||
self.assertEqual(stdout, 'Deleted 0 facts.\n')
|
|
||||||
|
|
||||||
def test_invoke_all_deleted(self):
|
|
||||||
self.builder.build(scan_count=20, host_count=10)
|
|
||||||
|
|
||||||
result, stdout, stderr = self.run_command('cleanup_facts', granularity='0d', older_than='0d')
|
|
||||||
self.assertEqual(stdout, 'Deleted 200 facts.\n')
|
|
||||||
|
|
||||||
def test_invoke_params_required(self):
|
|
||||||
result, stdout, stderr = self.run_command('cleanup_facts')
|
|
||||||
self.assertIsInstance(result, CommandError)
|
|
||||||
self.assertEqual(str(result), 'Both --granularity and --older_than are required.')
|
|
||||||
|
|
||||||
def test_module(self):
|
|
||||||
self.builder.add_fact('packages', TEST_FACT_PACKAGES)
|
|
||||||
self.builder.add_fact('services', TEST_FACT_SERVICES)
|
|
||||||
self.builder.build(scan_count=5, host_count=5)
|
|
||||||
|
|
||||||
result, stdout, stderr = self.run_command('cleanup_facts', granularity='0d', older_than='0d', module='packages')
|
|
||||||
self.assertEqual(stdout, 'Deleted 25 facts.\n')
|
|
||||||
|
|
||||||
class CommandTest(BaseTest):
|
|
||||||
def setUp(self):
|
|
||||||
super(CommandTest, self).setUp()
|
|
||||||
self.create_test_license_file()
|
|
||||||
|
|
||||||
@mock.patch('awx.main.management.commands.cleanup_facts.CleanupFacts.run')
|
|
||||||
def test_parameters_ok(self, run):
|
|
||||||
|
|
||||||
kv = {
|
|
||||||
'older_than': '1d',
|
|
||||||
'granularity': '1d',
|
|
||||||
'module': None,
|
|
||||||
}
|
|
||||||
cmd = Command()
|
|
||||||
cmd.handle(None, **kv)
|
|
||||||
run.assert_called_once_with(relativedelta(days=1), relativedelta(days=1), module=None)
|
|
||||||
|
|
||||||
def test_string_time_to_timestamp_ok(self):
|
|
||||||
kvs = [
|
|
||||||
{
|
|
||||||
'time': '2w',
|
|
||||||
'timestamp': relativedelta(weeks=2),
|
|
||||||
'msg': '2 weeks',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': '23d',
|
|
||||||
'timestamp': relativedelta(days=23),
|
|
||||||
'msg': '23 days',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': '11m',
|
|
||||||
'timestamp': relativedelta(months=11),
|
|
||||||
'msg': '11 months',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': '14y',
|
|
||||||
'timestamp': relativedelta(years=14),
|
|
||||||
'msg': '14 years',
|
|
||||||
},
|
|
||||||
]
|
|
||||||
for kv in kvs:
|
|
||||||
cmd = Command()
|
|
||||||
res = cmd.string_time_to_timestamp(kv['time'])
|
|
||||||
self.assertEqual(kv['timestamp'], res, "%s should convert to %s" % (kv['time'], kv['msg']))
|
|
||||||
|
|
||||||
def test_string_time_to_timestamp_invalid(self):
|
|
||||||
kvs = [
|
|
||||||
{
|
|
||||||
'time': '2weeks',
|
|
||||||
'msg': 'weeks instead of w',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': '2days',
|
|
||||||
'msg': 'days instead of d',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': '23',
|
|
||||||
'msg': 'no unit specified',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': None,
|
|
||||||
'msg': 'no value specified',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'time': 'zigzag',
|
|
||||||
'msg': 'random string specified',
|
|
||||||
},
|
|
||||||
]
|
|
||||||
for kv in kvs:
|
|
||||||
cmd = Command()
|
|
||||||
res = cmd.string_time_to_timestamp(kv['time'])
|
|
||||||
self.assertIsNone(res, kv['msg'])
|
|
||||||
|
|
||||||
# Mock run() just in case, but it should never get called because an error should be thrown
|
|
||||||
@mock.patch('awx.main.management.commands.cleanup_facts.CleanupFacts.run')
|
|
||||||
def test_parameters_fail(self, run):
|
|
||||||
kvs = [
|
|
||||||
{
|
|
||||||
'older_than': '1week',
|
|
||||||
'granularity': '1d',
|
|
||||||
'msg': 'Invalid older_than param value',
|
|
||||||
},
|
|
||||||
{
|
|
||||||
'older_than': '1d',
|
|
||||||
'granularity': '1year',
|
|
||||||
'msg': 'Invalid granularity param value',
|
|
||||||
}
|
|
||||||
]
|
|
||||||
for kv in kvs:
|
|
||||||
cmd = Command()
|
|
||||||
with self.assertRaises(CommandError):
|
|
||||||
cmd.handle(None, older_than=kv['older_than'], granularity=kv['granularity'])
|
|
||||||
|
|
||||||
class CleanupFactsUnitTest(BaseCommandMixin, BaseTest, MongoDBRequired):
|
|
||||||
def setUp(self):
|
|
||||||
super(CleanupFactsUnitTest, self).setUp()
|
|
||||||
|
|
||||||
self.builder = FactScanBuilder()
|
|
||||||
self.builder.add_fact('ansible', TEST_FACT_ANSIBLE)
|
|
||||||
self.builder.add_fact('packages', TEST_FACT_PACKAGES)
|
|
||||||
self.builder.build(scan_count=20, host_count=10)
|
|
||||||
|
|
||||||
'''
|
|
||||||
Create 10 hosts with 40 facts each. After cleanup, there should be 20 facts for each host.
|
|
||||||
Then ensure the correct facts are deleted.
|
|
||||||
'''
|
|
||||||
def test_cleanup_logic(self):
|
|
||||||
cleanup_facts = CleanupFacts()
|
|
||||||
fact_oldest = FactVersion.objects.all().order_by('timestamp').first()
|
|
||||||
granularity = relativedelta(years=2)
|
|
||||||
|
|
||||||
deleted_count = cleanup_facts.cleanup(self.builder.get_timestamp(0), granularity)
|
|
||||||
self.assertEqual(deleted_count, 2 * (self.builder.get_scan_count() * self.builder.get_host_count()) / 2)
|
|
||||||
|
|
||||||
# Check the number of facts per host
|
|
||||||
for host in self.builder.get_hosts():
|
|
||||||
count = FactVersion.objects.filter(host=host).count()
|
|
||||||
scan_count = (2 * self.builder.get_scan_count()) / 2
|
|
||||||
self.assertEqual(count, scan_count)
|
|
||||||
|
|
||||||
count = Fact.objects.filter(host=host).count()
|
|
||||||
self.assertEqual(count, scan_count)
|
|
||||||
|
|
||||||
# Ensure that only 2 facts (ansible and packages) exists per granularity time
|
|
||||||
date_pivot = self.builder.get_timestamp(0)
|
|
||||||
for host in self.builder.get_hosts():
|
|
||||||
while date_pivot > fact_oldest.timestamp:
|
|
||||||
date_pivot_next = date_pivot - granularity
|
|
||||||
kv = {
|
|
||||||
'timestamp__lte': date_pivot,
|
|
||||||
'timestamp__gt': date_pivot_next,
|
|
||||||
'host': host,
|
|
||||||
}
|
|
||||||
count = FactVersion.objects.filter(**kv).count()
|
|
||||||
self.assertEqual(count, 2, "should only be 2 FactVersion per the 2 year granularity")
|
|
||||||
count = Fact.objects.filter(**kv).count()
|
|
||||||
self.assertEqual(count, 2, "should only be 2 Fact per the 2 year granularity")
|
|
||||||
date_pivot = date_pivot_next
|
|
||||||
|
|
||||||
'''
|
|
||||||
Create 10 hosts with 40 facts each. After cleanup, there should be 30 facts for each host.
|
|
||||||
Then ensure the correct facts are deleted.
|
|
||||||
'''
|
|
||||||
def test_cleanup_module(self):
|
|
||||||
cleanup_facts = CleanupFacts()
|
|
||||||
fact_oldest = FactVersion.objects.all().order_by('timestamp').first()
|
|
||||||
granularity = relativedelta(years=2)
|
|
||||||
|
|
||||||
deleted_count = cleanup_facts.cleanup(self.builder.get_timestamp(0), granularity, module='ansible')
|
|
||||||
self.assertEqual(deleted_count, (self.builder.get_scan_count() * self.builder.get_host_count()) / 2)
|
|
||||||
|
|
||||||
# Check the number of facts per host
|
|
||||||
for host in self.builder.get_hosts():
|
|
||||||
count = FactVersion.objects.filter(host=host).count()
|
|
||||||
self.assertEqual(count, 30)
|
|
||||||
|
|
||||||
count = Fact.objects.filter(host=host).count()
|
|
||||||
self.assertEqual(count, 30)
|
|
||||||
|
|
||||||
# Ensure that only 1 ansible fact exists per granularity time
|
|
||||||
date_pivot = self.builder.get_timestamp(0)
|
|
||||||
for host in self.builder.get_hosts():
|
|
||||||
while date_pivot > fact_oldest.timestamp:
|
|
||||||
date_pivot_next = date_pivot - granularity
|
|
||||||
kv = {
|
|
||||||
'timestamp__lte': date_pivot,
|
|
||||||
'timestamp__gt': date_pivot_next,
|
|
||||||
'host': host,
|
|
||||||
'module': 'ansible',
|
|
||||||
}
|
|
||||||
count = FactVersion.objects.filter(**kv).count()
|
|
||||||
self.assertEqual(count, 1)
|
|
||||||
count = Fact.objects.filter(**kv).count()
|
|
||||||
self.assertEqual(count, 1)
|
|
||||||
date_pivot = date_pivot_next
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
Reference in New Issue
Block a user