Skip to content

Commit

Permalink
Add secure_random function
Browse files Browse the repository at this point in the history
  • Loading branch information
willsfeng authored and wills-feng committed Jun 11, 2024
1 parent 65bd5de commit 3786176
Show file tree
Hide file tree
Showing 4 changed files with 228 additions and 0 deletions.
12 changes: 12 additions & 0 deletions velox/docs/functions/presto/math.rst
Original file line number Diff line number Diff line change
Expand Up @@ -139,6 +139,18 @@ Mathematical Functions

Returns ``x`` rounded to ``d`` decimal places.

.. function:: secure_rand() -> double

This is an alias for :func:`secure_random()`.

.. function:: secure_random() -> double

Returns a cryptographically secure random value in the range 0.0 <= x < 1.0.

.. function:: secure_random(lower, upper) -> [same as input]

Returns a cryptographically secure random value in the range lower <= x < upper, where lower < upper.

.. function:: sign(x) -> [same as x]

Returns the signum function of ``x``. For both integer and floating point arguments, it returns:
Expand Down
52 changes: 52 additions & 0 deletions velox/functions/prestosql/Rand.h
Original file line number Diff line number Diff line change
Expand Up @@ -58,4 +58,56 @@ struct RandFunction {
}
};

template <typename T>
struct SecureRandFunction {
static constexpr bool is_deterministic = false;

template <typename InputType>
FOLLY_ALWAYS_INLINE void checkInput(
const InputType lower,
const InputType upper) {
VELOX_USER_CHECK_GT(
upper, lower, "upper bound must be greater than lower bound");
}

FOLLY_ALWAYS_INLINE void call(double& out) {
out = folly::Random::secureRandDouble01();
}

FOLLY_ALWAYS_INLINE void
call(double& out, const double lower, const double upper) {
checkInput(lower, upper);
out = folly::Random::secureRandDouble(lower, upper);
}

FOLLY_ALWAYS_INLINE void
call(float& out, const float lower, const float upper) {
checkInput(lower, upper);
out = float(folly::Random::secureRandDouble(lower, upper));
}

FOLLY_ALWAYS_INLINE void
call(int64_t& out, const int64_t lower, const int64_t upper) {
checkInput(lower, upper);
out = folly::Random::secureRand64(lower, upper);
}

FOLLY_ALWAYS_INLINE void
call(int32_t& out, const int32_t lower, const int32_t upper) {
checkInput(lower, upper);
out = int32_t(folly::Random::secureRand64(lower, upper));
}

FOLLY_ALWAYS_INLINE void
call(int16_t& out, const int16_t lower, const int16_t upper) {
checkInput(lower, upper);
out = int16_t(folly::Random::secureRand64(lower, upper));
}

FOLLY_ALWAYS_INLINE void
call(int8_t& out, const int8_t lower, const int8_t upper) {
checkInput(lower, upper);
out = int8_t(folly::Random::secureRand64(lower, upper));
}
};
} // namespace facebook::velox::functions
Original file line number Diff line number Diff line change
Expand Up @@ -92,6 +92,10 @@ void registerMathFunctions(const std::string& prefix) {
registerFunction<NanFunction, double>({prefix + "nan"});
registerFunction<RandFunction, double>({prefix + "rand", prefix + "random"});
registerUnaryIntegral<RandFunction>({prefix + "rand", prefix + "random"});
registerFunction<SecureRandFunction, double>(
{prefix + "secure_rand", prefix + "secure_random"});
registerBinaryNumeric<SecureRandFunction>(
{prefix + "secure_rand", prefix + "secure_random"});
registerFunction<FromBaseFunction, int64_t, Varchar, int64_t>(
{prefix + "from_base"});
registerFunction<ToBaseFunction, Varchar, int64_t, int64_t>(
Expand Down
160 changes: 160 additions & 0 deletions velox/functions/prestosql/tests/RandTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -23,6 +23,13 @@ namespace facebook::velox::functions {

namespace {

constexpr double kInf = std::numeric_limits<double>::infinity();
constexpr double kNan = std::numeric_limits<double>::quiet_NaN();
constexpr float kInfF = std::numeric_limits<float>::infinity();
constexpr float kNanF = std::numeric_limits<float>::quiet_NaN();
constexpr int64_t kLongMax = std::numeric_limits<int64_t>::max();
constexpr int64_t kLongMin = std::numeric_limits<int64_t>::min();

class RandTest : public functions::test::FunctionBaseTest {
protected:
template <typename T>
Expand All @@ -44,6 +51,18 @@ class RandTest : public functions::test::FunctionBaseTest {
std::optional<T> randWithTry(T n) {
return evaluateOnce<T>("try(rand(c0))", std::make_optional(n));
}

template <typename T>
std::optional<T> secureRandom(
std::optional<T> lower,
std::optional<T> upper) {
return evaluateOnce<T>("secure_random(c0, c1)", lower, upper);
}

template <typename T>
std::optional<T> secureRand(std::optional<T> lower, std::optional<T> upper) {
return evaluateOnce<T>("secure_rand(c0, c1)", lower, upper);
}
};

TEST_F(RandTest, zeroArg) {
Expand Down Expand Up @@ -74,5 +93,146 @@ TEST_F(RandTest, nonNullInt8) {
EXPECT_LT(rand(4), 4);
}

TEST_F(RandTest, secureRandZeroArg) {
auto result =
evaluateOnce<double>("secure_random()", makeRowVector(ROW({}), 1));
EXPECT_LT(result, 1.0);
EXPECT_GE(result, 0.0);

result = evaluateOnce<double>("secure_rand()", makeRowVector(ROW({}), 1));
EXPECT_LT(result, 1.0);
EXPECT_GE(result, 0.0);
}

TEST_F(RandTest, secureRandInt64) {
auto result =
secureRand<int64_t>((int64_t)-2147532562, (int64_t)4611791058295013614);
EXPECT_LT(result, 4611791058295013614);
EXPECT_GE(result, -2147532562);

result = secureRand<int64_t>((int64_t)0, (int64_t)46117910582950136);
EXPECT_LT(result, 46117910582950136);
EXPECT_GE(result, 0);

result = secureRand<int64_t>(
std::numeric_limits<int64_t>::min(), std::numeric_limits<int64_t>::max());
EXPECT_LT(result, std::numeric_limits<int64_t>::max());
EXPECT_GE(result, std::numeric_limits<int64_t>::min());
}

TEST_F(RandTest, secureRandInt32) {
auto result = secureRand<int32_t>((int32_t)8765432, (int32_t)2145613151);
EXPECT_LT(result, 2145613151);
EXPECT_GE(result, 8765432);

result = secureRand<int32_t>((int32_t)0, (int32_t)21456131);
EXPECT_LT(result, 21456131);
EXPECT_GE(result, 0);
}

TEST_F(RandTest, secureRandInt16) {
auto result = secureRand<int16_t>((int16_t)-100, (int16_t)23286);
EXPECT_LT(result, 23286);
EXPECT_GE(result, -100);

result = secureRand<int16_t>((int16_t)0, (int16_t)23286);
EXPECT_LT(result, 23286);
EXPECT_GE(result, 0);
}

TEST_F(RandTest, secureRandInt8) {
auto result = secureRand<int8_t>((int8_t)10, (int8_t)120);
EXPECT_LT(result, 120);
EXPECT_GE(result, 10);

result = secureRand<int8_t>((int8_t)0, (int8_t)120);
EXPECT_LT(result, 120);
EXPECT_GE(result, 0);
}

TEST_F(RandTest, secureRandDouble) {
auto result = secureRand<double>((double)10.5, (double)120.7895);
EXPECT_LT(result, 120.7895);
EXPECT_GE(result, 10.5);

result = secureRand<double>((double)0.0, (double)120.7895);
EXPECT_LT(result, 120.7895);
EXPECT_GE(result, 0.0);
}

TEST_F(RandTest, secureRandFloat) {
auto result = secureRand<float>((float)-10.5, (float)120.7);
EXPECT_LT(result, 120.7);
EXPECT_GE(result, -10.5);

result = secureRand<float>((float)0.0, (float)120.7);
EXPECT_LT(result, 120.7);
EXPECT_GE(result, 0.0);
}

TEST_F(RandTest, secureRandInvalid) {
VELOX_ASSERT_THROW(
secureRand<int64_t>((int64_t)-5, (int64_t)-10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int64_t>((int64_t)15, (int64_t)10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int32_t>((int32_t)5, (int32_t)-10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int32_t>((int32_t)15, (int32_t)10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int16_t>((int16_t)-5, (int16_t)-10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int16_t>((int16_t)15, (int16_t)10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int8_t>((int8_t)5, (int8_t)-10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<int8_t>((int8_t)15, (int8_t)10),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<double>(-5.7, -10.7),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<double>(15.6, 10.1),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<float>((float)-5.7, (float)-10.7),
"upper bound must be greater than lower bound");
VELOX_ASSERT_THROW(
secureRand<float>((float)15.6, (float)10.1),
"upper bound must be greater than lower bound");
}

TEST_F(RandTest, secureRandSpecialValues) {
auto result = secureRand<int64_t>(0, kLongMax);
EXPECT_LT(result, kLongMax);
result = secureRand<int64_t>(kLongMin, 0);
EXPECT_GE(result, kLongMin);

result = secureRand<double>(-kInf, 0);
EXPECT_GE(result, -kInf);
result = secureRand<double>(0.0, kInf);
EXPECT_LE(result, kInf);

result = secureRand<float>(-kInfF, 0);
EXPECT_GE(result, -kInfF);
result = secureRand<float>(0.0, kInfF);
EXPECT_LE(result, kInfF);

VELOX_ASSERT_THROW(
secureRand<double>(0.0, kNan),
"upper bound must be greater than lower bound");

VELOX_ASSERT_THROW(
secureRand<float>(0.0, kNanF),
"upper bound must be greater than lower bound");
}

} // namespace
} // namespace facebook::velox::functions

0 comments on commit 3786176

Please sign in to comment.