Skip to content

Commit 1bee0a6

Browse files
bugfix/empty-structs (#59)
Fixed equality bug due to null reference exceptions in empty/default structs.
1 parent 4f3e8c6 commit 1bee0a6

18 files changed

+143
-12
lines changed

OnixLabs.Core.UnitTests/Text/Base16Tests.cs

+15
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,21 @@ public void Base16ShouldNotChangeWhenModifyingObtainedByteArray()
5252
Assert.Equal(expected, actual);
5353
}
5454

55+
[Fact(DisplayName = "Base16 default values should be identical")]
56+
public void Base16DefaultValuesShouldBeIdentical()
57+
{
58+
// Given
59+
Base16 a = new();
60+
Base16 b = default;
61+
62+
// Then
63+
Assert.Equal(a, b);
64+
Assert.Equal(a.GetHashCode(), b.GetHashCode());
65+
Assert.True(a.Equals(b));
66+
Assert.True(a == b);
67+
Assert.False(a != b);
68+
}
69+
5570
[Fact(DisplayName = "Base16 values should be identical")]
5671
public void Base16ValuesShouldBeIdentical()
5772
{

OnixLabs.Core.UnitTests/Text/Base32Tests.cs

+15
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,21 @@ public void Base32ShouldNotChangeWhenModifyingObtainedByteArray()
5252
Assert.Equal(expected, actual);
5353
}
5454

55+
[Fact(DisplayName = "Base32 default values should be identical")]
56+
public void Base32DefaultValuesShouldBeIdentical()
57+
{
58+
// Given
59+
Base32 a = new();
60+
Base32 b = default;
61+
62+
// Then
63+
Assert.Equal(a, b);
64+
Assert.Equal(a.GetHashCode(), b.GetHashCode());
65+
Assert.True(a.Equals(b));
66+
Assert.True(a == b);
67+
Assert.False(a != b);
68+
}
69+
5570
[Fact(DisplayName = "Base32 values should be identical")]
5671
public void Base32ValuesShouldBeIdentical()
5772
{

OnixLabs.Core.UnitTests/Text/Base58Tests.cs

+15
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,21 @@ public void Base58ShouldNotChangeWhenModifyingObtainedByteArray()
5252
Assert.Equal(expected, actual);
5353
}
5454

55+
[Fact(DisplayName = "Base58 default values should be identical")]
56+
public void Base58DefaultValuesShouldBeIdentical()
57+
{
58+
// Given
59+
Base58 a = new();
60+
Base58 b = default;
61+
62+
// Then
63+
Assert.Equal(a, b);
64+
Assert.Equal(a.GetHashCode(), b.GetHashCode());
65+
Assert.True(a.Equals(b));
66+
Assert.True(a == b);
67+
Assert.False(a != b);
68+
}
69+
5570
[Fact(DisplayName = "Base58 values should be identical")]
5671
public void Base58ValuesShouldBeIdentical()
5772
{

OnixLabs.Core.UnitTests/Text/Base64Tests.cs

+15
Original file line numberDiff line numberDiff line change
@@ -52,6 +52,21 @@ public void Base64ShouldNotChangeWhenModifyingObtainedByteArray()
5252
Assert.Equal(expected, actual);
5353
}
5454

