mirror of
https://github.com/space-wizards/RobustToolbox.git
synced 2026-02-14 19:29:36 +01:00
* Make EntitySystemManager.DependencyCollection inject EntityQuery * Make BUIs inject systems and entity queries * Fix import * We parallelize those * RIDER I BEG YOU * Mocked unit tests are my passion * Perhaps we do not care about fractional milliseconds * Forgor to make it debug only * Use Parallel.For instead of ForEach * Rider I am going to become the joker * Fix EntMan resolve * Now with lazy resolve technology * Use GetOrAdd
226 lines
12 KiB
C#
226 lines
12 KiB
C#
using System;
|
|
using System.Collections.Generic;
|
|
using System.Diagnostics.CodeAnalysis;
|
|
using JetBrains.Annotations;
|
|
using Robust.Shared.IoC.Exceptions;
|
|
using Robust.Shared.Reflection;
|
|
using NotNull = System.Diagnostics.CodeAnalysis.NotNullAttribute;
|
|
|
|
namespace Robust.Shared.IoC
|
|
{
|
|
/// <summary>
|
|
/// The IoCManager handles Dependency Injection in the project.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// <para>
|
|
/// Dependency Injection is a concept where instead of saying "I need the <c>EntityManager</c>",
|
|
/// you say "I need something that implements <c>IEntityManager</c>".
|
|
/// This decouples the various systems into swappable components that have standardized interfaces.
|
|
/// </para>
|
|
/// <para>
|
|
/// This is useful for a couple of things.
|
|
/// Firstly, it allows the shared code to request the client or server code implicitly, without hacks.
|
|
/// Secondly, it's very useful for unit tests as we can replace components to test things.
|
|
/// </para>
|
|
/// <para>
|
|
/// To use the IoCManager, it first needs some types registered through <see cref="Register{TInterface, TImplementation}"/>.
|
|
/// These implementations can then be fetched with <see cref="Resolve{T}"/>, or through field injection with <see cref="DependencyAttribute" />.
|
|
/// </para>
|
|
/// <para>
|
|
/// This type is thread safe: registration and <see cref="BuildGraph"/> can run concurrently with resolves.
|
|
/// </para>
|
|
/// </remarks>
|
|
/// <seealso cref="IReflectionManager"/>
|
|
[NotContentImplementable]
|
|
public interface IDependencyCollection
|
|
{
|
|
IDependencyCollection FromParent(IDependencyCollection parentCollection);
|
|
|
|
/// <summary>
|
|
/// Enumerates over all registered types.
|
|
/// </summary>
|
|
IEnumerable<Type> GetRegisteredTypes();
|
|
|
|
/// <summary>
|
|
/// Registers an interface to an implementation, to make it accessible to <see cref="DependencyCollection.Resolve{T}"/>
|
|
/// <see cref="IDependencyCollection.BuildGraph"/> MUST be called after this method to make the new interface available.
|
|
/// </summary>
|
|
/// <typeparam name="TInterface">The type that will be resolvable.</typeparam>
|
|
/// <typeparam name="TImplementation">The type that will be constructed as implementation.</typeparam>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
/// <exception cref="InvalidOperationException">
|
|
/// Thrown if <paramref name="overwrite"/> is false and <typeparamref name="TInterface"/> has been registered before,
|
|
/// or if an already instantiated interface (by <see cref="DependencyCollection.BuildGraph"/>) is attempting to be overwritten.
|
|
/// </exception>
|
|
void Register<TInterface, [MeansImplicitUse] TImplementation>(bool overwrite = false)
|
|
where TImplementation : class, TInterface
|
|
where TInterface : class;
|
|
|
|
/// <summary>
|
|
/// Registers an interface to an implementation, to make it accessible to <see cref="DependencyCollection.Resolve{T}"/>
|
|
/// <see cref="IDependencyCollection.BuildGraph"/> MUST be called after this method to make the new interface available.
|
|
/// </summary>
|
|
/// <typeparam name="TInterface">The type that will be resolvable.</typeparam>
|
|
/// <typeparam name="TImplementation">The type that will be constructed as implementation.</typeparam>
|
|
/// <param name="factory">A factory method to construct the instance of the implementation.</param>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
/// <exception cref="InvalidOperationException">
|
|
/// Thrown if <paramref name="overwrite"/> is false and <typeparamref name="TInterface"/> has been registered before,
|
|
/// or if an already instantiated interface (by <see cref="DependencyCollection.BuildGraph"/>) is attempting to be overwritten.
|
|
/// </exception>
|
|
void Register<TInterface, TImplementation>(DependencyFactoryDelegate<TImplementation> factory, bool overwrite = false)
|
|
where TImplementation : class, TInterface
|
|
where TInterface : class;
|
|
|
|
|
|
/// <summary>
|
|
/// Registers a simple implementation without an interface.
|
|
/// </summary>
|
|
/// <param name="implementation">The type that will be resolvable.</param>
|
|
/// <param name="factory">A factory method to construct the instance of the implementation.</param>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
void Register(Type implementation, DependencyFactoryDelegate<object>? factory = null, bool overwrite = false);
|
|
|
|
/// <summary>
|
|
/// Registers a simple implementation without an interface.
|
|
/// </summary>
|
|
/// <param name="interfaceType">The type that will be resolvable.</param>
|
|
/// <param name="implementation">The type that will be resolvable.</param>
|
|
/// <param name="factory">A factory method to construct the instance of the implementation.</param>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
void Register(Type interfaceType, Type implementation, DependencyFactoryDelegate<object>? factory = null,
|
|
bool overwrite = false);
|
|
|
|
/// <summary>
|
|
/// Registers an interface to an existing instance of an implementation,
|
|
/// making it accessible to <see cref="IDependencyCollection.Resolve{T}"/>.
|
|
/// Unlike <see cref="IDependencyCollection.Register{TInterface, TImplementation}"/>,
|
|
/// <see cref="IDependencyCollection.BuildGraph"/> does not need to be called after registering an instance
|
|
/// if deferredInject is false.
|
|
/// </summary>
|
|
/// <typeparam name="TInterface">The type that will be resolvable.</typeparam>
|
|
/// <param name="implementation">The existing instance to use as the implementation.</param>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
void RegisterInstance<TInterface>(object implementation, bool overwrite = false) where TInterface : class;
|
|
|
|
/// <summary>
|
|
/// Registers an interface to an existing instance of an implementation,
|
|
/// making it accessible to <see cref="IDependencyCollection.Resolve{T}"/>.
|
|
/// Unlike <see cref="IDependencyCollection.Register{TInterface, TImplementation}"/>,
|
|
/// <see cref="IDependencyCollection.BuildGraph"/> does not need to be called after registering an instance.
|
|
/// </summary>
|
|
/// <param name="type">The type that will be resolvable.</param>
|
|
/// <param name="implementation">The existing instance to use as the implementation.</param>
|
|
/// <param name="overwrite">
|
|
/// If true, do not throw an <see cref="InvalidOperationException"/> if an interface is already registered,
|
|
/// replace the current implementation instead.
|
|
/// </param>
|
|
void RegisterInstance(Type type, object implementation, bool overwrite = false);
|
|
|
|
/// <summary>
|
|
/// Adds a callback to be called when attempting to resolve an unresolved type that matches the specified
|
|
/// base generic type, making it accessible to <see cref="IDependencyCollection.Resolve{T}"/>.
|
|
/// This instance will only be created the first time that it is attempted to be resolved.
|
|
/// </summary>
|
|
/// <param name="genericType">The base generic type of the type that will be resolvable.</param>
|
|
/// <param name="factory">The callback to call to get an instance of the implementation for that generic type.</param>
|
|
void RegisterBaseGenericLazy(Type genericType, DependencyFactoryBaseGenericLazyDelegate<object> factory);
|
|
|
|
/// <summary>
|
|
/// Clear all services and types.
|
|
/// Use this between unit tests and on program shutdown.
|
|
/// If a service implements <see cref="IDisposable"/>, <see cref="IDisposable.Dispose"/> will be called on it.
|
|
/// </summary>
|
|
void Clear();
|
|
|
|
/// <summary>
|
|
/// Resolve a dependency manually.
|
|
/// </summary>
|
|
/// <exception cref="UnregisteredTypeException">Thrown if the interface is not registered.</exception>
|
|
/// <exception cref="InvalidOperationException">
|
|
/// Thrown if the resolved type hasn't been created yet
|
|
/// because the object graph still needs to be constructed for it.
|
|
/// </exception>
|
|
[System.Diagnostics.Contracts.Pure]
|
|
T Resolve<T>();
|
|
|
|
/// <inheritdoc cref="Resolve{T}()"/>
|
|
void Resolve<T>([NotNull] ref T? instance);
|
|
|
|
/// <inheritdoc cref="Resolve{T}(ref T?)"/>
|
|
/// <summary>
|
|
/// Resolve two dependencies manually.
|
|
/// </summary>
|
|
void Resolve<T1, T2>([NotNull] ref T1? instance1, [NotNull] ref T2? instance2);
|
|
|
|
/// <inheritdoc cref="Resolve{T1, T2}(ref T1?, ref T2?)"/>
|
|
/// <summary>
|
|
/// Resolve three dependencies manually.
|
|
/// </summary>
|
|
void Resolve<T1, T2, T3>([NotNull] ref T1? instance1, [NotNull] ref T2? instance2, [NotNull] ref T3? instance3);
|
|
|
|
/// <inheritdoc cref="Resolve{T1, T2, T3}(ref T1?, ref T2?, ref T3?)"/>
|
|
/// <summary>
|
|
/// Resolve four dependencies manually.
|
|
/// </summary>
|
|
void Resolve<T1, T2, T3, T4>([NotNull] ref T1? instance1, [NotNull] ref T2? instance2, [NotNull] ref T3? instance3, [NotNull] ref T4? instance4);
|
|
|
|
/// <summary>
|
|
/// Resolve a dependency manually.
|
|
/// </summary>
|
|
/// <exception cref="UnregisteredTypeException">Thrown if the interface is not registered.</exception>
|
|
/// <exception cref="InvalidOperationException">
|
|
/// Thrown if the resolved type hasn't been created yet
|
|
/// because the object graph still needs to be constructed for it.
|
|
/// </exception>
|
|
[System.Diagnostics.Contracts.Pure]
|
|
object ResolveType(Type type);
|
|
|
|
/// <summary>
|
|
/// Resolve a dependency manually.
|
|
/// </summary>
|
|
bool TryResolveType<T>([NotNullWhen(true)] out T? instance);
|
|
|
|
/// <summary>
|
|
/// Resolve a dependency manually.
|
|
/// </summary>
|
|
bool TryResolveType(Type objectType, [MaybeNullWhen(false)] out object instance);
|
|
|
|
/// <summary>
|
|
/// Initializes the object graph by building every object and resolving all dependencies.
|
|
/// </summary>
|
|
/// <seealso cref="DependencyCollection.InjectDependencies"/>
|
|
void BuildGraph();
|
|
|
|
/// <summary>
|
|
/// Injects dependencies into all fields with <see cref="DependencyAttribute"/> on the provided object.
|
|
/// This is useful for objects that are not IoC created, and want to avoid tons of IoC.Resolve() calls.
|
|
/// </summary>
|
|
/// <remarks>
|
|
/// This does NOT initialize IPostInjectInit objects!
|
|
/// </remarks>
|
|
/// <param name="obj">The object to inject into.</param>
|
|
/// <param name="oneOff">If true, this object type is not expected to be injected commonly.</param>
|
|
/// <exception cref="UnregisteredDependencyException">
|
|
/// Thrown if a dependency field on the object is not registered.
|
|
/// </exception>
|
|
/// <seealso cref="DependencyCollection.BuildGraph"/>
|
|
void InjectDependencies(object obj, bool oneOff=false);
|
|
}
|
|
}
|