আমি কীভাবে সিগন্যালআর সংযোগটি হত্যা করতে পারি?


10

আমি কোনও ওয়েবসাইটে ডেটা স্থানান্তর করতে সিগন্যালআর ব্যবহার করছি। তবে সিগন্যালআর কেবলমাত্র সময়ের জন্য ডেটা প্রেরণ করতে সক্ষম হওয়া উচিত এবং যদি সময়সীমাটি পার হয়ে যায় তবে সংযোগটি হত্যা করা উচিত।

$.connection.hub.stop()কোনও অনুরোধ এখনও মুলতুবি থাকা এবং সম্পন্ন না হলে স্টপ-ফাংশন বাতিল করা হয়। তবে এই অনুরোধটি যতটুকু ডেটা প্রেরণ করা হয়েছে তা বাতিল করতে বাধ্য করা উচিত।

আমি কীভাবে সিগন্যালআর-সংযোগটি মেরে ফেলতে পারি ?

উত্তর:


6

যেমন আপনি এই মাইক্রোসফ্ট ডকুমেন্টেশনে টাইমআউট এবং রক্ষণাবেক্ষণ সেটিংস সম্পর্কে দেখতে পাচ্ছেন আপনি বিকল্পগুলিতে সংযোগের সময়টি নির্ধারণ করতে পারেন।

উদাহরণ:

protected void Application_Start(object sender, EventArgs e)
{
    // Make long-polling connections wait a maximum of 110 seconds for a
    // response. When that time expires, trigger a timeout command and
    // make the client reconnect.
    GlobalHost.Configuration.ConnectionTimeout = TimeSpan.FromSeconds(110);

    // Wait a maximum of 30 seconds after a transport connection is lost
    // before raising the Disconnected event to terminate the SignalR connection.
    GlobalHost.Configuration.DisconnectTimeout = TimeSpan.FromSeconds(30);

    // For transports other than long polling, send a keepalive packet every
    // 10 seconds. 
    // This value must be no more than 1/3 of the DisconnectTimeout value.
    GlobalHost.Configuration.KeepAlive = TimeSpan.FromSeconds(10);

    RouteTable.Routes.MapHubs();
}

সম্পাদনা করুন : যেহেতু আপনি ক্লায়েন্টের কাছ থেকে সংযোগটি হত্যা করতে চান না কেন, আপনি একটি CancellationTokenআচরণের কথা বলছেন তবে দুর্ভাগ্যক্রমে এটি এখনও সিগন্যালআর তে সমর্থনযোগ্য নয় কারণ আপনি এখানে এবং এখানে দেখতে পাচ্ছেন , দলটি এটি করতে চায় SignalRতবে এখনও আছে এটি সম্পর্কে কোন খবর।


যেমনটি আমি বলেছি ফ্রন্টএন্ড-সাইট থেকে অনুরোধটি সম্পন্ন হয়নি, তাই এখনও সীমান্ত থেকে কিছু তথ্য সিগন্যালআর-ব্যাকএন্ড / হাবের কাছে প্রেরণ করা দরকার। সুতরাং আমি একটি সীমা-সমাধানের সন্ধান করছি কারণ একটি শালীন পরিমাণের ডেটা প্রেরণ করা হয় এবং যদি সময়সীমা অতিক্রান্ত হয় তবে ডেটা সংক্রমণ করা হয়েছে কি না তা সংযোগটি সীমান্ত দিয়ে হত্যা করা উচিত। আমি কি খুঁজছি বুঝতে পারছেন?
স্নিকব্র্যাক

@ স্নিকব্র্যাক আপনি কি ক্লায়েন্ট-সাইডের মাধ্যমে সংযোগটি মেরে ফেলতে চান, আপনি এখনই ডেটা প্রেরণ করছেন, আমি ঠিক আছি?
কিরিল 1512

1
হ্যাঁ. আপনি সঠিক.
স্নিকব্র্যাক

@ স্নিকব্র্যাক আমার উত্তর আপডেট করেছে।
কিরিল 1512

@ স্নিকব্র্যাক আপনার প্রশ্নের সঠিক উত্তরটি, এটি বা এখানে অন্য উত্তরগুলি নির্বাচন করতে ভুলবেন না ...
Kiril1512

