Skip to content

Commit

Permalink
add inverse_weibull_cdf function
Browse files Browse the repository at this point in the history
  • Loading branch information
wills-feng committed Apr 29, 2024
1 parent a9a620d commit 51818bb
Show file tree
Hide file tree
Showing 4 changed files with 76 additions and 0 deletions.
5 changes: 5 additions & 0 deletions velox/docs/functions/presto/math.rst
Original file line number Diff line number Diff line change
Expand Up @@ -319,6 +319,11 @@ Probability Functions: inverse_cdf
probability (p): P(N < n). The a, b parameters must be positive real values (all of type DOUBLE).
The probability p must lie on the interval [0, 1].

.. function:: inverse_weibull_cdf(a, b, p) -> double

Compute the inverse of the Weibull cdf with given parameters ``a``, ``b`` for the probability ``p``.
The ``a``, ``b`` parameters must be positive double values. The probability ``p`` must be a double
on the interval [0, 1].

====================================
Statistical Functions
Expand Down
20 changes: 20 additions & 0 deletions velox/functions/prestosql/Probability.h
Original file line number Diff line number Diff line change
Expand Up @@ -249,5 +249,25 @@ struct WeibullCDFFunction {
}
};

template <typename T>
struct InverseWeibullCDFFunction {
VELOX_DEFINE_FUNCTION_TYPES(T);

FOLLY_ALWAYS_INLINE void call(double& result, double a, double b, double p) {
static constexpr double kInf = std::numeric_limits<double>::infinity();

VELOX_USER_CHECK((p >= 0) && (p <= 1), "p must be in the interval [0, 1]");
VELOX_USER_CHECK_GT(a, 0, "a must be greater than 0");
VELOX_USER_CHECK_GT(b, 0, "b must be greater than 0");

if (b == kInf) {
result = kInf;
} else {
// https://commons.apache.org/proper/commons-math/javadocs/api-3.6.1/org/apache/commons/math3/distribution/WeibullDistribution.html#inverseCumulativeProbability(double)
result = b * std::pow(-std::log1p(-p), 1.0 / a);
}
}
};

} // namespace
} // namespace facebook::velox::functions
Original file line number Diff line number Diff line change
Expand Up @@ -189,6 +189,8 @@ void registerSimpleFunctions(const std::string& prefix) {
Map<Varchar, double>>({prefix + "cosine_similarity"});
registerFunction<WeibullCDFFunction, double, double, double, double>(
{prefix + "weibull_cdf"});
registerFunction<InverseWeibullCDFFunction, double, double, double, double>(
{prefix + "inverse_weibull_cdf"});
}

} // namespace
Expand Down
49 changes: 49 additions & 0 deletions velox/functions/prestosql/tests/ProbabilityTest.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -453,5 +453,54 @@ TEST_F(ProbabilityTest, weibullCDF) {
weibullCDF(kDoubleMin, kNan, kDoubleMax), "b must be greater than 0");
}

TEST_F(ProbabilityTest, inverseWeibullCDF) {
const auto inverseWeibullCDF = [&](std::optional<double> a,
std::optional<double> b,
std::optional<double> p) {
return evaluateOnce<double>("inverse_weibull_cdf(c0, c1, c2)", a, b, p);
};

const auto roundToTwoDecimals = [](double value) {
return round(value * 100) / 100.0;
};

EXPECT_EQ(inverseWeibullCDF(1.0, 1.0, 0.), 0.0);
EXPECT_EQ(
roundToTwoDecimals(inverseWeibullCDF(1.0, 1.0, 0.632).value()), 1.00);
EXPECT_EQ(
roundToTwoDecimals(inverseWeibullCDF(1.0, 0.6, 0.91).value()), 1.44);

VELOX_ASSERT_THROW(
inverseWeibullCDF(1.0, 1.0, kNan), "p must be in the interval [0, 1]");
VELOX_ASSERT_THROW(
inverseWeibullCDF(999999999.9, 999999999.0, kInf),
"p must be in the interval [0, 1]");
VELOX_ASSERT_THROW(
inverseWeibullCDF(3, 5, -0.1), "p must be in the interval [0, 1]");
VELOX_ASSERT_THROW(
inverseWeibullCDF(3, 5, 1.1), "p must be in the interval [0, 1]");

EXPECT_EQ(inverseWeibullCDF(kDoubleMin, 1.0, 0.3), 0.0);
EXPECT_EQ(inverseWeibullCDF(kDoubleMax, 1.0, 0.4), 1.0);
EXPECT_EQ(inverseWeibullCDF(kInf, 999999999.9, 0.4), 9.999999999E8);
EXPECT_EQ(inverseWeibullCDF(kInf, 1.0, 0.1), 1.0);
EXPECT_EQ(inverseWeibullCDF(kDoubleMin, kDoubleMin, 0.9), kInf);
EXPECT_EQ(inverseWeibullCDF(kDoubleMax, kDoubleMax, 0.8), kDoubleMax);
VELOX_ASSERT_THROW(
inverseWeibullCDF(kNan, 1.0, 0.1), "a must be greater than 0");
VELOX_ASSERT_THROW(
inverseWeibullCDF(-1.0, 1.0, 0.1), "a must be greater than 0");
VELOX_ASSERT_THROW(inverseWeibullCDF(0, 3, 0.5), "a must be greater than 0");

EXPECT_EQ(inverseWeibullCDF(1.0, kDoubleMin, 0.5), 1.5423036715619055e-308);
EXPECT_EQ(inverseWeibullCDF(1.0, kDoubleMax, 0.7), kInf);
EXPECT_THAT(inverseWeibullCDF(1.0, kInf, 0.2), IsInf());
VELOX_ASSERT_THROW(
inverseWeibullCDF(1.0, kNan, 0.4), "b must be greater than 0");
VELOX_ASSERT_THROW(
inverseWeibullCDF(1.0, -1.0, 0.4), "b must be greater than 0");
VELOX_ASSERT_THROW(inverseWeibullCDF(3, 0, 0.5), "b must be greater than 0");
}

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

0 comments on commit 51818bb

Please sign in to comment.