// Copyright 2014 The Go Authors. All rights reserved. // Use of this source code is governed by a BSD-style // license that can be found in the LICENSE file. // Test that random number sequences generated by a specific seed // do not change from version to version. // // Do NOT make changes to the golden outputs. If bugs need to be fixed // in the underlying code, find ways to fix them that do not affect the // outputs. package rand_test import ( "bytes" "flag" "fmt" "go/format" "io" . "math/rand/v2" "os" "reflect" "strings" "testing" ) var update = flag.Bool("update", false, "update golden results for regression test") func TestRegress(t *testing.T) { var int32s = []int32{1, 10, 32, 1 << 20, 1<<20 + 1, 1000000000, 1 << 30, 1<<31 - 2, 1<<31 - 1} var uint32s = []uint32{1, 10, 32, 1 << 20, 1<<20 + 1, 1000000000, 1 << 30, 1<<31 - 2, 1<<31 - 1, 1<<32 - 2, 1<<32 - 1} var int64s = []int64{1, 10, 32, 1 << 20, 1<<20 + 1, 1000000000, 1 << 30, 1<<31 - 2, 1<<31 - 1, 1000000000000000000, 1 << 60, 1<<63 - 2, 1<<63 - 1} var uint64s = []uint64{1, 10, 32, 1 << 20, 1<<20 + 1, 1000000000, 1 << 30, 1<<31 - 2, 1<<31 - 1, 1000000000000000000, 1 << 60, 1<<63 - 2, 1<<63 - 1, 1<<64 - 2, 1<<64 - 1} var permSizes = []int{0, 1, 5, 8, 9, 10, 16} n := reflect.TypeOf(New(NewPCG(1, 2))).NumMethod() p := 0 var buf bytes.Buffer if *update { fmt.Fprintf(&buf, "var regressGolden = []any{\n") } for i := 0; i < n; i++ { if *update && i > 0 { fmt.Fprintf(&buf, "\n") } r := New(NewPCG(1, 2)) rv := reflect.ValueOf(r) m := rv.Type().Method(i) mv := rv.Method(i) mt := mv.Type() if mt.NumOut() == 0 { continue } for repeat := 0; repeat < 20; repeat++ { var args []reflect.Value var argstr string if mt.NumIn() == 1 { var x any switch mt.In(0).Kind() { default: t.Fatalf("unexpected argument type for r.%s", m.Name) case reflect.Int: if m.Name == "Perm" { x = permSizes[repeat%len(permSizes)] break } big := int64s[repeat%len(int64s)] if int64(int(big)) != big { // On 32-bit machine. // Consume an Int64 like on a 64-bit machine, // to keep the golden data the same on different architectures. r.Int64N(big) if *update { t.Fatalf("must run -update on 64-bit machine") } p++ continue } x = int(big) case reflect.Uint: if m.Name == "Uint" { continue } big := uint64s[repeat%len(uint64s)] if uint64(uint(big)) != big { r.Uint64N(big) // what would happen on 64-bit machine, to keep stream in sync if *update { t.Fatalf("must run -update on 64-bit machine") } p++ continue } x = uint(big) case reflect.Int32: x = int32s[repeat%len(int32s)] case reflect.Int64: x = int64s[repeat%len(int64s)] case reflect.Uint32: x = uint32s[repeat%len(uint32s)] case reflect.Uint64: x = uint64s[repeat%len(uint64s)] } argstr = fmt.Sprint(x) args = append(args, reflect.ValueOf(x)) } var out any out = mv.Call(args)[0].Interface() if m.Name == "Int" || m.Name == "IntN" { out = int64(out.(int)) } if m.Name == "Uint" || m.Name == "UintN" { out = uint64(out.(uint)) } if *update { var val string big := int64(1 << 60) if int64(int(big)) != big && (m.Name == "Int" || m.Name == "IntN" || m.Name == "Uint" || m.Name == "UintN") { // 32-bit machine cannot print 64-bit results val = "truncated" } else if reflect.TypeOf(out).Kind() == reflect.Slice { val = fmt.Sprintf("%#v", out) } else { val = fmt.Sprintf("%T(%v)", out, out) } fmt.Fprintf(&buf, "\t%s, // %s(%s)\n", val, m.Name, argstr) } else if p >= len(regressGolden) { t.Errorf("r.%s(%s) = %v, missing golden value", m.Name, argstr, out) } else { want := regressGolden[p] if m.Name == "Int" { want = int64(int(uint(want.(int64)) << 1 >> 1)) } if m.Name == "Uint" { want = uint64(uint(want.(uint64))) } if !reflect.DeepEqual(out, want) { t.Errorf("r.%s(%s) = %v, want %v", m.Name, argstr, out, want) } } p++ } } if *update { replace(t, "regress_test.go", buf.Bytes()) } } func TestUpdateExample(t *testing.T) { if !*update { t.Skip("-update not given") } oldStdout := os.Stdout defer func() { os.Stdout = oldStdout }() r, w, err := os.Pipe() if err != nil { t.Fatal(err) } defer r.Close() defer w.Close() go func() { os.Stdout = w Example_rand() os.Stdout = oldStdout w.Close() }() out, err := io.ReadAll(r) if err != nil { t.Fatal(err) } var buf bytes.Buffer fmt.Fprintf(&buf, "\t// Output:\n") for _, line := range strings.Split(string(out), "\n") { if line != "" { fmt.Fprintf(&buf, "\t// %s\n", line) } } replace(t, "example_test.go", buf.Bytes()) // Exit so that Example_rand cannot fail. fmt.Printf("UPDATED; ignore non-zero exit status\n") os.Exit(1) } // replace substitutes the definition text from new into the content of file. // The text in new is of the form // // var whatever = T{ // ... // } // // Replace searches file for an exact match for the text of the first line, // finds the closing brace, and then substitutes new for what used to be in the file. // This lets us update the regressGolden table during go test -update. func replace(t *testing.T, file string, new []byte) { first, _, _ := bytes.Cut(new, []byte("\n")) first = append(append([]byte("\n"), first...), '\n') data, err := os.ReadFile(file) if err != nil { t.Fatal(err) } i := bytes.Index(data, first) if i < 0 { t.Fatalf("cannot find %q in %s", first, file) } j := bytes.Index(data[i+1:], []byte("\n}\n")) if j < 0 { t.Fatalf("cannot find end in %s", file) } data = append(append(data[:i+1:i+1], new...), data[i+1+j+1:]...) data, err = format.Source(data) if err != nil { t.Fatal(err) } if err := os.WriteFile(file, data, 0666); err != nil { t.Fatal(err) } } var regressGolden = []any{ float64(0.5931317151369719), // ExpFloat64() float64(0.0680034588807843), // ExpFloat64() float64(0.036496967459790364), // ExpFloat64() float64(2.460335459645379), // ExpFloat64() float64(1.5792300208419903), // ExpFloat64() float64(0.9149501499404387), // ExpFloat64() float64(0.43463410545541104), // ExpFloat64() float64(0.5513632046504593), // ExpFloat64() float64(0.7426404617374481), // ExpFloat64() float64(1.2334925132631804), // ExpFloat64() float64(0.892529142200442), // ExpFloat64() float64(0.21508763681487764), // ExpFloat64() float64(1.0208588200798545), // ExpFloat64() float64(0.7650739736831382), // ExpFloat64() float64(0.7772788529257701), // ExpFloat64() float64(1.102732861281323), // ExpFloat64() float64(0.6982243043885805), // ExpFloat64() float64(0.4981788638202421), // ExpFloat64() float64(0.15806532306947937), // ExpFloat64() float64(0.9419163802459202), // ExpFloat64() float32(0.95955694), // Float32() float32(0.8076733), // Float32() float32(0.8135684), // Float32() float32(0.92872405), // Float32() float32(0.97472525), // Float32() float32(0.5485458), // Float32() float32(0.97740936), // Float32() float32(0.042272687), // Float32() float32(0.99663067), // Float32() float32(0.035181105), // Float32() float32(0.45059562), // Float32() float32(0.86597633), // Float32() float32(0.8954844), // Float32() float32(0.090798736), // Float32() float32(0.46218646), // Float32() float32(0.5955118), // Float32() float32(0.08985227), // Float32() float32(0.19820237), // Float32() float32(0.7443699), // Float32() float32(0.56461), // Float32() float64(0.6764556596678251), // Float64() float64(0.4613862177205994), // Float64() float64(0.5085473976760264), // Float64() float64(0.4297927436037299), // Float64() float64(0.797802349388613), // Float64() float64(0.3883664855410056), // Float64() float64(0.8192750264193612), // Float64() float64(0.3381816951746133), // Float64() float64(0.9730458047755973), // Float64() float64(0.281449117585586), // Float64() float64(0.6047654075331631), // Float64() float64(0.9278107175107462), // Float64() float64(0.16387541502137226), // Float64() float64(0.7263900707339023), // Float64() float64(0.6974917552729882), // Float64() float64(0.7640946923790318), // Float64() float64(0.7188183661358182), // Float64() float64(0.5856191500346635), // Float64() float64(0.9549597149363428), // Float64() float64(0.5168804691962643), // Float64() int64(4969059760275911952), // Int() int64(2147869220224756844), // Int() int64(5246770554000605320), // Int() int64(5471241176507662746), // Int() int64(4321634407747778896), // Int() int64(760102831717374652), // Int() int64(9221744211007427193), // Int() int64(8289669384274456462), // Int() int64(2449715415482412441), // Int() int64(3389241988064777392), // Int() int64(2986830195847294191), // Int() int64(8204908297817606218), // Int() int64(8134976985547166651), // Int() int64(2240328155279531677), // Int() int64(7311121042813227358), // Int() int64(5231057920893523323), // Int() int64(4257872588489500903), // Int() int64(158397175702351138), // Int() int64(1350674201389090105), // Int() int64(6093522341581845358), // Int() int32(1652216515), // Int32() int32(1323786710), // Int32() int32(1684546306), // Int32() int32(1710678126), // Int32() int32(503104460), // Int32() int32(88487615), // Int32() int32(1073552320), // Int32() int32(965044529), // Int32() int32(285184408), // Int32() int32(394559696), // Int32() int32(1421454622), // Int32() int32(955177040), // Int32() int32(2020777787), // Int32() int32(260808523), // Int32() int32(851126509), // Int32() int32(1682717115), // Int32() int32(1569423431), // Int32() int32(1092181682), // Int32() int32(157239171), // Int32() int32(709379364), // Int32() int32(0), // Int32N(1) int32(6), // Int32N(10) int32(8), // Int32N(32) int32(704922), // Int32N(1048576) int32(245656), // Int32N(1048577) int32(41205257), // Int32N(1000000000) int32(43831929), // Int32N(1073741824) int32(965044528), // Int32N(2147483646) int32(285184408), // Int32N(2147483647) int32(0), // Int32N(1) int32(6), // Int32N(10) int32(10), // Int32N(32) int32(283579), // Int32N(1048576) int32(127348), // Int32N(1048577) int32(396336665), // Int32N(1000000000) int32(911873403), // Int32N(1073741824) int32(1569423430), // Int32N(2147483646) int32(1092181681), // Int32N(2147483647) int32(0), // Int32N(1) int32(3), // Int32N(10) int64(4969059760275911952), // Int64() int64(2147869220224756844), // Int64() int64(5246770554000605320), // Int64() int64(5471241176507662746), // Int64() int64(4321634407747778896), // Int64() int64(760102831717374652), // Int64() int64(9221744211007427193), // Int64() int64(8289669384274456462), // Int64() int64(2449715415482412441), // Int64() int64(3389241988064777392), // Int64() int64(2986830195847294191), // Int64() int64(8204908297817606218), // Int64() int64(8134976985547166651), // Int64() int64(2240328155279531677), // Int64() int64(7311121042813227358), // Int64() int64(5231057920893523323), // Int64() int64(4257872588489500903), // Int64() int64(158397175702351138), // Int64() int64(1350674201389090105), // Int64() int64(6093522341581845358), // Int64() int64(0), // Int64N(1) int64(6), // Int64N(10) int64(8), // Int64N(32) int64(704922), // Int64N(1048576) int64(245656), // Int64N(1048577) int64(41205257), // Int64N(1000000000) int64(43831929), // Int64N(1073741824) int64(965044528), // Int64N(2147483646) int64(285184408), // Int64N(2147483647) int64(183731176326946086), // Int64N(1000000000000000000) int64(680987186633600239), // Int64N(1152921504606846976) int64(4102454148908803108), // Int64N(9223372036854775806) int64(8679174511200971228), // Int64N(9223372036854775807) int64(0), // Int64N(1) int64(3), // Int64N(10) int64(27), // Int64N(32) int64(665831), // Int64N(1048576) int64(533292), // Int64N(1048577) int64(73220195), // Int64N(1000000000) int64(686060398), // Int64N(1073741824) int64(0), // IntN(1) int64(6), // IntN(10) int64(8), // IntN(32) int64(704922), // IntN(1048576) int64(245656), // IntN(1048577) int64(41205257), // IntN(1000000000) int64(43831929), // IntN(1073741824) int64(965044528), // IntN(2147483646) int64(285184408), // IntN(2147483647) int64(183731176326946086), // IntN(1000000000000000000) int64(680987186633600239), // IntN(1152921504606846976) int64(4102454148908803108), // IntN(9223372036854775806) int64(8679174511200971228), // IntN(9223372036854775807) int64(0), // IntN(1) int64(3), // IntN(10) int64(27), // IntN(32) int64(665831), // IntN(1048576) int64(533292), // IntN(1048577) int64(73220195), // IntN(1000000000) int64(686060398), // IntN(1073741824) float64(0.37944549835531083), // NormFloat64() float64(0.07473804659119399), // NormFloat64() float64(0.20006841200604142), // NormFloat64() float64(-1.1253144115495104), // NormFloat64() float64(-0.4005883316435388), // NormFloat64() float64(-3.0853771402394736), // NormFloat64() float64(1.932330243076978), // NormFloat64() float64(1.726131393719264), // NormFloat64() float64(-0.11707238034168332), // NormFloat64() float64(-0.9303318111676635), // NormFloat64() float64(-0.04750789419852852), // NormFloat64() float64(0.22248301107582735), // NormFloat64() float64(-1.83630520614272), // NormFloat64() float64(0.7259521217919809), // NormFloat64() float64(0.8806882871913041), // NormFloat64() float64(-1.5022903484270484), // NormFloat64() float64(0.5972577266810571), // NormFloat64() float64(1.5631937339973658), // NormFloat64() float64(-0.3841235370075905), // NormFloat64() float64(-0.2967295854430667), // NormFloat64() []int{}, // Perm(0) []int{0}, // Perm(1) []int{1, 4, 2, 0, 3}, // Perm(5) []int{4, 3, 6, 1, 5, 2, 7, 0}, // Perm(8) []int{6, 5, 1, 8, 7, 2, 0, 3, 4}, // Perm(9) []int{9, 4, 2, 5, 6, 8, 1, 7, 0, 3}, // Perm(10) []int{5, 9, 3, 1, 4, 2, 10, 7, 15, 11, 0, 14, 13, 8, 6, 12}, // Perm(16) []int{}, // Perm(0) []int{0}, // Perm(1) []int{4, 2, 1, 3, 0}, // Perm(5) []int{0, 2, 3, 1, 5, 4, 6, 7}, // Perm(8) []int{2, 0, 8, 3, 4, 7, 6, 5, 1}, // Perm(9) []int{0, 6, 5, 3, 8, 4, 1, 2, 9, 7}, // Perm(10) []int{9, 14, 4, 11, 13, 8, 0, 6, 2, 12, 3, 7, 1, 10, 5, 15}, // Perm(16) []int{}, // Perm(0) []int{0}, // Perm(1) []int{2, 4, 0, 3, 1}, // Perm(5) []int{3, 2, 1, 0, 7, 5, 4, 6}, // Perm(8) []int{1, 3, 4, 5, 0, 2, 7, 8, 6}, // Perm(9) []int{1, 8, 4, 7, 2, 6, 5, 9, 0, 3}, // Perm(10) uint64(14192431797130687760), // Uint() uint64(11371241257079532652), // Uint() uint64(14470142590855381128), // Uint() uint64(14694613213362438554), // Uint() uint64(4321634407747778896), // Uint() uint64(760102831717374652), // Uint() uint64(9221744211007427193), // Uint() uint64(8289669384274456462), // Uint() uint64(2449715415482412441), // Uint() uint64(3389241988064777392), // Uint() uint64(12210202232702069999), // Uint() uint64(8204908297817606218), // Uint() uint64(17358349022401942459), // Uint() uint64(2240328155279531677), // Uint() uint64(7311121042813227358), // Uint() uint64(14454429957748299131), // Uint() uint64(13481244625344276711), // Uint() uint64(9381769212557126946), // Uint() uint64(1350674201389090105), // Uint() uint64(6093522341581845358), // Uint() uint32(3304433030), // Uint32() uint32(2647573421), // Uint32() uint32(3369092613), // Uint32() uint32(3421356252), // Uint32() uint32(1006208920), // Uint32() uint32(176975231), // Uint32() uint32(2147104640), // Uint32() uint32(1930089058), // Uint32() uint32(570368816), // Uint32() uint32(789119393), // Uint32() uint32(2842909244), // Uint32() uint32(1910354080), // Uint32() uint32(4041555575), // Uint32() uint32(521617046), // Uint32() uint32(1702253018), // Uint32() uint32(3365434230), // Uint32() uint32(3138846863), // Uint32() uint32(2184363364), // Uint32() uint32(314478343), // Uint32() uint32(1418758728), // Uint32() uint32(0), // Uint32N(1) uint32(6), // Uint32N(10) uint32(8), // Uint32N(32) uint32(704922), // Uint32N(1048576) uint32(245656), // Uint32N(1048577) uint32(41205257), // Uint32N(1000000000) uint32(43831929), // Uint32N(1073741824) uint32(965044528), // Uint32N(2147483646) uint32(285184408), // Uint32N(2147483647) uint32(789119393), // Uint32N(4294967294) uint32(2842909244), // Uint32N(4294967295) uint32(0), // Uint32N(1) uint32(9), // Uint32N(10) uint32(29), // Uint32N(32) uint32(266590), // Uint32N(1048576) uint32(821640), // Uint32N(1048577) uint32(730819735), // Uint32N(1000000000) uint32(522841378), // Uint32N(1073741824) uint32(157239171), // Uint32N(2147483646) uint32(709379364), // Uint32N(2147483647) uint64(14192431797130687760), // Uint64() uint64(11371241257079532652), // Uint64() uint64(14470142590855381128), // Uint64() uint64(14694613213362438554), // Uint64() uint64(4321634407747778896), // Uint64() uint64(760102831717374652), // Uint64() uint64(9221744211007427193), // Uint64() uint64(8289669384274456462), // Uint64() uint64(2449715415482412441), // Uint64() uint64(3389241988064777392), // Uint64() uint64(12210202232702069999), // Uint64() uint64(8204908297817606218), // Uint64() uint64(17358349022401942459), // Uint64() uint64(2240328155279531677), // Uint64() uint64(7311121042813227358), // Uint64() uint64(14454429957748299131), // Uint64() uint64(13481244625344276711), // Uint64() uint64(9381769212557126946), // Uint64() uint64(1350674201389090105), // Uint64() uint64(6093522341581845358), // Uint64() uint64(0), // Uint64N(1) uint64(6), // Uint64N(10) uint64(8), // Uint64N(32) uint64(704922), // Uint64N(1048576) uint64(245656), // Uint64N(1048577) uint64(41205257), // Uint64N(1000000000) uint64(43831929), // Uint64N(1073741824) uint64(965044528), // Uint64N(2147483646) uint64(285184408), // Uint64N(2147483647) uint64(183731176326946086), // Uint64N(1000000000000000000) uint64(680987186633600239), // Uint64N(1152921504606846976) uint64(4102454148908803108), // Uint64N(9223372036854775806) uint64(8679174511200971228), // Uint64N(9223372036854775807) uint64(2240328155279531676), // Uint64N(18446744073709551614) uint64(7311121042813227357), // Uint64N(18446744073709551615) uint64(0), // Uint64N(1) uint64(7), // Uint64N(10) uint64(2), // Uint64N(32) uint64(312633), // Uint64N(1048576) uint64(346376), // Uint64N(1048577) uint64(0), // UintN(1) uint64(6), // UintN(10) uint64(8), // UintN(32) uint64(704922), // UintN(1048576) uint64(245656), // UintN(1048577) uint64(41205257), // UintN(1000000000) uint64(43831929), // UintN(1073741824) uint64(965044528), // UintN(2147483646) uint64(285184408), // UintN(2147483647) uint64(183731176326946086), // UintN(1000000000000000000) uint64(680987186633600239), // UintN(1152921504606846976) uint64(4102454148908803108), // UintN(9223372036854775806) uint64(8679174511200971228), // UintN(9223372036854775807) uint64(2240328155279531676), // UintN(18446744073709551614) uint64(7311121042813227357), // UintN(18446744073709551615) uint64(0), // UintN(1) uint64(7), // UintN(10) uint64(2), // UintN(32) uint64(312633), // UintN(1048576) uint64(346376), // UintN(1048577) }