AyCode.Core/AyCode.Core.Tests/TestModels/SharedTestModels.cs

461 lines
14 KiB
C#

using AyCode.Core.Extensions;
using AyCode.Core.Interfaces;
using MessagePack;
using MongoDB.Bson.Serialization.Attributes;
using Newtonsoft.Json;
namespace AyCode.Core.Tests.TestModels;
#region Shared Enums
/// <summary>
/// Common status enum for all test entities
/// </summary>
public enum TestStatus
{
Pending = 0,
Active = 1,
Processing = 2,
Completed = 3,
Shipped = 4,
OnHold = 5
}
/// <summary>
/// Priority levels for tasks and projects
/// </summary>
public enum TestPriority
{
Low = 0,
Medium = 1,
High = 2,
Critical = 3
}
/// <summary>
/// User roles for testing
/// </summary>
public enum TestUserRole
{
User = 0,
Manager = 1,
Admin = 2
}
#endregion
#region Shared Reference Types (IId-based for $id/$ref testing)
/// <summary>
/// Shared tag/label - used across multiple entities for cross-reference testing.
/// Implements IId&lt;int&gt; for semantic $id/$ref serialization.
/// </summary>
public class SharedTag : IId<int>
{
public int Id { get; set; }
public string Name { get; set; } = "";
public string Color { get; set; } = "#000000";
public int Priority { get; set; }
public bool IsActive { get; set; } = true;
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
public string? Description { get; set; }
}
/// <summary>
/// Shared category - for hierarchical cross-reference testing.
/// </summary>
public class SharedCategory : IId<int>
{
public int Id { get; set; }
public string Name { get; set; } = "";
public string? Description { get; set; }
public int SortOrder { get; set; }
public bool IsDefault { get; set; }
public int? ParentCategoryId { get; set; }
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
public DateTime? UpdatedAt { get; set; }
}
/// <summary>
/// Shared user reference - appears in many places to test $ref deduplication.
/// </summary>
public class SharedUser : IId<int>
{
public int Id { get; set; }
public string Username { get; set; } = "";
public string Email { get; set; } = "";
public string FirstName { get; set; } = "";
public string LastName { get; set; } = "";
public bool IsActive { get; set; } = true;
public TestUserRole Role { get; set; } = TestUserRole.User;
public DateTime? LastLoginAt { get; set; }
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
public UserPreferences? Preferences { get; set; }
}
/// <summary>
/// User preferences - non-IId nested object
/// </summary>
public class UserPreferences
{
public string Theme { get; set; } = "light";
public string Language { get; set; } = "en-US";
public bool NotificationsEnabled { get; set; } = true;
public string? EmailDigestFrequency { get; set; }
}
#endregion
#region Non-IId Metadata (Newtonsoft numeric $id/$ref testing)
/// <summary>
/// Non-IId metadata class - uses Newtonsoft PreserveReferencesHandling (numeric $id/$ref).
/// Does NOT implement IId, so uses standard Newtonsoft reference tracking.
/// </summary>
public class MetadataInfo
{
public string Key { get; set; } = "";
public string Value { get; set; } = "";
public DateTime Timestamp { get; set; } = DateTime.UtcNow;
/// <summary>
/// Nested metadata for deep Newtonsoft reference testing
/// </summary>
public MetadataInfo? ChildMetadata { get; set; }
}
#endregion
#region 5-Level Test Hierarchy (Order -> Item -> Pallet -> Measurement -> Point)
/// <summary>
/// Level 1: Main order - root of the hierarchy
/// </summary>
public class TestOrder : IId<int>
{
public int Id { get; set; }
public string OrderNumber { get; set; } = "";
public TestStatus Status { get; set; } = TestStatus.Pending;
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
public DateTime? PaidDateUtc { get; set; }
public decimal TotalAmount { get; set; }
// Level 2 collection
public List<TestOrderItem> Items { get; set; } = [];
// Shared reference properties (for $id/$ref testing)
public SharedTag? PrimaryTag { get; set; }
public SharedTag? SecondaryTag { get; set; }
public SharedUser? Owner { get; set; }
public SharedCategory? Category { get; set; }
// Collection of shared references
public List<SharedTag> Tags { get; set; } = [];
// Non-IId metadata (for Newtonsoft $ref testing)
public MetadataInfo? OrderMetadata { get; set; }
public MetadataInfo? AuditMetadata { get; set; }
public List<MetadataInfo> MetadataList { get; set; } = [];
// NoMerge collection for testing replace behavior
[JsonNoMergeCollection]
public List<TestOrderItem> NoMergeItems { get; set; } = [];
// Parent reference - ignored by all serializers to prevent circular references
[JsonIgnore]
[IgnoreMember]
[BsonIgnore]
public object? Parent { get; set; }
}
/// <summary>
/// Level 2: Order item with pallets
/// </summary>
public class TestOrderItem : IId<int>
{
public int Id { get; set; }
public string ProductName { get; set; } = "";
public int Quantity { get; set; }
public decimal UnitPrice { get; set; }
public TestStatus Status { get; set; } = TestStatus.Pending;
// Level 3 collection
public List<TestPallet> Pallets { get; set; } = [];
// Shared references
public SharedTag? Tag { get; set; }
public SharedUser? Assignee { get; set; }
public MetadataInfo? ItemMetadata { get; set; }
// Parent reference - ignored by all serializers to prevent circular references
[JsonIgnore]
[IgnoreMember]
[BsonIgnore]
public TestOrder? ParentOrder { get; set; }
}
/// <summary>
/// Level 3: Pallet containing measurements
/// </summary>
public class TestPallet : IId<int>
{
public int Id { get; set; }
public string PalletCode { get; set; } = "";
public int TrayCount { get; set; }
public TestStatus Status { get; set; } = TestStatus.Pending;
public double Weight { get; set; }
// Level 4 collection
public List<TestMeasurement> Measurements { get; set; } = [];
// Shared references
public MetadataInfo? PalletMetadata { get; set; }
// Parent reference - ignored by all serializers to prevent circular references
[JsonIgnore]
[IgnoreMember]
[BsonIgnore]
public TestOrderItem? ParentItem { get; set; }
}
/// <summary>
/// Level 4: Measurement with multiple points
/// </summary>
public class TestMeasurement : IId<int>
{
public int Id { get; set; }
public string Name { get; set; } = "";
public double TotalWeight { get; set; }
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
// Level 5 collection
public List<TestMeasurementPoint> Points { get; set; } = [];
// Parent reference - ignored by all serializers to prevent circular references
[JsonIgnore]
[IgnoreMember]
[BsonIgnore]
public TestPallet? ParentPallet { get; set; }
}
/// <summary>
/// Level 5: Deepest level - measurement point
/// </summary>
public class TestMeasurementPoint : IId<int>
{
public int Id { get; set; }
public string Label { get; set; } = "";
public double Value { get; set; }
public DateTime MeasuredAt { get; set; } = DateTime.UtcNow;
// Parent reference - ignored by all serializers to prevent circular references
[JsonIgnore]
[IgnoreMember]
[BsonIgnore]
public TestMeasurement? ParentMeasurement { get; set; }
}
#endregion
#region Guid-based IId types
/// <summary>
/// Order with Guid Id - for testing Guid-based IId
/// </summary>
public class TestGuidOrder : IId<Guid>
{
public Guid Id { get; set; }
public string Code { get; set; } = "";
public List<TestGuidItem> Items { get; set; } = [];
public int Count { get; set; }
}
/// <summary>
/// Item with Guid Id
/// </summary>
public class TestGuidItem : IId<Guid>
{
public Guid Id { get; set; }
public string Name { get; set; } = "";
public int Qty { get; set; }
}
#endregion
#region Test-specific classes
/// <summary>
/// Simulates NopCommerce GenericAttribute - stores key-value pairs where DateTime values
/// are stored as strings in the database.
/// </summary>
public class TestGenericAttribute
{
public int Id { get; set; }
public string Key { get; set; } = "";
public string Value { get; set; } = "";
}
/// <summary>
/// DTO with GenericAttributes collection - simulates OrderDto with string-stored DateTime values.
/// This reproduces the production bug where Binary serialization was thought to corrupt DateTime strings.
/// </summary>
public class TestDtoWithGenericAttributes : IId<int>
{
public int Id { get; set; }
public string Name { get; set; } = "";
public List<TestGenericAttribute> GenericAttributes { get; set; } = [];
}
/// <summary>
/// Order with nullable collections for null vs empty testing
/// </summary>
public class TestOrderWithNullableCollections
{
public int Id { get; set; }
public string OrderNumber { get; set; } = "";
public List<TestOrderItem>? Items { get; set; }
public List<string>? Tags { get; set; }
}
/// <summary>
/// Class with all primitive types for WASM/serialization testing
/// </summary>
public class PrimitiveTestClass
{
public int IntValue { get; set; }
public long LongValue { get; set; }
public double DoubleValue { get; set; }
public decimal DecimalValue { get; set; }
public float FloatValue { get; set; }
public bool BoolValue { get; set; }
public string StringValue { get; set; } = "";
public Guid GuidValue { get; set; }
public DateTime DateTimeValue { get; set; }
public TestStatus EnumValue { get; set; }
public byte ByteValue { get; set; }
public short ShortValue { get; set; }
public int? NullableInt { get; set; }
public int? NullableIntNull { get; set; }
}
/// <summary>
/// Class with extended primitive types for full serializer coverage.
/// Includes DateTimeOffset, TimeSpan, Dictionary, null properties.
/// </summary>
public class ExtendedPrimitiveTestClass
{
public int Id { get; set; }
public string Name { get; set; } = "";
// Extended primitive types not covered in PrimitiveTestClass
public DateTimeOffset DateTimeOffsetValue { get; set; }
public TimeSpan TimeSpanValue { get; set; }
public uint UIntValue { get; set; }
public ulong ULongValue { get; set; }
public ushort UShortValue { get; set; }
public sbyte SByteValue { get; set; }
public char CharValue { get; set; }
// Dictionary property for WriteDictionary coverage in object context
public Dictionary<string, int>? Counts { get; set; }
public Dictionary<string, string>? Labels { get; set; }
// Nullable properties that will be null
public string? NullString { get; set; }
public TestOrderItem? NullObject { get; set; }
// Nested object for complex serialization
public SharedTag? Tag { get; set; }
}
/// <summary>
/// Class with array of objects containing null items for WriteNull coverage
/// </summary>
public class ObjectWithNullItems
{
public int Id { get; set; }
public List<object?> MixedItems { get; set; } = [];
}
#endregion
#region Property Mismatch DTOs (Server has more properties than Client)
/// <summary>
/// "Server-side" DTO with extra properties that the "client" doesn't know about.
/// Used to test SkipValue functionality when deserializing unknown properties.
/// </summary>
public class ServerCustomerDto : IId<int>
{
public int Id { get; set; }
// Common properties (both server and client have these)
public string FirstName { get; set; } = "";
public string LastName { get; set; } = "";
// Extra properties (only server has these - client will skip them)
public string Email { get; set; } = "";
public string Phone { get; set; } = "";
public string Address { get; set; } = "";
public string City { get; set; } = "";
public string Country { get; set; } = "";
public string PostalCode { get; set; } = "";
public string Company { get; set; } = "";
public string Department { get; set; } = "";
public string Notes { get; set; } = "";
public DateTime CreatedAt { get; set; } = DateTime.UtcNow;
public DateTime? LastLoginAt { get; set; }
public TestStatus Status { get; set; } = TestStatus.Active;
public bool IsVerified { get; set; }
public int LoginCount { get; set; }
public decimal Balance { get; set; }
}
/// <summary>
/// "Client-side" DTO with fewer properties than the server version.
/// When deserializing ServerCustomerDto data into this type,
/// the deserializer must skip unknown properties correctly
/// while still maintaining string intern table consistency.
/// </summary>
public class ClientCustomerDto : IId<int>
{
public int Id { get; set; }
// Only basic properties - client doesn't need all server fields
public string FirstName { get; set; } = "";
public string LastName { get; set; } = "";
}
/// <summary>
/// Server DTO with nested objects that client doesn't know about.
/// Tests skipping complex nested structures.
/// </summary>
public class ServerOrderWithExtras : IId<int>
{
public int Id { get; set; }
public string OrderNumber { get; set; } = "";
public decimal TotalAmount { get; set; }
// Nested object that client doesn't have
public ServerCustomerDto? Customer { get; set; }
// List of objects that client doesn't know about
public List<ServerCustomerDto> RelatedCustomers { get; set; } = [];
// Extra simple properties
public string InternalNotes { get; set; } = "";
public string ProcessingCode { get; set; } = "";
}
/// <summary>
/// Client version of the order - doesn't have Customer/RelatedCustomers properties.
/// </summary>
public class ClientOrderSimple : IId<int>
{
public int Id { get; set; }
public string OrderNumber { get; set; } = "";
public decimal TotalAmount { get; set; }
}
#endregion