55+
[Fact(DisplayName = "Base64 default values should be identical")]
56+
public void Base64DefaultValuesShouldBeIdentical()
57+
{
58+
// Given
59+
Base64 a = new();
60+
Base64 b = default;
61+
62+
// Then
63+
Assert.Equal(a, b);
64+
Assert.Equal(a.GetHashCode(), b.GetHashCode());
65+
Assert.True(a.Equals(b));
66+
Assert.True(a == b);
67+
Assert.False(a != b);
68+
}
69+
5570
[Fact(DisplayName = "Base64 values should be identical")]
5671
public void Base64ValuesShouldBeIdentical()
5772
{

OnixLabs.Core/Linq/Extensions.IEnumerable.cs

+16-1
Original file line numberDiff line numberDiff line change
@@ -90,8 +90,10 @@ public static void ForEach<T>(this IEnumerable<T> enumerable, Action<T> action)
9090
/// <param name="enumerable">The current <see cref="IEnumerable{T}"/> from which to compute a content hash code.</param>
9191
/// <typeparam name="T">The underlying type of the <see cref="IEnumerable{T}"/>.</typeparam>
9292
/// <returns>Returns the computed content hash code of the current <see cref="IEnumerable{T}"/>.</returns>
93-
public static int GetContentHashCode<T>(this IEnumerable<T> enumerable)
93+
public static int GetContentHashCode<T>(this IEnumerable<T>? enumerable)
9494
{
95+
if (enumerable is null) return default;
96+
9597
HashCode result = new();
9698
enumerable.ForEach(element => result.Add(element));
9799
return result.ToHashCode();
@@ -173,6 +175,19 @@ public static Optional<T> LastOrNone<T>(this IEnumerable<T> enumerable, Func<T,
173175
/// <returns>Returns <see langword="true"/> if none of the elements of the current <see cref="IEnumerable{T}"/> satisfy the specified predicate condition; otherwise, <see langword="false"/>.</returns>
174176
public static bool None<T>(this IEnumerable<T> enumerable, Func<T, bool> predicate) => !enumerable.Any(predicate);
175177

178+
/// <summary>Determines whether two sequences are <see langword="null"/>, or equal by comparing their elements by using a specified <see cref="T:IEqualityComparer{T}" />.</summary>
179+
/// <param name="first">An <see cref="T:IEnumerable{T}" /> to compare to <paramref name="second" />.</param>
180+
/// <param name="second">An <see cref="T:IEnumerable{T}" /> to compare to the first sequence.</param>
181+
/// <param name="comparer">An <see cref="T:IEqualityComparer{T}" /> to use to compare elements.</param>
182+
/// <typeparam name="T">The underlying type of the <see cref="IEnumerable{T}"/>.</typeparam>
183+
/// <returns> Returns <see langword="true" /> if the two source sequences are <see langword="null"/>, or of equal length and their corresponding elements compare equal according to <paramref name="comparer" />; otherwise, <see langword="false" />.</returns>
184+
public static bool SequenceEqualOrNull<T>(this IEnumerable<T>? first, IEnumerable<T>? second, EqualityComparer<T>? comparer = null)
185+
{
186+
if (first is null && second is null) return true;
187+
if (first is null || second is null) return false;
188+
return first.SequenceEqual(second, comparer ?? EqualityComparer<T>.Default);
189+
}
190+
176191
/// <summary>
177192
/// Obtains a single element of the current <see cref="IEnumerable{T}"/> that satisfies the specified predicate;
178193
/// otherwise <see cref="Optional{T}.None"/> if no such element is found,

OnixLabs.Core/OnixLabs.Core.csproj

+1-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,7 @@
1111
<NeutralLanguage>en</NeutralLanguage>
1212
<Copyright>Copyright © ONIXLabs 2020</Copyright>
1313
<RepositoryUrl>https://github.com/onix-labs/onixlabs-dotnet</RepositoryUrl>
14-
<PackageVersion>8.4.0</PackageVersion>
14+
<PackageVersion>8.5.0</PackageVersion>
1515
<PackageLicenseUrl></PackageLicenseUrl>
1616
</PropertyGroup>
1717
<PropertyGroup>

OnixLabs.Core/Text/Base16.Equatable.cs

+1-2
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,6 @@
1212
// See the License for the specific language governing permissions and
1313
// limitations under the License.
1414

15-
using System.Linq;
1615
using OnixLabs.Core.Linq;
1716

1817
namespace OnixLabs.Core.Text;
@@ -24,7 +23,7 @@ public readonly partial struct Base16
2423
/// </summary>
2524
/// <param name="other">An object to compare with the current object.</param>
2625
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Base16 other) => other.value.SequenceEqual(value);
26+
public bool Equals(Base16 other) => other.value.SequenceEqualOrNull(value);
2827

2928
/// <summary>
3029
/// Checks for equality between the current instance and another object.

OnixLabs.Core/Text/Base32.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Base32
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Base32 other) => other.value.SequenceEqual(value);
27+
public bool Equals(Base32 other) => other.value.SequenceEqualOrNull(value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

OnixLabs.Core/Text/Base58.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Base58
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Base58 other) => other.value.SequenceEqual(value);
27+
public bool Equals(Base58 other) => other.value.SequenceEqualOrNull(value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

OnixLabs.Core/Text/Base64.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Base64
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Base64 other) => other.value.SequenceEqual(value);
27+
public bool Equals(Base64 other) => other.value.SequenceEqualOrNull(value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

OnixLabs.Numerics/OnixLabs.Numerics.csproj

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
1111
<Copyright>Copyright © ONIXLabs 2020</Copyright>
1212
<RepositoryUrl>https://github.com/onix-labs/onixlabs-dotnet</RepositoryUrl>
13-
<PackageVersion>8.4.0</PackageVersion>
13+
<PackageVersion>8.5.0</PackageVersion>
1414
<LangVersion>12</LangVersion>
1515
<PackageLicenseUrl></PackageLicenseUrl>
1616
</PropertyGroup>

OnixLabs.Security.Cryptography.UnitTests/DigitalSignatureTests.cs

+19
Original file line numberDiff line numberDiff line change
@@ -65,6 +65,21 @@ public void DigitalSignatureValueShouldNotBeModifiedWhenAlteringTheObtainedByteA
6565
Assert.Equal(expected, actual);
6666
}
6767

68+
[Fact(DisplayName = "Identical default signature values should be considered equal")]
69+
public void IdenticalDefaultDigitalSignatureValuesShouldBeConsideredEqual()
70+
{
71+
// Given
72+
DigitalSignature left = new();
73+
DigitalSignature right = default;
74+
75+
// Then
76+
Assert.Equal(left, right);
77+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
78+
Assert.True(left.Equals(right));
79+
Assert.True(left == right);
80+
Assert.False(left != right);
81+
}
82+
6883
[Fact(DisplayName = "Identical signature values should be considered equal")]
6984
public void IdenticalDigitalSignatureValuesShouldBeConsideredEqual()
7085
{
@@ -74,8 +89,10 @@ public void IdenticalDigitalSignatureValuesShouldBeConsideredEqual()
7489

7590
// Then
7691
Assert.Equal(left, right);
92+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
7793
Assert.True(left.Equals(right));
7894
Assert.True(left == right);
95+
Assert.False(left != right);
7996
}
8097

8198
[Fact(DisplayName = "Different signature values should not be considered equal")]
@@ -87,7 +104,9 @@ public void DifferentDigitalSignatureValuesShouldNotBeConsideredEqual()
87104

88105
// Then
89106
Assert.NotEqual(left, right);
107+
Assert.NotEqual(left.GetHashCode(), right.GetHashCode());
90108
Assert.False(left.Equals(right));
109+
Assert.False(left == right);
91110
Assert.True(left != right);
92111
}
93112

