Después de hacer algunos (más o menos) socket
programación asíncrona de "bajo nivel" hace años (en forma de patrón asincrónico basado en eventos (EAP) ) y recientemente "subir" a un TcpListener
(modelo de programación asincrónica (APM) ) y luego tratando de pasar a async/await
(Patrón asíncrono basado en tareas (TAP) ) Casi lo he tenido con tener que molestarme con toda esta 'fontanería de bajo nivel'. Entonces estaba pensando; ¿por qué no intentarloRX
( extensiones reactivas ) ya que podría ajustarse más cómodamente a mi dominio problemático?
Una gran cantidad de código que escribo tiene que ver con muchos clientes que se conectan a través de Tcp a mi aplicación que luego inicia una comunicación bidireccional (asíncrona). El cliente o el servidor pueden decidir en cualquier momento que se debe enviar un mensaje y hacerlo, por lo que esta no es su request/response
configuración clásica , sino más bien una "línea" bidireccional en tiempo real abierta a ambas partes para enviar lo que quieran , cuando quieran. (¡Si alguien tiene un nombre decente para describir esto, me encantaría escucharlo!).
El "protocolo" difiere según la aplicación (y no es realmente relevante para mi pregunta). Sí, sin embargo, tengo una pregunta inicial:
- Dado que solo se está ejecutando un "servidor", pero tiene que realizar un seguimiento de muchas (generalmente miles) de conexiones (por ejemplo, clientes) que tienen (por falta de una mejor descripción) su propia "máquina de estado" para realizar un seguimiento de sus estados, etc., ¿qué enfoque preferirías? EAP / TAP / APM? ¿RX incluso se consideraría una opción? Si no, ¿por qué?
Por lo tanto, necesito trabajar Async ya que a) no es un protocolo de solicitud / respuesta, por lo que no puedo tener un hilo / cliente en una llamada de bloqueo de "mensaje en espera" o llamada de bloqueo de "envío de mensaje" (sin embargo, si el envío es bloqueando para ese cliente solo yo podría vivir con él) yb) necesito manejar muchas conexiones concurrentes. No veo forma de hacerlo (de manera confiable) usando llamadas de bloqueo.
La mayoría de mis aplicaciones están relacionadas con VoiP; ya sean mensajes SIP de clientes SIP o mensajes PBX (relacionados) de aplicaciones como FreeSwitch / OpenSIPS, etc. pero puede, en su forma más simple, tratar de imaginar un servidor de "chat" tratando de manejar muchos clientes de "chat". La mayoría de los protocolos están basados en texto (ASCII).
Entonces, después de haber implementado muchas permutaciones diferentes de las técnicas antes mencionadas, me gustaría simplificar mi trabajo creando un objeto que pueda simplemente crear instancias, decirle sobre qué IPEndpoint
escuchar y que me diga cuando ocurra algo de interés (lo cual generalmente use eventos para, por lo que algunos EAP generalmente se mezclan con las otras dos técnicas). La clase no debería molestarse en tratar de 'entender' el protocolo; simplemente debe manejar cadenas entrantes / salientes. Y así, teniendo mi ojo en RX esperando que eso (al final) simplificara el trabajo, creé un nuevo "violín" desde cero:
using System;
using System.Collections.Concurrent;
using System.Net;
using System.Net.Sockets;
using System.Reactive.Linq;
using System.Text;
class Program
{
static void Main(string[] args)
{
var f = new FiddleServer(new IPEndPoint(IPAddress.Any, 8084));
f.Start();
Console.ReadKey();
f.Stop();
Console.ReadKey();
}
}
public class FiddleServer
{
private TcpListener _listener;
private ConcurrentDictionary<ulong, FiddleClient> _clients;
private ulong _currentid = 0;
public IPEndPoint LocalEP { get; private set; }
public FiddleServer(IPEndPoint localEP)
{
this.LocalEP = localEP;
_clients = new ConcurrentDictionary<ulong, FiddleClient>();
}
public void Start()
{
_listener = new TcpListener(this.LocalEP);
_listener.Start();
Observable.While(() => true, Observable.FromAsync(_listener.AcceptTcpClientAsync)).Subscribe(
//OnNext
tcpclient =>
{
//Create new FSClient with unique ID
var fsclient = new FiddleClient(_currentid++, tcpclient);
//Keep track of clients
_clients.TryAdd(fsclient.ClientId, fsclient);
//Initialize connection
fsclient.Send("connect\n\n");
Console.WriteLine("Client {0} accepted", fsclient.ClientId);
},
//OnError
ex =>
{
},
//OnComplete
() =>
{
Console.WriteLine("Client connection initialized");
//Accept new connections
_listener.AcceptTcpClientAsync();
}
);
Console.WriteLine("Started");
}
public void Stop()
{
_listener.Stop();
Console.WriteLine("Stopped");
}
public void Send(ulong clientid, string rawmessage)
{
FiddleClient fsclient;
if (_clients.TryGetValue(clientid, out fsclient))
{
fsclient.Send(rawmessage);
}
}
}
public class FiddleClient
{
private TcpClient _tcpclient;
public ulong ClientId { get; private set; }
public FiddleClient(ulong id, TcpClient tcpclient)
{
this.ClientId = id;
_tcpclient = tcpclient;
}
public void Send(string rawmessage)
{
Console.WriteLine("Sending {0}", rawmessage);
var data = Encoding.ASCII.GetBytes(rawmessage);
_tcpclient.GetStream().WriteAsync(data, 0, data.Length); //Write vs WriteAsync?
}
}
Soy consciente de que, en este "violín", hay un pequeño detalle específico de implementación; en este caso, estoy trabajando con FreeSwitch ESL, por lo que "connect\n\n"
debería eliminarse en el violín cuando se refactoriza a un enfoque más genérico.
También soy consciente de que necesito refactorizar los métodos anónimos a métodos de instancia privada en la clase Servidor; No estoy seguro de qué convención (por ejemplo, " OnSomething
" por ejemplo) usar para sus nombres de método.
Esta es mi base / punto de partida / base (que necesita algunos "ajustes"). Tengo algunas preguntas sobre esto:
- Ver la pregunta anterior "1"
- ¿Estoy en el camino correcto? ¿O son injustas mis decisiones de "diseño"?
- Concurrencia: esto podría hacer frente a miles de clientes (analizando / manejando los mensajes reales a un lado)
- En excepciones: no estoy seguro de cómo obtener excepciones generadas dentro de los clientes "hasta" al servidor ("RX-wise"); ¿Cuál sería una buena manera?
- Ahora puedo obtener cualquier cliente conectado de mi clase de servidor (usando su
ClientId
), suponiendo que exponga a los clientes de una forma u otra, y llame a los métodos directamente. También puedo llamar a métodos a través de la clase Servidor (por ejemplo, elSend(clientId, rawmessage)
método (mientras que este último enfoque sería un método "conveniente" para enviar rápidamente un mensaje al otro lado). - No estoy muy seguro de dónde (y cómo) ir desde aquí:
- a) Necesito manejar los mensajes entrantes; ¿Cómo iba a configurar esto? Puedo obtener el flujo de curso, pero ¿ dónde manejaría la recuperación de los bytes recibidos? Creo que necesito algún tipo de "ObservableStream", ¿algo a lo que pueda suscribirme? ¿Pondría esto en el
FiddleClient
oFiddleServer
? - b) Suponiendo que quiero evitar el uso de eventos hasta que estas
FiddleClient
/FiddleServer
clases se implementen más específicamente para adaptar su manejo de protocolo específico de la aplicación, etc., usando clasesFooClient
/ más específicasFooServer
: ¿cómo pasaría de obtener los datos en las clases 'Fiddle' subyacentes a sus contrapartes más específicas?
- a) Necesito manejar los mensajes entrantes; ¿Cómo iba a configurar esto? Puedo obtener el flujo de curso, pero ¿ dónde manejaría la recuperación de los bytes recibidos? Creo que necesito algún tipo de "ObservableStream", ¿algo a lo que pueda suscribirme? ¿Pondría esto en el
Artículos / enlaces que ya leí / hojeé / utilicé como referencia:
- Consume un zócalo con la extensión reactiva
- Crear y suscribirse a secuencias observables simples
- Cómo agregar o asociar contexto a cada conexión / sesión de ObservableTcpListener
- Programación asincrónica con el marco reactivo y la biblioteca paralela de tareas - Parte 1 , 2 y 3 .
- ¿Es práctico usar Extensiones reactivas (Rx) para la programación de sockets?
- Un servidor web controlado por .NET Rx y un servidor web controlado por .NET Rx, Take 2
- Algunos tutoriales de Rx