# frozen_string_literal: false require 'test/unit' require 'securerandom' require_relative 'ruby/test_random_formatter' # This testcase does NOT aim to test cryptographically strongness and randomness. class TestSecureRandom < Test::Unit::TestCase include Random::Formatter::FormatterTest include Random::Formatter::NotDefaultTest def setup @it = SecureRandom end # This test took 2 minutes on my machine. # And 65536 times loop could not be enough for forcing PID recycle. if false def test_s_random_bytes_is_fork_safe begin require 'openssl' rescue LoadError return end SecureRandom.random_bytes(8) pid, v1 = forking_random_bytes assert(check_forking_random_bytes(pid, v1), 'Process ID not recycled?') end def forking_random_bytes r, w = IO.pipe pid = fork { r.close w.write SecureRandom.random_bytes(8) w.close } w.close v = r.read(8) r.close Process.waitpid2(pid) [pid, v] end def check_forking_random_bytes(target_pid, target) 65536.times do pid = fork { if $$ == target_pid v2 = SecureRandom.random_bytes(8) if v2 == target exit(1) else exit(2) end end exit(3) } pid, status = Process.waitpid2(pid) case status.exitstatus when 1 raise 'returned same sequence for same PID' when 2 return true end end false # not recycled? end end def test_with_openssl begin require 'openssl' rescue LoadError return end assert_equal(Encoding::ASCII_8BIT, @it.send(:gen_random_openssl, 16).encoding) 65.times do |idx| assert_equal(idx, @it.send(:gen_random_openssl, idx).size) end end def test_repeated_gen_random assert_nothing_raised NoMethodError, '[ruby-core:92633] [Bug #15847]' do @it.gen_random(1) @it.gen_random(1) end end end