Skip to content

Commit

Permalink
Undecorated service resolution succeeds, but decorated service resolu…
Browse files Browse the repository at this point in the history
…tion fails, #378. (#379)
  • Loading branch information
yallie authored Feb 25, 2021
1 parent 47f29a0 commit e42f5ea
Showing 1 changed file with 128 additions and 0 deletions.
128 changes: 128 additions & 0 deletions test/DryIoc.IssuesTests/GHIssue378_InconsistentResolutionFailure.cs
Original file line number Diff line number Diff line change
@@ -0,0 +1,128 @@
using System;
using System.Collections.Generic;
using System.ComponentModel.Composition;
using System.Text;
using DryIoc.MefAttributedModel;
using NUnit.Framework;

namespace DryIoc.IssuesTests
{
[TestFixture]
public class GHIssue378_InconsistentResolutionFailure
{
[Test, Ignore("Fails on DryIoc v4.7.3")]
public void Nested_scopes_undecorated_resolution_succeeds_but_decorated_resolution_fails()
{
var c = new Container().WithMef()
.With(rules => rules
.WithoutThrowIfDependencyHasShorterReuseLifespan()
.WithDefaultReuse(Reuse.Scoped));

// 1. default registrations
c.RegisterExports(typeof(DecoratedService), typeof(UndecoratedService), typeof(DependencyService));

// 2. IDecoratedService should be created via a factory method
var decoratorSetup = Setup.DecoratorWith(useDecorateeReuse: false);
var factoryMethod = new Func<Lazy<IDecoratedService>, IDecoratedService>(CreateDecoratedService).Method;
c.Register(typeof(IDecoratedService), reuse: Reuse.Transient, made: Made.Of(factoryMethod), setup: decoratorSetup);

// global scope is opened once on application startup
using (var globalScope = c.OpenScope())
{
// request scopes are created for each request
using (var requestScope = globalScope.OpenScope())
{
// succeeds
Assert.DoesNotThrow(() =>
{
var service = requestScope.Resolve<IUndecoratedService>();
service.Hello();
},
"Failed to use the undecorated service!");

// fails
Assert.DoesNotThrow(() =>
{
var decorated = requestScope.Resolve<IDecoratedService>();
decorated.Hello();
},
"Failed to use the decorated service!");
}
}
}

static IDecoratedService CreateDecoratedService(Lazy<IDecoratedService> lazy)
{
WriteLine("CreateDecoratedService is called!");
return lazy.Value;
}

private static void WriteLine(string s) => TestContext.Progress.WriteLine(s);

public interface IUndecoratedService
{
void Hello();
}

// Code smell: the singleton service uses a dependency of a shorter lifespan,
// but the container is set up to allow this
[Export(typeof(IUndecoratedService)), PartCreationPolicy(CreationPolicy.Shared)]
public class UndecoratedService : IUndecoratedService
{
public UndecoratedService(Lazy<IDependencyService> dep)
{
WriteLine("UndecoratedService is created");
Dependency = dep;
}

private Lazy<IDependencyService> Dependency { get; set; }

public void Hello()
{
WriteLine("UndecoratedService is accessing the Dependency...");

var cm = Dependency.Value;
WriteLine("Dependency returned: " + cm.Value);
}
}

public interface IDecoratedService
{
void Hello();
}

// Code smell: the singleton service uses a dependency of a shorter lifespan,
// but the container is set up to allow this
[Export(typeof(IDecoratedService)), PartCreationPolicy(CreationPolicy.Shared)]
public class DecoratedService : IDecoratedService
{
public DecoratedService(Lazy<IDependencyService> dep)
{
WriteLine("DecoratedService is created");
Dependency = dep;
}

private Lazy<IDependencyService> Dependency { get; set; }

public void Hello()
{
WriteLine("DecoratedService is accessing the Dependency...");

var cm = Dependency.Value;
WriteLine("Dependency returned: " + cm.Value);
}
}

public interface IDependencyService
{
string Value { get; }
}

// The only dependency of both decorated and undecorated services
[Export(typeof(IDependencyService))]
public class DependencyService : IDependencyService
{
public string Value => "Hello";
}
}
}

0 comments on commit e42f5ea

Please sign in to comment.