From 12a68a95538c634e40c2a6878bdf1730ab4e4b72 Mon Sep 17 00:00:00 2001 From: "Ilya.Usov" Date: Wed, 15 Jan 2025 15:29:06 +0100 Subject: [PATCH] Add IsInitialized + IsUninitialized api Fix tests Check IsInitialized in ValueLifetimed + add an assertion that the passed lifetime is initialized --- rd-net/Lifetimes/Lifetimes/Lifetime.cs | 35 ++++++++++++++----- rd-net/Lifetimes/Lifetimes/ValueLifetimed.cs | 4 ++- .../Test.Lifetimes/Lifetimes/LifetimeTest.cs | 25 ++++++++----- 3 files changed, 47 insertions(+), 17 deletions(-) diff --git a/rd-net/Lifetimes/Lifetimes/Lifetime.cs b/rd-net/Lifetimes/Lifetimes/Lifetime.cs index 5c01e6a3e..f7d2c272f 100644 --- a/rd-net/Lifetimes/Lifetimes/Lifetime.cs +++ b/rd-net/Lifetimes/Lifetimes/Lifetime.cs @@ -117,18 +117,26 @@ internal LifetimeDefinition Definition var def = myDefinition; if (def != null) return def; - if (LogErrorIfLifetimeIsNotInitialized) - { - Log.Root.Error("Lifetime is not initialized. " + - "This may cause a memory leak as default(Lifetime) is treated as Eternal. " + - "Please provide a properly initialized Lifetime or use `Lifetime?` if you need to handle both cases. " + - "Use Lifetime.Eternal explicitly if that behavior is intended."); - } - + Assertion.Assert(IsUninitialized); + + AssertInitialized(); return LifetimeDefinition.Eternal; } } + internal void AssertInitialized() + { + if (!Mode.IsAssertion) return; + + if (LogErrorIfLifetimeIsNotInitialized && IsUninitialized) + { + Log.Root.Error("Lifetime is not initialized. " + + "This may cause a memory leak as default(Lifetime) is treated as Eternal. " + + "Please provide a properly initialized Lifetime or use `Lifetime?` if you need to handle both cases. " + + "Use Lifetime.Eternal explicitly if that behavior is intended."); + } + } + //ctor internal Lifetime(LifetimeDefinition definition) { @@ -167,6 +175,17 @@ internal Lifetime(LifetimeDefinition definition) /// Whether current lifetime is equal to and never be terminated /// [PublicAPI] public bool IsEternal => Definition.IsEternal; + + + /// + /// Whether current lifetime is not properly initialized (created by default(Lifetime)) + /// + [PublicAPI] public bool IsUninitialized => !IsInitialized; + + /// + /// Whether current lifetime is properly initialized + /// + [PublicAPI] public bool IsInitialized => myDefinition != null; /// /// Is of this lifetime equal to diff --git a/rd-net/Lifetimes/Lifetimes/ValueLifetimed.cs b/rd-net/Lifetimes/Lifetimes/ValueLifetimed.cs index 90a0bc214..e6c70822b 100644 --- a/rd-net/Lifetimes/Lifetimes/ValueLifetimed.cs +++ b/rd-net/Lifetimes/Lifetimes/ValueLifetimed.cs @@ -20,13 +20,15 @@ public void Deconstruct(out Lifetime lifetime, out T? value) public ValueLifetimed(Lifetime lifetime, T value) { + lifetime.AssertInitialized(); + Lifetime = lifetime; Value = value; } public void ClearValueIfNotAlive() { - if (!Lifetime.IsAlive) + if (Lifetime is { IsInitialized: true, IsAlive: false }) Value = default(T); } } diff --git a/rd-net/Test.Lifetimes/Lifetimes/LifetimeTest.cs b/rd-net/Test.Lifetimes/Lifetimes/LifetimeTest.cs index 9455c3383..eae8a71d7 100644 --- a/rd-net/Test.Lifetimes/Lifetimes/LifetimeTest.cs +++ b/rd-net/Test.Lifetimes/Lifetimes/LifetimeTest.cs @@ -152,16 +152,25 @@ public void TestEternal() [Test] public void TestEquals() { - Lifetime eternal = default; - Assert.AreEqual(Lifetime.Eternal, eternal); - Assert.AreEqual(Lifetime.Eternal, Lifetime.Eternal); - Assert.AreEqual(eternal, eternal); + var old = Lifetime.LogErrorIfLifetimeIsNotInitialized; + Lifetime.LogErrorIfLifetimeIsNotInitialized = false; + try + { + Lifetime eternal = default; + Assert.AreEqual(Lifetime.Eternal, eternal); + Assert.AreEqual(Lifetime.Eternal, Lifetime.Eternal); + Assert.AreEqual(eternal, eternal); - Assert.True(Lifetime.Eternal == eternal); + Assert.True(Lifetime.Eternal == eternal); - Assert.AreNotEqual(Lifetime.Eternal, Lifetime.Terminated); - Assert.False(Lifetime.Eternal == Lifetime.Terminated); - Assert.False(eternal == Lifetime.Terminated); + Assert.AreNotEqual(Lifetime.Eternal, Lifetime.Terminated); + Assert.False(Lifetime.Eternal == Lifetime.Terminated); + Assert.False(eternal == Lifetime.Terminated); + } + finally + { + Lifetime.LogErrorIfLifetimeIsNotInitialized = old; + } } [Test]