2017-01-16 2 views
2

Я пытаюсь получить установку Autofac в моем проекте Web API 2. Я использовал его только пару раз. Я использовал Nuget и установил оба Autofac и Autofac.WebApi2. Тогда в моем классе Запуск Я сделал это:Autofac с Web API 2 - Ошибка конструктора без параметров

public partial class Startup 
{ 
    public void Configuration(IAppBuilder app) 
    { 

     // Get our http configuration 
     var config = new HttpConfiguration(); 

     // Register the Autofac middleware FIRST. This also adds 
     // Autofac-injected middleware registered with the container. 
     var container = ConfigureInversionOfControl(app, config); 

     // Register all areas 
     AreaRegistration.RegisterAllAreas(); 
     GlobalConfiguration.Configure(WebApiConfig.Register); 

     // Use our web api 
     app.UseWebApi(config); 
    } 

    /// <summary> 
    /// Configures Autofac DI/IoC 
    /// </summary> 
    /// <param name="app"></param> 
    /// <param name="config"></param> 
    private IContainer ConfigureInversionOfControl(IAppBuilder app, HttpConfiguration config) 
    { 

     // Create our container 
     var builder = new ContainerBuilder(); 

     // You can register controllers all at once using assembly scanning... 
     builder.RegisterApiControllers(Assembly.GetExecutingAssembly()); 

     // Register our module 
     builder.RegisterModule(new AutofacModule()); 

     // Build 
     var container = builder.Build(); 

     // Lets Web API know it should locate services using the AutofacWebApiDependencyResolver 
     config.DependencyResolver = new AutofacWebApiDependencyResolver(container); 

     // Return our container 
     return container; 
    } 
} 

Модуль я создал выглядит следующим образом:

public class AutofacModule : Module 
{ 
    protected override void Load(ContainerBuilder builder) 
    { 

     // Create our Singletons 
     builder.RegisterType<DatabaseContext>().As<DbContext>().InstancePerRequest(); 

     // Create our Services 
     builder.RegisterType<UnitOfWork>().As<IUnitOfWork>().InstancePerRequest(); 
     builder.RegisterType<AnswerService>().As<IAnswerService>().InstancePerRequest(); 
     builder.RegisterType<CategoryService>().As<ICategoryService>().InstancePerRequest(); 
     builder.RegisterType<FeedService>().As<IFeedService>().InstancePerRequest(); 
     builder.RegisterType<FilterService>().As<IFilterService>().InstancePerRequest(); 
     builder.RegisterType<QuestionGroupService>().As<IQuestionGroupService>().InstancePerRequest(); 
     builder.RegisterType<StateService>().As<IStateService>().InstancePerRequest(); 

     // Create our providers 
     //builder.RegisterType<AnswerProvider>().As<IAnswerProvider>().InstancePerLifetimeScope(); 
     builder.RegisterType<CategoryProvider>().As<ICategoryProvider>().InstancePerRequest(); 
     builder.RegisterType<FeedProvider>().As<IFeedProvider>().InstancePerRequest(); 
     builder.RegisterType<FilterProvider>().As<IFilterProvider>().InstancePerRequest(); 
     builder.RegisterType<QuestionGroupProvider>().As<IQuestionGroupProvider>().InstancePerRequest(); 
     builder.RegisterType<StateProvider>().As<IStateProvider>().InstancePerRequest(); 
    } 
} 

И я добавил поставщика в качестве параметра для моего API конструктора, как это:

// Readonly properties 
private readonly ICategoryProvider _provider; 

/// <summary> 
/// Default constructor 
/// </summary> 
public CategoriesController(ICategoryProvider provider) 
{ 
    _provider = provider; 
} 

Но, когда я пытаюсь вызвать API (с помощью почтальона) Я получаю эту ошибку:

