Flush the batch when exceeding a certain size, instead of when reaching a number of batched operations. Move batch to lowlevel to be able to use it in NamespacedKV. Increase the leveldb memory buffer from 4 to 16 MiB.
This commit is contained in:
parent
e2204d0071
commit
ca3ae64bbf
@ -73,7 +73,7 @@ func addToBlockMap(db *instance, folder []byte, fs []protocol.FileInfo) {
|
|||||||
name := []byte(f.Name)
|
name := []byte(f.Name)
|
||||||
for i, block := range f.Blocks {
|
for i, block := range f.Blocks {
|
||||||
binary.BigEndian.PutUint32(blockBuf, uint32(i))
|
binary.BigEndian.PutUint32(blockBuf, uint32(i))
|
||||||
keyBuf = t.db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
keyBuf = t.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
||||||
t.Put(keyBuf, blockBuf)
|
t.Put(keyBuf, blockBuf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -89,7 +89,7 @@ func discardFromBlockMap(db *instance, folder []byte, fs []protocol.FileInfo) {
|
|||||||
if !ef.IsDirectory() && !ef.IsDeleted() && !ef.IsInvalid() {
|
if !ef.IsDirectory() && !ef.IsDeleted() && !ef.IsInvalid() {
|
||||||
name := []byte(ef.Name)
|
name := []byte(ef.Name)
|
||||||
for _, block := range ef.Blocks {
|
for _, block := range ef.Blocks {
|
||||||
keyBuf = t.db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
keyBuf = t.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
||||||
t.Delete(keyBuf)
|
t.Delete(keyBuf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -80,7 +80,7 @@ func (db *instance) updateLocalFiles(folder []byte, fs []protocol.FileInfo, meta
|
|||||||
if ok {
|
if ok {
|
||||||
if !ef.IsDirectory() && !ef.IsDeleted() && !ef.IsInvalid() {
|
if !ef.IsDirectory() && !ef.IsDeleted() && !ef.IsInvalid() {
|
||||||
for _, block := range ef.Blocks {
|
for _, block := range ef.Blocks {
|
||||||
keyBuf = t.db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
keyBuf = db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
||||||
t.Delete(keyBuf)
|
t.Delete(keyBuf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -100,7 +100,7 @@ func (db *instance) updateLocalFiles(folder []byte, fs []protocol.FileInfo, meta
|
|||||||
l.Debugf("insert (local); folder=%q %v", folder, f)
|
l.Debugf("insert (local); folder=%q %v", folder, f)
|
||||||
t.Put(dk, mustMarshal(&f))
|
t.Put(dk, mustMarshal(&f))
|
||||||
|
|
||||||
gk = t.db.keyer.GenerateGlobalVersionKey(gk, folder, []byte(f.Name))
|
gk = db.keyer.GenerateGlobalVersionKey(gk, folder, []byte(f.Name))
|
||||||
keyBuf, _ = t.updateGlobal(gk, keyBuf, folder, protocol.LocalDeviceID[:], f, meta)
|
keyBuf, _ = t.updateGlobal(gk, keyBuf, folder, protocol.LocalDeviceID[:], f, meta)
|
||||||
|
|
||||||
keyBuf = db.keyer.GenerateSequenceKey(keyBuf, folder, f.Sequence)
|
keyBuf = db.keyer.GenerateSequenceKey(keyBuf, folder, f.Sequence)
|
||||||
@ -110,7 +110,7 @@ func (db *instance) updateLocalFiles(folder []byte, fs []protocol.FileInfo, meta
|
|||||||
if !f.IsDirectory() && !f.IsDeleted() && !f.IsInvalid() {
|
if !f.IsDirectory() && !f.IsDeleted() && !f.IsInvalid() {
|
||||||
for i, block := range f.Blocks {
|
for i, block := range f.Blocks {
|
||||||
binary.BigEndian.PutUint32(blockBuf, uint32(i))
|
binary.BigEndian.PutUint32(blockBuf, uint32(i))
|
||||||
keyBuf = t.db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
keyBuf = db.keyer.GenerateBlockMapKey(keyBuf, folder, block.Hash, name)
|
||||||
t.Put(keyBuf, blockBuf)
|
t.Put(keyBuf, blockBuf)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@ -19,7 +19,8 @@ import (
|
|||||||
|
|
||||||
const (
|
const (
|
||||||
dbMaxOpenFiles = 100
|
dbMaxOpenFiles = 100
|
||||||
dbWriteBuffer = 4 << 20
|
dbWriteBuffer = 16 << 20
|
||||||
|
dbFlushBatch = dbWriteBuffer / 4 // Some leeway for any leveldb in-memory optimizations
|
||||||
)
|
)
|
||||||
|
|
||||||
// Lowlevel is the lowest level database interface. It has a very simple
|
// Lowlevel is the lowest level database interface. It has a very simple
|
||||||
@ -127,3 +128,29 @@ func leveldbIsCorrupted(err error) bool {
|
|||||||
|
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
|
type batch struct {
|
||||||
|
*leveldb.Batch
|
||||||
|
db *Lowlevel
|
||||||
|
}
|
||||||
|
|
||||||
|
func (db *Lowlevel) newBatch() *batch {
|
||||||
|
return &batch{
|
||||||
|
Batch: new(leveldb.Batch),
|
||||||
|
db: db,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// checkFlush flushes and resets the batch if its size exceeds dbFlushBatch.
|
||||||
|
func (b *batch) checkFlush() {
|
||||||
|
if len(b.Dump()) > dbFlushBatch {
|
||||||
|
b.flush()
|
||||||
|
b.Reset()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (b *batch) flush() {
|
||||||
|
if err := b.db.Write(b.Batch, nil); err != nil {
|
||||||
|
panic(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@ -10,7 +10,6 @@ import (
|
|||||||
"encoding/binary"
|
"encoding/binary"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/syndtr/goleveldb/leveldb"
|
|
||||||
"github.com/syndtr/goleveldb/leveldb/util"
|
"github.com/syndtr/goleveldb/leveldb/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
@ -39,21 +38,12 @@ func NewNamespacedKV(db *Lowlevel, prefix string) *NamespacedKV {
|
|||||||
func (n *NamespacedKV) Reset() {
|
func (n *NamespacedKV) Reset() {
|
||||||
it := n.db.NewIterator(util.BytesPrefix(n.prefix), nil)
|
it := n.db.NewIterator(util.BytesPrefix(n.prefix), nil)
|
||||||
defer it.Release()
|
defer it.Release()
|
||||||
batch := new(leveldb.Batch)
|
batch := n.db.newBatch()
|
||||||
for it.Next() {
|
for it.Next() {
|
||||||
batch.Delete(it.Key())
|
batch.Delete(it.Key())
|
||||||
if batch.Len() > batchFlushSize {
|
batch.checkFlush()
|
||||||
if err := n.db.Write(batch, nil); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
batch.Reset()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if batch.Len() > 0 {
|
|
||||||
if err := n.db.Write(batch, nil); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
batch.flush()
|
||||||
}
|
}
|
||||||
|
|
||||||
// PutInt64 stores a new int64. Any existing value (even if of another type)
|
// PutInt64 stores a new int64. Any existing value (even if of another type)
|
||||||
|
|||||||
@ -104,18 +104,18 @@ func (db *schemaUpdater) updateSchema0to1() {
|
|||||||
var gk, buf []byte
|
var gk, buf []byte
|
||||||
|
|
||||||
for dbi.Next() {
|
for dbi.Next() {
|
||||||
|
t.checkFlush()
|
||||||
|
|
||||||
folder, ok := db.keyer.FolderFromDeviceFileKey(dbi.Key())
|
folder, ok := db.keyer.FolderFromDeviceFileKey(dbi.Key())
|
||||||
if !ok {
|
if !ok {
|
||||||
// not having the folder in the index is bad; delete and continue
|
// not having the folder in the index is bad; delete and continue
|
||||||
t.Delete(dbi.Key())
|
t.Delete(dbi.Key())
|
||||||
t.checkFlush()
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
device, ok := db.keyer.DeviceFromDeviceFileKey(dbi.Key())
|
device, ok := db.keyer.DeviceFromDeviceFileKey(dbi.Key())
|
||||||
if !ok {
|
if !ok {
|
||||||
// not having the device in the index is bad; delete and continue
|
// not having the device in the index is bad; delete and continue
|
||||||
t.Delete(dbi.Key())
|
t.Delete(dbi.Key())
|
||||||
t.checkFlush()
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
name := db.keyer.NameFromDeviceFileKey(dbi.Key())
|
name := db.keyer.NameFromDeviceFileKey(dbi.Key())
|
||||||
@ -128,7 +128,6 @@ func (db *schemaUpdater) updateSchema0to1() {
|
|||||||
gk = db.keyer.GenerateGlobalVersionKey(gk, folder, name)
|
gk = db.keyer.GenerateGlobalVersionKey(gk, folder, name)
|
||||||
buf = t.removeFromGlobal(gk, buf, folder, device, nil, nil)
|
buf = t.removeFromGlobal(gk, buf, folder, device, nil, nil)
|
||||||
t.Delete(dbi.Key())
|
t.Delete(dbi.Key())
|
||||||
t.checkFlush()
|
|
||||||
continue
|
continue
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -149,7 +148,6 @@ func (db *schemaUpdater) updateSchema0to1() {
|
|||||||
panic("can't happen: " + err.Error())
|
panic("can't happen: " + err.Error())
|
||||||
}
|
}
|
||||||
t.Put(dbi.Key(), bs)
|
t.Put(dbi.Key(), bs)
|
||||||
t.checkFlush()
|
|
||||||
symlinkConv++
|
symlinkConv++
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -210,7 +208,7 @@ func (db *schemaUpdater) updateSchema2to3() {
|
|||||||
if !need(f, ok, v) {
|
if !need(f, ok, v) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
nk = t.db.keyer.GenerateNeedFileKey(nk, folder, []byte(f.FileName()))
|
nk = t.keyer.GenerateNeedFileKey(nk, folder, []byte(f.FileName()))
|
||||||
t.Put(nk, nil)
|
t.Put(nk, nil)
|
||||||
t.checkFlush()
|
t.checkFlush()
|
||||||
return true
|
return true
|
||||||
@ -282,7 +280,7 @@ func (db *schemaUpdater) updateSchema6to7() {
|
|||||||
svl, err := t.Get(gk, nil)
|
svl, err := t.Get(gk, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// If there is no global list, we hardly need it.
|
// If there is no global list, we hardly need it.
|
||||||
t.Delete(t.db.keyer.GenerateNeedFileKey(nk, folder, name))
|
t.Delete(t.keyer.GenerateNeedFileKey(nk, folder, name))
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
var fl VersionList
|
var fl VersionList
|
||||||
@ -293,7 +291,7 @@ func (db *schemaUpdater) updateSchema6to7() {
|
|||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
if localFV, haveLocalFV := fl.Get(protocol.LocalDeviceID[:]); !need(global, haveLocalFV, localFV.Version) {
|
if localFV, haveLocalFV := fl.Get(protocol.LocalDeviceID[:]); !need(global, haveLocalFV, localFV.Version) {
|
||||||
t.Delete(t.db.keyer.GenerateNeedFileKey(nk, folder, name))
|
t.Delete(t.keyer.GenerateNeedFileKey(nk, folder, name))
|
||||||
}
|
}
|
||||||
return true
|
return true
|
||||||
})
|
})
|
||||||
|
|||||||
@ -12,13 +12,10 @@ import (
|
|||||||
"github.com/syndtr/goleveldb/leveldb/util"
|
"github.com/syndtr/goleveldb/leveldb/util"
|
||||||
)
|
)
|
||||||
|
|
||||||
// Flush batches to disk when they contain this many records.
|
|
||||||
const batchFlushSize = 64
|
|
||||||
|
|
||||||
// A readOnlyTransaction represents a database snapshot.
|
// A readOnlyTransaction represents a database snapshot.
|
||||||
type readOnlyTransaction struct {
|
type readOnlyTransaction struct {
|
||||||
*leveldb.Snapshot
|
*leveldb.Snapshot
|
||||||
db *instance
|
keyer keyer
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *instance) newReadOnlyTransaction() readOnlyTransaction {
|
func (db *instance) newReadOnlyTransaction() readOnlyTransaction {
|
||||||
@ -28,7 +25,7 @@ func (db *instance) newReadOnlyTransaction() readOnlyTransaction {
|
|||||||
}
|
}
|
||||||
return readOnlyTransaction{
|
return readOnlyTransaction{
|
||||||
Snapshot: snap,
|
Snapshot: snap,
|
||||||
db: db,
|
keyer: db.keyer,
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -37,7 +34,7 @@ func (t readOnlyTransaction) close() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (t readOnlyTransaction) getFile(folder, device, file []byte) (protocol.FileInfo, bool) {
|
func (t readOnlyTransaction) getFile(folder, device, file []byte) (protocol.FileInfo, bool) {
|
||||||
return t.getFileByKey(t.db.keyer.GenerateDeviceFileKey(nil, folder, device, file))
|
return t.getFileByKey(t.keyer.GenerateDeviceFileKey(nil, folder, device, file))
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t readOnlyTransaction) getFileByKey(key []byte) (protocol.FileInfo, bool) {
|
func (t readOnlyTransaction) getFileByKey(key []byte) (protocol.FileInfo, bool) {
|
||||||
@ -65,7 +62,7 @@ func (t readOnlyTransaction) getFileTrunc(key []byte, trunc bool) (FileIntf, boo
|
|||||||
}
|
}
|
||||||
|
|
||||||
func (t readOnlyTransaction) getGlobal(keyBuf, folder, file []byte, truncate bool) ([]byte, FileIntf, bool) {
|
func (t readOnlyTransaction) getGlobal(keyBuf, folder, file []byte, truncate bool) ([]byte, FileIntf, bool) {
|
||||||
keyBuf = t.db.keyer.GenerateGlobalVersionKey(keyBuf, folder, file)
|
keyBuf = t.keyer.GenerateGlobalVersionKey(keyBuf, folder, file)
|
||||||
|
|
||||||
bs, err := t.Get(keyBuf, nil)
|
bs, err := t.Get(keyBuf, nil)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@ -77,7 +74,7 @@ func (t readOnlyTransaction) getGlobal(keyBuf, folder, file []byte, truncate boo
|
|||||||
return keyBuf, nil, false
|
return keyBuf, nil, false
|
||||||
}
|
}
|
||||||
|
|
||||||
keyBuf = t.db.keyer.GenerateDeviceFileKey(keyBuf, folder, vl.Versions[0].Device, file)
|
keyBuf = t.keyer.GenerateDeviceFileKey(keyBuf, folder, vl.Versions[0].Device, file)
|
||||||
if fi, ok := t.getFileTrunc(keyBuf, truncate); ok {
|
if fi, ok := t.getFileTrunc(keyBuf, truncate); ok {
|
||||||
return keyBuf, fi, true
|
return keyBuf, fi, true
|
||||||
}
|
}
|
||||||
@ -90,14 +87,13 @@ func (t readOnlyTransaction) getGlobal(keyBuf, folder, file []byte, truncate boo
|
|||||||
// batch size.
|
// batch size.
|
||||||
type readWriteTransaction struct {
|
type readWriteTransaction struct {
|
||||||
readOnlyTransaction
|
readOnlyTransaction
|
||||||
*leveldb.Batch
|
*batch
|
||||||
}
|
}
|
||||||
|
|
||||||
func (db *instance) newReadWriteTransaction() readWriteTransaction {
|
func (db *instance) newReadWriteTransaction() readWriteTransaction {
|
||||||
t := db.newReadOnlyTransaction()
|
|
||||||
return readWriteTransaction{
|
return readWriteTransaction{
|
||||||
readOnlyTransaction: t,
|
readOnlyTransaction: db.newReadOnlyTransaction(),
|
||||||
Batch: new(leveldb.Batch),
|
batch: db.newBatch(),
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -106,19 +102,6 @@ func (t readWriteTransaction) close() {
|
|||||||
t.readOnlyTransaction.close()
|
t.readOnlyTransaction.close()
|
||||||
}
|
}
|
||||||
|
|
||||||
func (t readWriteTransaction) checkFlush() {
|
|
||||||
if t.Batch.Len() > batchFlushSize {
|
|
||||||
t.flush()
|
|
||||||
t.Batch.Reset()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (t readWriteTransaction) flush() {
|
|
||||||
if err := t.db.Write(t.Batch, nil); err != nil {
|
|
||||||
panic(err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
// updateGlobal adds this device+version to the version list for the given
|
// updateGlobal adds this device+version to the version list for the given
|
||||||
// file. If the device is already present in the list, the version is updated.
|
// file. If the device is already present in the list, the version is updated.
|
||||||
// If the file does not have an entry in the global list, it is created.
|
// If the file does not have an entry in the global list, it is created.
|
||||||
@ -142,7 +125,7 @@ func (t readWriteTransaction) updateGlobal(gk, keyBuf, folder, device []byte, fi
|
|||||||
// Inserted a new newest version
|
// Inserted a new newest version
|
||||||
global = file
|
global = file
|
||||||
} else {
|
} else {
|
||||||
keyBuf = t.db.keyer.GenerateDeviceFileKey(keyBuf, folder, fl.Versions[0].Device, name)
|
keyBuf = t.keyer.GenerateDeviceFileKey(keyBuf, folder, fl.Versions[0].Device, name)
|
||||||
if new, ok := t.getFileByKey(keyBuf); ok {
|
if new, ok := t.getFileByKey(keyBuf); ok {
|
||||||
global = new
|
global = new
|
||||||
} else {
|
} else {
|
||||||
@ -167,7 +150,7 @@ func (t readWriteTransaction) updateGlobal(gk, keyBuf, folder, device []byte, fi
|
|||||||
// The previous newest version is now at index 1
|
// The previous newest version is now at index 1
|
||||||
oldGlobalFV = fl.Versions[1]
|
oldGlobalFV = fl.Versions[1]
|
||||||
}
|
}
|
||||||
keyBuf = t.db.keyer.GenerateDeviceFileKey(keyBuf, folder, oldGlobalFV.Device, name)
|
keyBuf = t.keyer.GenerateDeviceFileKey(keyBuf, folder, oldGlobalFV.Device, name)
|
||||||
if oldFile, ok := t.getFileByKey(keyBuf); ok {
|
if oldFile, ok := t.getFileByKey(keyBuf); ok {
|
||||||
// A failure to get the file here is surprising and our
|
// A failure to get the file here is surprising and our
|
||||||
// global size data will be incorrect until a restart...
|
// global size data will be incorrect until a restart...
|
||||||
@ -187,7 +170,7 @@ func (t readWriteTransaction) updateGlobal(gk, keyBuf, folder, device []byte, fi
|
|||||||
// device according to the version list and global FileInfo given and updates
|
// device according to the version list and global FileInfo given and updates
|
||||||
// the db accordingly.
|
// the db accordingly.
|
||||||
func (t readWriteTransaction) updateLocalNeed(keyBuf, folder, name []byte, fl VersionList, global protocol.FileInfo) []byte {
|
func (t readWriteTransaction) updateLocalNeed(keyBuf, folder, name []byte, fl VersionList, global protocol.FileInfo) []byte {
|
||||||
keyBuf = t.db.keyer.GenerateNeedFileKey(keyBuf, folder, name)
|
keyBuf = t.keyer.GenerateNeedFileKey(keyBuf, folder, name)
|
||||||
hasNeeded, _ := t.Has(keyBuf, nil)
|
hasNeeded, _ := t.Has(keyBuf, nil)
|
||||||
if localFV, haveLocalFV := fl.Get(protocol.LocalDeviceID[:]); need(global, haveLocalFV, localFV.Version) {
|
if localFV, haveLocalFV := fl.Get(protocol.LocalDeviceID[:]); need(global, haveLocalFV, localFV.Version) {
|
||||||
if !hasNeeded {
|
if !hasNeeded {
|
||||||
@ -246,21 +229,21 @@ func (t readWriteTransaction) removeFromGlobal(gk, keyBuf, folder, device []byte
|
|||||||
if removedAt == 0 {
|
if removedAt == 0 {
|
||||||
// A failure to get the file here is surprising and our
|
// A failure to get the file here is surprising and our
|
||||||
// global size data will be incorrect until a restart...
|
// global size data will be incorrect until a restart...
|
||||||
keyBuf = t.db.keyer.GenerateDeviceFileKey(keyBuf, folder, device, file)
|
keyBuf = t.keyer.GenerateDeviceFileKey(keyBuf, folder, device, file)
|
||||||
if f, ok := t.getFileByKey(keyBuf); ok {
|
if f, ok := t.getFileByKey(keyBuf); ok {
|
||||||
meta.removeFile(protocol.GlobalDeviceID, f)
|
meta.removeFile(protocol.GlobalDeviceID, f)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if len(fl.Versions) == 0 {
|
if len(fl.Versions) == 0 {
|
||||||
keyBuf = t.db.keyer.GenerateNeedFileKey(keyBuf, folder, file)
|
keyBuf = t.keyer.GenerateNeedFileKey(keyBuf, folder, file)
|
||||||
t.Delete(keyBuf)
|
t.Delete(keyBuf)
|
||||||
t.Delete(gk)
|
t.Delete(gk)
|
||||||
return keyBuf
|
return keyBuf
|
||||||
}
|
}
|
||||||
|
|
||||||
if removedAt == 0 {
|
if removedAt == 0 {
|
||||||
keyBuf = t.db.keyer.GenerateDeviceFileKey(keyBuf, folder, fl.Versions[0].Device, file)
|
keyBuf = t.keyer.GenerateDeviceFileKey(keyBuf, folder, fl.Versions[0].Device, file)
|
||||||
global, ok := t.getFileByKey(keyBuf)
|
global, ok := t.getFileByKey(keyBuf)
|
||||||
if !ok {
|
if !ok {
|
||||||
panic("This file must exist in the db")
|
panic("This file must exist in the db")
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user