OnixLabs.Security.Cryptography.UnitTests/HashTests.cs

+19
Original file line numberDiff line numberDiff line change
@@ -85,6 +85,21 @@ public void HashValueShouldNotBeModifiedWhenAlteringTheObtainedByteArray()
8585
Assert.Equal(expected, actual);
8686
}
8787

88+
[Fact(DisplayName = "Identical default hash values should be considered equal")]
89+
public void IdenticalDefaultHashValuesShouldBeConsideredEqual()
90+
{
91+
// Given
92+
Hash left = new();
93+
Hash right = default;
94+
95+
// Then
96+
Assert.Equal(left, right);
97+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
98+
Assert.True(left.Equals(right));
99+
Assert.True(left == right);
100+
Assert.False(left != right);
101+
}
102+
88103
[Fact(DisplayName = "Identical hash values should be considered equal")]
89104
public void IdenticalHashValuesShouldBeConsideredEqual()
90105
{
@@ -94,8 +109,10 @@ public void IdenticalHashValuesShouldBeConsideredEqual()
94109

95110
// Then
96111
Assert.Equal(left, right);
112+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
97113
Assert.True(left.Equals(right));
98114
Assert.True(left == right);
115+
Assert.False(left != right);
99116
}
100117

101118
[Fact(DisplayName = "Different hash values should not be considered equal")]
@@ -107,7 +124,9 @@ public void DifferentHashValuesShouldNotBeConsideredEqual()
107124

