Fix due to the modified Close method in WebSocket.cs
This commit is contained in:
parent
33cddb9fa2
commit
a7eef35c96
Binary file not shown.
@ -100,7 +100,7 @@ namespace Example
|
||||
ws.OnClose += (sender, e) =>
|
||||
{
|
||||
enNfMessage(
|
||||
String.Format("[WebSocket] Close({0}:{1})", (ushort)e.Code, e.Code),
|
||||
String.Format("[WebSocket] Close({0})", e.Code),
|
||||
e.Reason,
|
||||
"notification-message-im");
|
||||
};
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -116,7 +116,7 @@ namespace Example1
|
||||
enNfMessage
|
||||
(
|
||||
"[AudioStreamer] disconnect",
|
||||
String.Format("WS: Close({0}:{1}): {2}", (ushort)e.Code, e.Code, e.Reason),
|
||||
String.Format("WS: Close({0}: {1})", e.Code, e.Reason),
|
||||
"notification-message-im"
|
||||
);
|
||||
};
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -13,7 +13,7 @@ namespace Example2
|
||||
|
||||
protected override void onClose(object sender, CloseEventArgs e)
|
||||
{
|
||||
Console.WriteLine("[Echo] Close({0}: {1})", (ushort)e.Code, e.Code);
|
||||
Console.WriteLine("[Echo] Close({0})", e.Code);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
@ -13,7 +13,7 @@ namespace Example3
|
||||
|
||||
protected override void onClose(object sender, CloseEventArgs e)
|
||||
{
|
||||
Console.WriteLine("[Echo] Close({0}: {1})", (ushort)e.Code, e.Code);
|
||||
Console.WriteLine("[Echo] Close({0})", e.Code);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
14
README.md
14
README.md
@ -4,7 +4,7 @@
|
||||
|
||||
## Usage ##
|
||||
|
||||
### The WebSocket client ###
|
||||
### WebSocket client ###
|
||||
|
||||
#### Step 1 ####
|
||||
|
||||
@ -101,7 +101,7 @@ ws.OnClose += (sender, e) =>
|
||||
};
|
||||
```
|
||||
|
||||
The `e.Code` (`WebSocketSharp.CloseEventArgs.Code`, its type is `WebSocketSharp.Frame.CloseStatusCode`) contains the close status code and the `e.Reason` (`WebSocketSharp.CloseEventArgs.Reason`, its type is `string`) contains the reason why closes, so you operate them.
|
||||
The `e.Code` (`WebSocketSharp.CloseEventArgs.Code`, its type is `ushort`) contains a status code indicating a reason for closure and the `e.Reason` (`WebSocketSharp.CloseEventArgs.Reason`, its type is `string`) contains a reason for closure, so you operate them.
|
||||
|
||||
#### Step 4 ####
|
||||
|
||||
@ -133,11 +133,13 @@ ws.Close(code, reason);
|
||||
|
||||
If you want to close the WebSocket connection explicitly, you can use the `Close` method.
|
||||
|
||||
The type of `code` is `WebSocketSharp.Frame.CloseStatusCode`, the type of `reason` is `string`.
|
||||
The `Close` method is overloaded.
|
||||
|
||||
The `Close` method is overloaded. (In addition, the `Close()` and `Close(code)` methods exist.)
|
||||
The types of `code` are `WebSocketSharp.Frame.CloseStatusCode` and `ushort`, the type of `reason` is `string`.
|
||||
|
||||
### The WebSocket server ###
|
||||
In addition, the `Close()` and `Close(code)` methods exist.
|
||||
|
||||
### WebSocket server ###
|
||||
|
||||
#### Step 1 ####
|
||||
|
||||
@ -251,7 +253,7 @@ Stopping the server.
|
||||
wssv.Stop();
|
||||
```
|
||||
|
||||
### The HTTP server with the WebSocket ###
|
||||
### HTTP server with the WebSocket ###
|
||||
|
||||
I modified the `System.Net.HttpListener`, `System.Net.HttpListenerContext` and some other classes of [Mono] to create the HTTP server that can upgrade the connection to the WebSocket connection when receives a WebSocket request.
|
||||
|
||||
|
@ -38,30 +38,27 @@ namespace WebSocketSharp
|
||||
private string _reason;
|
||||
private bool _wasClean;
|
||||
|
||||
public CloseStatusCode Code
|
||||
public ushort Code
|
||||
{
|
||||
get
|
||||
{
|
||||
return (CloseStatusCode)_code;
|
||||
get {
|
||||
return _code;
|
||||
}
|
||||
}
|
||||
|
||||
public string Reason
|
||||
{
|
||||
get
|
||||
{
|
||||
get {
|
||||
return _reason;
|
||||
}
|
||||
}
|
||||
|
||||
public bool WasClean
|
||||
{
|
||||
get
|
||||
{
|
||||
get {
|
||||
return _wasClean;
|
||||
}
|
||||
set
|
||||
{
|
||||
|
||||
set {
|
||||
_wasClean = value;
|
||||
}
|
||||
}
|
||||
|
@ -106,12 +106,6 @@ namespace WebSocketSharp {
|
||||
return ((int)code).GetStatusDescription();
|
||||
}
|
||||
|
||||
public static string GetHeaderValue(this string src, string separater)
|
||||
{
|
||||
int i = src.IndexOf(separater);
|
||||
return src.Substring(i + 1).Trim();
|
||||
}
|
||||
|
||||
// Derived from System.Net.HttpListenerResponse.GetStatusDescription method
|
||||
public static string GetStatusDescription(this int code)
|
||||
{
|
||||
@ -168,6 +162,27 @@ namespace WebSocketSharp {
|
||||
return String.Empty;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Gets the value from a <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
||||
/// </summary>
|
||||
/// <returns>
|
||||
/// A <see cref="string"/> that contains the value if can get; otherwise, <c>null</c>.
|
||||
/// </returns>
|
||||
/// <param name="nameAndValue">
|
||||
/// A <see cref="string"/> that contains a pair of name and value are separated by a separator string.
|
||||
/// </param>
|
||||
/// <param name="separator">
|
||||
/// A <see cref="string"/> that contains a separator string.
|
||||
/// </param>
|
||||
public static string GetValue(this string nameAndValue, string separator)
|
||||
{
|
||||
var i = nameAndValue.IndexOf(separator);
|
||||
if (i <= 0)
|
||||
return null;
|
||||
|
||||
return nameAndValue.Substring(i + 1).Trim();
|
||||
}
|
||||
|
||||
public static bool IsHostOrder(this ByteOrder order)
|
||||
{
|
||||
if (BitConverter.IsLittleEndian ^ (order == ByteOrder.LITTLE))
|
||||
@ -228,6 +243,18 @@ namespace WebSocketSharp {
|
||||
return false;
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Determines whether <paramref name="uri"/> is valid WebSocket URI.
|
||||
/// </summary>
|
||||
/// <returns>
|
||||
/// <c>true</c> if <paramref name="uri"/> is valid WebSocket URI; otherwise, <c>false</c>.
|
||||
/// </returns>
|
||||
/// <param name="uri">
|
||||
/// A <see cref="Uri"/> that contains a WebSocket URI.
|
||||
/// </param>
|
||||
/// <param name="message">
|
||||
/// A <see cref="string"/> that contains a error message if <paramref name="uri"/> is invalid WebSocket URI; otherwise, <c>String.Empty</c>.
|
||||
/// </param>
|
||||
public static bool IsValidWebSocketUri(this Uri uri, out string message)
|
||||
{
|
||||
if (!uri.IsAbsoluteUri)
|
||||
@ -246,7 +273,7 @@ namespace WebSocketSharp {
|
||||
var original = uri.OriginalString;
|
||||
if (original.Contains('#'))
|
||||
{
|
||||
message = "WebSocket URI must not contain the fragment identifier: " + original;
|
||||
message = "WebSocket URI must not contain a fragment component: " + original;
|
||||
return false;
|
||||
}
|
||||
|
||||
|
@ -297,23 +297,6 @@ namespace WebSocketSharp {
|
||||
get {
|
||||
return _readyState;
|
||||
}
|
||||
|
||||
private set {
|
||||
_readyState = value;
|
||||
|
||||
switch (value)
|
||||
{
|
||||
case WsState.OPEN:
|
||||
startMessageThread();
|
||||
OnOpen.Emit(this, EventArgs.Empty);
|
||||
break;
|
||||
case WsState.CLOSING:
|
||||
break;
|
||||
case WsState.CLOSED:
|
||||
closeConnection();
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
@ -379,13 +362,27 @@ namespace WebSocketSharp {
|
||||
string msg;
|
||||
if (!isValidRequest(req, out msg))
|
||||
{
|
||||
error(msg);
|
||||
onError(msg);
|
||||
close(CloseStatusCode.HANDSHAKE_FAILURE, msg);
|
||||
return;
|
||||
}
|
||||
|
||||
sendResponseHandshake();
|
||||
ReadyState = WsState.OPEN;
|
||||
onOpen();
|
||||
}
|
||||
|
||||
private bool canSendAsCloseFrame(PayloadData data)
|
||||
{
|
||||
if (data.Length >= 2)
|
||||
{
|
||||
var code = data.ToBytes().SubArray(0, 2).To<ushort>(ByteOrder.BIG);
|
||||
if (code == (ushort)CloseStatusCode.NO_STATUS_CODE ||
|
||||
code == (ushort)CloseStatusCode.ABNORMAL ||
|
||||
code == (ushort)CloseStatusCode.HANDSHAKE_FAILURE)
|
||||
return false;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
private void close(HttpStatusCode code)
|
||||
@ -394,7 +391,7 @@ namespace WebSocketSharp {
|
||||
return;
|
||||
|
||||
sendResponseHandshake(code);
|
||||
ReadyState = WsState.CLOSED;
|
||||
closeConnection();
|
||||
}
|
||||
|
||||
private void close(PayloadData data)
|
||||
@ -404,23 +401,31 @@ namespace WebSocketSharp {
|
||||
#endif
|
||||
lock(_forClose)
|
||||
{
|
||||
// Whether the closing handshake has been started already ?
|
||||
if (_readyState == WsState.CLOSING ||
|
||||
_readyState == WsState.CLOSED)
|
||||
return;
|
||||
|
||||
// Whether the closing handshake as server is started before the connection has been established ?
|
||||
if (_readyState == WsState.CONNECTING && !_isClient)
|
||||
{
|
||||
OnClose.Emit(this, new CloseEventArgs(data));
|
||||
close(HttpStatusCode.BadRequest);
|
||||
sendResponseHandshake(HttpStatusCode.BadRequest);
|
||||
onClose(new CloseEventArgs(data));
|
||||
|
||||
return;
|
||||
}
|
||||
|
||||
ReadyState = WsState.CLOSING;
|
||||
// Whether a close status code that must not be set for send is used ?
|
||||
if (!canSendAsCloseFrame(data))
|
||||
{
|
||||
onClose(new CloseEventArgs(data));
|
||||
return;
|
||||
}
|
||||
|
||||
_readyState = WsState.CLOSING;
|
||||
}
|
||||
|
||||
OnClose.Emit(this, new CloseEventArgs(data));
|
||||
var frame = createFrame(Fin.FINAL, Opcode.CLOSE, data);
|
||||
closeHandshake(frame);
|
||||
closeHandshake(data);
|
||||
#if DEBUG
|
||||
Console.WriteLine("WS: Info@close: Exit close method.");
|
||||
#endif
|
||||
@ -428,7 +433,12 @@ namespace WebSocketSharp {
|
||||
|
||||
private void close(CloseStatusCode code, string reason)
|
||||
{
|
||||
var data = new List<byte>(((ushort)code).ToBytes(ByteOrder.BIG));
|
||||
close((ushort)code, reason);
|
||||
}
|
||||
|
||||
private void close(ushort code, string reason)
|
||||
{
|
||||
var data = new List<byte>(code.ToBytes(ByteOrder.BIG));
|
||||
if (!String.IsNullOrEmpty(reason))
|
||||
{
|
||||
var buffer = Encoding.UTF8.GetBytes(reason);
|
||||
@ -439,42 +449,56 @@ namespace WebSocketSharp {
|
||||
if (payloadData.Length > 125)
|
||||
{
|
||||
var msg = "Close frame must have a payload length of 125 bytes or less.";
|
||||
error(msg);
|
||||
onError(msg);
|
||||
return;
|
||||
}
|
||||
|
||||
close(payloadData);
|
||||
}
|
||||
|
||||
private void closeConnection()
|
||||
private bool closeConnection()
|
||||
{
|
||||
if (_baseContext != null)
|
||||
{
|
||||
_baseContext.Response.Close();
|
||||
_wsStream = null;
|
||||
_baseContext = null;
|
||||
}
|
||||
_readyState = WsState.CLOSED;
|
||||
|
||||
if (_wsStream != null)
|
||||
try
|
||||
{
|
||||
_wsStream.Dispose();
|
||||
_wsStream = null;
|
||||
}
|
||||
if (_baseContext != null)
|
||||
{
|
||||
_baseContext.Response.Close();
|
||||
_wsStream = null;
|
||||
_baseContext = null;
|
||||
}
|
||||
|
||||
if (_tcpClient != null)
|
||||
if (_wsStream != null)
|
||||
{
|
||||
_wsStream.Dispose();
|
||||
_wsStream = null;
|
||||
}
|
||||
|
||||
if (_tcpClient != null)
|
||||
{
|
||||
_tcpClient.Close();
|
||||
_tcpClient = null;
|
||||
}
|
||||
|
||||
return true;
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
_tcpClient.Close();
|
||||
_tcpClient = null;
|
||||
onError(ex.Message);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
private void closeHandshake(WsFrame frame)
|
||||
private void closeHandshake(PayloadData data)
|
||||
{
|
||||
var args = new CloseEventArgs(data);
|
||||
var frame = createFrame(Fin.FINAL, Opcode.CLOSE, data);
|
||||
if (send(frame) && !Thread.CurrentThread.IsBackground)
|
||||
if (_exitMessageLoop != null)
|
||||
_exitMessageLoop.WaitOne(5 * 1000);
|
||||
|
||||
ReadyState = WsState.CLOSED;
|
||||
onClose(args);
|
||||
}
|
||||
|
||||
private void createClientStream()
|
||||
@ -551,22 +575,30 @@ namespace WebSocketSharp {
|
||||
string msg;
|
||||
if (!isValidResponse(res, out msg))
|
||||
{
|
||||
error(msg);
|
||||
onError(msg);
|
||||
close(CloseStatusCode.HANDSHAKE_FAILURE, msg);
|
||||
return;
|
||||
}
|
||||
|
||||
ReadyState = WsState.OPEN;
|
||||
onOpen();
|
||||
}
|
||||
|
||||
private void error(string message)
|
||||
private bool isValidCloseStatusCode(ushort code, out string message)
|
||||
{
|
||||
#if DEBUG
|
||||
var callerFrame = new StackFrame(1);
|
||||
var caller = callerFrame.GetMethod();
|
||||
Console.WriteLine("WS: Error@{0}: {1}", caller.Name, message);
|
||||
#endif
|
||||
OnError.Emit(this, new ErrorEventArgs(message));
|
||||
if (code < 1000)
|
||||
{
|
||||
message = "Close status codes in the range 0-999 are not used: " + code;
|
||||
return false;
|
||||
}
|
||||
|
||||
if (code > 4999)
|
||||
{
|
||||
message = "Out of reserved close status code range: " + code;
|
||||
return false;
|
||||
}
|
||||
|
||||
message = String.Empty;
|
||||
return true;
|
||||
}
|
||||
|
||||
private bool isValidRequest(RequestHandshake request, out string message)
|
||||
@ -714,9 +746,7 @@ namespace WebSocketSharp {
|
||||
{
|
||||
try
|
||||
{
|
||||
var eventArgs = receive();
|
||||
if (eventArgs != null)
|
||||
OnMessage.Emit(this, eventArgs);
|
||||
onMessage(receive());
|
||||
}
|
||||
catch (WsReceivedTooBigMessageException ex)
|
||||
{
|
||||
@ -741,13 +771,44 @@ namespace WebSocketSharp {
|
||||
_exitMessageLoop.Set();
|
||||
}
|
||||
|
||||
private void onClose(CloseEventArgs eventArgs)
|
||||
{
|
||||
if (closeConnection())
|
||||
eventArgs.WasClean = true;
|
||||
|
||||
OnClose.Emit(this, eventArgs);
|
||||
}
|
||||
|
||||
private void onError(string message)
|
||||
{
|
||||
#if DEBUG
|
||||
var callerFrame = new StackFrame(1);
|
||||
var caller = callerFrame.GetMethod();
|
||||
Console.WriteLine("WS: Error@{0}: {1}", caller.Name, message);
|
||||
#endif
|
||||
OnError.Emit(this, new ErrorEventArgs(message));
|
||||
}
|
||||
|
||||
private void onMessage(MessageEventArgs eventArgs)
|
||||
{
|
||||
if (eventArgs != null)
|
||||
OnMessage.Emit(this, eventArgs);
|
||||
}
|
||||
|
||||
private void onOpen()
|
||||
{
|
||||
_readyState = WsState.OPEN;
|
||||
startMessageThread();
|
||||
OnOpen.Emit(this, EventArgs.Empty);
|
||||
}
|
||||
|
||||
private bool ping(string data, int millisecondsTimeout)
|
||||
{
|
||||
var buffer = Encoding.UTF8.GetBytes(data);
|
||||
if (buffer.Length > 125)
|
||||
{
|
||||
var msg = "Ping frame must have a payload length of 125 bytes or less.";
|
||||
error(msg);
|
||||
onError(msg);
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -939,7 +1000,7 @@ namespace WebSocketSharp {
|
||||
_readyState == WsState.CLOSED)
|
||||
{
|
||||
var msg = "Connection isn't established or has been closed.";
|
||||
error(msg);
|
||||
onError(msg);
|
||||
return false;
|
||||
}
|
||||
|
||||
@ -958,7 +1019,7 @@ namespace WebSocketSharp {
|
||||
{
|
||||
_unTransmittedBuffer.Add(frame);
|
||||
var msg = "Current data can not be sent because there is untransmitted data.";
|
||||
error(msg);
|
||||
onError(msg);
|
||||
}
|
||||
|
||||
return false;
|
||||
@ -966,7 +1027,7 @@ namespace WebSocketSharp {
|
||||
catch (Exception ex)
|
||||
{
|
||||
_unTransmittedBuffer.Add(frame);
|
||||
error(ex.Message);
|
||||
onError(ex.Message);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
@ -986,7 +1047,7 @@ namespace WebSocketSharp {
|
||||
if (_readyState != WsState.OPEN)
|
||||
{
|
||||
var msg = "Connection isn't established or has been closed.";
|
||||
error(msg);
|
||||
onError(msg);
|
||||
return;
|
||||
}
|
||||
|
||||
@ -1105,18 +1166,19 @@ namespace WebSocketSharp {
|
||||
#region Public Methods
|
||||
|
||||
/// <summary>
|
||||
/// Sends a Close frame using the connection and closes the connection and releases all associated resources.
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
public void Close()
|
||||
{
|
||||
Close(CloseStatusCode.NORMAL);
|
||||
var data = new PayloadData(new byte[]{});
|
||||
close(data);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sends a Close frame using the connection and closes the connection and releases all associated resources.
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
/// <param name="code">
|
||||
/// A <see cref="WebSocketSharp.Frame.CloseStatusCode"/>.
|
||||
/// A <see cref="WebSocketSharp.Frame.CloseStatusCode"/> that contains a status code indicating a reason for closure.
|
||||
/// </param>
|
||||
public void Close(CloseStatusCode code)
|
||||
{
|
||||
@ -1124,16 +1186,48 @@ namespace WebSocketSharp {
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sends a Close frame using the connection and closes the connection and releases all associated resources.
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
/// <param name="code">
|
||||
/// A <see cref="WebSocketSharp.Frame.CloseStatusCode"/>.
|
||||
/// A <see cref="ushort"/> that contains a status code indicating a reason for closure.
|
||||
/// </param>
|
||||
public void Close(ushort code)
|
||||
{
|
||||
Close(code, String.Empty);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
/// <param name="code">
|
||||
/// A <see cref="WebSocketSharp.Frame.CloseStatusCode"/> that contains a status code indicating a reason for closure.
|
||||
/// </param>
|
||||
/// <param name="reason">
|
||||
/// A <see cref="string"/> that contains the reason why closes.
|
||||
/// A <see cref="string"/> that contains a reason for closure.
|
||||
/// </param>
|
||||
public void Close(CloseStatusCode code, string reason)
|
||||
{
|
||||
Close((ushort)code, reason);
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
/// <param name="code">
|
||||
/// A <see cref="ushort"/> that contains a status code indicating a reason for closure.
|
||||
/// </param>
|
||||
/// <param name="reason">
|
||||
/// A <see cref="string"/> that contains a reason for closure.
|
||||
/// </param>
|
||||
public void Close(ushort code, string reason)
|
||||
{
|
||||
string msg;
|
||||
if (!isValidCloseStatusCode(code, out msg))
|
||||
{
|
||||
onError(msg);
|
||||
return;
|
||||
}
|
||||
|
||||
close(code, reason);
|
||||
}
|
||||
|
||||
@ -1163,13 +1257,13 @@ namespace WebSocketSharp {
|
||||
}
|
||||
catch (Exception ex)
|
||||
{
|
||||
error(ex.Message);
|
||||
onError(ex.Message);
|
||||
close(CloseStatusCode.HANDSHAKE_FAILURE, "An exception has occured.");
|
||||
}
|
||||
}
|
||||
|
||||
/// <summary>
|
||||
/// Sends a Close frame using the connection and closes the connection and releases all associated resources.
|
||||
/// Closes the connection and releases all associated resources after sends a Close control frame.
|
||||
/// </summary>
|
||||
/// <remarks>
|
||||
/// Call <see cref="Dispose"/> when you are finished using the <see cref="WebSocketSharp.WebSocket"/>. The
|
||||
|
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Binary file not shown.
Loading…
Reference in New Issue
Block a user