Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Castle.Windsor.Extensions.DependencyInjection: support parallel containers (see #563) #577

Merged
Merged
Show file tree
Hide file tree
Changes from 25 commits
Commits
Show all changes
28 commits
Select commit Hold shift + click to select a range
d0b3975
Encapsulate single root scope in WindsorScopeFactory
rvdginste Nov 30, 2020
d46be5b
Add test with external container
generik0 Dec 2, 2020
9d8a864
Merge remote-tracking branch 'castle.windsor/master' into feature/sup…
generik0 Dec 7, 2020
d3aecbf
Change BeginScope to utilize the self contained current scope
generik0 Dec 7, 2020
558d545
Improve readability of null handling for scopes
generik0 Dec 7, 2020
07ff3e3
Cleanup
generik0 Dec 8, 2020
1e1f6ef
Remove unneded field rootScope in favour of private set property
generik0 Dec 9, 2020
6d8fb32
Clean up modifiers and code
generik0 Dec 9, 2020
f7d06b5
Move scope classes to scop folder
generik0 Dec 9, 2020
6192d33
Fix namespaces for moved scope classes
generik0 Dec 9, 2020
ba6a8cb
Move ForcedScope no longer to be an inner class for ExtensionContaine…
generik0 Dec 9, 2020
00ccc0e
Change Current property to handle nulls with exceptiion. Current must…
generik0 Dec 9, 2020
af05190
Remove var, only keep discard
generik0 Dec 9, 2020
4df5c19
Cleanup
generik0 Dec 9, 2020
c12a810
Remove unneded passing of RootScope as it is in the parent.
generik0 Dec 9, 2020
ecfa403
Rename CurrentOrThrow to Current
generik0 Dec 10, 2020
36ac910
Add check that the disposing scope is current, before changing the pa…
generik0 Dec 10, 2020
3a3a63e
Move the set of the curr3ent.Value to the ctor from the ExtensionCont…
generik0 Dec 10, 2020
bdf3507
Merge remote-tracking branch 'castle.windsor/master' into feature/sup…
generik0 Dec 10, 2020
32ce9c5
Improve ordering for readability
generik0 Dec 10, 2020
f6740c4
Rename ExtensionContainerScope to Base
generik0 Dec 10, 2020
414887d
Make ExtensionContainerRootScope and ExtensionContainerRoot use a base
generik0 Dec 10, 2020
e039c17
Change BeginScope and BeginRootScope to set current
generik0 Dec 10, 2020
5c7f535
Move Current into its own static class
generik0 Dec 10, 2020
6d4d75e
Cleanup
generik0 Dec 10, 2020
b9753fb
Cleanup
generik0 Dec 10, 2020
34b0d9f
Fix TransientMarker to be readonly
generik0 Dec 15, 2020
0c04c83
Remove unneeded exception when the getter already throws
generik0 Dec 15, 2020
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

namespace Castle.Windsor.Extensions.DependencyInjection.Tests
{
using System;

using Microsoft.Extensions.DependencyInjection;
using Microsoft.Extensions.DependencyInjection.Specification;

public class WindsorScopedServiceProviderCustomWindsorContainerTests : DependencyInjectionSpecificationTests
{
protected override IServiceProvider CreateServiceProvider(IServiceCollection serviceCollection)
{
var factory = new WindsorServiceProviderFactory(new WindsorContainer());
var container = factory.CreateBuilder(serviceCollection);
return factory.CreateServiceProvider(container);
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@

<PropertyGroup>
<TargetFrameworks>netstandard2.0</TargetFrameworks>
<LangVersion>9.0</LangVersion>
</PropertyGroup>

<Import Project="..\..\buildscripts\common.props"></Import>
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -39,7 +39,7 @@ public static ComponentRegistration<TService> ScopedToNetServiceScope<TService>(
public static ComponentRegistration<TService> LifestyleNetTransient<TService>(this ComponentRegistration<TService> registration) where TService : class
{
return registration
.Attribute(ExtensionContainerScope.TransientMarker).Eq(Boolean.TrueString)
.Attribute(ExtensionContainerScopeBase.TransientMarker).Eq(Boolean.TrueString)
.LifeStyle.ScopedToNetServiceScope(); //.NET core expects new instances but release on scope dispose
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -14,19 +14,16 @@

namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
internal class ExtensionContainerRootScope : ExtensionContainerScope
internal class ExtensionContainerRootScope : ExtensionContainerScopeBase
{
internal static ExtensionContainerRootScope RootScope {get; private set;}
private ExtensionContainerRootScope() : base(null)
{

}

public static ExtensionContainerRootScope BeginRootScope()
{
var scope = new ExtensionContainerRootScope();
ExtensionContainerScope.current.Value = scope;
RootScope = scope;
ExtensionContainerScopeCache.Current = scope;
return scope;
}

internal override ExtensionContainerScopeBase RootScope => this;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -15,24 +15,19 @@
namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;

using Castle.MicroKernel.Context;
using Castle.MicroKernel.Lifestyle.Scoped;

internal class ExtensionContainerRootScopeAccessor : IScopeAccessor
{
public ILifetimeScope GetScope(CreationContext context)
{
if (ExtensionContainerRootScope.RootScope == null)
{
throw new InvalidOperationException("No root scope");
}

return ExtensionContainerRootScope.RootScope;
return ExtensionContainerScopeCache.Current.RootScope ?? throw new InvalidOperationException("No root scope available");
}

public void Dispose()
{
}
}
}
}
Original file line number Diff line number Diff line change
@@ -1,4 +1,4 @@
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
Expand All @@ -14,92 +14,32 @@

namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;
using System.Threading;

using Castle.Core;
using Castle.MicroKernel;
using Castle.MicroKernel.Lifestyle.Scoped;

internal class ExtensionContainerScope : ILifetimeScope, IDisposable
internal class ExtensionContainerScope : ExtensionContainerScopeBase
{
public static ExtensionContainerScope Current => current.Value;
public static string TransientMarker = "Transient";
protected static readonly AsyncLocal<ExtensionContainerScope> current = new AsyncLocal<ExtensionContainerScope>();
private readonly ExtensionContainerScope parent;
private readonly IScopeCache scopeCache;

protected ExtensionContainerScope(ExtensionContainerScope parent)
{
scopeCache = new ScopeCache();
if(parent == null)
{
this.parent = ExtensionContainerRootScope.RootScope;
}
else
{
this.parent = parent;
}
}
private readonly ExtensionContainerScopeBase parent;

public static ExtensionContainerScope BeginScope(ExtensionContainerScope parent)
protected ExtensionContainerScope()
{
var scope = new ExtensionContainerScope(parent);
current.Value = scope;
return scope;
parent = ExtensionContainerScopeCache.Current;
}

internal override ExtensionContainerScopeBase RootScope { get; set; }

public void Dispose()
{
var disposableCache = scopeCache as IDisposable;
if (disposableCache != null)
{
disposableCache.Dispose();
}

current.Value = parent;
}

public Burden GetCachedInstance(ComponentModel model, ScopedInstanceActivationCallback createInstance)
internal static ExtensionContainerScopeBase BeginScope()
{
lock (scopeCache)
{

// Add transient's burden to scope so it gets released
if (model.Configuration.Attributes.Get(TransientMarker) == bool.TrueString)
{
var transientBurden = createInstance((_) => {});
scopeCache[transientBurden] = transientBurden;
return transientBurden;
}

var scopedBurden = scopeCache[model];
if (scopedBurden != null)
{
return scopedBurden;
}
scopedBurden = createInstance((_) => {});
scopeCache[model] = scopedBurden;
return scopedBurden;
}
var scope = new ExtensionContainerScope { RootScope = ExtensionContainerScopeCache.Current.RootScope };
ExtensionContainerScopeCache.Current = scope;
return scope;
}

/// <summary>
/// Forces a specific <see name="ExtensionContainerScope" /> for 'using' block. In .NET scope is tied to an instance of <see name="System.IServiceProvider" /> not a thread or async context
/// </summary>
internal class ForcedScope : IDisposable
public override void Dispose()
{
private readonly ExtensionContainerScope previousScope;
public ForcedScope(ExtensionContainerScope scope)
{
previousScope = ExtensionContainerScope.Current;
ExtensionContainerScope.current.Value = scope;
}
public void Dispose()
if (ExtensionContainerScopeCache.current.Value == this)
{
ExtensionContainerScope.current.Value = previousScope;
ExtensionContainerScopeCache.current.Value = parent;
}
base.Dispose();
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -23,11 +23,7 @@ internal class ExtensionContainerScopeAccessor : IScopeAccessor
{
public ILifetimeScope GetScope(CreationContext context)
{
if(ExtensionContainerScope.Current == null)
{
throw new InvalidOperationException("No scope available");
}
return ExtensionContainerScope.Current;
return ExtensionContainerScopeCache.Current ?? throw new InvalidOperationException("No scope available");
jonorossi marked this conversation as resolved.
Show resolved Hide resolved
}

public void Dispose()
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,66 @@
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;

using Castle.Core;
using Castle.MicroKernel;
using Castle.MicroKernel.Lifestyle.Scoped;

internal abstract class ExtensionContainerScopeBase : ILifetimeScope
{
public static string TransientMarker = "Transient";
jonorossi marked this conversation as resolved.
Show resolved Hide resolved
private readonly IScopeCache scopeCache;

protected ExtensionContainerScopeBase()
{
scopeCache = new ScopeCache();
}

internal virtual ExtensionContainerScopeBase RootScope { get; set; }

public virtual void Dispose()
{
if (scopeCache is IDisposable disposableCache)
{
disposableCache.Dispose();
}
}

public Burden GetCachedInstance(ComponentModel model, ScopedInstanceActivationCallback createInstance)
{
lock (scopeCache)
{
// Add transient's burden to scope so it gets released
if (model.Configuration.Attributes.Get(TransientMarker) == bool.TrueString)
{
var transientBurden = createInstance(_ => {});
scopeCache[transientBurden] = transientBurden;
return transientBurden;
}

var scopedBurden = scopeCache[model];
if (scopedBurden != null)
{
return scopedBurden;
}
scopedBurden = createInstance((_) => {});
scopeCache[model] = scopedBurden;
return scopedBurden;
}
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;
using System.Threading;

internal static class ExtensionContainerScopeCache
{
internal static readonly AsyncLocal<ExtensionContainerScopeBase> current = new AsyncLocal<ExtensionContainerScopeBase>();
/// <summary>Current scope for the thread. Initial scope will be set when calling BeginRootScope from a ExtensionContainerRootScope instance.</summary>
/// <exception cref="InvalidOperationException">Thrown when there is no scope available.</exception>
internal static ExtensionContainerScopeBase Current
{
get => current.Value ?? throw new InvalidOperationException("No scope available");
set => current.Value = value;
}
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,38 @@
// Copyright 2004-2020 Castle Project - http://www.castleproject.org/
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;

/// <summary>
/// Forces a specific <see name="ExtensionContainerScope" /> for 'using' block. In .NET scope is tied to an instance of <see name="System.IServiceProvider" /> not a thread or async context
/// </summary>
internal class ForcedScope : IDisposable
{
private readonly ExtensionContainerScopeBase scope;
private readonly ExtensionContainerScopeBase previousScope;
internal ForcedScope(ExtensionContainerScopeBase scope)
{
previousScope = ExtensionContainerScopeCache.Current;
this.scope = scope;
ExtensionContainerScopeCache.Current = scope;
}
public void Dispose()
{
if(ExtensionContainerScopeCache.Current != scope) return;
ExtensionContainerScopeCache.Current = previousScope;
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -12,10 +12,10 @@
// See the License for the specific language governing permissions and
// limitations under the License.

namespace Castle.Windsor.Extensions.DependencyInjection
namespace Castle.Windsor.Extensions.DependencyInjection.Scope
{
using System;

using Microsoft.Extensions.DependencyInjection;

internal class ServiceScope : IServiceScope
Expand Down
Loading