1

হাবের জীবনকাল ইভেন্ট সম্পর্কে এই মাইক্রোসফ্ট ডকুমেন্টটি পড়ুন । আপনি এই সেটিংসের জন্য ডিফল্ট মান পরিবর্তন করতে পারেন, সেগুলি Application_Startআপনার Global.asaxফাইলে সেট করুন । তবে এইভাবে আপনি ক্লায়েন্টের সম্পূর্ণ নিয়ন্ত্রণ করতে পারবেন না। সুতরাং আপনি জাভাস্ক্রিপ্ট setTimeoutফাংশনটি ব্যবহার করেন এবং যখন কোনও নতুন ব্যবহারকারী কানেক্ট.আইটি হতে পারে GlobalHost.Configuration.DisconnectTimeoutবা যে কোনও সময় আপনি চান তখন টাইম ফর্ম সার্ভারটি শেষ করে দিন । আমি ডেমো প্রকল্পের সাথে একটি পূর্ণ উদাহরণ দিই। প্রকৃতপক্ষে রিয়েল-টাইম হোল্ডিং টিকিটের জন্য আমি খুব বড় টিকিট পদ্ধতিতে এই যুক্তিটি ব্যবহার করি। (দয়া করে সমস্ত ইনলাইন মন্তব্য পড়ুন)

মডেল:

public class MyModel
{
    public int Id { get; set; }

    public string Name { get; set; }


    public static string Send(MyModel my)
    {
        //Do Somthing           
        return $"Data Sending to {my.Name}...";
    }
    public static string Stop(string name)
    {
        //Do Somthing

        return $"ForceStop {name}.";
    }
    public static string Delete()
    {
        //Do Somthing

        return "Deleted";
    }
}

হাব:

[HubName("myHub")]
public class MyHub : Hub
{
    int connectionTimeOut = 10;//sec

    [HubMethodName("connect")]
    public void Connect()
    {  
            //apply logic if any when user connected or reload page
            //set connection Time Out as you need
        connectionTimeOut= 10;// GlobalHost.Configuration.DisconnectTimeout

       Clients.Client(Context.ConnectionId).onNewUserConnected(connectionTimeOut);
    }
    [HubMethodName("startSendingServer")]
    public void StartSending(int id, string name)//pass anything you need
    {
        //apply logic if any when start sending data

        var my = new MyModel
        {
            Id = id,
            Name = name
        };
        var status = MyModel.Send(my);//example

        var result = new
        {
            status,
            name
        };

        Clients.Client(Context.ConnectionId).startSendingClient(result);

    }

    [HubMethodName("forceStopServer")]
    public void ForceStop(string name)//pass anything you need
    {
        //apply logic if any when force stop sending data
        var status = MyModel.Stop(name);
        Clients.Client(Context.ConnectionId).forceStopClint(status);
    }


    public override Task OnDisconnected(bool stopCalled)
    {

        //apply logic if any when connection Disconnected

        var status = MyModel.Delete();//example
        if (stopCalled)
        {
            //  status=String.Format("Client {0} explicitly closed the connection.", Context.ConnectionId)
            //your code here
        }
        else
        {
            // status=String.Format("Client {0} timed out .", Context.ConnectionId);
            //your code here
            //Clients.Client(Context.ConnectionId).onUserDisconnected(status);
        }

        return base.OnDisconnected(stopCalled);
    }


}

TestView:

<div class="row">
    <div class="col-md-12">
        <h1> Status: <span id="hubStatus"></span></h1>
        <br />
        <h4> Countdown : <span id="counter"></span></h4>
        <br />

        <button id="btnHub" class="btn btn-primary btn-lg">Start Sending Data</button>
    </div>
</div>
@section scripts{
    <script src="~/Scripts/app/hub.js"></script>
}

hub.js:

var proxyTimer = null;
var sendTimeLimit = 1;//sec
var sessionTime = sendTimeLimit * 1000;

$(function () {
    var myProxy = $.connection.myHub;
    $.connection.hub.start().done(function () {
        registerServerEvents(myProxy);
    });

    clientMethods(myProxy);
});

