Signalr Hubs和Twitter Streaming

所以我'm trying to use Signalr with the Twitter Streaming API, and specifically, for this I' m使用Tweetinvi C#API(http://tweetinvi.codeplex.com/) .

该应用程序的目的是使用某些关键字实时过滤推文到页面 .

TweetInvi库是一种享受,我有一个命令行应用程序成功打印出包含某些关键字的推文 .

我用法的基本概要如下:

我有一个带有单个页面的MVC Web应用程序,带有文本输入和一个按钮(用于更新过滤器),然后调用Signalr Hub中的Hub方法,如果没有已存在的流,则启动流并停止它点击第二个按钮 .

所有这一切都很好,除了信号器部分 .

public class TweetHub : Hub
{
    private IStreamManager _streamManager;

    public void AddTweet(String tweet, double lat, double lon)
    {
        Clients.All.addTweet(tweet, lat, lon);
    }

    public void StartStream(String[] filters)
    {
        string accessToken = ConfigurationManager.AppSettings["AccessToken"];
        string accessTokenSecret = ConfigurationManager.AppSettings["AccessTokenSecret"];
        string consumerKey = ConfigurationManager.AppSettings["ConsumerKey"];
        string consumerSecret = ConfigurationManager.AppSettings["ConsumerSecret"];

        IToken token = new Token(accessToken, accessTokenSecret, consumerKey, consumerSecret);

        if (_streamManager != null && _streamManager.StreamIsOpen())
        {
            _streamManager.StopStream();
            _streamManager.StartStream(token, filters, tweet => AddTweet(tweet.Text, tweet.LocationCoordinates.Lattitude, tweet.LocationCoordinates.Longitude));
        }
        else if (_streamManager != null && !_streamManager.StreamIsOpen())
        {
            _streamManager.StartStream(token, filters, tweet => AddTweet(tweet.Text, tweet.LocationCoordinates.Lattitude, tweet.LocationCoordinates.Longitude));
        }
        else
        {
            _streamManager = new StreamManager();
            _streamManager.StartStream(token, filters, tweet => AddTweet(tweet.Text, tweet.LocationCoordinates.Lattitude, tweet.LocationCoordinates.Longitude));
        }
    }

    public void StopStream()
    {
        if (_streamManager != null && _streamManager.StreamIsOpen())
        {
            _streamManager.StopStream();
        }
    }
}

那是我的Signalr Hub的代码 . 正如我所说,使用js我可以触发启动和停止流方法 .

这是我的StreamManager类的代码:

public class StreamManager : IStreamManager
{
    private StreamClient _streamClient;
    private bool _streamOpen = false;

    public void StartStream(IToken token, String[] filters, Action<ITweet> action)
    {
        if (_streamClient == null)
            _streamClient = new StreamClient(token, filters, new FilteredStream());

        _streamClient.StartStream(action);
        _streamOpen = true;
    }

    public void StopStream()
    {
        if (_streamClient != null)
        {
            _streamClient.StopStream();
            _streamOpen = false;
        }
    }

    public bool StreamIsOpen()
    {
        return _streamOpen;
    }

    public void Dispose()
    {
        if (_streamOpen)
        {
            StopStream();
        }
        _streamClient.Dispose();
        _streamClient = null;
    }
}

我的StreamClient类的代码:

public class StreamClient : IStreamClient
{
    private IFilteredStream _filteredStream;
    private IToken _token;
    private bool _streamOpen = false;


    public StreamClient(IToken token, String[] filters, IFilteredStream filteredStream)
    {
        _token = token;
        _filteredStream = filteredStream;
        AddFilters(filters);
    }

    private void AddFilters(String[] filters)
    {
        for (int i = 0; i < filters.Length; ++i)
        {
            _filteredStream.AddTrack(filters[i]);
        }
    }

    public void StartStream(Action<ITweet> action)
    {
        _filteredStream.StartStream(_token, action);
        _streamOpen = true;

    }

    public void StartStream(Func<ITweet, bool> predicateFunc)
    {
        _filteredStream.StartStream(_token, predicateFunc);
        _streamOpen = true;
    }

    public void StopStream()
    {
        _filteredStream.StopStream();
        _streamOpen = false;
    }

    public void Dispose()
    {
        Dispose(true);
        GC.SuppressFinalize(this);
    }

    protected virtual void Dispose(bool disposing)
    {
        if (disposing)
        {
            // free managed resources
            if (_streamOpen)
            {
                _filteredStream.StopStream();
                _filteredStream = null;
                _token = null;
            }
        }
    }

上面的代码是直接调用Tweetinvi库的地方 .

我的问题是,当我将Hub方法作为Action参数传递给StreamManager的StartStream方法时,AddTweet方法永远不会被命中 .

正如我所说,当使用命令提示符应用程序作为客户端时,这一切都正常,并使用此代码:

static void Main(string[] args)
{

    string accessToken = ConfigurationManager.AppSettings["AccessToken"];
    string accessTokenSecret = ConfigurationManager.AppSettings["AccessTokenSecret"];
    string consumerKey = ConfigurationManager.AppSettings["ConsumerKey"];
    string consumerSecret = ConfigurationManager.AppSettings["ConsumerSecret"];


    IToken token = new Token(accessToken, accessTokenSecret, consumerKey, consumerSecret);

    String[] filters = new string[2]
    {
            "test",
            "twitter"
    };

    StreamClient streamClient = new StreamClient(token, filters, new FilteredStream());
        streamClient.StartStream(tweet => TestMethod());
}

public static void TestMethod()
{
    Console.WriteLine("test");
}

这非常有效,并且在收到这些关键字后打印出推文 .

这让我相信这是我使用Signalr的方式的一个问题,信号器方法永远不会被击中,因为流肯定会被打开,我只是有一个偷偷摸摸的怀疑,这是与生命周期的关系枢纽和我使用它的方式 .

我怀疑这是因为,虽然我的Hub中的StartStream方法被称为正常,并且更新了被单击的按钮,当我再次单击以调用StopStream时,StopStream方法被命中,但我的“_streamManager”成员变量为null,它如果集线器在其生命周期内保持状态,则不应该如此,我想它不会 .

无论是那个或它被处理掉,然后流不再存在 .

我对Signalr没有足够的经验来正确调试 .

在此先感谢您的帮助 .

回答(1)

2 years ago

集线器类的实例是瞬态的 . 这意味着每次调用该集线器类中的方法或发生连接事件时,SignalR都会创建集线器类的实例(例如onConnected,onDisconnected) . 当被调用的方法完成其工作时,该中心实例被处理掉 . 阅读本文:http://www.asp.net/signalr/overview/signalr-20/hubs-api/hubs-api-guide-server#transience

因此,您不应该尝试维护有关集线器类中的连接的状态信息 . 在你的情况下,将是“_streamManager” . 因此,我认为您应该将所有业务逻辑移动到另一个类(不是从Hub派生) . 将它们封装在方法中并从signalR方法中调用它们 .

如果需要从服务器代码中调用集线器中的方法,则应首先获取集线器上下文 . 看看如何做到这一点:http://www.asp.net/signalr/overview/signalr-20/hubs-api/hubs-api-guide-server#callfromoutsidehub

希望这可以帮助!