Самостоятельный хостинг Nancyfx не может остановить HttpListener

У меня есть приложение, в котором есть служба отдыха на основе NancyFx, и я нахожу его очень классным фреймворком. Но у меня очень большая проблема с его остановкой. Мой хост-класс выглядит так:

public class RestHost : IStartStopAble
{
    private readonly string _uri;
    private readonly IWindsorContainer _container;
    private NancyHost _host;
    private Thread _wsThread;
    private WindsorBootstrapper _windsorBootstrapper;

    public RestHost(string uri, IWindsorContainer container)
    {
        _uri = uri;
        _container = container;
    }

    public void Start()
    {
        var uri = new Uri(_uri);

        _windsorBootstrapper = new WindsorBootstrapper(_container);
        _host = new NancyHost(uri, _windsorBootstrapper);

        _host.Start();
    }

    public void Stop()
    {
        _host.Dispose();
    }
}

Как видите, я использую Windsor вместе с Нэнси. Все отлично работает, я вызываю Start, Stop и конструктор из основного потока, но когда я удаляю свой хост, я ловлю это:

 HttpListener: HttpListenerException: The I/O operation has been aborted because of either a thread exit or an application request

исключение. Я нашел эту статью: http://maykov.blogspot.ru/2009/02/c-httplistener-httplistenerexception-io.html но я использую основной поток и не понимаю, как я могу "Сохранить" поток, в котором создается HttpListener, а также как я могу его передать к Нэнси.

Я буду очень благодарен за любую помощь, но я даже не знаю, что я могу сделать с исключением HttpListener в ядре .Net framework.


person Alex Voskresenskiy    schedule 28.09.2015    source источник
comment
Вместо этого следует просто использовать собственный хост OWIN.   -  person Phill    schedule 30.09.2015
comment
@Phill Я использую серверы Linux, работает ли там OWIN?   -  person Alex Voskresenskiy    schedule 30.09.2015
comment
Пока что-то не изменилось, работает...   -  person Phill    schedule 30.09.2015


Ответы (2)


Возможно, вы помните SynchronizationContext:

public void Start()
{
    // init _host ...

    _syncContext = SynchronizationContext.Current;
    _host.Start();
}

public void Stop()
{
    if (_syncContext == SynchronizationContext.Current)
    {
        _host.Dispose();
    }
    else
    {
        _syncContext.Post((state) =>
        {
            _host.Dispose();
        }
        , null);
    }
}
person xmedeko    schedule 09.02.2016
comment
Спасибо за ваш ответ! Это работает. В любом случае, теперь я размещаю свой сервис в качестве демона, и Linux правильно управляет моим процессом. - person Alex Voskresenskiy; 10.02.2016
comment
На всякий случай, если у кого-то возникнет аналогичная проблема: в моем приложении Нэнси работала внутри приложения WPF и должна была завершиться при закрытии окна, но вместо этого приложение зависло. Это решение устранило проблему. - person Impworks; 06.03.2018

У меня это не работает из-за работающего в данный момент Quartz Scheduler. В NancyBootstrapper нет виртуального метода для завершения процесса. Итак, вот мое решение:

В NancyBootStrapper.ApplicationStartup я устанавливаю планировщик в поле статического класса (что меня не устраивает) с именем GlobalModule.TaskScheduler:

GlobalModule.TaskScheduler = ScheduleTasksAndStartScheduler(container);
GlobalModule.Logger = _logger;

Вот глобальный модуль:

public static class GlobalModule
{
    public static ILogger Logger;
    public static ITaskScheduler TaskScheduler;

    public static void StopScheduler()
    {
        if (TaskScheduler == null) return;
        try
        {
            TaskScheduler.Stop();
        }
        catch (Exception ex)
        {
            if (Logger != null)
            {
                Logger.Error("TaskScheduler.Stop()", ex);
            }
        }
    }
}

И, наконец, Program.Stop:

private static void Stop(NancyHost host)
    {
        GlobalModule.StopScheduler();
        if (_syncContext == SynchronizationContext.Current)
        {
            host.Dispose();
        }
        else
        {
            _syncContext.Post((state) =>
            {
                host.Dispose();
            }
            , null);
        }
        try
        {
            host.Stop();
        }
        catch (Exception ex)
        {
            Console.WriteLine(string.Format("Exception: {0}\r\n Stack Trace: {1}", ex.Message, ex.StackTrace));
        }
    }

С уважением...

person efaruk    schedule 29.07.2016