Open in new tab

Probly is a Python-like mini-language for probabilistic estimation. It's based on Starlark and implemented in Go.

Start with Example 1

Probly syntax

You may use any Starlark syntax. There are only the following differences to Starlark:

  • A variable may follow a probability distribution, in addition to usual types like numbers or dictionaries
  • The Starlark math module is imported by default, so you can directly use it e.g. math.sqrt(2). Probly also has a built-in sum function not available in Starlark.

This page will show you the probability distributions of all the numeric (scalar or distribution) global variables in your program (except those starting with an underscore). The values are taken at the end of the program's execution.

Probability distributions

Name to
p10 to p90
pm
plus/minus
td
times/divide
Quantiles Notes
Normal mean sd 2
LogNormal mu sigma 2 Alternatively: mean, sd
Beta alpha beta
PERT min mode max [lambd] Like the triangular, but smoother (Wikipedia)
Uniform a b 2 a need not be less than b
LogUniform a b 2 a need not be less than b
Bernoulli p
Binomial n p
Discrete x_1 p_1 x_2 p_2 ... Generic discrete distribution over any finite set of values

math

These mathematical functions and constants are available in the math module:

  • pow(x, y) - Returns x raised to the power of y
  • exp(x)
  • sqrt(x)
  • log(x, [base]) - Natural logarithm by default if base is not specified
  • e
  • pi
  • Ceil, floor, and sign manipulation:
    • ceil(x)
    • floor(x)
    • fabs(x) - Returns the absolute value of x as float
    • copysign(x, y) - Returns a value with the magnitude of x and the sign of y
  • mod(x, y) - Returns x modulo y
  • remainder(x, y)
  • round(x) - Returns the nearest integer, rounding half away from zero
  • Trigonometry (in radians unless otherwise specified):
    • acos(x)
    • asin(x)
    • atan(x)
    • atan2(y, x) - Returns atan(y / x). The result is between -pi and pi
    • cos(x)
    • sin(x)
    • tan(x)
    • degrees(x) - Converts angle x from radians to degrees
    • radians(x) - Converts angle x from degrees to radians
    • acosh(x)
    • asinh(x)
    • atanh(x)
    • cosh(x)
    • sinh(x)
    • tanh(x)
  • hypot(x, y) - Returns the Euclidean norm, sqrt(x^2 + y^2); the distance from the origin to (x, y)
  • gamma(x) - Returns the Gamma function at x

Starlark syntax

This code provides an example of the syntax of Starlark:

# Define a number
number = 18

# Define a list
numbers = [1, 2, 3, 4, 5]

# List comprehension
halves = [n / 2 for n in numbers]

# Define a function
def is_even(n):
    """Return True if n is even."""
    return n % 2 == 0

# Define a dictionary
people = {
    "Alice": 22,
    "Bob": 40,
    "Charlie": 55,
    "Dave": 14,
}

names = ", ".join(people.keys())  # Alice, Bob, Charlie, Dave

# Modify a variable in a loop
sum_even_ages = 0
for age in people.values():
    if is_even(age):
        sum_even_ages += age

# Append to a list in a loop
over_30_names = []
for name, age in people.items():
    if age > 30:
        over_30_names.append(name)

If you've ever used Python, this should look very familiar. In fact, the code above is also valid Python code. Still, this short example shows most of the language. Starlark is a very small language that implements a limited subset of Python.

For our purposes, one notable difference to Python is that the exponentiation operator ** is not supported. You have to use math.pow.

You can also look at the Starlark language specification.

Speed

Though not designed for speed, Probly is fast enough for practical purposes: around 10 milliseconds for 3,000 samples, for most examples on this page. This is due to being implemented in Go.

The time taken to return results on this page is spent overwhelmingly in web application code, not in Probly evaluation.

Interestingly, Probly is still slower than Python code that uses entirely numpy array operations, which are very well optimised. This should only begin to matter at very large scales, or if latency is critical.

