Метод запроса / ответа на вызов дуплексного сервиса заморозить клиента до истечения времени ожидания - PullRequest
0 голосов
/ 25 августа 2018

Я создал дуплексную службу WCF, которая возвращает данные с внешнего устройства клиенту и позволяет также запрашивать / отвечать на вызовы.

Моя проблема заключается в том, что вызовы на запрос / ответ иногда замораживают клиента до истечения времени ожидания.

Это интерфейсы:

[ServiceContract(CallbackContract = typeof(ITimerServiceCallback), SessionMode = SessionMode.Required)]
public interface ITimerService
{
    [OperationContract]
    void StartTimer();

    [OperationContract]
    void StopTimer();

    [OperationContract]
    void DoOp();
}

public interface ITimerServiceCallback
{
    [OperationContract(IsOneWay = true)]
    void ReportTick(string now);
}

Это реализация:

[ServiceBehavior(InstanceContextMode = InstanceContextMode.PerSession)]
    public class TimerService : ITimerService
    {
        public TimerService()
        {
            _timer = new System.Timers.Timer() { Interval = 500 };
            _timer.Elapsed += Timer_Elapsed;
        }

        private void Timer_Elapsed(object sender, ElapsedEventArgs e)
        {
            _context.GetCallbackChannel<ITimerServiceCallback>().ReportTick(DateTime.Now.ToString());
        }

        private readonly System.Timers.Timer _timer;
        private OperationContext _context;

        public void StartTimer()
        {
            _context = OperationContext.Current;
            _timer.Start();
        }

        public void StopTimer()
        {
            _timer.Stop();
        }

        public void DoOp()
        {
            System.Threading.Thread.Sleep(200);
        }
    }

Шаг для воспроизведения проблемы: * вызов StartTimer * вызов DoOp (один илибольше времени, пока клиент не замерзнет)

Через одну минуту у меня исключение тайм-аута.

1 Ответ

0 голосов
/ 28 августа 2018

Если я использую async / await со стороны клиента, все работает нормально.

ex: метод DoOp вызывается следующим образом:

private async void _btnDoOp_Click(object sender, EventArgs e)
{
    _btnNoOp.Enabled = false;
    try {
        Task task = Task.Run(() => _client.DoOperation());
        await task;
    } catch (Exception ex) {
        MessageBox.Show(ex.Message);
    } finally {
        _btnNoOp.Enabled = true;
    }
}
...