2018-10-30 23:42:04 -07:00
|
|
|
#!/usr/bin/env nomsu -V4.8.10
|
2018-05-15 18:55:55 -07:00
|
|
|
#
|
2018-01-11 18:51:21 -08:00
|
|
|
This file defines some common math literals and functions
|
|
|
|
|
2018-02-02 15:48:28 -08:00
|
|
|
use "core/metaprogramming.nom"
|
|
|
|
use "core/text.nom"
|
|
|
|
use "core/operators.nom"
|
|
|
|
use "core/control_flow.nom"
|
2018-06-14 21:59:25 -07:00
|
|
|
use "core/collections.nom"
|
2018-01-11 18:51:21 -08:00
|
|
|
|
2018-11-08 15:23:22 -08:00
|
|
|
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
|
|
|
|
|
2018-01-11 18:51:21 -08:00
|
|
|
# Literals:
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
2018-09-14 19:17:09 -07:00
|
|
|
assume (all of [inf, NaN, pi, tau, golden ratio, e]) or barf "\
|
|
|
|
..math constants failed"
|
2018-07-22 13:59:08 -07:00
|
|
|
%nan = (NaN)
|
|
|
|
assume (%nan != %nan) or barf "NaN failed"
|
2018-10-30 23:42:04 -07:00
|
|
|
[infinity, inf] all compile to (Lua value "math.huge")
|
|
|
|
[not a number, NaN, nan] all compile to (Lua value "(0/0)")
|
|
|
|
[pi, Pi, PI] all compile to (Lua value "math.pi")
|
|
|
|
[tau, Tau, TAU] all compile to (Lua value "(2*math.pi)")
|
|
|
|
(golden ratio) compiles to (Lua value "((1+math.sqrt(5))/2)")
|
|
|
|
(e) compiles to (Lua value "math.exp(1)")
|
2018-01-11 18:51:21 -08:00
|
|
|
|
|
|
|
# Functions:
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume (("5" as a number) == 5)
|
2018-10-30 23:42:04 -07:00
|
|
|
[% as a number, % as number] all compile to (..)
|
|
|
|
Lua value "tonumber(\(% as lua expr))"
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume (..)
|
|
|
|
all of [..]
|
|
|
|
abs 5, | 5 |, sqrt 5, √ 5, sine 5, cosine 5, tangent 5, arc sine 5, arc cosine 5
|
|
|
|
arc tangent 5, arc tangent 5 / 10, hyperbolic sine 5, hyperbolic cosine 5
|
|
|
|
hyperbolic tangent 5, e^ 5, ln 5, log base 2 of 5, floor 5, ceiling 5, round 5
|
|
|
|
..or barf "math functions failed"
|
2018-10-30 23:42:04 -07:00
|
|
|
[absolute value %, | % |, abs %] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.abs(\(% as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[square root %, square root of %, √ %, sqrt %] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.sqrt(\(% as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[sine %, sin %] all compile to (Lua value "math.sin(\(% as lua expr))")
|
|
|
|
[cosine %, cos %] all compile to (Lua value "math.cos(\(% as lua expr))")
|
|
|
|
[tangent %, tan %] all compile to (Lua value "math.tan(\(% as lua expr))")
|
|
|
|
[arc sine %, asin %] all compile to (Lua value "math.asin(\(% as lua expr))")
|
|
|
|
[arc cosine %, acos %] all compile to (Lua value "math.acos(\(% as lua expr))")
|
|
|
|
[arc tangent %, atan %] all compile to (Lua value "math.atan(\(% as lua expr))")
|
|
|
|
[arc tangent %y / %x, atan2 %y %x] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.atan2(\(%y as lua expr), \(%x as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[hyperbolic sine %, sinh %] all compile to (..)
|
|
|
|
Lua value "math.sinh(\(% as lua expr))"
|
|
|
|
|
|
|
|
[hyperbolic cosine %, cosh %] all compile to (..)
|
|
|
|
Lua value "math.cosh(\(% as lua expr))"
|
|
|
|
|
|
|
|
[hyperbolic tangent %, tanh %] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.tanh(\(% as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[e^ %, exp %] all compile to (Lua value "math.exp(\(% as lua expr))")
|
|
|
|
[natural log %, ln %, log %] all compile to (..)
|
|
|
|
Lua value "math.log(\(% as lua expr))"
|
|
|
|
|
|
|
|
[log % base %base, log base %base of %] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.log(\(% as lua expr), \(%base as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
(floor %) compiles to (Lua value "math.floor(\(% as lua expr))")
|
|
|
|
[ceiling %, ceil %] all compile to (Lua value "math.ceil(\(% as lua expr))")
|
|
|
|
[round %, % rounded] all compile to (..)
|
|
|
|
Lua value "math.floor(\(% as lua expr) + .5)"
|
2018-07-30 15:05:41 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((463 to the nearest 100) == 500) or barf "rounding failed"
|
|
|
|
assume ((2.6 to the nearest 0.25) == 2.5) or barf "rounding failed"
|
2018-10-30 23:42:04 -07:00
|
|
|
externally (%n to the nearest %rounder) means (..)
|
2018-01-11 18:51:21 -08:00
|
|
|
=lua "(\%rounder)*math.floor((\%n / \%rounder) + .5)"
|
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
# Any/all
|
|
|
|
externally [all of %items, all %items] all mean:
|
|
|
|
for % in %items:
|
|
|
|
unless %: return (no)
|
|
|
|
return (yes)
|
2018-10-30 23:42:04 -07:00
|
|
|
[all of %items, all %items] all compile to:
|
2018-07-17 23:08:13 -07:00
|
|
|
unless (%items.type is "List"):
|
2018-11-06 15:13:55 -08:00
|
|
|
return %tree
|
2018-11-02 14:38:24 -07:00
|
|
|
%clauses = (((% as lua expr)::text) for % in %items)
|
2018-09-10 16:26:08 -07:00
|
|
|
return (Lua value "(\(%clauses::joined with " and "))")
|
2018-10-30 23:42:04 -07:00
|
|
|
[not all of %items, not all %items] all parse as (not (all of %items))
|
2018-11-06 15:13:55 -08:00
|
|
|
|
|
|
|
externally [any of %items, any %items] all mean:
|
|
|
|
for % in %items:
|
|
|
|
if %: return (yes)
|
|
|
|
return (no)
|
2018-10-30 23:42:04 -07:00
|
|
|
[any of %items, any %items] all compile to:
|
2018-07-17 23:08:13 -07:00
|
|
|
unless (%items.type is "List"):
|
2018-11-06 15:13:55 -08:00
|
|
|
return %tree
|
2018-11-02 14:38:24 -07:00
|
|
|
%clauses = (((% as lua expr)::text) for % in %items)
|
2018-09-10 16:26:08 -07:00
|
|
|
return (Lua value "(\(%clauses::joined with " or "))")
|
2018-10-30 23:42:04 -07:00
|
|
|
[none of %items, none %items] all parse as (not (any of %items))
|
2018-11-06 15:13:55 -08:00
|
|
|
|
|
|
|
# Sum/product
|
|
|
|
externally [sum of %items, sum %items] all mean:
|
|
|
|
%total = 0
|
|
|
|
for % in %items: %total += %
|
|
|
|
return %total
|
2018-10-30 23:42:04 -07:00
|
|
|
[sum of %items, sum %items] all compile to:
|
2018-07-17 23:08:13 -07:00
|
|
|
unless (%items.type is "List"):
|
2018-11-06 15:13:55 -08:00
|
|
|
return %tree
|
2018-11-02 14:38:24 -07:00
|
|
|
%clauses = (((% as lua expr)::text) for % in %items)
|
2018-09-10 16:26:08 -07:00
|
|
|
return (Lua value "(\(%clauses::joined with " + "))")
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
externally [product of %items, product %items] all mean:
|
|
|
|
%prod = 1
|
|
|
|
for % in %items: %prod *= %
|
|
|
|
return %prod
|
|
|
|
[product of %items, product %items] all compile to:
|
|
|
|
unless (%items.type is "List"):
|
|
|
|
return %tree
|
|
|
|
%clauses = (((% as lua expr)::text) for % in %items)
|
|
|
|
return (Lua value "(\(%clauses::joined with " * "))")
|
|
|
|
|
|
|
|
externally [avg of %items, average of %items] all mean (..)
|
|
|
|
(sum of %items) / (size of %items)
|
|
|
|
|
|
|
|
# Shorthand for control flow
|
2018-10-30 23:42:04 -07:00
|
|
|
[if all of %items %body, if all of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (all of %items) %body
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[unless all of %items %body, unless all of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (not (all of %items)) %body
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[if any of %items %body, if any of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (any of %items) %body
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[unless any of %items %body, unless any of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (not (any of %items)) %body
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[if none of %items %body, if none of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (not (any of %items)) %body
|
2018-10-30 23:42:04 -07:00
|
|
|
|
|
|
|
[unless none of %items %body, unless none of %items then %body] all parse as (..)
|
2018-09-26 12:45:08 -07:00
|
|
|
if (any of %items) %body
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[if all of %items %body else %else, if all of %items then %body else %else] all parse \
|
|
|
|
..as (if (all of %items) %body else %else)
|
|
|
|
|
|
|
|
[..]
|
|
|
|
unless all of %items %body else %else, unless all of %items then %body else %else
|
|
|
|
..all parse as (if (not (all of %items)) %body else %else)
|
|
|
|
|
|
|
|
[if any of %items %body else %else, if any of %items then %body else %else] all parse \
|
|
|
|
..as (if (any of %items) %body else %else)
|
|
|
|
|
|
|
|
[..]
|
|
|
|
unless any of %items %body else %else, unless any of %items then %body else %else
|
|
|
|
..all parse as (if (not (any of %items)) %body else %else)
|
|
|
|
|
|
|
|
[if none of %items %body else %else, if none of %items then %body else %else] all \
|
|
|
|
..parse as (if (not (any of %items)) %body else %else)
|
|
|
|
|
|
|
|
[..]
|
|
|
|
unless none of %items %body else %else, unless none of %items then %body else %else
|
|
|
|
..all parse as (if (any of %items) %body else %else)
|
|
|
|
|
2018-11-06 15:13:55 -08:00
|
|
|
# Min/max
|
|
|
|
externally [min of %items, smallest of %items, lowest of %items] all mean:
|
|
|
|
%best = (nil)
|
|
|
|
for % in %items:
|
|
|
|
if ((%best == (nil)) or (% < %best)):
|
|
|
|
%best = %
|
|
|
|
return %best
|
|
|
|
|
|
|
|
externally [max of %items, biggest of %items, largest of %items, highest of %items] all mean:
|
|
|
|
%best = (nil)
|
|
|
|
for % in %items:
|
|
|
|
if ((%best == (nil)) or (% > %best)):
|
|
|
|
%best = %
|
|
|
|
return %best
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-07-22 13:59:08 -07:00
|
|
|
test:
|
|
|
|
assume ((min of [3, -4, 1, 2] by % = (% * %)) == 1)
|
|
|
|
assume ((max of [3, -4, 1, 2] by % = (% * %)) == -4)
|
2018-10-30 23:42:04 -07:00
|
|
|
(min of %items by %item = %value_expr) parses as (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
result of:
|
2018-11-06 15:13:55 -08:00
|
|
|
%best = (nil)
|
|
|
|
%best_key = (nil)
|
2018-07-17 23:08:13 -07:00
|
|
|
for %item in %items:
|
2018-07-18 01:27:56 -07:00
|
|
|
%key = %value_expr
|
|
|
|
if ((%best == (nil)) or (%key < %best_key)):
|
2018-11-06 15:13:55 -08:00
|
|
|
%best = %item
|
|
|
|
%best_key = %key
|
2018-05-27 18:28:23 -07:00
|
|
|
return %best
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
(max of %items by %item = %value_expr) parses as (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
result of:
|
2018-11-06 15:13:55 -08:00
|
|
|
%best = (nil)
|
|
|
|
%best_key = (nil)
|
2018-07-17 23:08:13 -07:00
|
|
|
for %item in %items:
|
2018-07-18 01:27:56 -07:00
|
|
|
%key = %value_expr
|
|
|
|
if ((%best == (nil)) or (%key > %best_key)):
|
2018-11-06 15:13:55 -08:00
|
|
|
%best = %item
|
|
|
|
%best_key = %key
|
2018-05-27 18:28:23 -07:00
|
|
|
return %best
|
2018-01-11 18:51:21 -08:00
|
|
|
|
|
|
|
# Random functions
|
2018-11-08 15:23:22 -08:00
|
|
|
externally (seed random with %) means:
|
2018-09-14 19:17:09 -07:00
|
|
|
lua> "\
|
|
|
|
..math.randomseed(\%);
|
|
|
|
for i=1,20 do math.random(); end"
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
(seed random) parses as (seed random with (=lua "os.time()"))
|
|
|
|
[random number, random, rand] all compile to (Lua value "math.random()")
|
|
|
|
[random int %n, random integer %n, randint %n] all compile to (..)
|
2018-07-17 23:08:13 -07:00
|
|
|
Lua value "math.random(\(%n as lua expr))"
|
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
[random from %low to %high, random number from %low to %high, rand %low %high] all \
|
|
|
|
..compile to (Lua value "math.random(\(%low as lua expr), \(%high as lua expr))")
|
2018-07-17 23:08:13 -07:00
|
|
|
|
2018-10-30 23:42:04 -07:00
|
|
|
externally [..]
|
|
|
|
random choice from %elements, random choice %elements, random %elements
|
|
|
|
..all mean (=lua "\%elements[math.random(#\%elements)]")
|