Files
RobustToolbox/Robust.Shared/IoC/IDependencyCollection.cs
DrSmugleaf fe1648d290 Make EntitySystemManager.DependencyCollection inject EntityQuery, make BUIs inject systems and entity queries (#6394)
* 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
2026-02-05 21:35:52 +01:00

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);
}
}