Limitations

It's not currently possible to obtain and manipulate properties of a distribution within an Probly program, like so:

x = Normal(1 to 10)
y = x.std()  # Not possible

Supporting this would require some fundamental changes to the implementation of Probly, which is currently very simplistic.

Prior work

The to binary operator was inspired by Squiggle.

Example Dissolving the Fermi Paradox (Sandberg et al.)

This example follows Section 3 of Dissolving the Fermi Paradox (Sandberg and colleagues, 2018).

From the abstract:

[The Fermi paradox] arises from the use of Drake-like equations, which implicitly assume certainty regarding highly uncertain parameters. ... When the model is recast to represent realistic distributions of uncertainty, we find a substantial ex ante probability of there being no other intelligent life in our observable universe, and thus that there should be little surprise when we fail to detect any signs of it.

The results of our simulation should approximately match Figure 2 of the paper:

The mean for N is [...] 27 million, but the median is now only 0.32 – less than one civilization per galaxy like our own. The probability of N<1 is now 52%.

Distribution details

n

Mean 11.6 E 6
Std. dev. 119 E 6
Variance 14.1 E 15
Quantile
0.05 0
0.25 136 E -12
0.50 5.62
0.75 9 780
0.95 11.5 E 6

longevity

Mean 559 E 6
Std. dev. 1.60 E 9
Variance 2.57 E 18
Quantile
0.05 240
0.25 12 200
0.50 1.30 E 6
0.75 104 E 6
0.95 4.30 E 9

detectable

Mean 0.212
Std. dev. 0.247
Variance 0.061 2
Quantile
0.05 0.012 5
0.25 0.032 3
0.50 0.104
0.75 0.302
0.95 0.785

intelligence

Mean 0.140
Std. dev. 0.222
Variance 0.049 2
Quantile
0.05 0.001 39
0.25 0.005 12
0.50 0.030 5
0.75 0.168
0.95 0.689

life

Mean 0.537
Std. dev. 0.492
Variance 0.242
Quantile
0.05 0
0.25 41.9 E -15
0.50 1.00
0.75 1.00
0.95 1.00

r_life

Mean 3.06 E 78
Std. dev. 167 E 78
Variance 27.9 E 159
Quantile
0.05 12.0 E -36
0.25 41.9 E -15
0.50 40.9
0.75 12.8 E 15
0.95 4.19 E 36

habitable

Mean 0.389
Std. dev. 0.246
Variance 0.060 4
Quantile
0.05 0.115
0.25 0.179
0.50 0.319
0.75 0.550
0.95 0.880

planet

Mean 0.395
Std. dev. 0.254
Variance 0.064 3
Quantile
0.05 0.112
0.25 0.177
0.50 0.319
0.75 0.575
0.95 0.904

star_formation

Mean 21.0
Std. dev. 25.0
Variance 624
Quantile
0.05 1.26
0.25 3.21
0.50 9.69
0.75 29.1
0.95 80.1

Simulation data

CSV

Download CSV

Preview

star_formation planet habitable r_life ... intelligence detectable longevity n
0 65.7 0.326 0.215 4.37 E 6 ... 0.003 12 0.287 1.69 E 6 6 960
1 38.1 0.162 0.142 10.6 E -42 ... 0.240 0.074 3 2.25 E 6 0
2 30.4 0.113 0.100 35.5 ... 0.418 0.038 8 165 000 920
... ... ... ... ... ... ... ... ... ...
2997 20.2 0.108 0.390 12.9 E 39 ... 0.003 33 0.834 793 000 1 880
2998 18.9 0.226 0.173 763 E 15 ... 0.004 90 0.976 10.2 E 6 35 900
2999 11.5 0.106 0.143 3.18 E 18 ... 0.047 4 0.011 3 829 E 6 77 300

API

Get the simulation data (and more) in a machine-readable format: /api/sim/NHCVSg9X58hdePRUnJMBvJ/