Skip to content

Commit dee17e0

Browse files
committed
Reverts Random.Decimal() implementation from PR 300 to previous v29 implementation. #319 #320
1 parent 749aa8a commit dee17e0

File tree

5 files changed

+91
-34
lines changed

5 files changed

+91
-34
lines changed

HISTORY.md

+7-2
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,15 @@
1+
## v30.0.4
2+
Release Date: 2020-08-15
3+
4+
* Issue 319: The `Random.Decimal()` implementation reverted to previous v29 implementation. Avoids `System.OverflowException` when calling `Random.Decimal(0, decimal.MaxValue)`. The v30 implementation moved to `Bogus.Extensions` namespace as `Random.Decimal2()` which can generate more decimal precision.
5+
16
## v30.0.3
2-
Release Date: 2020-08-13
7+
Release Date: 2020-08-13, UNPUBLISHED FROM NUGET
38

49
* Added `f.Address.CountryOfUnitedKingdom()` extension method in `Bogus.Extensions.UnitedKingdom`.
510

611
## v30.0.2
7-
Release Date: 2020-08-05
12+
Release Date: 2020-08-05, UNPUBLISHED FROM NUGET
813

914
* Deterministic sequences may have changed.
1015
* Promoted v30.0.1-beta-4 to v30.0.2 release.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
using System;
2+
using Bogus.Extensions;
3+
using FluentAssertions;
4+
using Xunit;
5+
6+
namespace Bogus.Tests.GitHubIssues
7+
{
8+
public class Issue319 : SeededTest
9+
{
10+
[Fact]
11+
public void can_generate_decimal_edge_case()
12+
{
13+
var r = new Randomizer();
14+
15+
Action a = () =>
16+
{
17+
r.Decimal(0m, decimal.MaxValue);
18+
r.Decimal(0m, decimal.MaxValue);
19+
r.Decimal(0m, decimal.MaxValue);
20+
r.Decimal(0m, decimal.MaxValue);
21+
};
22+
a.Should().NotThrow();
23+
}
24+
25+
[Fact]
26+
public void decimal2_should_throw_on_edge_case()
27+
{
28+
var r = new Randomizer();
29+
Action a = () =>
30+
{
31+
r.Decimal2(0, decimal.MaxValue);
32+
};
33+
34+
a.Should().Throw<OverflowException>();
35+
}
36+
}
37+
}

Source/Bogus.Tests/RandomizerTest.cs

+1-1
Original file line numberDiff line numberDiff line change
@@ -194,7 +194,7 @@ public void generate_double_with_min_and_max()
194194
[Fact]
195195
public void generate_decimal_with_min_and_max()
196196
{
197-
r.Decimal(2.2m, 5.2m).Should().Be(3.8697355489728032005903907232m);
197+
r.Decimal(2.2m, 5.2m).Should().Be(4.0105668499183690m);
198198
}
199199

200200
[Fact]
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
namespace Bogus.Extensions
2+
{
3+
public static class ExtensionsForRandomizer
4+
{
5+
/// <summary>
6+
/// Get a random decimal, between 0.0 and 1.0.
7+
/// </summary>
8+
/// <param name="min">Minimum, default 0.0</param>
9+
/// <param name="max">Maximum, default 1.0</param>
10+
public static decimal Decimal2(this Randomizer r, decimal min = 0.0m, decimal max = 1.0m)
11+
{
12+
// Decimal: 128 bits wide
13+
// bit 0: sign bit
14+
// bit 1-10: not used
15+
// bit 11-15: scale (values 29, 30, 31 not used)
16+
// bit 16-31: not used
17+
// bit 32-127: mantissa (96 bits)
18+
19+
// Max value: 00000000 FFFFFFFF FFFFFFFF FFFFFFFF
20+
// = 79228162514264337593543950335
21+
22+
// Max value with max scaling: 001C0000 FFFFFFFF FFFFFFFF FFFFFFFF
23+
// = 7.9228162514264337593543950335
24+
25+
// Step 1: Generate a value with uniform distribution between 0 and this value.
26+
// This ensures the greatest level of precision in the distribution of bits;
27+
// the resulting value, after it is adjusted into the caller's desired range,
28+
// should not skip any possible values at the least significant end of the
29+
// mantissa.
30+
31+
int lowBits = r.Number(int.MinValue, int.MaxValue);
32+
int middleBits = r.Number(int.MinValue, int.MaxValue);
33+
int highBits = r.Number(int.MinValue, int.MaxValue);
34+
35+
const int Scale = 28;
36+
37+
decimal result = new decimal(lowBits, middleBits, highBits, isNegative: false, Scale);
38+
39+
// Step 2: Scale the value and adjust it to the desired range. This may decrease
40+
// the accuracy by adjusting the scale as necessary, but we get the best possible
41+
// outcome by starting with the most precise scale.
42+
return result * (max - min) / 7.9228162514264337593543950335m + min;
43+
}
44+
}
45+
}

Source/Bogus/Randomizer.cs

+1-31
Original file line numberDiff line numberDiff line change
@@ -194,37 +194,7 @@ public double Double(double min = 0.0d, double max = 1.0d)
194194
/// <param name="max">Maximum, default 1.0</param>
195195
public decimal Decimal(decimal min = 0.0m, decimal max = 1.0m)
196196
{
197-
// Decimal: 128 bits wide
198-
// bit 0: sign bit
199-
// bit 1-10: not used
200-
// bit 11-15: scale (values 29, 30, 31 not used)
201-
// bit 16-31: not used
202-
// bit 32-127: mantissa (96 bits)
203-
204-
// Max value: 00000000 FFFFFFFF FFFFFFFF FFFFFFFF
205-
// = 79228162514264337593543950335
206-
207-
// Max value with max scaling: 001C0000 FFFFFFFF FFFFFFFF FFFFFFFF
208-
// = 7.9228162514264337593543950335
209-
210-
// Step 1: Generate a value with uniform distribution between 0 and this value.
211-
// This ensures the greatest level of precision in the distribution of bits;
212-
// the resulting value, after it is adjusted into the caller's desired range,
213-
// should not skip any possible values at the least significant end of the
214-
// mantissa.
215-
216-
int lowBits = Number(int.MinValue, int.MaxValue);
217-
int middleBits = Number(int.MinValue, int.MaxValue);
218-
int highBits = Number(int.MinValue, int.MaxValue);
219-
220-
const int Scale = 28;
221-
222-
decimal result = new decimal(lowBits, middleBits, highBits, isNegative: false, Scale);
223-
224-
// Step 2: Scale the value and adjust it to the desired range. This may decrease
225-
// the accuracy by adjusting the scale as necessary, but we get the best possible
226-
// outcome by starting with the most precise scale.
227-
return result * (max - min) / 7.9228162514264337593543950335m + min;
197+
return Convert.ToDecimal(Double()) * (max - min) + min;
228198
}
229199

230200
/// <summary>

0 commit comments

Comments
 (0)