C# 使用Autofac将信号器IHubContext注入服务层
在运行Framework4.72而不是.NETCore的应用程序中,我试图将SignalriHubContext注入到WebAPI2.x服务中。我的解决方案分为三个项目:web、服务和数据。信号集线器位于web层。我有在服务层运行的背景代码,完成后我需要它通过集线器发送mesage。此后台任务不是由控制器启动的 My Global.asax相当标准:C# 使用Autofac将信号器IHubContext注入服务层,c#,asp.net-web-api,dependency-injection,signalr,autofac,C#,Asp.net Web Api,Dependency Injection,Signalr,Autofac,在运行Framework4.72而不是.NETCore的应用程序中,我试图将SignalriHubContext注入到WebAPI2.x服务中。我的解决方案分为三个项目:web、服务和数据。信号集线器位于web层。我有在服务层运行的背景代码,完成后我需要它通过集线器发送mesage。此后台任务不是由控制器启动的 My Global.asax相当标准: protected void Application_Start() { GlobalConfiguration.Configure(W
protected void Application_Start()
{
GlobalConfiguration.Configure(WebApiConfig.Register);
// Set JSON serializer to use camelCase
var json = GlobalConfiguration.Configuration.Formatters.JsonFormatter;
json.SerializerSettings.ContractResolver = new CamelCasePropertyNamesContractResolver();
DIConfig.Setup();
FilterConfig.RegisterGlobalFilters(GlobalFilters.Filters);
var logConfigFilePath = Server.MapPath("~/log4net.config");
log4net.Config.XmlConfigurator.ConfigureAndWatch(new System.IO.FileInfo(logConfigFilePath));
}
我的DIConfig包含:
internal static void Setup()
{
var config = System.Web.Http.GlobalConfiguration.Configuration;
var builder = new ContainerBuilder();
builder.Register(c => new ShopAPDbContext()).AsImplementedInterfaces().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType<ShopAPRepository>().As<IShopAPRepository>().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType<ShopAPService>().As<IShopAPService>().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.AddAutoMapper(typeof(InvoiceMappingProfile).Assembly);
builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
var container = builder.Build();
DependencyResolver.SetResolver(new AutofacDependencyResolver(container));
config.DependencyResolver = new AutofacWebApiDependencyResolver(container);
Hangfire.GlobalConfiguration.Configuration.UseAutofacActivator(container);
}
我的服务层的构造函数如下所示:
public ShopAPService()
{
_shopAPRepository = new ShopAPRepository();
_mapper = new Mapper((IConfigurationProvider)typeof(InvoiceMappingProfile).Assembly);
_hubContext = null; // what here?
}
public ShopAPService(IShopAPRepository shopAPRepository, IMapper mapper, IHubContext hubContext)
{
_shopAPRepository = shopAPRepository;
_mapper = mapper;
_hubContext = hubContext;
}
我知道我需要将IHubContext的一个实例传递到服务中,但到目前为止我还没有成功。正如我所知,第一个构造函数是从控制器以外的任何地方调用服务时使用的构造函数
第一版 好的,我知道所有的东西都应该放在一个容器里。根据反馈并查看这些链接,我创建了一个容器并将其传递。以下是我修改后的启动:
public class Startup
{
public void Configuration(IAppBuilder app)
{
//var config = System.Web.Http.GlobalConfiguration.Configuration;
var container = GetDependencyContainer();
RegisterWebApi(app, container);
RegisterSignalR(app, container);
GlobalHost.DependencyResolver = new AutofacDependencyResolver(container);
HubConfiguration config = new HubConfiguration();
config.Resolver = new AutofacDependencyResolver(container);
//config.DependencyResolver = new AutofacWebApiDependencyResolver(container);
Hangfire.GlobalConfiguration.Configuration.UseAutofacActivator(container);
HangFireDashboardConfig.Configure(app);
}
private IContainer GetDependencyContainer()
{
return AutofacConfig.RegisterModules();
}
private void RegisterWebApi(IAppBuilder app, IContainer container)
{
var configuration = new HttpConfiguration
{
DependencyResolver = new AutofacWebApiDependencyResolver(container)
};
WebApiConfig.Register(configuration);
app.UseAutofacMiddleware(container);
app.UseAutofacWebApi(configuration);
app.UseWebApi(configuration);
}
private void RegisterSignalR(IAppBuilder app, IContainer container)
{
var configuration = new HubConfiguration
{
Resolver = new AutofacDependencyResolver(container)
};
app.MapSignalR(configuration);
}
}
我的AutofacConfig构建容器并执行大部分注册:
internal class AutofacConfig
{
public static IContainer RegisterModules()
{
var builder = new ContainerBuilder();
builder.Register(c => new ShopAPDbContext()).AsImplementedInterfaces().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType<ShopAPRepository>().As<IShopAPRepository>().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType<ShopAPService>().As<IShopAPService>().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterAutoMapper(typeof(InvoiceMappingProfile).Assembly);
builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
// Register Autofac resolver into container to be set into HubConfiguration later
builder.RegisterType<AutofacDependencyResolver>().As<IDependencyResolver>().SingleInstance();
// Register ConnectionManager as IConnectionManager so that you can get hub context via IConnectionManager injected to your service
builder.RegisterType<ConnectionManager>().As<IConnectionManager>().SingleInstance();
builder.RegisterHubs(Assembly.GetExecutingAssembly());
var container = builder.Build();
return container;
}
}
内部类AutofacConfig
{
公共静态IContainer注册表模块()
{
var builder=new ContainerBuilder();
builder.Register(c=>newshopapdbcontext()).AsImplementedInterfaces().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType().As().InstancePerBackgroundJob().InstancePerLifetimeScope();
builder.RegisterType().As().InstancePerBackgroundJob().InstancePerLifetimeScope();
注册自动映射器(typeof(InvoiceMappingProfile).Assembly);
RegisterAppController(Assembly.getExecutionGassembly());
//将Autofac解析器注册到容器中,以便稍后设置到HUB配置中
builder.RegisterType
我的服务方法是从HangFire调用的,没有对中心的引用。如何在无参数构造函数中引用它
public partial class ShopAPService : IShopAPService
{
private static readonly ILog _log = LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
readonly IShopAPRepository _shopAPRepository;
readonly IMapper _mapper;
private IHubContext _hubContext;
public ShopAPService()
{
_shopAPRepository = new ShopAPRepository();
_mapper = new Mapper((IConfigurationProvider)typeof(InvoiceMappingProfile).Assembly);
_hubContext = connectionManager.GetHubContext<MessageHub>();
}
public ShopAPService(IShopAPRepository shopAPRepository, IMapper mapper, IHubContext hubContext)
{
_shopAPRepository = shopAPRepository;
_mapper = mapper;
_hubContext = hubContext;
}
}
public部分类shoppservice:ishoppservice
{
private static readonly ILog\u log=LogManager.GetLogger(System.Reflection.MethodBase.GetCurrentMethod().DeclaringType);
readonly IShopAPRepository(商店预存);;
只读图像映射器;
私有IHubContext hubContext;
公共购物服务()
{
_shoppaprepository=新shoppaprepository();
_映射器=新映射器((IConfigurationProvider)typeof(InvoiceMappingProfile).Assembly);
_hubContext=connectionManager.GetHubContext();
}
public ShoppaService(iShopPrepository ShoppaPrespository、IMapper映射器、IHubContext hubContext)
{
_shopAPRepository=shopAPRepository;
_映射器=映射器;
_hubContext=hubContext;
}
}
您无法直接解析IHubContext
。但您可以解析此接口的通用实现。详细信息请参阅
我只是创建了非常简单的OWIN
(Stratup.cs)实现。
我已经安装了nugget软件包,并使用了注册方法和映射方法。这是标准的方法,解决类型化Hub的实现是可行的
但是,如果希望解析上下文更正确,则需要再添加两个注册:AutofacDependencyResolver和ConnectionManager(更多信息可用)
请查看完整样本:
using Autofac;
using Autofac.Integration.SignalR;
using Autofac.Integration.WebApi;
using Microsoft.AspNet.SignalR;
using Microsoft.Owin;
using Owin;
using System.Reflection;
using System.Web.Http;
[assembly: OwinStartup(typeof(Startup))]
namespace Sample
{
public class Startup
{
public void Configuration(IAppBuilder app)
{
var container = GetDependencyContainer();
RegisterWebApi(app, container);
RegisterSignalR(app, container);
}
private IContainer GetDependencyContainer()
{
var builder = new ContainerBuilder();
AutofacConfig.RegisterModules(builder);
builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
builder.RegisterHubs(Assembly.GetExecutingAssembly());
// Register Autofac resolver into container to be set into HubConfiguration later
builder.RegisterType<AutofacDependencyResolver>().As<IDependencyResolver>().SingleInstance();
// Register ConnectionManager as IConnectionManager so that you can get hub context via IConnectionManager injected to your service
builder.RegisterType<ConnectionManager>().As<IConnectionManager>().SingleInstance();
var container = builder.Build();
return container;
}
private void RegisterWebApi(IAppBuilder app, IContainer container)
{
var configuration = new HttpConfiguration
{
DependencyResolver = new AutofacWebApiDependencyResolver(container)
};
WebApiConfig.Register(configuration);
app.UseAutofacMiddleware(container);
app.UseAutofacWebApi(configuration);
app.UseWebApi(configuration);
}
private void RegisterSignalR(IAppBuilder app, IContainer container)
{
var configuration = new HubConfiguration
{
Resolver = new AutofacDependencyResolver(container)
};
app.MapSignalR(configuration);
}
}
}
注册后,您可以使用集线器注入上下文。类似于它(提供2个选项,您只能使用一个您想要的):
公共帐户控制器(IConnectionManager连接管理器,MaintenanceHub MaintenanceHub)
{
var context=connectionManager.GetHubContext();
var hub=维护hub;
}
AutofacConfig在哪里?@alexandar如何将集线器注入另一个类中的服务?@ConnieDeCinko关于AutofacConfig:注册Autofac模块(数据库模块、基础设施模块等)的只是我的业务逻辑。您对此类不感兴趣。您可以将其删除。@ConnieDeCinko请查看启动类。其中包含AutofacDependencyResolver和ConnectionManager的注册。您可以将IConnectionManager插入到服务类中。您可以从IConnectionManager中提取集线器上下文,类似于:var Context=connectionManager.GetHubContext();
我已经更新了我的代码,但是我仍然无法引用服务中没有任何参数调用的中心。当HangFire完成后台任务时,将调用其中的方法。
using Autofac;
using Autofac.Integration.SignalR;
using Autofac.Integration.WebApi;
using Microsoft.AspNet.SignalR;
using Microsoft.Owin;
using Owin;
using System.Reflection;
using System.Web.Http;
[assembly: OwinStartup(typeof(Startup))]
namespace Sample
{
public class Startup
{
public void Configuration(IAppBuilder app)
{
var container = GetDependencyContainer();
RegisterWebApi(app, container);
RegisterSignalR(app, container);
}
private IContainer GetDependencyContainer()
{
var builder = new ContainerBuilder();
AutofacConfig.RegisterModules(builder);
builder.RegisterApiControllers(Assembly.GetExecutingAssembly());
builder.RegisterHubs(Assembly.GetExecutingAssembly());
// Register Autofac resolver into container to be set into HubConfiguration later
builder.RegisterType<AutofacDependencyResolver>().As<IDependencyResolver>().SingleInstance();
// Register ConnectionManager as IConnectionManager so that you can get hub context via IConnectionManager injected to your service
builder.RegisterType<ConnectionManager>().As<IConnectionManager>().SingleInstance();
var container = builder.Build();
return container;
}
private void RegisterWebApi(IAppBuilder app, IContainer container)
{
var configuration = new HttpConfiguration
{
DependencyResolver = new AutofacWebApiDependencyResolver(container)
};
WebApiConfig.Register(configuration);
app.UseAutofacMiddleware(container);
app.UseAutofacWebApi(configuration);
app.UseWebApi(configuration);
}
private void RegisterSignalR(IAppBuilder app, IContainer container)
{
var configuration = new HubConfiguration
{
Resolver = new AutofacDependencyResolver(container)
};
app.MapSignalR(configuration);
}
}
}
public class MaintenanceHub : Hub
{
public MaintenanceHub(IMaintenanceLogProvider maintenanceLogProvider)
{
maintenanceLogProvider.TaskProgressStatusEvent += (s, e) => GetTaskLogStatus(e);
}
public void GetTaskLogStatus(LongTaskProgressStatus taskProgressStatus)
{
Clients.All.getTaskLogStatus(taskProgressStatus);
}
}
public AccountController(IConnectionManager connectionManager, MaintenanceHub maintenanceHub)
{
var context = connectionManager.GetHubContext<MaintenanceHub>();
var hub = maintenanceHub;
}