Przykład aplikacji SignalR Console

84

Czy istnieje mały przykład aplikacji konsoli lub winform używającej signalR do wysyłania wiadomości do koncentratora .net? Wypróbowałem przykłady .net i spojrzałem na wiki, ale nie ma dla mnie sensu związek między hubem (.net) a klientem (aplikacją konsoli) (nie mogłem znaleźć tego przykładu). Czy aplikacja potrzebuje tylko adresu i nazwy koncentratora, aby się połączyć?

Gdyby ktoś mógł podać małą ciekawostkę kodu pokazującego aplikację łączącą się z hubem i wysyłającą „Hello World” lub coś, co otrzymuje hub .net?

PS. Mam standardowy przykład czatu w centrum, który działa dobrze, jeśli spróbuję przypisać do niego nazwę koncentratora w Cs, przestaje działać, tj. [HubName ("test")], czy znasz przyczynę tego ?.

Dzięki.

Bieżący kod aplikacji konsoli.

static void Main(string[] args)
{
    //Set connection
    var connection = new HubConnection("http://localhost:41627/");
    //Make proxy to hub based on hub name on server
    var myHub = connection.CreateProxy("chat");
    //Start connection
    connection.Start().ContinueWith(task =>
    {
        if (task.IsFaulted)
        {
            Console.WriteLine("There was an error opening the connection:{0}", task.Exception.GetBaseException());
        }
        else
        {
            Console.WriteLine("Connected");
        }
    }).Wait();

    //connection.StateChanged += connection_StateChanged;

    myHub.Invoke("Send", "HELLO World ").ContinueWith(task => {
        if(task.IsFaulted)
        {
            Console.WriteLine("There was an error calling send: {0}",task.Exception.GetBaseException());
        }
        else
        {
            Console.WriteLine("Send Complete.");
        }
    });
 }

Serwer centralny. (inny obszar roboczy projektu)

public class Chat : Hub
{
    public void Send(string message)
    {
        // Call the addMessage method on all clients
        Clients.addMessage(message);
    }
}

Informacje Wiki na ten temat to http://www.asp.net/signalr/overview/signalr-20/hubs-api/hubs-api-guide-net-client

user685590
źródło
OK, właściwie to zadziałało, pomyślałem, że otrzymuję ten sam wynik, po prostu dodałem kilka punktów zatrzymania i Console.ReadLine (); na końcu. Wydać okrzyk radości!.
user685590

Odpowiedzi:

110

Przede wszystkim należy zainstalować SignalR.Host.Self w aplikacji serwera i SignalR.Client w aplikacji klienckiej przez nuget:

PM> Install-Package SignalR.Hosting.Self -Version 0.5.2

PM> Install-Package Microsoft.AspNet.SignalR.Client

Następnie dodaj poniższy kod do swoich projektów;)

(uruchom projekty jako administrator)

Aplikacja konsoli serwera:

using System;
using SignalR.Hubs;

namespace SignalR.Hosting.Self.Samples {
    class Program {
        static void Main(string[] args) {
            string url = "http://127.0.0.1:8088/";
            var server = new Server(url);

            // Map the default hub url (/signalr)
            server.MapHubs();

            // Start the server
            server.Start();

            Console.WriteLine("Server running on {0}", url);

            // Keep going until somebody hits 'x'
            while (true) {
                ConsoleKeyInfo ki = Console.ReadKey(true);
                if (ki.Key == ConsoleKey.X) {
                    break;
                }
            }
        }

        [HubName("CustomHub")]
        public class MyHub : Hub {
            public string Send(string message) {
                return message;
            }

            public void DoSomething(string param) {
                Clients.addMessage(param);
            }
        }
    }
}

Aplikacja konsoli klienta:

using System;
using SignalR.Client.Hubs;