function registerServerEvents(proxyHub) {
    proxyHub.server.connect();
    $(document).on("click", "#btnHub", function (e) {

        $("#hubStatus").html("Sending..");
        $("#btnHub").text("Count Down Start...");

        //Logic Before start sending data.
        var id = 1;
        var name = "AzR";        
        proxyHub.server.startSendingServer(id,name);

       // $.connection.hub.disconnected(function () {
      //  setTimeout(function () { $.connection.hub.start(); }, 5000); // Restart connection after 5 seconds.
       //});

        $.connection.hub.disconnected(function () {
            $("#hubStatus").html("Disconnected");// you can restart on here.     
            $("#btnHub").text("Stat Again after reload window");

        });

    });
}



function clientMethods(proxyHub) {

    //proxyHub.on('onConnected', function (sendTimeLimit) {
    //    sendTimeLimit = sendTimeLimit;
    //});

    proxyHub.on('onNewUserConnected', function (serverItem) {
        sendTimeLimit = serverItem;
        sessionTime = sendTimeLimit * 1000;
    });


    proxyHub.on('startSendingClient', function (serverItem) {

        //Logic after start sending data.
        var name = serverItem.name;
        var status = serverItem.status;
        $("#hubStatus").html(status);
        $("#counter").html(sendTimeLimit);
        timeCounter();
        startTimer(proxyHub, name );
    });

    proxyHub.on('forceStopClint', function (serverItem) {


        clearClintPendingTask(serverItem);//Logic before proxy stop.
        $("#btnHub").text("Force Stop...");
        $.connection.hub.stop();
    });

    proxyHub.on('onUserDisconnected', function (serverItem) {
        //Logic after proxy Disconnected (time out).
        $("#hubStatus").html(serverItem);
        $("#btnHub").text("Stat Again after reload window");
   });
}

//Logic before proxy stop.
function clearClintPendingTask(status) {
    //do all you need
    $("#hubStatus").html(status); 
    stopTimer();
}

function startTimer(proxyHub,data) {
    stopTimer();
    proxyTimer = setTimeout(function () {
        proxyHub.server.forceStopServer(data);
    }, sessionTime);
}

function stopTimer() {
    if (proxyTimer) {
        clearTimeout(proxyTimer);
        proxyTimer = null;
    }
}

function timeCounter() {
    var counter = sendTimeLimit;
    var interval = setInterval(function () {
        counter--;
        $("#counter").html(counter);
        if (counter == 0) {
            //Do something
            $("#counter").html("Countdown ended!");
            // Stop the counter
            clearInterval(interval);
        }
    }, 1000);
}

(পরিক্ষিত)


0

আপনার একটি সময়সীমা নির্ধারণ করতে হবে। সার্ভারে আপনি ডিসকানেক্ট টাইমআউটটি সেট করতে পারেন:

GlobalHost.Configuration.DisconnectTimeout = TimeSpan.FromMinutes(30);

https://zzz.buzz/2016/05/11/setting-timeout-for-signalr-for-easier-debugging/


যেমনটি আমি বলেছি ফ্রন্টএন্ড-সাইট থেকে অনুরোধটি সম্পন্ন হয়নি, তাই এখনও সীমান্ত থেকে কিছু তথ্য সিগন্যালআর-ব্যাকএন্ড / হাবের কাছে প্রেরণ করা দরকার। তাই আমি একটি সীমা-সমাধান খুঁজছি।
স্নিকব্র্যাক

0

আপডেট সম্পাদনা, নীচে বিকল্প 3 দেখুন। বাকি সমস্ত সময়সীমার উপর নির্ভর করছে, আমি একটি জোর করে সংযোগ বিচ্ছিন্ন করে পোস্ট করেছি।

আপনি যদি ফোর্স সংযোগ বিচ্ছিন্ন করার চেষ্টা করে থাকেন - আপনি সংযুক্ত ব্যবহারকারীদের তালিকা পেতে এবং কল করতে পারেন ForceLogOut সার্ভারের পাশে ফাংশনটি , কোড কোডে আমি এটি কোথাও দেখেছি, আমি আশা করি এটি সহায়তা করবে। আপনি যদি কেবলমাত্র কিছু ব্যবহারকারীকে জোরপূর্বক লগআউট / হত্যা করতে চান তবে কেবল সেই সংযোগটি লুপ করুন এবং হত্যা করুন।