108125
// Then
109126
Assert.NotEqual(left, right);
127+
Assert.NotEqual(left.GetHashCode(), right.GetHashCode());
110128
Assert.False(left.Equals(right));
129+
Assert.False(left == right);
111130
Assert.True(left != right);
112131
}
113132

OnixLabs.Security.Cryptography.UnitTests/SaltTests.cs

+19
Original file line numberDiff line numberDiff line change
@@ -66,6 +66,21 @@ public void SaltValueShouldNotBeModifiedWhenAlteringTheObtainedByteArray()
6666
Assert.Equal(expected, actual);
6767
}
6868

69+
[Fact(DisplayName = "Identical default salt values should be considered equal")]
70+
public void IdenticalDefaultSaltValuesShouldBeConsideredEqual()
71+
{
72+
// Given
73+
Salt left = new();
74+
Salt right = default;
75+
76+
// Then
77+
Assert.Equal(left, right);
78+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
79+
Assert.True(left.Equals(right));
80+
Assert.True(left == right);
81+
Assert.False(left != right);
82+
}
83+
6984
[Fact(DisplayName = "Identical salt values should be considered equal")]
7085
public void IdenticalSaltValuesShouldBeConsideredEqual()
7186
{
@@ -75,8 +90,10 @@ public void IdenticalSaltValuesShouldBeConsideredEqual()
7590

7691
// Then
7792
Assert.Equal(left, right);
93+
Assert.Equal(left.GetHashCode(), right.GetHashCode());
7894
Assert.True(left.Equals(right));
7995
Assert.True(left == right);
96+
Assert.False(left != right);
8097
}
8198

8299
[Fact(DisplayName = "Different salt values should not be considered equal")]
@@ -88,7 +105,9 @@ public void DifferentSaltValuesShouldNotBeConsideredEqual()
88105

89106
// Then
90107
Assert.NotEqual(left, right);
108+
Assert.NotEqual(left.GetHashCode(), right.GetHashCode());
91109
Assert.False(left.Equals(right));
110+
Assert.False(left == right);
92111
Assert.True(left != right);
93112
}
94113

OnixLabs.Security.Cryptography/DigitalSignature.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct DigitalSignature
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(DigitalSignature other) => value.SequenceEqual(other.value);
27+
public bool Equals(DigitalSignature other) => value.SequenceEqualOrNull(other.value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

OnixLabs.Security.Cryptography/Hash.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Hash
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Hash other) => value.SequenceEqual(other.value);
27+
public bool Equals(Hash other) => value.SequenceEqualOrNull(other.value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

OnixLabs.Security.Cryptography/OnixLabs.Security.Cryptography.csproj

+1-1
Original file line numberDiff line numberDiff line change
@@ -10,7 +10,7 @@
1010
<GeneratePackageOnBuild>true</GeneratePackageOnBuild>
1111
<Copyright>Copyright © ONIXLabs 2020</Copyright>
1212
<RepositoryUrl>https://github.com/onix-labs/onixlabs-dotnet</RepositoryUrl>
13-
<PackageVersion>8.4.0</PackageVersion>
13+
<PackageVersion>8.5.0</PackageVersion>
1414
<LangVersion>12</LangVersion>
1515
<PackageLicenseUrl></PackageLicenseUrl>
1616
</PropertyGroup>

OnixLabs.Security.Cryptography/Salt.Equatable.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -24,7 +24,7 @@ public readonly partial struct Salt
2424
/// </summary>
2525
/// <param name="other">An object to compare with the current object.</param>
2626
/// <returns>Returns <see langword="true"/> if the current object is equal to the other parameter; otherwise, <see langword="false"/>.</returns>
27-
public bool Equals(Salt other) => value.SequenceEqual(other.value);
27+
public bool Equals(Salt other) => value.SequenceEqualOrNull(other.value);
2828

2929
/// <summary>
3030
/// Checks for equality between the current instance and another object.

0 commit comments

Comments
 (0)