namespace SignalRConsoleApp {
    internal class Program {
        private static void Main(string[] args) {
            //Set connection
            var connection = new HubConnection("http://127.0.0.1:8088/");
            //Make proxy to hub based on hub name on server
            var myHub = connection.CreateHubProxy("CustomHub");
            //Start connection

            connection.Start().ContinueWith(task => {
                if (task.IsFaulted) {
                    Console.WriteLine("There was an error opening the connection:{0}",
                                      task.Exception.GetBaseException());
                } else {
                    Console.WriteLine("Connected");
                }

            }).Wait();

            myHub.Invoke<string>("Send", "HELLO World ").ContinueWith(task => {
                if (task.IsFaulted) {
                    Console.WriteLine("There was an error calling send: {0}",
                                      task.Exception.GetBaseException());
                } else {
                    Console.WriteLine(task.Result);
                }
            });

            myHub.On<string>("addMessage", param => {
                Console.WriteLine(param);
            });

            myHub.Invoke<string>("DoSomething", "I'm doing something!!!").Wait();


            Console.Read();
            connection.Stop();
        }
    }
}
Mehrdad Bahrainy
źródło
Możesz użyć powyższego kodu w aplikacji Windows, ale czy to naprawdę konieczne ?! Nie jestem pewien, co masz na myśli, możesz powiadomić w systemie Windows na inne sposoby.
Mehrdad Bahrainy
1
klient współpracuje z serwerem 0.5.2 do 1.0.0-alpha2 np. Install-Package Microsoft.AspNet.SignalR.Client -version 1.0.0- alpha2 nuget.org/packages/Microsoft.AspNet.SignalR.Client/1.0.0-alpha2 (wersje kodu i SignalR powinny współpracować z .net 4.0 przy użyciu VS2010 SP1) próbowałem dowiedzieć się, dlaczego nie mogę go uruchomić, ostatecznie wypróbowałem klienta z wczesnymi wersjami SignalR.
Nick Giles
miły, bardzo pomocny
Dika Arta Karunia
4
Należy pamiętać, że .On<T>()przed wywołaniem connection.Start()metody należy dodać detektory zdarzeń ( wywołania metod).
nicolocodev
Zasługuje na to, aby podać ten link: docs.microsoft.com/en-us/aspnet/signalr/overview/ ...
Mohammed Noureldin
24

Przykład dla SignalR 2.2.1 (maj 2017)

serwer

Pakiet instalacyjny Microsoft.AspNet.SignalR.SelfHost - wersja 2.2.1

[assembly: OwinStartup(typeof(Program.Startup))]
namespace ConsoleApplication116_SignalRServer
{
    class Program
    {
        static IDisposable SignalR;

        static void Main(string[] args)
        {
            string url = "http://127.0.0.1:8088";
            SignalR = WebApp.Start(url);

            Console.ReadKey();
        }

        public class Startup
        {
            public void Configuration(IAppBuilder app)
            {
                app.UseCors(CorsOptions.AllowAll);

                /*  CAMEL CASE & JSON DATE FORMATTING
                 use SignalRContractResolver from
                /programming/30005575/signalr-use-camel-case

                var settings = new JsonSerializerSettings()
                {
                    DateFormatHandling = DateFormatHandling.IsoDateFormat,
                    DateTimeZoneHandling = DateTimeZoneHandling.Utc
                };

                settings.ContractResolver = new SignalRContractResolver();
                var serializer = JsonSerializer.Create(settings);
                  
               GlobalHost.DependencyResolver.Register(typeof(JsonSerializer),  () => serializer);                
            
                 */

                app.MapSignalR();
            }
        }

        [HubName("MyHub")]
        public class MyHub : Hub
        {
            public void Send(string name, string message)
            {
                Clients.All.addMessage(name, message);
            }
        }
    }
}

Klient

(prawie taka sama jak odpowiedź Mehrdada Bahrainy)

Pakiet instalacyjny Microsoft.AspNet.SignalR.Client - wersja 2.2.1