{ 
    "message": "An error has occurred.", 
    "exceptionMessage": "An error occurred when trying to create a controller of type 'CategoriesController'. Make sure that the controller has a parameterless public constructor.", 
    "exceptionType": "System.InvalidOperationException", 
    "stackTrace": " at System.Web.Http.Dispatcher.DefaultHttpControllerActivator.Create(HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor, Type controllerType)\r\n at System.Web.Http.Controllers.HttpControllerDescriptor.CreateController(HttpRequestMessage request)\r\n at System.Web.Http.Dispatcher.HttpControllerDispatcher.<SendAsync>d__1.MoveNext()", 
    "innerException": { 
     "message": "An error has occurred.", 
     "exceptionMessage": "Type 'Piiick.Api.Controllers.CategoriesController' does not have a default constructor", 
     "exceptionType": "System.ArgumentException", 
     "stackTrace": " at System.Linq.Expressions.Expression.New(Type type)\r\n at System.Web.Http.Internal.TypeActivator.Create[TBase](Type instanceType)\r\n at System.Web.Http.Dispatcher.DefaultHttpControllerActivator.GetInstanceOrActivator(HttpRequestMessage request, Type controllerType, Func`1& activator)\r\n at System.Web.Http.Dispatcher.DefaultHttpControllerActivator.Create(HttpRequestMessage request, HttpControllerDescriptor controllerDescriptor, Type controllerType)" 
    } 
} 

Я сейчас застрял, я не вижу, где проблема вообще. Поставщик категории имеет конструктор, как это:

/// <summary> 
/// Default constructor 
/// </summary> 
public CategoryProvider(IUnitOfWork unitOfWork, ICategoryService service) 
{ 

    // Map our properties 
    _unitOfWork = unitOfWork; 
    _service = service; 
} 

и CategoryService имеет конструктор, как это:

/// <summary> 
/// The default constructor 
/// </summary> 
/// <param name="unitOfWork"></param> 
public CategoryService(IUnitOfWork unitOfWork) 
    : base(unitOfWork) 
{ 
} 

и, наконец, UnitOfWork имеет конструктор, как это:

/// <summary> 
/// Default constructor 
/// </summary> 
/// <param name="context">The database context</param> 
public UnitOfWork(DbContext context) 
{ 
    _context = context; 
    _repositories = new Dictionary<Type, object>(); 
} 

Из чего я могу сказать, все настроено c orrectly. Может ли кто-нибудь заметить то, что я забыл сделать?

+0

Вы используете OWIN? Также, пожалуйста, покажите свой код, который вызывает «ConfigureInversionOfControl». –

+0

Должно быть что-то еще, вы не показываете как, я помещаю ваш код в Visual Studio, и он отлично работает. Вы упомянули исключение, которое вы указали, когда вы пытаетесь использовать зависимость, не зарегистрированную в контейнере.Кроме того, свойство 'InnerException' исключений должно содержать информацию о том, какая зависимость отсутствует. – tdragon

+0

Пожалуйста, покажите полные сведения об исключении: сообщение, тип и трассировку стека исключения и ** все ** его внутренние исключения. – Steven

ответ

3

Я нашел проблему, я использовал HttpConfiguration для установки зависимого преобразователя, это должно было быть GlobalConfiguration.Configuration.

Так что, когда я изменил свою линию от:

config.DependencyResolver = new AutofacWebApiDependencyResolver(container); 

в

GlobalConfiguration.Configuration.DependencyResolver = new AutofacWebApiDependencyResolver(container); 

Все начали работать.

0

Попробуйте регистрации сборок, которые были загружены в контекст выполнения вашего домена приложения только перед созданием вашего контейнера:

var assemblies = AppDomain.CurrentDomain.GetAssemblies(); builder.RegisterAssemblyModules(assemblies);

/// <summary> 
/// Configures Autofac DI/IoC 
/// </summary> 
/// <param name="app"></param> 
/// <param name="config"></param> 
private IContainer ConfigureInversionOfControl(IAppBuilder app, HttpConfiguration config) 
{ 

    // Create our container 
    var builder = new ContainerBuilder(); 

    // You can register controllers all at once using assembly scanning... 
    builder.RegisterApiControllers(Assembly.GetExecutingAssembly()); 

    // Register our module 
    builder.RegisterModule(new AutofacModule()); 

    // [!!!] ** Register assemblies of the current domain ** 
    var assemblies = AppDomain.CurrentDomain.GetAssemblies(); 
    builder.RegisterAssemblyModules(assemblies); 

    // Build 
    var container = builder.Build(); 

    // Lets Web API know it should locate services using the AutofacWebApiDependencyResolver 
    config.DependencyResolver = new AutofacWebApiDependencyResolver(container); 

    // Return our container 
    return container; 
} 

И не забудьте определили:

public class AutofacModule : Module 
{ 
    protected override void Load(ContainerBuilder builder) 
    { 

     ... 

     builder.RegisterControllers(typeof(WebApiApplication).Assembly); 
     builder.RegisterApiControllers(typeof(WebApiApplication).Assembly); 
    } 
} 
+0

метод RegisterController предназначен только для MVC? – r3plica

Смежные вопросы