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

Reduce Serializer and Index Complexity #107

Merged
merged 11 commits into from
Oct 22, 2024
34 changes: 19 additions & 15 deletions src/NexusMods.MnemonicDB.Abstractions/Attribute.cs
Original file line number Diff line number Diff line change
@@ -1,34 +1,30 @@
using System;
using System.Buffers;
using System.Collections.Generic;
using System.Diagnostics;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Text;
using NexusMods.MnemonicDB.Abstractions.DatomIterators;
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.Exceptions;
using NexusMods.MnemonicDB.Abstractions.Internals;
using NexusMods.MnemonicDB.Abstractions.Models;
using Reloaded.Memory.Extensions;

namespace NexusMods.MnemonicDB.Abstractions;

/// <summary>
/// Interface for a specific attribute
/// </summary>
/// <typeparam name="TValueType"></typeparam>
public abstract class Attribute<TValueType, TLowLevelType> : IAttribute<TValueType>
public abstract class Attribute<TValueType, TLowLevelType, TSerializer> :
IWritableAttribute<TValueType>,
IReadableAttribute<TValueType>
where TSerializer : IValueSerializer<TLowLevelType>
{
protected Attribute(
ValueTag lowLevelType,
string ns,
string name,
bool isIndexed = false,
bool noHistory = false,
Cardinality cardinality = Cardinality.One)
{
LowLevelType = lowLevelType;

Id = Symbol.Intern(ns, name);
Cardinalty = cardinality;
IsIndexed = isIndexed;
Expand All @@ -44,9 +40,9 @@ protected Attribute(
/// Converts a high-level value to a low-level value
/// </summary>
protected abstract TValueType FromLowLevel(TLowLevelType value, AttributeResolver resolver);

/// <inheritdoc />
public ValueTag LowLevelType { get; }
public ValueTag LowLevelType => TSerializer.ValueTag;

/// <inheritdoc />
public Symbol Id { get; }
Expand Down Expand Up @@ -92,12 +88,20 @@ public ReadDatom Resolve(in Datom datom, AttributeResolver resolver)
return new ReadDatom(in prefix, ReadValue(datom.ValueSpan, datom.Prefix.ValueTag, resolver), this);
}

[MethodImpl(MethodImplOptions.AggressiveInlining | MethodImplOptions.AggressiveOptimization)]
private void AssertTag(ValueTag tag)
{
if (tag != LowLevelType)
throw new InvalidCastException($"Invalid value tag for attribute {Id.Name}");
}

/// <summary>
/// Reads the high level value from the given span
/// </summary>
public TValueType ReadValue(ReadOnlySpan<byte> span, ValueTag tag, AttributeResolver resolver)
{
return FromLowLevel(tag.Read<TLowLevelType>(span), resolver);
AssertTag(tag);
return FromLowLevel(TSerializer.Read(span), resolver);
}

/// <summary>
Expand Down Expand Up @@ -180,7 +184,7 @@ public override string ToString()
/// <summary>
/// Typed datom for this attribute
/// </summary>
public ReadDatom(in KeyPrefix prefix, TValueType v, Attribute<TValueType, TLowLevelType> a)
public ReadDatom(in KeyPrefix prefix, TValueType v, Attribute<TValueType, TLowLevelType, TSerializer> a)
{
Prefix = prefix;
TypedAttribute = a;
Expand All @@ -190,7 +194,7 @@ public ReadDatom(in KeyPrefix prefix, TValueType v, Attribute<TValueType, TLowLe
/// <summary>
/// The typed attribute for this datom
/// </summary>
public readonly Attribute<TValueType, TLowLevelType> TypedAttribute;
public readonly Attribute<TValueType, TLowLevelType, TSerializer> TypedAttribute;

/// <summary>
/// The abstract attribute for this datom
Expand Down Expand Up @@ -218,7 +222,7 @@ public ReadDatom(in KeyPrefix prefix, TValueType v, Attribute<TValueType, TLowLe
/// <inheritdoc />
public void Retract(ITransaction tx)
{
tx.Add(E, (Attribute<TValueType, TLowLevelType>)A, V, true);
tx.Add(E, (Attribute<TValueType, TLowLevelType, TSerializer>)A, V, true);
}

/// <inheritdoc />
Expand Down
1 change: 0 additions & 1 deletion src/NexusMods.MnemonicDB.Abstractions/AttributeCache.cs
Original file line number Diff line number Diff line change
Expand Up @@ -50,7 +50,6 @@ public AttributeCache()
/// <summary>
/// Resets the cache, causing it to re-query the database for the latest definitions.
/// </summary>
/// <param name="idb"></param>
public void Reset(IDb db)
{
var symbols = db.Datoms(AttributeDefinition.UniqueId);
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// Used to mark the cardinality of an attribute in the database
/// </summary>
public sealed class CardinalityAttribute(string ns, string name) : ScalarAttribute<Cardinality, byte>(ValueTag.UInt8, ns, name)
public sealed class CardinalityAttribute(string ns, string name) : ScalarAttribute<Cardinality, byte, UInt8Serializer>(ns, name)
{
/// <inheritdoc />
protected override byte ToLowLevel(Cardinality value) => (byte)value;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,22 +1,20 @@
using System.Collections.Generic;
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.IndexSegments;
using NexusMods.MnemonicDB.Abstractions.IndexSegments;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.Query;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that represents a collection of values
/// </summary>
public abstract class CollectionAttribute<TValue, TLowLevel>(ValueTag tag, string ns, string name)
: Attribute<TValue, TLowLevel>(tag, ns, name, cardinality: Cardinality.Many)
public abstract class CollectionAttribute<TValue, TLowLevel, TSerializer>(string ns, string name)
: Attribute<TValue, TLowLevel, TSerializer>(ns, name, cardinality: Cardinality.Many)
where TSerializer : IValueSerializer<TLowLevel>
{

/// <summary>
/// Gets all values for this attribute on the given entity
/// </summary>
public Values<TValue, TLowLevel> Get(IHasIdAndIndexSegment ent)
public Values<TValue> Get(IHasIdAndIndexSegment ent)
{
var segment = ent.IndexSegment;
var dbId = ent.Db.AttributeCache.GetAttributeId(Id);
Expand All @@ -30,16 +28,16 @@ public Values<TValue, TLowLevel> Get(IHasIdAndIndexSegment ent)
{
i++;
}
return new Values<TValue, TLowLevel>(segment, start, i, this, ent.Db.Connection.AttributeResolver);
return new Values<TValue>(segment, start, i, this, ent.Db.Connection.AttributeResolver);
}
return new Values<TValue, TLowLevel>(segment, 0, 0, this, ent.Db.Connection.AttributeResolver);
return new Values<TValue>(segment, 0, 0, this, ent.Db.Connection.AttributeResolver);
}

/// <summary>
/// Gets all values for this attribute on the given entity, this performs a lookup in the database
/// so prefer using the overload with IHasIdAndIndexSegment if you already have the segment.
/// </summary>
protected Values<TValue, TLowLevel> Get(IHasEntityIdAndDb ent)
protected Values<TValue> Get(IHasEntityIdAndDb ent)
{
var segment = ent.Db.Get(ent.Id);
var dbId = ent.Db.AttributeCache.GetAttributeId(Id);
Expand All @@ -53,9 +51,9 @@ protected Values<TValue, TLowLevel> Get(IHasEntityIdAndDb ent)
{
i++;
}
return new Values<TValue, TLowLevel>(segment, start, i, this, ent.Db.Connection.AttributeResolver);
return new Values<TValue>(segment, start, i, this, ent.Db.Connection.AttributeResolver);
}
return new Values<TValue, TLowLevel>(segment, 0, 0, this, ent.Db.Connection.AttributeResolver);
return new Values<TValue>(segment, 0, 0, this, ent.Db.Connection.AttributeResolver);
}

/// <summary>
Expand Down
Original file line number Diff line number Diff line change
@@ -1,5 +1,6 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

Expand All @@ -9,7 +10,7 @@ namespace NexusMods.MnemonicDB.Abstractions.Attributes;
/// </summary>
/// <param name="ns"></param>
/// <param name="name"></param>
public class MarkerAttribute(string ns, string name) : Attribute<Null, Null>(ValueTag.Null, ns, name)
public class MarkerAttribute(string ns, string name) : Attribute<Null, Null, NullSerializer>(ns, name)
{
/// <inheritdoc />
protected override Null ToLowLevel(Null value) => value;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that references another entity.
/// </summary>
public class ReferenceAttribute(string ns, string name) : ScalarAttribute<EntityId, EntityId>(ValueTag.Reference, ns, name)
public class ReferenceAttribute(string ns, string name) : ScalarAttribute<EntityId, EntityId, EntityIdSerializer>(ns, name)
{
/// <inheritdoc />
protected override EntityId ToLowLevel(EntityId value) => value;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,12 +1,13 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// Represents a collection of references to other entities.
/// </summary>
public class ReferencesAttribute(string ns, string name) : CollectionAttribute<EntityId, EntityId>(ValueTag.Reference, ns, name)
public class ReferencesAttribute(string ns, string name) : CollectionAttribute<EntityId, EntityId, EntityIdSerializer>(ns, name)
{
/// <inheritdoc />
protected override EntityId ToLowLevel(EntityId value) => value;
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -8,8 +8,10 @@ namespace NexusMods.MnemonicDB.Abstractions.Attributes;
/// <summary>
/// An attribute that represents a scalar value, where there is a 1:1 ratio between the attribute and the value.
/// </summary>
public abstract class ScalarAttribute<TValue, TLowLevel>(ValueTag tag, string ns, string name) :
Attribute<TValue, TLowLevel>(tag, ns, name) where TValue : notnull
public abstract class ScalarAttribute<TValue, TLowLevel, TSerializer>(string ns, string name) :
Attribute<TValue, TLowLevel, TSerializer>(ns, name)
where TSerializer : IValueSerializer<TLowLevel>
where TValue : notnull
{
/// <summary>
/// True if the attribute is optional, and not required by models
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that represents a string.
/// </summary>
public sealed class StringAttribute(string ns, string name) : ScalarAttribute<string, string>(ValueTag.Utf8, ns, name)
public sealed class StringAttribute(string ns, string name) : ScalarAttribute<string, string, Utf8Serializer>(ns, name)
{
/// <inheritdoc />
protected override string ToLowLevel(string value) => value;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that encodes a symbol.
/// </summary>
public class SymbolAttribute(string ns, string name) : ScalarAttribute<Symbol, string>(ValueTag.Ascii, ns, name)
public class SymbolAttribute(string ns, string name) : ScalarAttribute<Symbol, string, AsciiSerializer>(ns, name)
{
/// <inheritdoc />
protected override string ToLowLevel(Symbol value) => value.Id;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,13 +1,14 @@
using System;
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.Models;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that holds a timestamp
/// </summary>
public class TimestampAttribute(string ns, string name) : ScalarAttribute<DateTimeOffset, long>(ValueTag.Int64, ns, name)
public class TimestampAttribute(string ns, string name) : ScalarAttribute<DateTimeOffset, long, Int64Serializer>(ns, name)
{
/// <inheritdoc />
protected override long ToLowLevel(DateTimeOffset value) => value.ToUnixTimeMilliseconds();
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,11 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// UInt64 attribute (ulong)
/// </summary>
public sealed class ULongAttribute(string ns, string name) : ScalarAttribute<ulong, ulong>(ValueTag.UInt64, ns, name)
public sealed class ULongAttribute(string ns, string name) : ScalarAttribute<ulong, ulong, UInt64Serializer>(ns, name)
{
/// <inheritdoc />
protected override ulong ToLowLevel(ulong value) => value;
Expand Down
Original file line number Diff line number Diff line change
@@ -1,11 +1,12 @@
using NexusMods.MnemonicDB.Abstractions.ElementComparers;
using NexusMods.MnemonicDB.Abstractions.ValueSerializers;

namespace NexusMods.MnemonicDB.Abstractions.Attributes;

/// <summary>
/// An attribute that represents a value tag value
/// </summary>
public sealed class ValuesTagAttribute(string ns, string name) : ScalarAttribute<ValueTag, byte>(ValueTag.UInt8, ns, name)
public sealed class ValuesTagAttribute(string ns, string name) : ScalarAttribute<ValueTag, byte, UInt8Serializer>(ns, name)
{
/// <inheritdoc />
protected override byte ToLowLevel(ValueTag value) => (byte)value;
Expand Down
30 changes: 14 additions & 16 deletions src/NexusMods.MnemonicDB.Abstractions/Datom.cs
Original file line number Diff line number Diff line change
Expand Up @@ -47,7 +47,11 @@ public byte[] ToArray()
/// <summary>
/// The KeyPrefix of the datom
/// </summary>
public KeyPrefix Prefix => _prefix;
public KeyPrefix Prefix
{
get => _prefix;
init => _prefix = value;
}

/// <summary>
/// The valuespan of the datom
Expand Down Expand Up @@ -95,7 +99,9 @@ public Datom Clone()
/// <inheritdoc />
public override string ToString()
{
return $"[E: {E}, A: {A}, T: {T}, IsRetract: {IsRetract}, Value: {Prefix.ValueTag.Read<object>(ValueSpan)}]";
if (_prefix.Index == IndexType.None)
return $"[E: {E}, A: {A}, T: {T}, IsRetract: {IsRetract}, Value: {Prefix.ValueTag.Read<object>(ValueSpan)}]";
return $"[Index: {_prefix.Index}, E: {E}, A: {A}, T: {T}, IsRetract: {IsRetract}, Value: {Prefix.ValueTag.Read<object>(ValueSpan)}]";
}

/// <summary>
Expand All @@ -118,22 +124,14 @@ public IReadDatom Resolved(IConnection conn)
/// Returns -1 if this datom is less than the other, 0 if they are equal, and 1 if this datom is greater than the other.
/// in relation to the given index type.
/// </summary>
public int Compare(Datom other, IndexType indexType)
{
return indexType switch
{
IndexType.TxLog => DatomComparators.TxLogComparator.Compare(this, other),
IndexType.EAVTCurrent or IndexType.EAVTHistory => DatomComparators.EAVTComparator.Compare(this, other),
IndexType.AEVTCurrent or IndexType.AEVTHistory => DatomComparators.AEVTComparator.Compare(this, other),
IndexType.AVETCurrent or IndexType.AVETHistory => DatomComparators.AVETComparator.Compare(this, other),
IndexType.VAETCurrent or IndexType.VAETHistory => DatomComparators.VAETComparator.Compare(this, other),
_ => ThrowArgumentOutOfRange(indexType)
};
}
public int Compare(Datom other) => GlobalComparer.Compare(in this, in other);

private static int ThrowArgumentOutOfRange(IndexType indexType)
/// <summary>
/// Return a copy of this datom with the given index set as the index
/// </summary>
public Datom WithIndex(IndexType index)
{
throw new ArgumentOutOfRangeException(nameof(indexType), indexType, "Unknown index type");
return new Datom(_prefix with {Index = index}, _valueBlob);
}

/// <summary>
Expand Down
Loading
Loading