namespace ConsoleApplication116_SignalRClient
{
    class Program
    {
        private static void Main(string[] args)
        {
            var connection = new HubConnection("http://127.0.0.1:8088/");
            var myHub = connection.CreateHubProxy("MyHub");

            Console.WriteLine("Enter your name");    
            string name = Console.ReadLine();

            connection.Start().ContinueWith(task => {
                if (task.IsFaulted)
                {
                    Console.WriteLine("There was an error opening the connection:{0}", task.Exception.GetBaseException());
                }
                else
                {
                    Console.WriteLine("Connected");

                    myHub.On<string, string>("addMessage", (s1, s2) => {
                        Console.WriteLine(s1 + ": " + s2);
                    });

                    while (true)
                    {
                        Console.WriteLine("Please Enter Message");
                        string message = Console.ReadLine();

                        if (string.IsNullOrEmpty(message))
                        {
                            break;
                        }

                        myHub.Invoke<string>("Send", name, message).ContinueWith(task1 => {
                            if (task1.IsFaulted)
                            {
                                Console.WriteLine("There was an error calling send: {0}", task1.Exception.GetBaseException());
                            }
                            else
                            {
                                Console.WriteLine(task1.Result);
                            }
                        });
                    }
                }

            }).Wait();

            Console.Read();
            connection.Stop();
        }
    }
}
ADOConnection
źródło
4
Nie działa dla mnie ... zgłasza zerowy wyjątek ref w WebApp.Start ()
Fᴀʀʜᴀɴ Aɴᴀᴍ
Może przypadkiem teraz, jak ustawić ustawienia serializacji json (np. CamelCase) globalnie na tym własnym serwerze sygnalizacyjnym?
Xaris Fytrakis
2
@XarisFytrakis super łatwe, jeśli aktualizuję anwser, potrzebujesz narzędzia do rozpoznawania kontraktów stąd: stackoverflow.com/questions/30005575/signalr-use-camel-case oraz DateFormatHandling = DateFormatHandling.IsoDateFormat, jeśli używasz go z js.
ADOConnection
@ADOConnection Dzięki za szybką odpowiedź. Problem występuje teraz podczas wywoływania metody z klienta .net. Na przykład, jeśli jestem w klasie Hub, wywołaj to: HubContext.Clients.All.UpdateMetric (new {Data = "xxx", Something = "yyy"}, nazwa użytkownika); Otrzymuję odpowiedź json z prawidłowymi ustawieniami serializacji (obudowa Camel). Ale jeśli wywołam to z przekazanymi danymi od klienta (klient asp.net) w ten sposób: public void UpdateMetric (metryki obiektu, nazwa użytkownika w postaci ciągu) {HubContext.Clients.All.UpdateMetric (metryki, nazwa użytkownika); wynik na kliencie to nie Camel Cased.
Xaris Fytrakis
2

To jest dla dot net core 2.1 - po wielu próbach i błędach w końcu udało mi się to działać bezbłędnie:

var url = "Hub URL goes here";

var connection = new HubConnectionBuilder()
    .WithUrl($"{url}")
    .WithAutomaticReconnect() //I don't think this is totally required, but can't hurt either
    .Build();

//Start the connection
var t = connection.StartAsync();

//Wait for the connection to complete
t.Wait();

//Make your call - but in this case don't wait for a response 
//if your goal is to set it and forget it
await connection.InvokeAsync("SendMessage", "User-Server", "Message from the server");

Ten kod pochodzi z typowego klienta czatu dla biednych sygnalistów. Problem, na który natknęliśmy się ja i wiele innych osób, polega na ustanowieniu połączenia przed próbą wysłania wiadomości do centrum. Ma to kluczowe znaczenie, dlatego ważne jest, aby poczekać na zakończenie zadania asynchronicznego - co oznacza, że ​​synchronizujemy je, czekając na zakończenie zadania.

dyslexicanaboko
źródło
faktycznie możesz połączyć start i czekać jako połączenie.
StartAsync.Wait