mirror of
https://github.com/bitwarden/server
synced 2025-12-17 16:53:23 +00:00
Document queries and scenes
This commit is contained in:
@@ -1,15 +1,60 @@
|
|||||||
namespace Bit.Seeder;
|
namespace Bit.Seeder;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Base interface for query operations in the seeding system. The base interface should not be used directly, rather use `IQuery<TRequest, TResult>`.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Queries are synchronous, read-only operations that retrieve data from the seeding context.
|
||||||
|
/// Unlike scenes which create data, queries fetch existing data based on request parameters.
|
||||||
|
/// They follow a type-safe pattern using generics to ensure proper request/response handling
|
||||||
|
/// while maintaining a common non-generic interface for dynamic invocation.
|
||||||
|
/// </remarks>
|
||||||
public interface IQuery
|
public interface IQuery
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the type of request this query expects.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The request type that this query can process.</returns>
|
||||||
Type GetRequestType();
|
Type GetRequestType();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Executes the query based on the provided request object.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object containing parameters for the query operation.</param>
|
||||||
|
/// <returns>The query result data as an object.</returns>
|
||||||
object Execute(object request);
|
object Execute(object request);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generic query interface for synchronous, read-only operations with specific request and result types.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="TRequest">The type of request object this query accepts.</typeparam>
|
||||||
|
/// <typeparam name="TResult">The type of data this query returns.</typeparam>
|
||||||
|
/// <remarks>
|
||||||
|
/// Use this interface when you need to retrieve existing data from the seeding context based on
|
||||||
|
/// specific request parameters. Queries are synchronous and do not modify data - they only read
|
||||||
|
/// and return information. The explicit interface implementations allow dynamic invocation while
|
||||||
|
/// maintaining type safety in the implementation.
|
||||||
|
/// </remarks>
|
||||||
public interface IQuery<TRequest, TResult> : IQuery where TRequest : class where TResult : class
|
public interface IQuery<TRequest, TResult> : IQuery where TRequest : class where TResult : class
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Executes the query based on the provided strongly-typed request and returns typed result data.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object containing parameters for the query operation.</param>
|
||||||
|
/// <returns>The typed query result data.</returns>
|
||||||
TResult Execute(TRequest request);
|
TResult Execute(TRequest request);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the request type for this query.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The type of TRequest.</returns>
|
||||||
Type IQuery.GetRequestType() => typeof(TRequest);
|
Type IQuery.GetRequestType() => typeof(TRequest);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adapts the non-generic Execute to the strongly-typed version.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object to cast and process.</param>
|
||||||
|
/// <returns>The typed result cast to object.</returns>
|
||||||
object IQuery.Execute(object request) => Execute((TRequest)request);
|
object IQuery.Execute(object request) => Execute((TRequest)request);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,19 +1,59 @@
|
|||||||
namespace Bit.Seeder;
|
namespace Bit.Seeder;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Base interface for seeding operations. The base interface should not be used directly, rather use `IScene<Request>`.
|
||||||
|
/// </summary>
|
||||||
|
/// <remarks>
|
||||||
|
/// Scenes are components in the seeding system that create and configure test data. They follow
|
||||||
|
/// a type-safe pattern using generics to ensure proper request/response handling while maintaining
|
||||||
|
/// a common non-generic interface for dynamic invocation.
|
||||||
|
/// </remarks>
|
||||||
public interface IScene
|
public interface IScene
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the type of request this scene expects.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The request type that this scene can process.</returns>
|
||||||
Type GetRequestType();
|
Type GetRequestType();
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Seeds data based on the provided request object.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object containing parameters for the seeding operation.</param>
|
||||||
|
/// <returns>A scene result containing any returned data, mangle map, and entity tracking information.</returns>
|
||||||
Task<SceneResult<object?>> SeedAsync(object request);
|
Task<SceneResult<object?>> SeedAsync(object request);
|
||||||
}
|
}
|
||||||
|
|
||||||
/// <summary>
|
/// <summary>
|
||||||
/// Generic scene interface for seeding operations with a specific request type. Does not return a value beyond tracking entities and a mangle map.
|
/// Generic scene interface for seeding operations with a specific request type. Does not return a value beyond tracking entities and a mangle map.
|
||||||
/// </summary>
|
/// </summary>
|
||||||
/// <typeparam name="TRequest"></typeparam>
|
/// <typeparam name="TRequest">The type of request object this scene accepts.</typeparam>
|
||||||
|
/// <remarks>
|
||||||
|
/// Use this interface when your scene needs to process a specific request type but doesn't need to
|
||||||
|
/// return any data beyond the standard mangle map for ID transformations and entity tracking.
|
||||||
|
/// The explicit interface implementations allow this scene to be invoked dynamically through the
|
||||||
|
/// base IScene interface while maintaining type safety in the implementation.
|
||||||
|
/// </remarks>
|
||||||
public interface IScene<TRequest> : IScene where TRequest : class
|
public interface IScene<TRequest> : IScene where TRequest : class
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Seeds data based on the provided strongly-typed request.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object containing parameters for the seeding operation.</param>
|
||||||
|
/// <returns>A scene result containing the mangle map and entity tracking information.</returns>
|
||||||
Task<SceneResult> SeedAsync(TRequest request);
|
Task<SceneResult> SeedAsync(TRequest request);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the request type for this scene.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The type of TRequest.</returns>
|
||||||
Type IScene.GetRequestType() => typeof(TRequest);
|
Type IScene.GetRequestType() => typeof(TRequest);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adapts the non-generic SeedAsync to the strongly-typed version.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object to cast and process.</param>
|
||||||
|
/// <returns>A scene result wrapped as an object result.</returns>
|
||||||
async Task<SceneResult<object?>> IScene.SeedAsync(object request)
|
async Task<SceneResult<object?>> IScene.SeedAsync(object request)
|
||||||
{
|
{
|
||||||
var result = await SeedAsync((TRequest)request);
|
var result = await SeedAsync((TRequest)request);
|
||||||
@@ -21,10 +61,36 @@ public interface IScene<TRequest> : IScene where TRequest : class
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Generic scene interface for seeding operations with a specific request type that returns typed data.
|
||||||
|
/// </summary>
|
||||||
|
/// <typeparam name="TRequest">The type of request object this scene accepts. Must be a reference type.</typeparam>
|
||||||
|
/// <typeparam name="TResult">The type of data this scene returns. Must be a reference type.</typeparam>
|
||||||
|
/// <remarks>
|
||||||
|
/// Use this interface when your scene needs to return specific data that can be used by subsequent
|
||||||
|
/// scenes or test logic. The result is wrapped in a SceneResult that also includes the mangle map
|
||||||
|
/// and entity tracking information. The explicit interface implementations allow dynamic invocation
|
||||||
|
/// while preserving type safety in the implementation.
|
||||||
|
/// </remarks>
|
||||||
public interface IScene<TRequest, TResult> : IScene where TRequest : class where TResult : class
|
public interface IScene<TRequest, TResult> : IScene where TRequest : class where TResult : class
|
||||||
{
|
{
|
||||||
|
/// <summary>
|
||||||
|
/// Seeds data based on the provided strongly-typed request and returns typed result data.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object containing parameters for the seeding operation.</param>
|
||||||
|
/// <returns>A scene result containing the typed result data, mangle map, and entity tracking information.</returns>
|
||||||
Task<SceneResult<TResult>> SeedAsync(TRequest request);
|
Task<SceneResult<TResult>> SeedAsync(TRequest request);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Gets the request type for this scene.
|
||||||
|
/// </summary>
|
||||||
|
/// <returns>The type of TRequest.</returns>
|
||||||
Type IScene.GetRequestType() => typeof(TRequest);
|
Type IScene.GetRequestType() => typeof(TRequest);
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Adapts the non-generic SeedAsync to the strongly-typed version.
|
||||||
|
/// </summary>
|
||||||
|
/// <param name="request">The request object to cast and process.</param>
|
||||||
|
/// <returns>A scene result with the typed result cast to object.</returns>
|
||||||
async Task<SceneResult<object?>> IScene.SeedAsync(object request) => (SceneResult<object?>)await SeedAsync((TRequest)request);
|
async Task<SceneResult<object?>> IScene.SeedAsync(object request) => (SceneResult<object?>)await SeedAsync((TRequest)request);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,9 @@ using Bit.Infrastructure.EntityFramework.Repositories;
|
|||||||
|
|
||||||
namespace Bit.Seeder.Queries;
|
namespace Bit.Seeder.Queries;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Retrieves all emergency access invite urls for the provided email.
|
||||||
|
/// </summary>
|
||||||
public class EmergencyAccessInviteQuery(
|
public class EmergencyAccessInviteQuery(
|
||||||
DatabaseContext db,
|
DatabaseContext db,
|
||||||
IDataProtectorTokenFactory<EmergencyAccessInviteTokenable> dataProtectorTokenizer)
|
IDataProtectorTokenFactory<EmergencyAccessInviteTokenable> dataProtectorTokenizer)
|
||||||
|
|||||||
@@ -4,6 +4,9 @@ using Bit.Seeder.Factories;
|
|||||||
|
|
||||||
namespace Bit.Seeder.Scenes;
|
namespace Bit.Seeder.Scenes;
|
||||||
|
|
||||||
|
/// <summary>
|
||||||
|
/// Creates a single user using the provided account details.
|
||||||
|
/// </summary>
|
||||||
public class SingleUserScene(UserSeeder userSeeder, IUserRepository userRepository) : IScene<SingleUserScene.Request>
|
public class SingleUserScene(UserSeeder userSeeder, IUserRepository userRepository) : IScene<SingleUserScene.Request>
|
||||||
{
|
{
|
||||||
public class Request
|
public class Request
|
||||||
|
|||||||
Reference in New Issue
Block a user