সার্ভার সাইড


public class User
{
    public string Name { get; set; }
    public HashSet<string> ConnectionIds { get; set; }
}

public class ExtendedHub : Hub
{        
   private static readonly ConcurrentDictionary<string, User> ActiveUsers  = 
      new ConcurrentDictionary<string, User>(StringComparer.InvariantCultureIgnoreCase);
    public IEnumerable<string> GetConnectedUsers()
    {
        return ActiveUsers.Where(x => {

            lock (x.Value.ConnectionIds)
            {
                return !x.Value.ConnectionIds.Contains
                        (Context.ConnectionId, StringComparer.InvariantCultureIgnoreCase);
            }

        }).Select(x => x.Key);
    }           

    public void forceLogOut(string to)
    {
        User receiver;
        if (ActiveUsers.TryGetValue(to, out receiver))
        {
            IEnumerable<string> allReceivers;
            lock (receiver.ConnectionIds)
            {
                allReceivers = receiver.ConnectionIds.Concat(receiver.ConnectionIds);      
            }

            foreach (var cid in allReceivers)
            {
             // ***************** log out/KILL connection for whom ever your want here
                Clients.Client(cid).Signout();
            }
        }
    }
}

মক্কেলের পক্ষে

 // 1- Save your connection variable when you start it, and later on you can use it to stop.
var myHubProxy = $.connection.myHub 
// 2- Use it when you need to stop it, IF NOT YOU WILL GET AN ERROR
myHubProxy.client.stopClient = function() {
    $.connection.hub.stop();
};

// With a button for testing
$('#SomeButtonKillSignalr').click(function () {
            $.connection.hub.stop();                
        });

অপশন 3 এর সাথে আপডেট হয়েছে : অনুরোধের ভিত্তিতে ... অন্য সমাধানগুলি সময়ের বাইরে নির্ভর করে তবে আপনি নিজেও সংযোগটি ডিসপোজ করে সরাসরি এটিকে জোর করতে পারেন

আমি সিগন্যালআর কোডটি খোলার চেষ্টা করেছি এবং ভিতরে আপনি দেখতে পাচ্ছেন DisposeAndRemoveAsync ক্লায়েন্ট সংযোগের প্রকৃত সমাপ্তি ।

1- আপনি পরিবর্তন বা কল করতে পারেন DisposeAndRemoveAsync আপনার সংযোগটি ।

2- তারপরে কল করুন RemoveConnection(connection.ConnectionId);

public async Task DisposeAndRemoveAsync(HttpConnectionContext connection)
        {
            try
            {
                // this will force it
                await connection.DisposeAsync();
            }
            catch (IOException ex)
            {
                _logger.ConnectionReset(connection.ConnectionId, ex);
            }
            catch (WebSocketException ex) when (ex.InnerException is IOException)
            {
                _logger.ConnectionReset(connection.ConnectionId, ex);
            }
            catch (Exception ex)
            {
                _logger.FailedDispose(connection.ConnectionId, ex);
            }
            finally
            {
                // Remove it from the list after disposal so that's it's easy to see
                // connections that might be in a hung state via the connections list
                RemoveConnection(connection.ConnectionId);
            }
        }

সাবধানতা, এটি সম্পন্ন হওয়ার পরে কোনও নিজেকে পরিষ্কার করুন।


আমি যেমন বলেছি- $.connection.hub.stop()ফাংশনটি একটি ত্রুটি ছুড়ে ফেলে কারণ অনুরোধটি পুরোপুরি ব্যাকএন্ডে প্রেরণ করা হয়নি। সুতরাং আমি এমন একটি সমাধান খুঁজছি যা বর্তমানে চলমান অনুরোধ উপস্থিত থাকলেও বর্তমানে সক্রিয় সংযোগটিকে হত্যা করে।
স্নিকব্র্যাক
আমাদের সাইট ব্যবহার করে, আপনি স্বীকার করেছেন যে আপনি আমাদের কুকি নীতি এবং গোপনীয়তা নীতিটি পড়েছেন এবং বুঝতে পেরেছেন ।
Licensed under cc by-sa 3.0 with attribution required.