-
#532:
-
Removes the
potemkin
dependency by importing only what we need directly intosicmutils.util.def
. This makes sense since our versions add afork
call so that they work for ClojureScript as well. -
Moves all
examples
into the tests so that we don't ship them with the library. These will eventually be converted to Clerk notebooks. -
Removes the
hiccup
dependency. -
Upgrades
test.chuck
and removes all:include-macros true
calls for that library. Onlysame/ish
requires them now! -
Capitalizes the "Script" in ClojureScript everywhere it appears.
-
-
#531:
-
Fixes a typo in one of the rules in
sicmutils.simplify.rules/expand-multiangle
, closing #530 -
Adds proper self-require forms to all
cljc
namespaces with macros; this removes the need for:include-macros true
or:refer-macros
and makes the life of consumers simpler. -
Upgrades sci, test.check, core.match, timbre, clojurescript, shadow-cljs dependencies to remove various warnings.
-
Drops cljsjs dependencies in favor of
deps.cljs
entries -
Forces the subexpression walk in
pattern.rule/try-subexpressions
withdoall
-
Adds type hints to all remaining
Complex
calls in js, to fix issues with advanced compilation (thanks to @mhuebert for finding this) -
Adds
sicmutils.structure/symbol-set
-
Adds
IHash
implementations forjs/BigInt
,goog.math.Long
,goog.map.Integer
andRatio
types
-
-
#515:
-
tidies up the
square
andcube
speedups thanks to a tip from GJS -
Converts more
(mul x x)
tosquare
in the derivatives of thesicmutils.generic
namespace. -
fixes a bug in the
numeric-zero?
check of exponent's derivative. -
#514:
-
Modifies
sicmutils.calculus.derivative/taylor-series
to return a properPowerSeries
instance, which the user can call with somedx
to get back the old behavior.The new version can take any number of arguments in addition to
f
. Supplying no arguments returns the expansion at 0; if you supply many arguments (totally fine!), you'll need to wrap yourdx
components in a vector before supplying them to the returnedPowerSeries
. -
sicmutils.series/function->
works the same way now, and functions identically, but with a different implementation. (previously it took a single expansion point under a keyword argument:x0
.) -
The new
sicmutils.calculus.derivative/symbolic-taylor-series
is a port ofTaylor-series-coefficients
fromscmutils
. It has the same contract astaylor-series
, except that the full expansion is performed symbolically, and the original arguments are substituted in after expansion and simplification. -
Other changes:
-
Installs
1
as theone-like
andidentity-like
return values for structures and vectors. A true identity element would be an identity element compatible with all entries of the structure; but as defined now,1
is a fine choice and matches thescmutils
implementation. -
new
sicmutils.differential/map-coefficients
, which makessimplify
slightly more efficient by filtering terms . -
more efficient
sicmutils.expression/variables-in
, maybe 30% faster for big expressions; this makes a difference in the simplifier! -
sicmutils.expression/substitute
now works for properLiteral
instances. Before it only worked for unwrapped literals. -
matrix walks made slightly faster by caching a row or column before traversal
-
-
-
#512:
-
adds
sicmutils.mechanics.routhian
, with implementations ofLagrangian->Routhian
,Routh-equations
,Routhian->acceleration
,Routhian->state-derivative
,Lagrangian-state->Routhian-state
andRouthian-state->Lagrangian-state
. -
adds missing
sicmutils.mechanics.{routhian,time-evolution,noether}
tosicmutils.env.sci
-
-
#509:
-
Fixes a bug with
down*Matrix
multiplication, and adds tests for correctness. -
Adds
sicmutils.matrix.{symmetric?,antisymmetric?}
predicates -
The mechanics port continues with
sicmutils.mechanics.rigid
:-
T-rigid-body
moves toT-body-Euler
with an alias back to its original name. Same situation forEuler-state->L-body
=>L-body-Euler
andEuler-state->L-space
=>L-space-Euler
. -
New functions:
three-vector-components->antisymmetric
,T-body
,L-body
,L-space
,Euler->omega
,Euler->omega-body
,quaternion-state->omega-body
,quaternion-state->omega-space
,qw-state->L-body
,qw-state->L-space
,T-quaternion-state
-
-
-
#511 focuses on adding more rotations and efficiency to
sicmutils.quaternion
. Specifically:-
magnitude-sq
andmagnitude
are now more efficient. -
New functions to get to and from quaternions and various matrix representations:
from-rotation-matrix
,->rotation-matrix
,from-complex-matrix
,->complex-matrix
,from-4x4-matrix
,->4x4-matrix
-
New instances
ONE-matrix
,I-matrix
,J-matrix
,K-matrix
, matrix representations of the corresponding quaternion elements. -
Similar to the matrix elements, we also now have
ONE-tensor
,I-tensor
,J-tensor
,K-tensor
.
-
-
#503:
-
Changes the default implementation of
square
andcube
for differentials to use(expt <> 2)
etc instead of(* <> <>)
.This is a big deal! For certain expressions there's a huge blowup when you square a big symbolic term, and taking the derivative of it TWICE is very messy.
With this change, differentials use the chain rule to calculuate the derivative of
$x^2$ as$2x*x'$ , instead of using the product rule and achieving a SECOND differentiatation of the same form and another multiplication:$xx' + x'x$ .Before a judicious
simplify
call I added, this change dropped the runtime of thesicmutils.sicm.ch3-test
suite down by 6x. After the simplify change insicmutils.examples.top
the tests were still 40% faster in that namespace. -
Fixes a bug where the
RationalFunction
cube implementation actually calledsquare
. -
adds
sicmutils.mechanics.lagrange/Lagrangian
for building function signatures of Lagrangians. -
adds the
sicmutils.mechanics.time-evolution
namespace -
adds
sicmutils.mechanics.lagrange/L-axisymmetric-top
, more efficient than the version insicmutils.examples.top
-
Fleshes out
sicmutils.mechanics.hamilton
:-
New functions:
H-state?
,compatible-H-state?
,state->p
,momenta
,P
,literal-Hamiltonian-state
,L-state->H-state
,H-state->L-state
,H-state->matrix
,matrix->H-state
,make-Hamiltonian
,D-phase-space
,Hamiltonian->Lagrangian-procedure
,Hamiltonian->Lagrangian
,flow-derivative
,flow-transform
,standard-map-inverse
,F->K
,J-func
,T-func
,canonical-H?
,canonical-K?
,linear-function->multiplier
,Phi
,Phi*
,qp-canonical?
,polar-canonical-inverse
,two-particle-center-of-mass
,two-particle-center-of-mass-canonical
,transpose-function
,multiplicative-transpose
,symplectic-two-form
,canonical-transform?
,J-matrix
,symplectic?
-
F->CH
moves toF->CT
(F->CT
is now an alias) -
Legendre-transform-fn
becomesLegendre-transform-procedure
and gains more correctness tests, toggled on and off by the*validate-Legendre-transform?*
dynamic variable.
-
-
-
#508 adds
sicmutils.mechanics.noether
namespace, withNoether-integral
. -
#506 tidies up the build by removing unneeded reader conditionals and replacing renames like
core-=
with a proper require ofclojure.core
. -
#502 begins the port of the remaining items in the scmutils
mechanics
package over the Clojure. This PR focuses onsicmutils.mechanics.lagrange
, which contains functions from many files in the originalmechanics
folder.-
momentum-tuple
moves here fromsicmutils.mechanics.hamilton
-
New functions
->L-state
,->local
,->state
,state->n-dof
,time
,state->{q,qdot,qddot}
,coordinates
,velocities
,accelerations
,Q
,Qdot
,Qdotdot
,literal-Lagrangian-state
path->state-path
(alias for the existingGamma
),Rayleigh-dissipation
,qv->local-path
,Lagrange-equations-first-order
, (withLagrange-equations-1
alias),Lagrangian->power-loss
,T3-spherical
,L3-central
,Dt-procedure
and the wrapping operatorDt
,Euler-lagrange-operator
(withLagrange-equations-operator
andLE
aliases),generalized-LE
. -
Many of these are aliased into
sicmutils.env
. Ask if you think more should be there! -
many new built-in Lagrangians:
L-Kepler-polar
,L-coupled-harmonic
,L-sliding-pend
,L-pendulum
,L-two-particle
-
Lagrange-equations
,Lagrangian->acceleration
,Lagrangian->state-derivative
now take a dissipation function -
local-state-derivative
aliases the 1-arity version ofLagrangian->state-derivative
-
New
rectangular->polar
,polar->rectangular
,spherical->rectangular
andrectangular->spherical
that operate on coordinates, with associatedr->p
(new),p->r
,s->r
andr->s
(new).
-
-
#501 moves
elliptic-integrals
fromsicmutils.special.elliptical-test
sicmutils.special.elliptical
, as it's needed by the upcomingsicmutils.mechanics.pendulum
namespace.
-
#497:
-
sicmutils.expression.compile/compile-state-fn
and its non-memoized version can now take an explicit:mode
argument; this will override the dynamically bound*mode*
.Invalid modes supplied via
:mode
will causecompile-state-fn
to throw an exception. -
Fixes a bug where non-numeric operations
up
anddown
were applied at compile time, throwing an error.
-
-
#498: replace all long-form GPL headers with
"SPDX-License-Identifier: GPL-3.0"
. -
#496:
-
replaces the function values in
sicmutils.expression.compile
with symbols; I hadn't realized before that substituting in symbolicMath/sqrt
, for example, was possible, vs a#(Math/sqrt %)
function value. Compiled functions are now faster!A simulation run of the double pendulum example in the clerk-demo repository now runs in 350ms vs the former 2.2 seconds, a major win.
-
Function compilation now pre-simplifies numerical forms encountered inside a function, like
(/ 1 2)
, instead of letting them be evaluated on every fn call. -
All numerical forms encountered in function compilation are now converted to either
double
on the JVM orjs/Number
in javascript; this way noBigInt
values etc are left around. -
In
sicmutils.expression.compile
:-
gains a new, validating
compiler-mode
function for fetching the compiler function. -
set-compiler-mode!
now actually works. It never did! -
New
:source
compile mode that returns a source code form. You can either calleval
on this or callsci-eval
to get an SCI-evaluated function with all proper bindings in place. -
compile-state-fn
now takes an optional options map, with support for:flatten?
and:generic-params?
keywords. These can be used to tune the shape of the function returned bycompile-state-fn
.
-
-
-
#485:
-
Bumps the JDK version for Github Actions to 17 from 8.
-
Adds a development dependency on Nextjournal's Clerk library, and begins the process of massaging various namespaces into proper literate essays display-able with Clerk. To run these, start a REPL and follow the instructions in
dev/user.clj
.sicmutils.calculus.derivative
andsicmutils.differential
now render as proper literate essays, with all TeX bugs fixed.
-
Bumps the shadow-cljs dependency to version
2.17.4
, and the includedcljs
version to1.11.4
.sicmutils.collection
properly handles the new cljsIntegerRange
class. -
sicmutils.polynomial.factor
now memoizespoly->factored-expression
by default. If polynomial GCD fails inside that function the computation now proceeds with a warning instead of failing.
-
-
#492 updates the
clj-kondo
linters to emit custom warnings with all metadata from the original token, not just:row
and:col
. This fixes the ability to override or ignore individual warnings. -
#490 adds
sicmutils.numerical.roots.bisect
with implementations of bisection search, secant search and a mixed method found inscmutils
. These all live under abisect
function.The data structure returned is similar to the minimization functions in the
sicmutils.numeric.{unimin, multimin}
namespaces. As more root-finding methods come online this should all standardize nicely. -
#491 adds
sicmutils.mechanics.rotation/M->Euler
, for converting from a rotation matrix to a triple of Euler angles. Now we can successfully round trip. -
#489:
-
Installs
g/log
andg/exp
forjs/BigInt
instances, enablingg/log2
andg/log10
in the mix. -
Removes most
js*
calls usingcoercive-=
andjs-mod
. This form is internal and should be avoided.
-
-
#484 adds
sicmutils.polynomial/from-power-series
, for generating a polynomial instance from some prefix of a (univariate) power series.
-
#481:
-
fixes a long-standing (test-only) bug in
sicmutils.polynomial-test
around palindromic polynomials -
Adds a new
x-degree
argument tosicmutils.polynomial/univariate->dense
, for padding the result with zeros in the case that you want to guarantee a certain dense degree in the result.
-
-
#480:
sicmutils.numerical.quadrature/definite-integral
now coerces the result of non-compiled integrands in cljs to double. This prevents the @kloimhardt bug where certain paths would produceBigInt
instances and fail in quadrature calls. -
#477:
-
Adds tight integration with the
clj-kondo
linter via an exported clj-kondo configuration in theresources
directory. All macros in the library now offer pleasant linting to users. This is especially helpful for the macros inpattern.rule
, which now can offer live feedback to pattern-matching authors.See
doc/linting.md
for details on various warnings reported, and installation instructions for the clj-kondo config.Thanks to @borkdude for all of his help getting this working, and making this amazing project!
-
All linter errors and warnings are now addressed, fixed and silenced for the entire codebase, both
test
andsrc
directories. -
A new Github Action will run the linter for every PR and push to master, and annotate PRs with linter warnings and errors.
-
com.gfredericks/test.chuck
dev dependency upgraded to0.2.13
to grab its clj-kondo exported config. -
I found the following bugs with the help of the linter:
-
Deleted the unused
sicmutils.differential/d:apply
. -
Fixed a bug with
sicmutils.expression.render/->JavaScript
not using the second argument toremainder
. -
deleted
sicmutils.numerical.quadrature.common
in favor ofsicmutils.generic/infinite?
-
Fixed a broken integrator in
sicmutils.numerical.quadrature.simpson38
, and fixed the tests to actually stress this code. -
Fixed a bug where
sicmutils.numerical.quadrature.substitute/exponential-upper
was not actually using its input function! -
unused
simplify
argument removed fromsicmutils.simplify.rules/non-negative-factors!
and all uses. -
Bug fix in
sicmutils.special.elliptic/jacobi-elliptic-functions
; deep in the gnarly fn, one of the branches returned nil instead of its required values. Thank you, linter! -
pattern.rule
patterns can now handle spliced and unquote-spliced inputs in their symbol position. -
sicmutils.pattern/template
will no longer error in the 1-arity case when some form contains a binding entry like(? (fn [m] ...))
. Instead, the function will be passed an empty map.
-
-
-
#474:
g/quotient
now supports inexact inputs, as LONG as the inputs are equal up to sign. So(g/quotient 1.2 -1.2)
now returns-1
instead of throwing.
-
#469:
-
sicmutils.matrix
gains:-
literal-column-matrix
,literal-row-matrix
for generating slightly tidier matrices of literal entries. (Seeliteral-matrix
for the prior option.) -
structure->matrix
converts 2 tensors into explicit matrices. -
s:solve-linear-left
,s:solve-linear-right
,s:divide-by-structure
act on 2 tensors. These live in the matrix namespace since they depend on conversions to and from tensors and matrices. -
make-diagonal
for generating diagonal matrices with a constant element along the diagonal. -
s->m
,s:transpose
ands:inverse
all gain new 2-arities that provides a sane default forls
. -
More efficient matrix
invert
anddeterminant
routines, plus functions to generate type specific custom matrix inversion and determinant routines viaclassical-adjoint-formula
,general-determinant
. -
Linear equation solving via
solve
,rsolve
andcramers-rule
.
-
-
Implements new generics for matrices and structures:
-
diagonal matrices respond true to
v/=
with a scalar if all entries along the diagonal are equal to that scalar. -
square matrices can now
g/+
andg/-
with scalars; the scalarc
is converted(* c I)
, whereI
is an identity matrix of the same dimension as the square matrix. -
(g/acot M)
now expands the matrixM
into a nice power series, more efficient than the previous default. -
Thanks to
solve
andcramers-rule
, the followingg/div
combinations now work:matrix/scalar
,scalar/square-matrix
,column-matrix/square-matrix
,row-matrix/square-matrix
,up/square-matrix
,down/square-matrix
,matrix/square-matrix
. -
new
solve-linear
implementations between square matrices andup
down
, row and column matrices, and between structures and scalars. -
new
solve-linear-right
between row-matrix+square-matrix, down+square-matrix and scalar+structure.
-
-
Fixes an infinite loop with
sicmutils.matrix/some
. -
Renames
square-structure->
totwo-tensor->
, andsquare-structure-operation
totwo-tensor-operation
. These functions now work with rectangular 2 tensors, not just square. -
sicmutils.structure
gainsdown-of-ups?
,up-of-downs?
,two-up?
,two-down?
,two-tensor?
andtwo-tensor-info
for working with "2 tensors", ie, structures that contain structural entries of matching orientation and size. -
New
g/acot
generic method installed for Operator instances.
-
-
#471:
-
installs the complex GCD implementation into the generic system and modifies it to work with real/complex pairs.
-
tweaks the default gcd implementation so that two identical values
x
, even if they are floating point, will returnx
from(gcd x x)
. (default gcd insicmutils.euclid
can handle cases now where the terms are equal and of opposite sign.) -
adds
exact-divide
handling of non-integral numbers when the inputs are either equal or of opposite sign.
-
-
#398 adds a
sicmutils.generic/gcd
implementation for complex numbers, closing the long-standing #58. Thanks to @adamhaber for this! -
#461 adds
sicmutils.quaternion
, with a full arithmetic implementation and the beginnings of a rotation API. Quaternions are implemented like vectors of length 4, and implement all appropriate Clojure protocols. All arithmetic is compatible with all scalars and complex numbers.-
Accessors:
get-r
,real-part
,get-i
,get-j
,get-k
,complex-1
,complex-2
,->complex-pair
,->vector
,three-vector
-
Predicates:
real?
,zero?
,one?
,pure?
,unit?
-
Constants:
ZERO
,ONE
,I
,J
,K
-
Reader literal
#sicm/quaternion
takes a 4-vector or a single entry. -
Constructors:
make
,from-complex
,spherical
,semipolar
,multipolar
,cylindrospherical
,cylindrical
-
More:
arity
,evaluate
,partial-derivative
,magnitude-sq
,normalize
,commutator
-
Transcendental functions:
exp
log
,cos
,sin
,tan
sinh
,cosh
,tanh
. Many more transcendentals will work, thanks to their default implementations. -
Arithmetic and generics:
+
,-
,*
,/
,dot-product
,cross-product
,conjugate
,magnitude
,expt
,sqrt
,simplify
,infinite?
,solve-linear-right
,solve-linear
-
Rotation-related:
from-angle-normal-axis
,from-angle-axis
,pitch
,roll
,yaw
,->angle-axis
-
-
#468:
-
adds
sicmutils.polynomial/touchard
, implementing a constructor for the type of polynomial known as a "complete Bell polynomial" or "Touchard polynomial". -
adds
sicmutils.special.factorial/bell
for computing the nth Bell number -
sicmutils.series/bell-series
returns an infinite sequence of bell numbers.
-
-
#276 adds an
integration-opts
tosicmutils.mechanics.lagrange/Lagrangian-action
. All options are passed on todefinite-integral
. By default,parametric-path-action
passes:compile? false
, since we do NOT want to compile the polynomial. -
#463 adds a new 1-arity to
sicmutils.matrix/characteristic-polynomial
that returns an actual polynomial instance. Creating this polynomial once and calling it many times is much more efficient. Closes #209.-
expt
called with a negative base and non-integral power now properly returns a complex number instead of##NaN
. -
symbolic
=
now behaves correctly and accumulates an expression of nestedand
s, vs before. The previous behavior would convert(= 'a 'b 'c')
to(= (= 'a 'b) 'c')
, which is NOT correct. (if(= 'a 'b)
is true, then the expression evaluates tofalse
, since(= true 'c')
is false.) -
adds
sicmutils.series/function->
, for generating a Maclaurin series from a function.
-
-
#449:
-
All missing trigonometric functions have been filled in
sicmutils.generic
and aliased insicmutils.env
:- Inverse cotangent:
acot
- inverse secant:
asec
- inverse cosecant:
acsc
- hyperbolic (inverse hyperbolic) cotangent:
coth
andacoth
- hyperbolic (and inverse hyperbolic) secant:
sech
andasech
- hyperbolic (and inverse hyperbolic) cosecant:
csch
andacsch
All of these have default implementations and derivatives defined. They'll work out of the box for all types with
atan
defined (and potentiallyexp
,sqrt
andlog
.)Thanks to John D Cook's 'Bootstrapping a minimal math library' for his inspiration on the defaults and implementation order of these new functions.
- Inverse cotangent:
-
expt
gains a new default implementation for non-native-integral powers, makingexpt
work for any type withexp
,log
andmul
defined. -
sqrt
gains a default implementation for all types implementingexp
,mul
andlog
. -
All trig functions now have derivatives and docstrings.
-
New
sinc
,tanc
,sinhc
,tanhc
functions live insicmutils.generic
and are aliased intosicmutils.env
. These are generically defined as(/ (sin x) x)
,(/ (tan x) x)
(and similar withsinh
andtanh
), with correct definitions for 0 and infinite-valued inputs.These functions all support derivatives as well.
-
New default
acot
implementation insicmutils.series
.
-
-
#450:
-
Adds
sicmutils.series/harmonic-series
, the infinite series of harmonic numbers -
moves
sicmutils.numerical.elliptic
to thesicmutils.special
package, assicmutils.special.elliptic
. -
New
sicmutils.special.factorial
namespace!sicmutils.util.permute/factorial
moves here, and the forgotten duplicatesicmutils.generic/factorial
is now gone.- New functions:
falling-factorial
,rising-factorial
,double-factorial
,multi-factorial
,subfactorial
,binomial-coefficient
,stirling-first-kind
,stirling-second-kind
.
- New functions:
-
New
sicmutils.util.permute/multichoose
function, implementing the definition described here. -
better
number-of-combinations
impl insicmutils.util.permute
, usingsicmutils.special.factorial/falling-factorial
-
sci bindings for
sicmutils.special.factorial
,sicmutils.util.permute
.
-
-
#458:
-
Default implementation of
g/negative?
returningfalse
for literal numbers and symbols. This was required to getg/abs
working for polynomials and rational functions with symbolic coefficients. -
Polynomials and rational functions now correctly unwrap
Literal
coefficients in->expression
. Without this, the resulting expressions would not correctly respond tosimplify
calls. -
Slight efficiency improvement in
sicmutils.polynomial.gcd/->content+primitive
. -
sicmutils.rational-function/from-points
now correctly builds its function. Before, it was unhygienic; if'x
appeared in the coefficients the results would be incorrect.
-
-
#456:
-
sicmutils.mechanics.lagrange/{Γ,Γ-bar}
are removed in favor of the existingGamma
andGamma-bar
functions. Thesicmutils.env
aliases are gone as well. -
sicmutils.mechanics.lagrange/Lagrange-interpolation-function
now returns an actual polynomial instance. Because polynomials supportIFn
and respond to the derivative operatorD
, this makes thefind-path
example on pages 22/23 of SICM run about 5x faster. -
Richardson extrapolation is now implemented as a functional fold. The exposition in
sicmutils.polynomial.richardson
discusses this; the namespaces gainsrichardson-fold
,richardson-sum
andrichardson-scan
.
-
-
#455 makes
sicmutils.util.aggregate/scan
andsicmutils.algebra.fold/fold->scan-fn
slightly more efficient by dropping the first element of the returned sequence before mapping thepresent
function. -
#453:
- Adds
sicmutils.polynomial/from-points
andsicmutils.rational-function/from-points
for generatingPolynomial
andRationalFunction
instances from sequences of points.
- Adds
-
#451:
-
new
sicmutils.algebra.fold
namespace:-
New folds:
kahan-babushka-neumaier
(aliased askbn
),kahan-babushka-klein
and andkbk-n
macro for generating higher-orderkahan-babushka-klein
variants.generic-sum-fold
folds usingsicmutils.generic/+
. -
sicmutils.util.aggregate/kahan-fold
now lives here, namedkahan
. -
fold->sum-fn
andfold->scan-fn
generate functions likesicmutils.util.aggregate.{sum,scan}
specialized to the supplied fold. See the docstrings for the multiple arities supported -
fold primitives:
count
,constant
,min
,max
. -
fold combinator
join
allows compound folds to be built out of primitive folds.
-
-
Upgrades to
sicmutils.util.aggregate
:-
scanning-sum
renamed toscan
-
halt-at
deleted in favor of the built-inhalt-when
that I didn't know about! -
scan
andsum
now both use a dynamic binding,*fold*
, to set the fold they use for aggregation. By default, this is set to the newkahan-babushka-neumaier-fold
. -
The three-arity version of
sum
now uses transducers, saving a pass over the input range. -
pairwise-sum
implements pairwise summation, an error-limiting technique for summing vectors. Use the dynamic binding*cutoff*
to set wherepairwise-sum
bails out to normal summation.
-
-
Upgrades to
sicmutils.rational-function.polynomial
:-
The folds in this namespace now follow the fold contract laid out in
sicmutils.algebra.fold
, implementing all three arities correctly. -
I realized that the fold implementation here should /not/ return a full row every time it processes a previous row; a far better
present
implementation would return the best estimate so far. Then you could build ascan
from that fold to see the estimates evolve lazily as new points are added. This has better performance, it turns out, than the original method! -
added a bunch to the exposition to make the advantages clear.
-
-
Upgrades to
sicmutils.rational-function.interpolate
:-
fold
interface upgraded, similar to the polynomial interpolation notes. -
New
bulirsch-stoer-fold
,bulirsch-stoer-sum
andbulirsch-stoer-scan
functions. These are similar to themodified-**
versions but use thebulirsch-stoer
algorithm, instead ofmodified-bulirsch-stoer
. -
modified-bulirsch-stoer-fold-fn
renamed tomodified-bulirsch-stoer-fold
, to match the naming scheme of other "folds" in the library. -
modified-bulirsch-stoer-fold
renamed tomodified-bulirsch-stoer-sum
, to match the convention that "reducing a sequence with a fold" is called "summing" the sequence. I can see this changing down the road...
See
context-opts
for instructions on how to enablesicmutils.algebra.fold/kbk-n
in the SCI environment (you'll need to turn on access tojs/Math
orjava.lang.Math
). -
-
Fixed a type inference warning in ClojureScript in
sicmutils.complex
. -
Added support for
sicmutils.util.def
and itsfork
macro to the default SCI environment provided by SICMUtils. Helpful for macro-writing! -
sicmutils.numerical.quadrature.adaptive
now uses the dynamically boundsicmutils.util.aggregate/*fold*
to accumulate its numerical integral pieces, instead of a hardcodedkahan-sum
. -
sicmutils.numerical.quadrature.bulirsch-stoer
now uses the functional scan versions of polynomial and rational function interpolation, as these are a bit faster than the originals! -
sicmutils.util.stream/scan
deleted in favor ofsicmutils.util.aggregate/scan
with a dynamic binding for*fold*
to customize.
-
-
#448:
-
new
g/infinite?
generic with implementations for all numeric types, complex numbers,differential
instances. Defaults tofalse
for all other types. (Also aliased intosicmutils.env/infinite?
). -
The infix, TeX and JavaScript renderers (
->infix
,->TeX
and->JavaScript
) all properly render##Inf
and##-Inf
. Infix uses the Unicode symbol ∞, while->TeX
uses the LaTeX command\infty
. Javascript'sInfinity
stands in for##Inf
in generated JS code. -
Complex numbers now respond
true
tog/negative?
if their imaginary component is zero and real component is negative, false otherwise. -
g/+
,g/-
,g//
now no longer short circuit if there is a NUMERIC zero on either side. This was causing bugs in cases where we allow, say, a scalar to be added to a quaternion, and auto-convert the scalar right there (so it adds only to the real part). OR in cases, like in the matrix PR, where we convert the scalar in addition to<scalar>*I*
.- This caused some problems with
sicmutils.matrix
tests that were not well typed.
- This caused some problems with
-
The default
expt
implementation is now available as a function to call directly (sicmutils.generic/default-expt
) without going through the dispatch system.
-
-
#447 contains a grab-bag of fixes and additions, many related to complex numbers:
-
Use
Math/E
instead of(Math/exp 1)
for euler's constant insicmutils.env
. -
Fix bug in
sicmutils.calculus.indexed
, in a case where either input was missing anup
ordown
index type. -
symbolic
dot-product
andinner-product
-
inner-product
now defaults todot-product
for scalar instances. This is correct for all numeric types we currently have, sincecomplex
is the only tough case, and it has real coefficients. -
simplify now does NOT freeze expressions before simplifying. This allows complex numbers to survive simplification, since they freeze to
(complex <re> <im>)
.-
big rewrite in
sicmutils.simplify.rules
, to convert all of the frozen matchers like(complex 1 2)
into matchers that actually bind to a complex number. -
more rules in
complex-trig
, it can now handle bigger products inside ofsin
andcos
multiplied byI
.
-
-
Various improvements to
sicmutils.complex
:-
complex implementations for
dot-product
between complex and real types -
Fixed reflection warnings with
ComplexFormat
in complex parsing code -
complex
zero?
now returns true for inputs like(complex -0.0 -0.0)
, where a negative zero lives in the real or imaginary slots -
new
sicmutils.complex/-I
binding, set to(g/negate c/I)
-
g/expt
for complex numbers optimizes the inputs equal toI
by returning exact 1, -1,I
or-I
depending on the input. This applies tog/square
andg/cube
as well.
-
-
-
#445 fixes a bug where structures and other seq-able types were interpreted as sequence matchers.
In
pattern.match
and all rules, things that respond true tosequential?
but notseq?
orvector?
(many of the sicmutils types, like structures and the upcoming Quaternion type) were being converted toseq
and treated as sequence matchers vs literal matchers. This no longer happens, and structures etc are treated as literal matchers. -
#443:
-
Implements
IKVReduce
andReversible
for structures. This enablesrseq
andreduce-kv
to work with structures. -
Removes a
reduced
shortcut condition insicmutils.generic/*
that was causing multiplications of the form(* 0 0 (up 0 0))
to shortcut and return0
instead of the appropriate structural form. -
the
atan
implementation for symbolic numbers is now careful not to return a floating point number in the case of a 0 argument in the second position. Additionally, it now returns symbolicpi
or 0 in the case of0
in the y argument for positive and negativex
argument, respectively, and symbolic(/ pi 2)
or(- (/ pi 2))
for a 0x
argument and respective positive or negativey
argument.
-
-
#442 fixes #441 by upgrading the implementations of
sicmutils.util.permute/{factorial,number-of-combinations}
to be able to handle large inputs. Thanks to @swapneils for the report. -
#440:
-
Modifies
(g/exp 0)
to return an exact 1, vs the previous1.0
. -
Fixes a bug in
sicmutils.rules/exp-contract
leftover from the port from Scheme. Thanks to @adamhaber for pointing this out!
-
-
#438:
-
converts
doall
calls torun!
,dorun
,doseq
ormapv
where applicable. In cases where we were trying to force side effects (mostly in the tests), this change prevents the environment from retaining the full sequence. This will save memory! -
adds missing tests from
connection.scm
tosicmutils.calculus.connection-test
, stressing pages 205 - 213 from MTW, Gravitation.
-
-
#434: allow pattern matching forms to successfully bind to
nil
orfalse
. -
#397:
sicmutils.calculus.manifold/typical-coords
now returns generated coordinate symbols that start with the same symbol as the coordinate system's prototype, like:
(typical-coords R2-polar)
;;=> (up x065308 x165309)
(typical-coords
(with-coordinate-prototype R2-polar (up 'r 'theta)))
;;=> (up r65312 theta65313)
-
#396:
-
fixes a bug in the SCI version of
define-coordinates
which didn't allow any rebinding of manifolds. -
Removes the
bindings
key fromsicmutils.env.sci/context-opts
. babashka/sci#637 is a bug with variable rebinding that occurs when:bindings
is in play. Instead of relying on this key, evaluate(require '[sicmutils.env :refer :all])
against your SCI environment to get all bindings. -
bumps the default version of SCI to 0.2.7.
-
-
#348:
-
Adds a new single arity version of
sicmutils.util.permute/permutation-parity
, which returns the parity of a permutation relative to its sorted version. -
sicmutils.complex/complex
can now take a single string argument in both Clojure and ClojureScript. -
Expands the complex number literal parser to take these forms, in addition to the previously-supported string argument:
-
#sicm/complex [1.2 3.6] ;; 1.2+3.6i
#sicm/complex [1.2] ;; 1.2
#sicm/complex 1.4 ;; 1.4
#sicm/complex "1.2 + 3.6i" ;; 1.2+3.6i
-
#394 fixes a bug with derivatives of functions that returned a map... but where the map was actually meant to represent some other type, by holding a
:type
key. We do this for manifold families and manifold points, as two examples. Now, instead of recursing into the values, the system will correctly throw an error. (You can fix this by using adefrecord
instead of a map and implementingsicmutils.differential/IPerturbed
.) -
#393:
-
Forms like
(let-coordinates [(up x y) R2-rect] ...)
will now work even ifup
is not present in the environment. Previously this syntax was valid, but only ifup
had been imported. -
Adds the
sicmutils.calculus.coordinate/define-coordinates
macro, also aliased intosicmutils.env
. This macro allows you to write forms like
-
(define-coordinates (up t x y z) spacetime-rect)
(define-coordinates [r theta] R2-polar)
and install set of bindings for a manifold's coordinate functions, basis
vector fields and basis form fields into a namespace. This is used liberally
in Functional Differential Geometry. (You might still prefer let-coordinates
for temporary binding installation.)
-
Converts many of the
sicmutils.fdg
test namespaces to use the newdefine-coordinates
macro, making for a presentation closer to the book's. -
Fixes a ClojureScript warning in
sicmutils.util
warning due to redefinition ofclojure.core/uuid
-
#386:
-
Aliases
sicmutils.mechanics.hamilton/phase-space-derivative
intosicmutils.env
, and addssicmutils.sr.frames/base-frame-maker
. The latter function makes it easier to write reference frames likethe-ether
, as with thehome
variable in chapter 11 of FDG. -
Adds all code listings from chapters 10 and 11 of FDG as
sicmutils.fdg.{ch9,ch10}-test
.
-
-
#384:
-
Adds
sicmutils.fdg.ch9-test
, with tests for all forms from FDG's 9th chapter. -
Tests from
sicmutils.fdg.einstein-test
now all work, and quite fast. The functions in this namespace comprise some of the exercises from FDG chapter 9. (Einstein's Field Equations hung until this PR... getting these working is a huge achievement for me, and, in some sense, the final milestone of the Big Port from scmutils.) -
Adds
sicmutils.function/memoize
, a metadata-and-function-arity preserving version ofclojure.core/memoize
. -
in
sicmutils.calculus.indexed
,with-argument-types
andwith-index-types
now both correctly set the arity of the returned function, in addition to the argument types or indices.sicmutils.function/arity
will now work correctly with indexed or typed functions. -
Adds new
manifold?
andmanifold-family?
functions insicmutils.env
andsicmutils.calculus.manifold
. These are enabled by new:type :sicmutils.calculus.manifold/{manifold,manifold-family}
keys in the appropriate structures in the manifold namespace. Manifolds and manifold families will now respond with these keywords tosicmutils.value/kind
. -
The
sicmutils.calculus.manifold/ICoordinateSystem
now has auuid
function, for internal comparison of coordinate systems. This is here so that points can cache coordinate system representations by UUID. Before this change, changing the coordinate prototype, or attaching metadata to a coordinate system would break its cache entry in manifold points. (This was the killer for the Einstein Field Equations!) -
sicmutils.calculus.manifold/{coordinate-prototype,with-coordinate-prototype}
now store and retrieve the coordinate prototype from metadata. This plus the previous change allows manifold points to correctly cache their coordinate representations. -
sicmutils.calculus.manifold/manifold
acts as identity on manifolds now. Previously it only worked on coordinate systems.
-
-
#382:
-
Makes the
name
argument tosicmutils.operator/make-operator
optional.name
now defaults to'???
. -
adds tests for all code forms in Chapter 8 of FDG.
-
-
#376 adds more type hints to the
ratio.cljc
namespace. This fully solves the advanced compilation issues we were seeing. -
#374: Demos, thanks to @sigmaxipi!
-
#379 fixes typos in a couple of the equations in
richardson.cljc
, closing #377. Thanks to @leifp for the report. -
Features, tests and bugfixes from #381:
-
sicmutils.calculus.coordinate/generate
moves tosicmutils.calculus.manifold/c:generate
; this supports a bugfix where 1-dimensional manifolds likeR1-rect
, akathe-real-line
, return a coordinate prototype of a single element liket
instead of a structure with a single entry, like(up t)
. Thanks to @phasetr for the bug report that led to this fix, and @gjs for finding and fixing the bug. -
same.ish/Approximate
implemented forsicmutils.structure/Structure
, allowingish?
comparison ofup
anddown
structures with approximate entries. Requiresicmutils.generator
for this feature. (NOTE: because protocols are implemented for the LEFT argument,(ish? <vector> (down ...))
will still return true if the values are approximately equal, even though a<vector>
is technically anup
and should NOT equal adown
. Do an explicit conversion toup
usingsicmutils.structure/vector->up
if this distinction is important.) -
same.ish/Approximate
now defers tosicmutils.value/=
for equality betweenSymbol
and other types. This letsish?
handle equality between symbols like'x
and literal expressions that happen to wrap a single symbol. -
Cartan->Cartan-over-map
now does NOT compose(differential map)
with its internal Cartan forms. This fixed a bug in a code listing in section 7.3 of FDG. -
Section 7.3 of FDG implemented as tests in
sicmutils.fdg.ch7-test
. -
Many new tests and explorations ported over from
covariant-derivative.scm
. These live insicmutils.calculus.covariant-test
. -
timeout exceptions resulting from full GCD are now caught in tests using
sicmutils.simplify/hermetic-simplify-fixture
. Previously, setting a low timeout where simplification failed would catch and move on in normal work, but fail in tests where fixtures were applied.
-
Yet another incremental release, this time to bump the Fraction.js
dependency.
The new cljsjs
dependency has code compatible with advanced compilation.
- #372 bumps the
Fraction.js
dependency to4.1.1
.
This is an incremental bugfix release to get ClojureScript advanced compilation into shape.
-
#371:
-
fixes a subtle bug with extern inference on
fraction.js/bigfraction.js
. Thanks to @sigmaxipi for this report! -
removes overridden factory constructors like
->Polynomial
. I had originally done this for functions that held a metadata field, so that the user could leave it out and have it default tonil
... but advanced Closure compilation can't understand thens-unmap
call, so it has to go. -
Many unary functions on
Operator
,Structure
,Series
,PowerSeries
,Polynomial
andRationalFunction
now preserve metadata. Binary functions between two instances of any of these still return a new object with metadata ==nil
.
-
(If you have any questions about how to use any of the following, please ask us at our Github Discussions page!)
This release focused on improving the expressiveness and performance of the three simplification engines in SICMUtils:
-
sicmutils.polynomial
andsicmutils.rational-function
are now quite well fleshed out, with full polynomial and rational function APIs and many generics. -
The polynomial and rational function simplifiers work by round-tripping expressions through these types, depending on each namespace to emit symbolic expressions in "canonical form". This process is now much faster! On one important Bianchi Identity benchmark in
sicmutils.fdg.bianchi-test
, one test that formerly took close to 30 minutes now runs in 30 seconds, and all see a 60-fold improvement. -
By default, these simplifiers emit expressions with all terms multiplied out; the new
factor
function insicmutils.env
lets you factor expressions, overriding this default. -
The rule-based simplifier is now based on a powerful pattern matching engine, implemented in
pattern.match
andpattern.rule
.sicmutils.simplify.rules
now contains every rule and possible customization from the original scmutils codebase.
There is a lot in this release, all motivated by performance. Please read on for the detailed notes, and enjoy version 0.19.0!
-
#353 introduces a powerful new simplifier, ported from the
new-simplify
procedure insimplify/rules.scm
of the scmutils library. There are now a BUNCH of new rulesets and rule simplifiers insicmutils.simplify.rules
!The next step with these is to massage them into separate bundles of rules that users can mix and match into custom simplifiers for objects like abstract matrices, abstract bra and ket structures, up and down, booleans (for representing equations and inequalities) and so on.
-
#349 introduces a new pattern matching system, built out of matcher combinators. All of the rules in
sicmutils.simplify.rules
now use the new syntax offered by the library. Some notes:-
pattern.match
defines a number of "matcher combinators"; these are functions that take a map of bindings, a data input and a success continuation and either succeed by calling their continuation, or fail. Out of the box, the library providesfail
,pass
,with-frame
,update-frame
,predicate
,frame-predicate
,eq
,bind
,match-when
,match-if
,or
,and
,not
,segment
andsequence
. -
Additionally, any combinator that takes another combinator can ALSO take a pattern form like
'?x
. Seepattern.syntax
for the full, rich range of syntax allowed. These are all functions, so you'll have to quote your symbols at this stage. -
Passing a matcher combinator to
pattern.match/matcher
to generate a matcher object. This is a function from somedata
input to a map of bindings on success, or an explicitpattern.match/failure
object on failure. Test for failure withpattern.match/failed?
. -
A combination of a matcher and a "consequence function" is called a "rule". A consequence is a function that takes a binding map and either returns a new result or fails by returning
nil
orfalse
. (Don't worry, you can succeed with these values too by wrapping them insicmutils.rule/succeed
.)Rules are the heart of the whole simplification mechanism in sicmutils! To learn about how to build these, see the documentation for
pattern*
,pattern
,consequence
,template
,rule*
andrule
. -
pattern.rule
gives you some starter rules, and many combinators you can use to build more and more powerful and complex sets of rules. These arepass
,fail
,predicate
,return
,branch
,choice*
,choice
,pipe*
,pipe
,n-times
,attempt
,guard
,iterated
,while
,until
,fixed-point
andtrace
. -
Rules are nice for rewriting entire expressions recursively, from the bottom up or top down. This is called "term rewriting". A big motivation for this rewrite was to make it easy to build custom term rewriters for types like abstract matrices or abstract up and down structures. You can use your rules to rewrite structures recursively with
bottom-up
,top-down
,iterated-bottom-up
anditerated-top-down
.ruleset*
,ruleset
,rule-simplifier
andterm-rewriting
capture some common patterns the library uses to go from rules => term rewriters. -
If you want ideas about how to use the pattern matching library to rewrite expressions, see
sicmutils.simplify.rules
for many examples.
-
-
#354 adds SCI support for all macros and functions in the new pattern matching namespaces, and adds these to the namespaces exposed via
sicmutils.env.sci
.
-
#341 takes on a large rewrite of the rational function and polynomial simplfiers. One goal of this project was to improve the performance of the Bianchi Identities in
sicmutils.fdg.bianchi-test
, and I'm happy to say that they are now a good bit faster than the original scmutils implementation.sicmutils.polynomial
andsicmutils.rational-function
are now solid data structures of their own, with many operations installed into the generic system. These are now valuable and useful outside of their role in the simplifier.This was a large project, and many small improvements and bugfixes snuck in. Here is the full list:
-
v/kind
now works forsorted-map
instances. -
GCD in ClojureScript is now fast and efficient between all combinations of
js/BigInt
andjs/Number
, and in Clojure between all combinations ofclojure.lang.BigInt
,BigInteger
,Long
andInteger
. -
on the JVM, GCD now works properly with rational numbers. Previously anything non-integral would return
1
; now(gcd 1/2 1/3)
properly returns1/6
. -
g/exact-divide
now succeeds for all non-exact::v/scalar
types (symbols, floats, etc) either if the denominator is zero, or if the two arguments are equal. Else, it throws, just like before. -
A multi-arity call to
sicmutils.generic/*
now stops if it encounters a 0, rather than attempting to multiply all remaining items by 0. -
The default function for
sicmutils.generic/lcm
protects against overflow by dividing only a single one of its argumentsa
andb
by(gcd a b)
. -
(g/lcm 0 0)
now properly returns 0. -
New
sicmutils.util.aggregate/{monoid,group}
functions let you build multi-arity aggregations out of binary combination functions, with an option to bail early at "annihilator" values, like 0 for multiplication. -
New multi-arity
lcm
andgcd
implementations for symbolic expressions appropriately handle0
and1
on either side, as well as the case where both arguments are equal. -
In the
sicmutils.numsymb
namespace, thanks tomonoid
andgroup
, the'*
,'/
,'-
,'+
,'or
,'and
,'gcd
,'lcm
and'=
operations now have efficient multi-arity implementations that stop computing when they receive an annihilator, like0
for multiplication ortrue
foror
. Access these via(sicmutils.numsymb/symbolic-operator <symbol>)
. -
sicmutils.series/PowerSeries
gainsarg-scale
andarg-shift
functions; these are identical tosicmutils.function/arg-{scale,shift}
, but preserve thePowerSeries
type. (#367 proposes making these functions generic.) -
New
sicmutils.ratio/IRational
protocol, withnumerator
anddenominator
functions implemented for ratios and for theRationalFunction
data type. These two are now exposed insicmutils.env
. -
sicmutils.simplify.rules/*divide-numbers-through-simplify?*
is nowtrue
by default; numbers in the denominator will now automatically pull up into the numerator. All tests now reflect this setting. -
Any analyzer generated from
sicmutils.expression.analyze
can now act on both bare, unwrapped expressions (raw lists etc) and onsicmutils.expression.Literal
instances. This means that you can now callsicmutils.simplify/{*rf-simplify*,*poly-simplify*}
as functions and canonicalize some form with either simplifier without triggering a full simplification. A small win, but ice. -
sicmutils.polynomial.factor
got a major rewrite, and now exposes a few functions likepoly->factored-expression
,factor-expression
andfactor
.factor
is tremendously useful! Callfactor
(it's aliased intosicmutils.env
) on any expression to factor out all possible terms. This makes it much easier to see where there is some cancellation lurking, in, say, some expression you know should equal zero (a residual).
-
bugfix:
sicmutils.expression.Literal
instances now compare their contained expression viasicmutils.value/=
. -
sicmutils.rules/constant-elimination
can now eliminate constants from expressions with any arity, not just binary forms.
Now, the three big namespaces...
sicmutils.polynomial
,sicmutils.rational-function
andsicmutils.polynomial.gcd
all got a big overhaul.-
sicmutils.polynomial
notes:-
Polynomial
uses a new sparse representation for its "power product" term; this, plus an arithmetic rewrite, makes the whole system much faster for larger numbers of variables (for all #s, really). -
Polynomial
instances implement many more Clojure(script) protocols. They can hold metadata; they can be evaluated as functions of their indeterminates, andseq
now returns a sequence of terms. -
Polynomial
extendssicmutils.function/IArity
anddifferential/IPerturbed
, so you can usesicmutils.function/arity
, and take derivatives of functions that return polynomials. -
In their arithmetic,
Polynomial
instances will drop down to bare coefficients whenever some multiplication or addition removes all indeterminates. All binary arithmetic exposed in the namespace can handle non-Polynomial
instances on either or both sides, so this is fine. Coefficients are treated as constant polynomials. -
The namespace holds many new functions. Some choice ones are:
-
constructors:
make
,constant
,linear
,c*xn
,identity
, andnew-variables
-
accessor functions:
arity
,degree
,coefficients
,leading-term
,leading-coefficient
,leading-exponents
,leading-base-coefficient
,trailing-coefficient
,lowest-degree
-
predicates:
monomial?
,monic?
,univariate?
,multivariate?
,negative?
-
functions to generate new polynomials:
map-coefficients
,map-exponents
,scale
,scale-l
,normalize
,reciprocal
,drop-leading-term
,contract
andextend
alongsidecontractible?
,lower-arity
,raise-arity
,with-lower-arity
,arg-scale
,arg-shift
-
arithmetic:
negate
,abs
,add
,sub
,mul
,square
,cube
,expt
,divide
along withdivisible?
,evenly-divide
,pseudo-remainder
, and lots of functions installed into the generic arithmetic system. -
different ways to evaluate polynomials:
evaluate
,horner-with-error
-
calculus!
partial-derivative
andpartial-derivatives
are alive and well, and work with theD
operator. -
Functions to get in and out of polynomials from other types:
univariate->dense
,->power-series
,expression->
,->expression
-
-
-
sicmutils.polynomial.gcd
also got a rewrite; it's fairly clear to read now, and prepared for the eventual addition of the sparse multivariate GCD routine that scmutils uses. There are some efficiency gains here too that let us turn a number of tests back on, or demote them from:long
markers. -
sicmutils.rational-function
notes:-
RationalFunction
instances implement many more Clojure(script) protocols. They can hold metadata; they can be evaluated as functions of their indeterminates, andseq
now returns a pair ofnumerator
,denominator
. -
RationalFunction
extendssicmutils.function/IArity
andsicmutils.ratio/IRational
, so our genericarity
,numerator
anddenominator
work on these instances. -
Here are some new functions from the
RationalFunction
namespace:-
constructor:
make
, drops to polynomial or coefficient where needed just likePolynomial
functions -
functions to generate new rational functions:
arg-scale
,arg-shift
-
predicates:
negative?
-
arithmetic:
negate
,abs
,add
,sub
,mul
,square
,cube
,expt
,invert
,div
,gcd
, and many functions installed into the generic arithmetic system. -
evaluation via
evaluate
-
calculus!
partial-derivative
andpartial-derivatives
are alive and well, and work with theD
operator. -
Functions to get in and out of rational functions from symbolic expressions:
expression->
,->expression
.
-
-
-
-
#358:
-
Adds a more efficient
literal-derivative
implementation tosicmutils.abstract.function
, making the Bianchi identity benchmarks run 40% faster. -
In ClojureScript,
Range
instances now implementsicmutils.value.Value
andsicmutils.differential.IPerturbed
, allowing them to be returned from derivative-taking functions -
Major, unexpected performance improvement - it turns out
sicmutils.value/number?
was quite slow in Clojure (less so in ClojureScript). Changing this function from anisa?
check to a series of explicitinstance?
checks cut the build time in half. This makes the numeric tower less extensible... but it wasn't terribly extensible to start with, and needs some attention to make it so. A big win! -
The Bianchi identity benchmarks have all been updated to reflect the big performance improvements achieved here, thanks to the wonderful Tufte profiling library from @ptaoussanis. The remaining very slow piece in the simplifier is the implementation of
g/add
for polynomial instances. #341 will improve this situation.
-
-
#360 introduces a number of performance improvements to the
sicmutils.differential.Differential
implementation, primarily interms:+
andterms:*
. thanks again to @ptaoussanis and the Tufte profiling library for helping me track these down. -
#357:
-
Adds the ability to do incremental simplification, every time an operation is performed involving a symbolic expression. Bind
sicmutils.numsymb/*incremental-simplifier*
to a function from raw expression -> raw expression, likesicmutils.simplify/simplify-expression
or any of the rules insicmutils.simplify.rules
to enable this behavior. -
Expands the
sicmutils.expression.analyze
API with the functionsdefault-simplifier
,expression-simplifier
,initializer
,expression-analyzer
andauxiliary-variable-fetcher
. See the API documentation for detailed notes on how to do interactive expression analysis and simplification with these new tools. -
by default, each simplification pass uses both rational function and polynomial canonicalization. This brings the simplifier into line with the scmutils simplifier.
-
-
#353:
-
Adds a new
sicmutils.util.logic
namespace with anassume!
function that allows rules to log assumptions when some simplification like(sqrt (square x))
might have to choose one of multiple possible simplifications ((non-negative? x)
, in this example).This function simply logs the assumption for now, instead of performing any checks. now. Turn off assumption logging with the dynamic variable
*log-assumptions?*
in that namespace. -
new
sicmutils.value/almost-integral?
returns true if its argument is VERY close to an integral value, false otherwise.
-
-
Efficient
symmetric-difference
implementation insicmutils.util.vector-set
(#346)
-
#369:
-
Removes JVM dependencies on Guava and nrepl.
-
Removes
sicmutils.env/sicmutils-repl-init
; this is only used bylein repl
, and we now accomplish the same task with the:repl-options
entry inproject.clj
. -
Makes
sicmutils.polynomial.{factor,gcd}
available to SCI via thesicmutils.env.sci
namespace -
moves a few namespaces to more valid locations, now that the rational function and polynomial namespaces are tidied:
-
sicmutils.numerical.interpolate.polynomial
->sicmutils.polynomial.interpolate
-
sicmutils.numerical.interpolate.richardson
->sicmutils.polynomial.richardson
-
sicmutils.numerical.interpolate.rational
->sicmutils.rational-function.interpolate
-
-
-
#358:
-
Converts the ClojureScript test build and REPL command from
lein-cljsbuild
toshadow-cljs
. This enables more formerly-slow tests for ClojureScript; these are now fast enough to run, thanks to the performance improvements described below. -
Upgrades our Timbre logging dependency to version 5.1.2, and SCI to 0.2.5
-
-
#353:
-
expression->stream
,expression->string
,print-expression
,pe
move fromsicmutils.simplify
tosicmutils.expression
, and are now aliased insicmutils.env
. -
pattern.rule/guard
now fails if its rule argument fails; previously it wrapped the result inattempt
, and would return its original input on failure. -
fixed a heisenbug in
sicmutils.expression.analyze/make-analyzer
where, in ClojureScript, using expressions containing ajs/BigInt
as a hashmap key caused certain simplifications to fail. (This is vague, but the bug was really subtle.) The fix was to make sure we freeze keys in the symbol cache. This is now noted in the function body.
-
(If you have any questions about how to use any of the following, please ask us at our Github Discussions page!)
This release focused on porting over all of the material required to run every piece of code from Sussman and Wisdom's "Functional Differential Geometry". The namespaces are lightly documented; the situation is better than the original library, but will only get better as I work through the material and add commentary.
There is a huge amount of functionality and material here! We can run many examples from general and special relativity, and the tests are full of exercises from the classic "Gravitation" book by Misner, Thorne and Wheeler (MTW).
Notable changes from the rest of the library:
-
Operator
instances are slightly more efficient with their addition and multiplication, handlingzero?
andone?
cases appropriately -
Structure
s can now hold metadata -
We've extended the SICMUtils generics to Clojure's Map and Set data structures. These can now combine with
+
. Maps are treated as sparse infinite-dimensional vector spaces, and can multiply with symbolic or numeric scalars. -
ModInt
instances are now correctly equal to numbers (when those numbers mod down to theModInt
instance's residue).
The next major change will be an overhaul of the simplifier to make it work fast enough to solve Einstein's field equations in a reasonable amount of time, maybe even in the browser. Polynomial GCD is slow, but #341 will make it fast.
On to the detailed notes!
-
From #339:
-
The new
sicmutils.calculus.covariant/Lie-D
can compute the Lie derivative for coordinates. -
sicmutils.calculus.frame
lets us create relativistic reference frames for investigating special relativity problems. This namespace aliases the following functions intosicmutils.env
: 'frame?',make-event
,event?
,claim
,coords->event
,event->coords
,ancestor-frame
,frame-name
,frame-owner
andframe-maker
. -
sicmutils.calculus.hodge-star
implements the Hodge star operator from chapter 10 of Functional Differential Geometry, plus Gram Schmidt orthonormalization. This namespace aliases the following functions intosicmutils.env
:Gram-Schmidt
,orthonormalize
andHodge-star
. -
sicmutils.calculus.indexed
ports over the scmutils work on indexed objects and typed functions. This namespace aliases the following functions intosicmutils.env
:argument-types
,with-argument-types
,index-types
,with-index-types
,typed->indexed
,indexed->typed
,typed->structure
,structure->typed
,i:outer-product
andi:contract
. -
sicmutils.calculus.manifold
gainscoordinate-system?
, which (predictably) returns true if its argument is a coordinate system, false otherwise.chart
andpoint
also take relativistic reference frames in addition to coordinate systems; the returned function converts to and from coordinates and events, rather than coordinates and manifold points. -
Div
,Grad
,Curl
andLap
move fromsicmutils.calculus.derivative
tosicmutils.calculus.vector-calculus
. This namespace also contains versions of these operators from Functional Differential Geometry. This namespace aliases the following functions intosicmutils.env
:divergence
,curl
,gradient
andLaplacian
(along with the others mentioned). -
lots of new namespaces available in
sicmutils.env.sci
, soon to be deployed to Nextjournal:sicmutils.calculus.{hodge-star, indexed, vector-calculus}
, andsicmutils.sr.{boost,frames}
. -
sicmutils.sr.boost
describes boosts from special relativity, covered in chapter 11 of Functional Differential Geometry. This namespace aliases the following functions intosicmutils.env
:make-four-tuple
,four-tuple->ct
,four-tuple->space
,proper-time-interval
,proper-space-interval
,general-boost
,general-boost2
andextended-rotation
. -
sicmutils.sr.frames
implements relativistic reference frames from special relativity, covered in chapter 11 of Functional Differential Geometry. This namespace aliases the following functions intosicmutils.env
:make-SR-coordinates
,SR-coordinates?
,SR-name
,make-SR-frame
,the-ether
,boost-direction
,v:c
,coordinate-origin
,add-v:cs
andadd-velocities
.
-
-
From #338:
-
sicmutils.fdg.bianchi-test
verifies the Bianchi identities; this was a challenge posed by GJS, and getting it working exposed a few bugs and triggered the rest of the work in this PR. Thank you, GJS! -
covariant-derivative
now properly handles the case of functions with argument types attached. -
added
covariant-differential
tosicmutils.calculus.covariant
. -
aliased all functions from various namespaces in
sicmutils.calculus
intosicmutils.env
. -
adds
sicmutils.calculus.metric
, with the following functions exposed insicmutils.env
:-
coordinate-system->metric-components
,coordinate-system->metric
,coordinate-system->inverse-metric
,literal-metric
,components->metric
,metric->components
,metric->inverse-components
,metric-over-map
,lower
,vector-field->oneform-field
,drop1
,raise
,oneform-field->vector-field
,raise1
,drop2
,raise2
,trace2down
,trace2up
,sharpen
,S2-metric
-
sicmutils.calculus.metric/invert
is exposed asmetric:invert
to match the scmutils naming scheme.
-
-
adds
sicmutils.calculus.connection
, with the following functions exposed insicmutils.env
:make-Christoffel-1
,metric->Christoffel-1
,metric->Christoffel-2
,literal-Christoffel-1
,literal-Christoffel-2
,metric->connection-1
,metric->connection-2
,literal-Cartan
,structure-constant
-
-
#337:
-
adds
sicmutils.calculus.curvature
, with these new functions and many tests from the classic "Gravitation" book:Riemann-curvature
,Riemann
,Ricci
,torsion-vector
,torsion
andcurvature-components
-
form fields now have NO identity operator, since they multiply by wedge, not composition.
-
-
#328 adds many utilities for "Functional Differential Geometry".
-
vector fields, in
sicmutils.calculus.vector-field
:-
new functions:
basis-components->vector-field
,vector-field->basis-components
-
vector fields now implement
v/zero?
andv/zero-like
by returning proper vector fields.
-
-
form fields, in
sicmutils.calculus.vector-field
:-
new functions:
nform-field?
,basis-components->oneform-field
,oneform-field->basis-components
andfunction->oneform-field
(aliased asdifferential-of-function
) -
Alt
,alt-wedge
provide alternate wedge product definitions -
form fields now implement
v/zero?
andv/zero-like
by returning proper form fields that retain their rank. -
form fields now correctly multiply via
*
by usingsicmutils.calculus.form-field/wedge
, instead of composition.
-
-
maps between manifolds, in
sicmutils.calculus.map
:-
new function:
pushforward-function
-
differential
becomesdifferential-of-map
, aliased back asdifferential
-
-
sicmutils.calculus.covariant
gains new functions:Cartan?
,Christoffel?
,Cartan->Christoffel
,symmetrize-Christoffel
,symmetrize-Cartan
,Cartan->Cartan-over-map
,geodesic-equation
,parallel-transport-equation
. -
sicmutils.calculus.covariant/vector-field-Lie-derivative
can now handle structural inputs.
-
-
From #342:
-
Added
sicmutils.calculus.derivative/D-as-matrix
andsicmutils.matrix/as-matrix
, ported from scmutils. -
converted
sicmutils.modint.ModInt
to adeftype
; this allowsModInt
instances to be=
to non-ModInt
numbers on the right, if the right side is equal to the residue plus any integer multiple of the modulus.v/=
gives us this behavior with numbers on the LEFT too, andModInt
on the right.- This change means that
:i
and:m
won't return the residue and modulus anymore.sicmutils.modint
gains newresidue
andmodulus
functions to access these attributes.
- This change means that
-
The JVM version of sicmutils gains more efficient
gcd
implementations forInteger
andLong
(in addition to the existing nativeBigInteger
gcd
), thanks to our existing Apache Commons-Math dependency. -
sicmutils.structure/dual-zero
aliasescompatible-zero
to match the scmutils interface. Both are now aliased intosicmutils.env
. -
Structure
instances can now hold metadata (#339).
-
-
From #339:
-
In
sicmutils.mechanics.rotation
:-
gains aliases for
R{xyz}
inrotate-x
,rotate-y
androtate-z
. -
R{x,y,z}-matrix
now aliasrotate-{x,y,z}-matrix
. -
Added new functions
angle-axis->rotation-matrix
and the mysterious, undocumentedwcross->w
from scmutils -
rotate-{x,y,z}-tuple
are now aliased intosicmutils.env
.
-
-
Operator
instances now ignore the right operator in operator-operator addition if the left operator passes av/zero?
test. Contexts are still appropriately merged. -
in
sicmutils.simplify.rules
, thesqrt-contract
ruleset now takes a simplifier argument and attempts to use it to simplify expressions internal to a square root. As an example, if two square roots in a product simplify to the same expression, we can drop the wrapping square root; otherwise multiplication is pushed under the root as before.- Added a missing rule in
simplify-square-roots
that handles roots of exponents with odd powers.
- Added a missing rule in
-
sicmutils.matrix
changes:-
generate
has a new 2-arity version; if you supply a single dimension the returned matrix is square. -
diagonal?
returns true if its argument is a diagonal matrix, false otherwise.
-
-
A new namespace,
sicmutils.util.permute
:-
factorial
moved here fromsicmutils.generic
. It's still aliased intosicmutils.env
. -
new functions:
permutations
,combinations
,cartesian-product
,list-interchanges
,permutation-parity
,permutation-interchanges
,permute
,sort-and-permute
,subpermute
,number-of-permutations
,number-of-combinations
. See the tests for usage examples.
-
-
-
From #338:
(* <structure> <operator>)
multiplication pushes operator multiplication into the structure, rather than converting a structure into an operator.
-
#337:
-
If you combine
Operator
instances with non-equal:subtype
fields, the returned operator now keeps the parent subtype (or throws if one is not a subtype of the other). -
Operator
instances now ignore anyidentity?
-passing operator on the left or right side of operator-operator multiplication. Contexts are still appropriately merged. -
Similarly,
Operator
addition ignoreszero?
operators on the left or right side, and subtraction ignoreszero?
operators on the right right.
-
-
#328:
-
Closes #249; operators now verify compatible contexts on multiplication.
-
Operator
instances can now provides customzero?
,one?
,identity?
,zero-like
,one-like
andidentity-like
implementations by setting a function of a single (operator-typed) argument to a keyword like:zero?
in their context. the identity operator returnstrue
foridentity?
, andfalse
forone?
so that it isn't stripped by theg/*
function. -
structures implement the 0-arity case of IFn now.
-
-
#335 implements
g/make-rectangular
,g/make-polar
g/real-part
andg/imag-part
for clojure's Map data structure. Maps are treated as sparse vectors, any missing key on either side ofmake-rectangular
ormake-polar
is treated as a 0 (rather than an error because the keys don't match, as in vectors). -
#334 adds implementations of
g/add
and thesicmutils.value.Value
protocol for clojure's Set data structure. Addition is defined as set union, and(zero-like <set>)
returns the empty set. -
#334 implements
g/add
,g/negate
andg/sub
for Clojure's Map data structure. Map addition is defined as a merge usingg/add
on clashing values;g/sub
is the same, but any values on the right side not on the left side are negated.Maps can also be multiplied with scalars (commutatively) or divided (scalar on the right side only) by scalars. This, plus the commutative group property declared above, mean that Clojure's maps are sparse vector spaces over anything that responds true to
sicmutils.value/scalar?
... currently anything in the numeric tower up to complex, along with symbolic expressions andDifferential
instances.
(If you have any questions about how to use any of the following, please ask us at our Github Discussions page!)
This release starts the work of porting all of GJS and JW's "Functional Differential Geometry" to SICMUtils. The Differential Geometry section below describes the many new manifolds, coordinate systems and functions for interacting with these that we've gained.
The main big change in 0.17.0 that simplify
no longer changes the type of its
input; simplified expressions remain expressions.
solve-linear
, solve-linear-left
and solve-linear-right
round out the
stable of generics ported from scmutils.
They're not fully installed yet, but we've laid the groundwork for a new literal boolean type. This can represent equalities and inequalities, and will be excellent for equation solving.
Enjoy the release!
-
#330 adds
g/real-part
andg/imag-part
implementations forsicmutils.structure.Structure
andsicmutils.matrix.Matrix
instances. These pass through to the entries in the structure or matrix. #331 adds similar implementations forg/make-rectangular
andg/make-polar
. -
#327 adds
sicmutils.structure/sumr
, also aliased intosicmutils.env
Given some functionf
and any number of isomorphicstructures
,sumr
returns the sum of the results of applyingf
to each associated set of entries in eachstructure
. -
#319 adds
- symbolic boolean implementations for
sym:=
,sym:and
,sym:or
andsym:not
with infix, latex and JavaScript renderers. sym:derivative
, for purely symbolic derivatives
The boolean operators will act just like
=
,and
andor
on booleans, and appropriately respond if just one side is a boolean. If both sides are symbolic, These return a form like(= a b)
,(and a b)
or(or a b)
.The functions currently live in
sicmutils.numsymb
only; access them via(numsymb/symbolic-operator <sym>)
, where<sym>
is one of'=
,'and
,'or
,'not
or'derivative
. - symbolic boolean implementations for
-
#304 aliases
sicmutils.operator/anticommutator
,sicmutils.util/bigint?
and intosicmutils.env
-
implements
v/=
properly for sequences,Differential
,Complex
,Structure
andMatrix
instances -
in
sicmutils.env
,v/=
now overridesclojure.core/=
.v/=
should act identically toclojure.core/=
everywhere; the difference is that its behavior is customizable, so we can makeDifferential
instances equal to numbers, or complex numbers with a 0 imaginary part equal to real numbers with the same real part.v/=
may not drop recursively down into, say, Clojure maps. Please open an issue if you find a case like this! -
BIG CHANGE:
Literal
andStructure
instances now KEEP their type underg/simplify
. If you want to get the expression back out of itsLiteral
wrapper, usesicmutils.expression/expression-of
, also aliased intosicmutils.env
.This means that you can no longer make comparisons like this:
-
;; this worked before, and was used all over the tests (probably not in much
;; user code!)
(clojure.core/= '(* 3 x)
(simplify (+ 'x 'x 'x)))
;;=> false
Instead, use v/=
(which is now aliased into sicmutils.env
):
;; `v/=` will do the right thing by unwrapping the literal expression on the
;; right:
(v/= '(+ x y) (+ 'x 'y))
;;=> true
-
#305 adds
g/solve-linear
andg/solve-linear-left
implementations betweensicmutils.structure/Structure
instances. -
#207:
-
adds missing implementations of
g/floor
,g/ceiling
,g/integer-part
andg/fractional-part
for functions, both literal and abstract. -
adds
g/solve-linear
,g/solve-linear-left
,g/solve-linear-right
.(g/solve-linear-right a b)
returnsx
such thata = x*b
, whileg/solve-linear
(and its alias,g/solve-linear-left
) returnsx
such thata*x = b
. These functions are implemented for:sicmutils.series.{Series, PowerSeries}
- all numeric types
- functions, operators
sicmutils.modint.ModInt
sicmutils.differential.Differential
, so you can differentiate through this operation
-
-
#309:
sicmutils.util/bigint
is aliased assicmutils.env/bigint
in ClojureScript only. This is available natively in Clojure. -
#308 and #310 add:
-
sicmutils.ratio/{numerator,denominator,ratio?,rationalize}
and are now aliased intosicmutils.env
in ClojureScript. These are available natively in Clojure.sicmutils.complex/complex?
is aliased intosicmutils.env
for both platforms. -
Proper superscript support in
->infix
and->TeX
renderers.
-
-
#306: Added the mathematical constants
phi
ande
bound to, respectively,sicmutils.env/{phi,euler}
.
-
#326 is a large PR that marks the start of a big push toward full implementation of the ideas in "Functional Differential Geometry". Here is the full list of changes:
-
sicmutils.calculus.basis
gains a new::coordinate-basis
type, along withcoordinate-basis?
,basis->coordinate-system
,basis->dimension
,contract
andmake-constant-vector-field
from scmutils. More functions moved here. -
In
sicmutils.calculus.coordinate
,let-coordinates
andusing-coordinates
can now handle namespaced coordinate systems likem/R2-rect
in their coordinate system position! Their docstrings are far better too. -
sicmutils.calculus.vector-field/coordinate-basis-vector-fields
was renamed tocoordinate-system->vector-basis
. -
sicmutils.calculus.form-field/coordinate-basis-oneform-fields
was renamed tocoordinate-system->oneform-basis
. -
sicmutils.calculus.manifold
gets a LOT of restructuring, and many new manifolds out of the box. Here's the full list of new functions:manifold-type
,patch-names
,coordinate-system-names
,manifold-point?
,typical-coords
,typical-point
,transfer-point
,corresponding-velocities
zero-manifold-function
,one-manifold-function
,constant-manifold-function
And new manifolds and coordinate systems. Here's the full list of manifolds that are now present:
- From the
Rn
family:R1
,R2
,R3
,R4
,spacetime
- From
S2-type
:S2
- From
Sn
:S1
,S2p
,S3
- From
SO3-type
:SO3
And coordinate systems, prefixed by their manifold in all cases:
R1-rect
,the-real-line
(alias forR1-rect
)R2-rect
,R2-polar
,R3-rect
,R3-cyl
,R3-spherical
,R4-rect
,R4-cyl
,spacetime-rect
,spacetime-sphere
-
-
#329 fixes a bug where the simplifier couldn't handle expressions like
(sqrt (literal-number 2))
, where literal numbers with no symbols were nested inside of symbolic expressions. -
#321 changes the default
TeX
rendering style fordown
tuples back to horizontal, undoing #283. @kloimhardt made a solid case that becausedown
tuples represent row vectors, it's not helpful for building knowledge and intuition to only distinguish these with differently-shaped braces. Intuition-builders win! -
#320:
Operator
gains a new simplifier for itsname
field; the simplifier applies the associative rule to products and sums of operators, collapses (adjacent) products down into exponents, and removes instances ofidentity
(the multiplicative identity for operators).Operator
multiplication (function composition) is associative but NOT commutative, so the default simplifier is not appropriate.Before this change:
sicmutils.env> (series/exp-series D)
#object[sicmutils.series.Series
"(+ identity
(* D identity)
(* (/ 1 2) (* D (* D identity)))
(* (/ 1 6) (* D (* D (* D identity)))) ...)"]
After:
sicmutils.env> (series/exp-series D)
#object[sicmutils.series.Series
"(+ identity
D
(* (/ 1 2) (expt D 2))
(* (/ 1 6) (expt D 3)) ...)"]
- #315:
g/log2
andg/log10
on symbolic expressions now stay exact, instead of evaluating(log 2)
or(log 10)
and getting a non-simplifiable real number in the denominator:
(g/log2 'x)
;;=> (/ (log x) (log 2))
(g/log10 'x)
;;=> (/ (log x) (log 10))
-
#304:
-
implements
v/=
properly for sequences,Differential
,Complex
,Structure
andMatrix
instances -
in
sicmutils.env
,v/=
now overridesclojure.core/=
.v/=
should act identically toclojure.core/=
everywhere; the difference is that its behavior is customizable, so we can makeDifferential
instances equal to numbers, or complex numbers with a 0 imaginary part equal to real numbers with the same real part.v/=
may not drop recursively down into, say, Clojure maps. Please open an issue if you find a case like this! -
BIG CHANGE:
Literal
andStructure
instances now KEEP their type underg/simplify
. If you want to get the expression back out of itsLiteral
wrapper, usesicmutils.expression/expression-of
, also aliased intosicmutils.env
.This means that you can no longer make comparisons like this:
-
;; this worked before, and was used all over the tests (probably not in much
;; user code!)
(clojure.core/= '(* 3 x)
(simplify (+ 'x 'x 'x)))
;;=> false
Instead, use v/=
(which is now aliased into sicmutils.env
):
;; `v/=` will do the right thing by unwrapping the literal expression on the
;; right:
(v/= '(+ x y) (+ 'x 'y))
;;=> true
-
#207 fixes a bug where
sicmutils.function/compose
would fail when provided with no arguments. Now it appropriately returnsidentity
. -
#310:
g/make-rectangular
andg/make-polar
now return non-Complex numbers on the JVM if you pass0
for the (respectively) imaginary or angle components. Previously this behavior only occurred on an integer 0; now it happens with 0.0 too, matching CLJS. -
#308 and #310:
->infix
now renders any symbol named as an upper and lowercase greek characters ('alpha
,'Phi
etc) as their proper unicode characters.'ldots
renders to '...', and'ell
renders to a pretty "ℓ", matching the TeX renderer.
(If you have any questions about how to use any of the following, please ask us at our Github Discussions page!)
This release contains a few correctness fixes, a number of new
sicmutils.generic
function implementations contributed by @pangloss, and a
large expansion of the namespaces available to SCI-hosted environments.
The themes of the release are:
- Many new functions and functionality for existing types
- Upgraded rendering for forms like nested partials
- Better and better documentation!
- Easier interop with interactive hosts via SCI
A major goal was to develop SICMUtils into an environment that could host all of the exercises from the SICM textbook. We have many of those hosted at the https://github.com/sicmutils/sicm-exercises repository, and they all work and generate correctly rendered TeX!
The goals for the next release roll over from 0.15.0:
we'll focus on getting SICMUtils integrated with 2D and 3D rendering libraries like three.js, babylon.js and Quil. The long-term goal is for SICMUtils to support the sort of workflow I described in "The Dynamic Notebook".
Out-of-the-box charting and animation primitives are missing and sorely needed. Onward!
Detailed release notes:
-
#278 adds new generic
floor
,ceiling
,integer-part
andfractional-part
generic functions, along with:- implementations for all types in the numeric tower - ratios, integers,
reals, complex, and
Differential
(so derivatives of these functions work!) - symbolic expression implementations
- symbolic implementations for
modulo
andremainder
- new support for these four generics plus
modulo
andremainder
in function compilation viasicmutils.expression.compile
(#295) - rendering support by
->infix
,->TeX
,->Javascript
(#295)
Thank you to @pangloss for this major contribution!
- implementations for all types in the numeric tower - ratios, integers,
reals, complex, and
-
division between two
Structure
instancesa
andb
now (as of #297) returns a new structure instance(/ a b)
that matches the contract(= a (* b (/ a b)))
. Previously, the division would not necessarily contract withb
to returna
, resulting in problems with the double pendulum exercise of #296. -
#149 adds a
sicmutils.modint/modint?
predicate, andsicmutils.modint/chinese-remainder
. The latter efficiently performs the Chinese Remainder algorithm for solving systems of linear congruences. Available viasicmutils.env/chinese-remainder
. -
clojure.lang.Var
implements thesicmutils.value/Value
protocol, allowing it to respond appropriately with its name tov/freeze
(#298). -
Install
sicmutils.generic/{quotient,modulo,remainder,partial-derivative}
intosicmutils.env
(#273). Thanks to @pangloss for pointing out that these were missing! -
#284 added:
-
new functions
sicmutils.mechanics.lagrange/acceleration-tuple
for creating the acceleration entry in a local tuple -
sicmutils.mechanics.lagrange/acceleration
for extracting the acceleration component of a local tuple -
An upgraded
sicmutils.mechanics.lagrange/F->C
to handle local tuples of arbitrary length. This version ofF->C
is more general than the version from the textbook that was previously included.
These are all aliased in
sicmutils.env
, along with a newΓ-bar
alias forsicmutils.mechanics.lagrange/Γ-bar
. -
-
#282 modifies the
sicmutils.value/freeze
implementation for Clojure vector to freeze vectors into the same representation as anup
structure. This makes rendering these forms much more simple and matches thescmutils
behavior. -
sicmutils.structure.Structure
implementsclojure.lang.{Indexed, IReduce}
on the JVM, allowing it to act more like a vector (#282). (The CLJS implementation already did this.)(vec (up 1 2 3))
now works correctly. -
Series
,PowerSeries
andOperator
can hold metadata and respond properly tometa
andwith-meta
(#265).sicmutils.series/{->Series, ->PowerSeries}
andsicmutils.operator/->Operator
all take a new arity for metadata.
-
g/simplify
called with an argumentx
of typeSeries
,PowerSeries
,Matrix
,Operator
,Complex
andsicmutils.abstract.function/Function
now return an instance of typex
, performing appropriate simplifications if possible. before #297 and #298, these operation would return bare symbols or sequences.A future release will make this change for
Structure
andLiteral
too, once #255 is resolved.
-
#286 adds a batch of rules to
sicmutils.simplify.rules/canonicalize-partials
that act to gather up nestedpartial
(derivative) applications into products and exponentiated partials.->TeX
and->infix
both produce better-looking forms with this change.This example shows how
g/simplify
can organize a nested application of many partial derivatives into a product:
(let [f (literal-function 'f (-> (UP Real Real) Real))]
(simplify
(((partial 0)
((partial 1)
((partial 0) f))) (up 'x 'y))))
;;=> (((* (expt (partial 0) 2) (partial 1)) f) (up x y))
-
#283 changes the default
TeX
rendering style fordown
tuples to vertical vs horizontal. -
Symbols like
'qprime
ending withprime
orprimeprime
will render asq'
orq''
respectively inTeX
, rather than the fully-spelled-out\mathsf{qprime}
(#282). -
#280 adds a new
:equation
keyword argument tosicmutils.render/->TeX
. If you pass a truthy value to:equation
, the result will be wrapped in an equation environment.:equation <string>
will insert a\\label{<string>}
entry inside the equation environment.sicmutils.env/->tex-equation
is identical to#(sicmutils.render/->TeX (g/simplify %) :equation true)
; If you pass a:label
keyword argument to->tex-equation
it will be forwarded to->TeX
, creating the expected label entry.
-
#279: Function aliases in
sicmutils.env
now properly mirror over docstrings and otherVar
metadata, thanks to Potemkin'simport-def
. This doesn't quite work in ClojureScript since we can't useresolve
inside of a macro (special form!). -
Add a proper namespace to
demo.clj
, to make it easier to use outside oflein repl
(#264).
-
#289 adds many namespaces to
sicmutils.env.sci
:-
sicmutils.{complex,expression,modint,numsymb,polynomial,ratio,rational-function,util,value}
-
sicmutils.abstract.number
-
sicmutils.expression.analyze
-
sicmutils.numerical.elliptic
-
sicmutils.util.{aggregate,stream}
-
#289 also introduces
sicmutils.function/*strict-arity-checks*
to allow the user to toggle whether or not to throw exceptions if the system thinks that arities are incompatible. It turns out that inside of an SCI environment, the usual tricks for detecting arities fail, causing errors in many expressions. To get around this,*strict-arity-checks*
is FALSE by default.
-
-
In JVM Clojure (as of #298),
sicmutils.expression.compile
defaults toclojure.core/eval
to compile functions, while ClojureScript defaults to SCI. The performance is much faster for numerical routines and worth the slightly different default behavior.To use to SCI compilation on the JVM, wrap your form in a binding:
(require '[sicmutils.expression.compile :as compile]) (binding [compile/*mode* :sci] (my-compiler-triggering-function))
To set the mode permanently, use
compile/set-compiler-mode!
:(compile/set-compiler-mode! :sci) (my-compiler-triggering-function)
The options allowed as of
0.16.0
are:sci
and:native
. -
#292 fixes a
StackOverflow
that would sometimes appear when comparing symbolic expressions to non-expressions.(= (literal-number x) y)
now returns true if(= x y)
(AND, in clj, ify
is not a collection!), false otherwise. #255 is currently blocking pass-through equality with collections on the JVM. Thanks to @daslu for the report here! -
sicmutils.modint/make
now verifies with a precondition that its two arguments are bothv/integral?
(#298). We need this constraint now thatg/modulo
is defined for more types. -
#285 fixes a bug that prevented
sin / cos
from simplifying into atan
in the numerator, and makesseq:-
slightly more efficient (closing heisenbug #151).
(If you have any questions about how to use any of the following, please ask us at our Github Discussions page!)
This release was focused on a small number of themes:
Automatic differentiation:
The goal for this cycle's automatic differentiation (AD) work was to expand the
set of functions and types that can play with AD. AD now works on functions that
return all Clojure sequences, maps, and vectors, in addition to SICMUtils types
like Operator
, Series
, PowerSeries
and Structure
. The system is now
fully extensible, so if you want to differentiate functions that return custom
records or Java collections, it's now no problem.
SICMUtils can now differentiate functions in ClojureScript that use comparison
operations like <
, =
, <=
and friends. Clojure can't quite do this yet, but
you can differentiate through v/compare
and v/=
calls.
We can also differentiate functions that return other functions with no trouble; only a few libraries can do this, and the behavior is subtle. Hold tight for comprehensive docs describing this behavior.
New Div
, Grad
, Curl
and Lap
operators build on this foundation.
SCI Integration
To support safe execution inside of a browser-based Notebook or REPL environment, SICMUtils now has full support for @borkdude's SCI, the Small Clojure Interpreter, via the sicmutils.sci namespace. Every function and macro in the library now works in SCI. (Thanks for @borkdude and @mk for your help and contributions.
Rendering
@hcarvalhoalves made a number of contributions to the LaTeX and infix renderers;
PowerSeries
and Series
now render beautifully, as well as =
and the
various inequality symbols. Expect a lot more action here as we move into more
browser-based notebook environments.
There's a lot more that went into this release; give the detailed notes below a look for more details.
What's coming next?
The next release will focus on getting SICMUtils integrated with 2D and 3D rendering libraries like three.js, babylon.js and Quil. The long-term goal is for SICMUtils to support the sort of workflow I described in "The Dynamic Notebook". This will require a big push on generic, pluggable representations for the various types and expressions in the library.
Thanks again to @hcarvalhoalves and @mk for their contributions, and to @borkdude for his help with SCI!
On to the detailed release notes:
-
New, literate
Differential
implementation lives at atsicmutils.differential
(#221) (see this page for a readable version.) Notable changes to the original impl atsicmutils.calculus.derivative
include:-
We've changed our terminology from GJS's
finite-part
,infinitesimal-part
,make-x+dx
to the more modernprimal-part
,tangent-part
,bundle-element
that the Automatic Differentiation community has adopted. His comment is that he doesn't take terms from mathematics unless he's sure that he's using it in the correct way; the safer way is to stick with his terms, but we go boldly forth with the masses. -
A new
sicmutils.differential.IPerturbed
protocol makes it possible to extend the Automatic Differentiation (AD) system to be able to handle different Functor-shaped return values, like Java or JS lists and objects. See the cljdoc page on Automatic Differentiation for more detail.-
#222 implements
d/IPerturbed
for Clojure maps, vectors and sequences; all are now valid return types for functions you pass toD
. -
#222 also implements
d/IPerturbed
for SICMUtilsMatrix
,Structure
,Series
,PowerSeries
andOperator
. -
#223 implements
d/IPerturbed
for Clojure functions and multimethods, handling the attendant subtlety that fixes "Alexey's Amazing Bug".
-
-
sicmutils.differential/{lift-1,lift-2,lift-n}
allow you to make custom operations differentiable, provided you can supply a derivative. -
Differential
implementssicmutils.function/arity
,IFn
, and can be applied to arguments if its coefficients are function values.Differential
instances alsov/freeze
andg/simplify
properly (by pushing these actions into their coefficients). -
New
compare
andequiv
implementations allowDifferential
instances to compare themselves with other objects using only their primal parts; this makes it possible to use functions like<=
,>
,=
to do control flow during automatic differentiation. (Usecompare-full
andeq
if you want to do full equality comparisons on primal and tangent components.) -
related,
g/abs
is now implemented forDifferential
instances, making this function available in functions passed toD
. -
proper
numerical?
,one?
andidentity?
implementations. The latter two only respondtrue
if there are NO tangent components; This means thatone?
and(= % 1)
will not agree. -
The new implementation fixes a subtle bug with nested, higher order automatic differentiation - it's too subtle for the CHANGELOG, so please the "amazing" bug sections in
sicmutils.calculus.derivative-test
for proper exposition.
-
-
#223 converts the implementation of
sicmutils.calculus.derivative/D
to use the newDifferential
type; this fixes "Alexey's Amazing Bug" and allowsD
to operate on higher order functions. For some functionf
that returns another function,((D f) x)
will return a function that keepsx
"alive" for the purposes of differentiation inside its body. Seesicmutils.calculus.derivative-test/amazing-bug
for an extended example. -
sicmutils.generic/partial-derivative
gains aKeyword
extension, so it can respond properly to:name
and:arity
calls (#221). -
D
(orsicmutils.generic/partial-derivative
) applied to a matrix of functions now takes the elementwise partials of every function in the matrix. (#218) -
#253 moves the derivative implementations (where relevant) onto the metadata of generic functions. You can access these by calling
(<generic-function> :dfdx)
or(<generic-function> :dfdy)
, depending on whether the generic is unary or binary. #253 also changes the name of macrosicmutils.generic/def-generic-function
tosicmutils.generic/defgeneric
.
-
sicmutils.expression/Literal
instances now usepr-str
to generate a string representation; this allows this type to wrap lazy-sequence expressions such as those returned fromg/simplify
(#259) -
sicmutils.expression.render/->infix
andsicmutils.expression.render/->TeX
now handle equality/inequality symbols (=
,>=
,>
, ...) as infix (#257). -
sicmutils.expression.render/*TeX-sans-serif-symbols*
binding to control if symbols longer than 1 char should have\mathsf
applied (#258). -
->infix
,->TeX
and->JavaScript
insicmutils.expression.render
can now accept unfrozen and unsimplifiedExpression
instances (#241). This makes it a bit more convenient to use->infix
and->TeX
at the REPL, or in a Notebook environment. Additionally, the return values of renderers are always coerced to strings. (Previously,(->infix 10)
would return a number directly.) -
up
anddown
tuples fromsicmutils.structure
gain a properprint-method
implementation (#229); these now render as(up 1 2 3)
and(down 1 2 3)
, instead of the former more verbose representation (when usingpr
.) -
sicmutils.render/->infix
andsicmutils.render/->TeX
will renderSeries
andPowerSeries
as an infinite sum (showing the first four terms). In the case of unnapliedPowerSeries
, it will represent the unbound variable as_
(#260).
sicmutils.modint
gains more efficient implementations forinverse
,quotient
,exact-divide
andexpt
on the JVM (#251).
-
beefed up the Javascript numeric tower to allow objects like
sicmutils.differential/Differential
,sicmutils.expression/Expression
and friends that WRAP numbers to compare properly using cljs-native<
,<=
,=
,>=
and>
(#236) -
new
sicmutils.value/compare
function exposed insicmutils.env
returns a valid comparison bit between native numbers and numbers wrapped inDifferential
orExpression
in both JVM Clojure and ClojureScript (#236). The behavior matchesclojure.core/compare
for all reals on the JVM; it doesn't in ClojureScript because nativecompare
can't handlegoog.math.{Long,Integer}
orjs/BigInt
.
-
#219 introduces a number of changes to
Operator
's behavior:-
Operator
is now adeftype
(not adefrecord
); the keyword lookup for its:name
,:arity
,:context
and:o
fields have been replaced by, respectively,o/name
,sicmutils.function/arity
,o/context
ando/procedure
functions. This change happened to allowOperator
to implement protocols likeILookup
. -
Native
get
andget-in
now act onOperator
. Given an operator functionf
,get
andget-in
compose#(get % k)
, or similar withf
. This deferred action matches the effect of all sicmutils generics on functions. -
Combining an operator and a non-operator via
+
and-
, the non-operator was previously lifted into an operator that multiplied itself by the new operator's argument. As of #219, this "multiplication" uses the operator definition of multiplication - meaning, the new operator composes the non-operator with its argument. Where does this matter?Previously adding the non-operator
sicmutils.function/I
to the identity operatorI
would act like this:(((g/+ o/identity f/I) f/I) 10) ;; => 110 == (+ 10 (* 10 10))
Because
f/I
multiplied itself by its argument... resulting in(* f/I f/I) == g/square
.After the change, you see this:
(((g/+ o/identity f/I) f/I) 10) ;; => 20
because
f/I
composes with its argument. -
sicmutils.operator/identity-operator
has been renamed tosicmutils.operator/identity
-
o/make-operator
now takes an explicitcontext
map, instead of a multi-arity implementation with key-value pairs. -
Operator
now implementsg/negate
. -
g/cross-product
is no longer implemented forOperator
. operators were introduced by GJS to act like "differential operators", can only add, negate and multiply (defined as composition). We will probably relax this in the future, and add more functions likeg/cross-product
that compose with the operator's output; but for now we're cleaning house, since this function isn't used anywhere. -
In this same spirit,
Operator
instances can now only be divided by scalars (not functions anymore), reflecting the ring structure of a differential operator.
-
-
#224 adds new
Div
,Grad
,Curl
andLap
operators insicmutils.calculus.derivative
and installs them intosicmutils.env
. #224 also removes theg/transpose
implementation forOperator
instances, and exposessicmutils.calculus.derivative/taylor-series
tosicmutils.env
. -
#222 adds
v/Value
implementations for Clojure sequences and maps. Maps and vectors implementf/Arity
and return[:between 1 2].
zero?and
zero-likework on sequence entries and map values. Maps can specify their
v/kindreturn value with a
:typekey, and some of the calculus implementations do already make use of this feature.
g/partial-derivative` on a Clojure Map passes through to its values. -
As of #232,
sicmutils.expression.compile/compile-univariate-fn
is nowcompile-fn
(same change for the non-cachedcompile-fn*
in the same namespace). The new implementation can compile arguments of any arity, not just arity == 1. The new version takes an arity parametern
that defaults to(sicmutils.function/arity f)
. -
sicmutils.function/arity
is now a protocol method, under thesicmutils.function/IArity
protocol (#218). In addition to functions,arity
now correctly responds to:sicmutils.matrix/Matrix
: callingarity
on a matrix assumes that the matrix has function elements; the returned arity is the most general arity that all functions will respond to.sicmutils.operator/Operator
: returns the arity of the operator's wrapped function.sicmutils.series/Series
:arity
on aSeries
assumes that the series contains functions as entries, and returns, conservatively, the arity of the first element of the series.sicmutils.series/PowerSeries
:arity
returns[:exactly 1]
, sincePowerSeries
are currently single variable.- vectors, and
sicmutils.structure/Structure
:arity
on these collections assumes that the collection contains functions as entries, and returns the most general arity that is compatible with all of the function elements.
-
New single-arity case for
sicmutils.structure/opposite
returns an identical structure with flipped orientation (#220). acts asidentity
for non-structures. -
Added missing
identity?
,identity-like
for complex and rational numbers (#236) -
sicmutils.env/ref
now accepts function and operators (#219).(ref f 0 1)
, as an example, returns a new functiong
that acts likef
but calls(ref result 0 1)
on the result. -
The slightly more general
sicmutils.env/component
replacessicmutils.structure/component
in thesicmutils.env
namespace (#219).((component 0 1) x) == (ref x 0 1)
. -
New functions
sicmutils.function/{get,get-in}
added that act like theclojure.core
versions; but given a functionf
, they compose#(get % k)
, or similar withf
. This deferred action matches the effect of all sicmutils generics on functions. (#218) -
sicmutils.function/I
aliasesclojure.core/identity
(#218). #219 exposesI
insicmutils.env
. -
sicmutils.env.sci
contains an SCI context and namespace mapping sufficient to evaluate all of sicmutils, macros and all, inside of an SCI environment (#216). Huge thanks to @borkdude for support and @mk for implementing this! -
sicmutils.numerical.elliptic
gains a full complement of elliptic integral utilities (#211):- Carlson symmetric forms of the elliptic integrals:
carlson-rd
,carlson-rc
,carlson-rj
(carlson-rf
was already present) - Legendre elliptic integrals of the second and third forms, as the two-arity
forms of
elliptic-e
andelliptic-pi
(elliptic-f
already existed) - the complete elliptic integrals via
elliptic-k
(first kind) and the single-arity forms ofelliptic-e
andelliptic-pi
k-and-deriv
returns a pair of the complete elliptical integral of the first form,elliptic-k
, and its derivative with respect tok
.jacobi-elliptic-functions
ported fromscmutils
and Press's Numerical Recipes
- Carlson symmetric forms of the elliptic integrals:
-
The operator returned by
sicmutils.calculus.derivative/partial
now has a proper name field like(partial 0)
, instead of:partial-derivative
(#223). -
#223 fixes a problem where
(operator * structure)
would return a structure of operators instead of an operator that closed over the multiplication.::s/structure
is now properly a::o/co-operator
, matching its status as a::f/cofunction
. -
Fix a bug where
f/arity
would throw an exception with multiple-arity functions on the JVM (#240). It now responds properly with[:between min-arity max-arity]
, or[:at-least n]
if there is a variadic case too. -
#238 converts
sicmutils.abstract.function/Function
from adefrecord
to adeftype
, fixing a subtle bug where (empty f) was getting called in a nested derivative test. -
fixed bug with
g/dimension
for row and column matrices (#214). previously they returned1
in both cases; now they return the total number of entries. -
#253 adds proper
:arglists
metadata for all generic functions.
- After the work below,
v/nullity?
renamed tov/zero?
, andv/unity?
renamed tov/one?
(#180). This affects the names listed in the CHANGELOG entries below.
-
expose
bootstrap-repl!
to ClojureScript, so that this is available in self-hosted CLJS (sicmutils#157) -
modified
infix.cljc
to wrap forms indisplaystyle
and add proper carriage returns inside structures (sicmutils#157) -
add
multidimensional-minimize
to thesicmutils.env
namespace (sicmutils#157) -
add more
sqrt
simplification rules to allow square roots to cancel out across a division boundary, with or without products in the numerator and denominator (sicmutils#160) -
fix NPE bug that appears in nelder-mead, when callback isn't supplied (sicmutils#162)
-
Add
sqrt-expand
andsqrt-contract
, to allow simplifications to push inside of square roots (sicmutils#163) -
speed up power series multiplication by skipping work when either head term is zero (sicmutils#166)
-
File moves:
sicmutils.polynomial-gcd
=>sicmutils.polynomial.gcd
sicmutils.polynomial-factor
=>sicmutils.polynomial.factor
sicmutils.rules
=>sicmutils.simplify.rules
sicmutils.analyze
=>sicmutils.expression.analyze
sicmutils.infix
=>sicmutils.expression.render
sicmutils.numerical.compile
=>sicmutils.expression.compile
-
sicmutils.env/one?
now exposes/aliasessicmutils.value/unity?
#154 -
Fixed #93 by adding an explicit
g/invert
implementation for polynomials in the rational fn namespace. The fix lives in #169. -
added
sicmutils.value/sqrt-machine-epsilon
(#170) -
fixed issues in
function.cljc
andoperator.cljc
where the ClojureScriptIFn
-invoke
arguments shadowed either thethis
operator, or some parameter name in the deftype (#169) -
g/sqrt
now maintains precision with ClojureScript's rational numbers.(g/sqrt #sicm/ratio 9/4)
for example returns#sicm/ratio 3/2
. (#168) -
g/determinant
andg/transpose
now act as identity for everything in the numeric tower, plus symbolic expressions (#168) -
sicmutils.expression.Expression
is nowsicmutils.expression.Literal
; it has a newmeta
field, and is adeftype
instead of adefrecord
. (#168)- To get the internal expression, use
x/expression-of
instead of:expression
. - to access the
type
field, usex/literal-type
instead of:type
- To get the internal expression, use
-
2-arity
g/atan
,g/cross-product
andg/gcd
now work for functions (#168) -
Literal
now responds appropriately tov/unity?
andv/nullity?
if it wraps a numerical "0" or "1".v/exact?
now returns true if the literal wraps an exact number (#168) -
x/variables-in
now works with wrapped expressions; no more need to explicitly unwrap (#168) -
x/walk-expression
renamedx/evaluate
(#168) -
The new
x/substitute
performs substitutions on an unwrapped expression (#168) -
x/compare
returns a comparator that works with unwrapped symbolic expression trees (#168). The rules are that that types have the following ordering: -
empty sequence is < anything (except another empty seq)
-
real < symbol < string < sequence
-
sequences compare element-by-element
-
Any types NOT in this list compare using hashes
-
g/transpose
now works properly for functions that act as linear maps. The defining relation is:
(= (((transpose f) g) 'x)
(g (f x)))
-
added
g/determinant
implementation to functions (#171) -
Moved all
literal-function
machinery and definitions tosicmutils.abstract.function
(#171).sicmutils.function
now contains only the generic method implementations for clojure functions and multimethods. -
Switched inheritance order for functions;
:sicmutils.abstract.function/function
(used to be:sicmutils.function/function
) now inherits from::v/function
instead of the other way around. (#171) -
Enhanced the
g/simplify
behavior for core functions that overlap with generic functions (+
,-
,*
,/
,mod
,quot
,rem
,neg?
). These now freeze to the same symbols as their generic counterparts. (#173) -
Add support for the hyperbolic trig functions
sinh
,cosh
,tanh
,atanh
,asinh
andacosh
tosicmutils.expression.render/->Javascript
. (#174) -
Add support for the hyperbolic trig functions
atanh
,asinh
andacosh
tosicmutils.expression.compile
. (#175) -
matrix.cljc
gainsm/nth-col
andm/diagonal
(#178 introduces:) -
As of #178 introduces:, we have three new kinds for matrices. Square matrices return
::m/square-matrix
, and columns and rows return::m/column-matrix
and::row-matrix
respectively. These all derive from::m/matrix
. This makes it easier to register methods or test specifically for these cases. We've also addedm/column?
andm/row?
predicates to check for these cases. -
#185 specializes all matrix operations that return power series (trig operations and
g/exp
to::square-matrix
). -
#184 modifies
v/exact?
on functions;((v/exact? f) x) == (v/exact? (f x))
now, instead of false as before.literal-function
forms now have a correctv/one-like
implementation. -
clojure Vars now respond to function algebra (#184). All functions implement
g/negative?
,g/abs
,g/quotient
,g/remainder
,g/modulo
,g/dimension
andg/exact-divide
, responding to the appropriate arities. -
sicmutils.complex/complex
can now take any real type in its constructor, vs only numbers (#184). -
modint
instances now implementv/freeze?
:(sicmutils.modint/make 1 2)
freezes to that(modint 1 2)
. (#185). -
v/eq
renamed tov/=
. (#186). -
v/zero-like
on matrices now fills entries with appropriatev/zero-like
versions of their existing types (#188) -
v/Value
gainsidentity-like
andidentity
(#188). These are aliased intosicmutils.env
. Implementations are installed on:- all numeric types, symbolic expressions,
Differential
(they return 1 of the appropriate type) - native and abstract functions, vars (they return an identity function)
- operators (return an identity operator, same as
one-like
) - matrices (identity matrix, only works with
::m/square-matrix
) Polynomial
(only works on monomials for now, returns an identity polynomial)RationalFunction
(returns the identity poly divided by unit poly, so only works on monomials by extension)ModInt
(returns the same asone-like
)Series
andPowerSeries
(returns[0 1 0 0 0 0...]
). This is slightly suspect in the case ofSeries
, sinceSeries
, unlikePowerSeries
, are general infinite sequences and not necessarily interpreted as polynomials. This decision followsscmutils
convention.
- all numeric types, symbolic expressions,
-
sicmutils.complex/I
aliasesi
(#189) -
matrix.cljc
has a newby-cols
(analogous tom/by-rows
), androw
to generate a row matrix (analagous tocolumn
). #197 Also inmatrix.cljc
:num-rows
,num-cols
access the row or column number without inspecting the deftype variables directlyfmap-indexed
, likefmap
but receivesi
andj
indices as second and third arguments.with-substituted-row
, for swapping out a single row in a matrixsubmatrix
generates a submatrix from low and high row and colsmatrix-some
renamed tosome
: make sure to use a namespace prefix to avoid clashing withclojure.core/some
.- new-matrix constructor
by-cols
(analogous toby-rows
, takes a sequence of columns) row
constructor takes a sequence of values and returns a row matrix.by-rows*
,by-cols*
,row*
andcolumn*
are non-variadic versions of those functions. If you already have a sequence of rows, columns or elements, prefer these.up->row-matrix
=>down->row-matrix
androw-matrix->up
=>row-matrix->down
. A row is analogous to adown
, so we make a change to reflect this.g/cross-product
between twodown
structures now returns adown
.make-zero
generates a zero-valued matrix of the supplied dimensions.make-diagonal
generates a diagonal matrix containing the values of the supplied sequence.m/identity-like
returns an identity matrix (given a square matrix) with entries of identical type, but set appropriately to zero or one. This is installed asv/one-like
andv/identity-like
.v/identity?
now returns true for identity matrices, false otherwise.v/one?
returnsfalse
for identity matrices! If it didn't,(* 2 (I 10))
would return2
, sinceone?
signals multiplicative identity.
-
sicmutils.structure/up
andsicmutils.structure/down
now have analogouss/up*
ands/down*
functions. These behave identically, but are non-variadic. If you already have a sequence you'd like to transform, prefer these (#197). -
sicmutils.value/kind-predicate
takes some item and returns a predicate that returns true if its argument has the same type (or inherits from it) (#197). -
sicmutils.function/arg-shift
andsicmutils.function/arg-scale
take functions and return new functions that shift and scale their arguments (respectively) by the originally supplied shifts (#197). -
sicmutils.generic/factorial
computes the factorial of the supplied integern
. (#197). -
Many new functions and constants exposed in
sicmutils.env
via #197:-pi
joinspi
as a constants:generate
,m:generate
,vector:generate
to generate matrices, structures and vectorsconstant-series
, fromseries/constant
seq:print
andseq:pprint
matrix-by-cols
,row-matrix
,v:make-basis-unit
- aliases for
sicmutils.function
'sarity
,arg-shift
,arg-scale
dimension
,factorial
aliased fromsicmutils.generic
derivative
aliased fromsicmutils.calculus.derivative
submatrix
,up->column-matrix
,down->row-matrix
,row-matrix->{down,vector}
,column-matrix->{up,vector}
aliased fromsicmutils.matrix
D-numeric
fromsicmutils.numerical.derivative
brent-min
,brent-max
,golden-section-min
,golden-section-max
nelder-mead
sum
from `sicmutils.util.aggregatekind-predicate
fromsicmutils.value
-
Structures and matrices both gain the ability to do native
get-in
,assoc-in
andempty
. These work as expected, like a potentially nested vector. (#193) -
matrix.cljc
gainsup->row-matrix
,up->column-matrix
,row-matrix->up
,column-matrix->up
(#193) -
structure.cljc
gains many features in (#193):kronecker
andbasis-unit
for generating potentially infinite basis sequences- the ability to conj new items onto a structure:
(conj (up 1 2) 3) => (up 1 2 3)
- The structure-preserving
map-chain
takes a 2-arg function and presents it with each element of a deeply nested structure, along with a vector of its "chain", the path into its location. The fn's return becomes the new item at that location. structure->prototype
generates a same-shape structure as its argument, with symbolic entries that display their location (preserving orientation).typical-object
returns a structure of the same shape and orientation ass
, generated by substituting gensymmed symbols in for each entry.compatible-zero
returns a structure compatible for multiplication withs
down to 0.transpose-outer
returns a new structure with the same orientation as the first element ofs
, filled with elements of the same orientation ass
. Each element is generating by taking the first element of each entry ins
, the the second, etc... In that sense this is similar to a traditional matrix transpose.dot-product
takes the dot product of two structures. They must be the same top-level orientation and dimension; beyond that, their entries are pairwise-multiplied and summed.inner-product
is the same, but the left structure is conjugated first.outer-product
now works multiple levels deep.vector-outer-product
andvector-inner-product
are similar, but only enforce the top-level length; all internal structures are NOT flattened and must be compatible forg/*
.compatible-for-contraction?
now searches recursively down into a structure; previously it only checked the top level.- The new
*allow-incompatible-multiplication*
dynamic variable is set totrue
by default. Set it false to force a setting where, when you multiply structures, they must be:- opposite orientation
- every element of the right entry must be compatible for contraction with the left
- structure multiplication with scalars, etc now respects ordering, just in case any multiplication is not commutative.
sicmutils.generators
now holds generators forup
,down
, andstructure
generators; these produce potentially deeply nested structures.up1
,down1
andstructure1
generate only one level deep. Mix and match! Seestructure_test.cljc
for many examples of how to use these.
literal-matrix
fn generates a symbolic matrix (sicmutils#169)literal
,literal-up
andliteral-down
generate symbolic structures (sicmutils#169)
This release (courtesy of
#168) brings
the numeric tower in line with the scmutils tower. Prior to this release, all
numbers, including complex, descended from ::x/numerical-expression
. Symbolic
expressions also derived from this type! The problem this causes is that all
of generic implementations for the number types default to the symbolic
functions.
If I don't specify a g/sec
method for numbers, for example, then (g/sec 12)
returns a symbolic (/ 1 (cos 12))
, instead of actually evaluating the
expression.
The fix comes from these changes:
-
::v/number
now means, "the numeric tower ascending from integer -> rational -> real -> complex numbers. All of these types now respondtrue
tov/number?
(prior to this release, Complex numbers did NOT!) -
::v/real
now means, "anything in the numeric tower except Complex". These all respond true tov/real?
-
::x/numeric-expression
has changed to::x/numeric
, and now means "anything that responds to::"v/number
, plus symbolic expressions, which now clearly represent any number in the numeric tower. Query for these withv/scalar?
I can now make some comments that clear up my former misunderstandings:
-
The
sicmutils.abstract.number
(I'll call thisan
here) namespace is responsible for installing generic implementations of all numeric methods for symbolic expressions and "literal numbers". -
the
an/literal-number
constructor promotes a number, symbol or symbolic expression up to:xx/numeric
, which means that any operation you perform on it will pass it through the symbolic expressions defined insicmutils.numsymb
. A few notes on these expressions:-
They will try to preserve exactness, but if they can't - ie, if you do something like
(cos (an/literal-number 2.2))
- the system will return-.588
. If you call(cos (an/literal-number 2))
, you'll get the expression(cos 2)
, preserving exactness. -
Symbols are automatically interpreted as "literal numbers".
-
The only ways to make a proper symbolic expression that works with the generics are:
- Use the explicit
an/literal-number
constructor - pass a symbol to any generic arithmetic function
- perform any unary or binary arithmetic operation on an existing symbolic expression.
- Use the explicit
-
an/abstract-number?
returns true for symbolic expressions, anything wrapped inliteral-number
or symbols. -
literal-number?
only returns true for explicitly wrapped things and symbolic expressions, not symbols. -
use
v/real?
,v/number?
andv/scalar?
to query the numeric tower.
-
-
If you want to compare literal numbers and an expression like
(an/literal-number 12)
, usev/=
. In ClojureScript, this will work with the built in=
as well, since equality is implemented with a protocol that we can extend. For example:
(v/= 12 (literal-number 12))
;;=> true
(= 12 (literal-number 12))
;; true in cljs, false in clj
If you keep the literal on the left side of =
, this will work in both systems,
since we've overridden the =
implementation for literals:
(= (literal-number 12) 12)
;;=> true in both languages
This paves the way for the other abstract types that exist in scmutils
, like
matrices, up and down tuples.
This release brings us closer to the interface provided by scmutils
.
PR #193 brings:
g/dot-product
, for scalars, differentials, structures, functions and row/column matricesg/inner-product
for scalars, structures, functions and row/column matricesg/outer-product
for functions, structures of length 3 and matrices, between a row and a column onlyg/cross-product
now works for row/column matrices in addition to structures (and functions that accept these)
PR sicmutils#169 brings:
g/exp2
,g/exp10
for exponents with base 2 and 10g/log2
, for base 2 logarithmsg/log10
for base 10 logsg/gcd
andg/lcm
are now exposed insicmutils.env
#178 introduces:
g/dimension
for scalars (always 1), structures and matrices (square, column and row)g/trace
returns the trace for square matrices and square structures
We now expose the following additional trigonometric functions in
sicmutils.generic
(courtesy of
sicmutils#154):
cosh
: hyperbolic cosinesinh
: hyperbolic sinetanh
: hyperbolic tangent, ie sinh/coshsech
: hyperbolic secant, ie 1/coshcsch
: hyperbolic secant, ie 1/sinhacosh
: inverse hyperbolic cosine, ie,(= x (cosh (acosh x)))
asinh
: inverse hyperbolic sine, ie,(= x (sinh (asinh x)))
atanh
: inverse hyperbolic tangent, ie,(= x (tanh (atanh x)))
These three methods existed in sicmutils.env
, but not as extensible generics.
Now they're fully extensible:
cot
: cotangent, ie 1/tansec
: secant, ie 1/coscsc
: cosecant, ie 1/sin
These all work with:
- real and complex numbers
- power series (missing a few implementations, operators and matrices are missing the same ones for this reason)
- matrices (square matrices return their power series expansions)
- operators (power series expansion of the operator)
- functions (where they create composition)
- symbolic expressions
- Derivatives and dual numbers! The new functions all work with
D
, the forward-mode automatic differentiation operator.
Additionally, four methods that lived in sicmutils.generic
are now exposed as
generics:
real-part
imag-part
angle
conjugate
These now work on:
-
all numeric types, including symbolic expressions.
-
functions
-
structures (only
magnitude
andconjugate
)magnitude
formerly didn't handle structures containing complex numbers by taking a proper inner product. This is fixed as of #168
-
PR #189 introduces:
g/make-rectangular
, (build a complex number from real and imaginary parts)g/make-polar
(build a complex number from radius and angle)- note that these work with real numbers, symbolic numbers, functions and any combination of these.
These work with functions, real numbers and symbolic expressions (and any mix of the three).
The main announcement for this release is ClojureScript Support!. Implementing this resulted in a few upgrades along the way:
- more powerful numerics, specifically
definite-integral
and native minimization routines - a generic numeric tower for ClojureScript
- Many more tests! The test coverage was great before, and it's stayed high as we've added new implementations.
- added explicit code coverage metrics via Codecov:
Here are more explicit details on the release.
generic.cljc
now includes a default implementation of:
expt
given a validmul
- default
sub
, given a validadd
andnegate
- default
div
, given a validmul
andinvert
Expression
andOperator
both have betterprint-method
implementations, so the repl experience feels more likescmutils
Operator
now has anexpn
method, which acts likeg/exp
on an operator but expands each term in ordern
.- many, many more tests!
Full conversion of SICMUtils to ClojureScript. All functionality from v0.12.1 now works in both Clojure and ClojureScript!
Most of the conversion was straightforward. The major missing piece was a numeric tower implementation for ClojureScript (complex numbers, ratios) that bring it up to parity with Clojure:
- Add the
bigfraction
implementation from fraction.js sicmutils.ratio for cross-platform ratio support (#99) - Adds CLJS complex number support through complex.js (#41)
js/BigInt
,goog.math.Long
andgoog.math.Integer
implementations round out the numeric tower (#45)
The numerical routines in SICMUtils depend heavily on Apache Commons, which of course only exists in Java. We had to implement much of the numerics code in native Clojure. It's fast, efficient and functional. Give it a read if you're curious about how these algorithms work.
-
New, native minimization routines have replaced the Apache Commons implementations:
-
Univariate Minimizers
- Port scipy's auto-bracketing + scmutils version (#104)
- Port golden section search from scipy (#105)
- Implement Brent's method for fn minimization in native clj (#106)
-
Multivariate
- pure Clojure implementation of Nelder-Mead (#102)
-
-
Native
definite-integral
numerics implementation, written as a series of computational essays:-
Basics:
- Riemann Sums, all the way up through efficient, incremental, "accelerated" versions of these easy-to-understand methods:
- Midpoint method, same development but shorter since it reuses functional abstractions. Also incremental, efficient, accelerated
- Trapezoid Method, same idea but for closed intervals.
-
Sequence Acceleration / Extrapolation Methods
- Polynomial interpolation: the general thing that "richardson extrapolation" is doing below. Historically cool and used to accelerate arbitrary integration sequences
- Rational Function extrapolation: used in bulirsch-stoer integration and ODE solving.
- "Richardson extrapolation" is a special case, where we get more efficient by assuming that the x values for the polynomial interpolation go 1, 1/2, 1/4... and that we're extrapolating to 0.
-
Higher-order Calculus:
- Numerical derivatives: derivatives using three kinds of central difference formulas... accelerated using Richardson extrapolation, with a nice technique for guarding against underflow.
- Simpson's Method... fit a parabola to every slice. OR, "accelerate" the trapezoid method with one step of Richarsdson extrapolation!
- Simpson's 3/8 Method: Same idea, but accelerate a sequence that triples its slices every iteration.
- Boole's Rule: trapezoid method plus two steps of Richardson extrapolation. (Are you starting to see the pattern??)
- Romberg Integration: midpoint OR trapezoid, with as many steps of Richardson extrapolation as we can take!
- Milne's Rule, MIDPOINT method, one step of extrapolation!
- Bulirsch-Stoer integration... midpoint or trapezoid, with rational function extrapolation, as many steps as we can handle AND some custom step sizes.
-
Combinators:
- Variable Substitutions: implemented as functional wrappers that take an integrator and return a modified integrator.
- Improper Integrals: a template for a combinator that enables infinite endpoints on any integrator, using variable substitution on an appropriate, tunable range.
- Adaptive Integration: a combinator that turns any of the integrators above into an "adaptive" integrator that's able to focus in on difficult regions.
-
And finally, "Numerical Quadrature", the namespace/essay that ties it all together.
-
-
sicmutils.numerical.compile
uses SCI, the Small Clojure Interpreter, to generate compiled numerical code (#133) -
Implemented ODE solving using @littleredcomputer's odex-js library (#135)
data_readers.cljc provides 3 new data reader literals:
#sicm/ratio
Use this with a ratio literal, like #sicm/ratio 1/2
, or with a string like
#sicm/ratio "1/4"
. If the denominator is 1
this literal will return a
js/BigInt
in ClojureScript, or a Long in Clojure.
#sicm/bigint
Use with a number literal, like, #sicm/bigint 10
, or a string like
#sicm/bigint "10000012"
to generate a js/BigInt
in ClojureScript, or a
clojure.lang.BigInt
in Clojure.
#sicm/complex
Currently this only works with a string like #sicm/complex "1 + 2i"
. In the
future it might work with a pair of (real, complex)
, like:
#sicm/complex [1 2]
The Power Series implementation in series.cljc
received an overhaul. The
implementation now follows Doug McIlroy's beautiful paper, "Power Series, Power
Serious".
Doug also has a 10-line version in Haskell on his
website.
The API now offers two types:
-
Series
, which represents a generic infinite series of arbitrary values, and -
PowerSeries
, a series that represents a power series in a single variable; in other words, a series where the nth entry is interpreted as the coefficient of$x^n$ :$$[a b c d ...] ==
$a + bx + cx^2 + dx^3 + ...$ $
series/series?
responds true to both. series/power-series?
only responds
true to a PowerSeries
.
To turn a PowerSeries
into a Series
, call it as a function with a single
argument, or pass the series and one argument to series/value
to evaluate the
series using the above equation.
To turn a Series
into a PowerSeries
, call series/->function
. None of the
functions discussed below can mix series types; you'll have to do the conversion
explicitly.
Each type supports the following generic operations:
*
,+
,-
,/
between series and non-seriesg/negate
,g/invert
,g/sqrt
,g/expt
work as expected.g/add
between series and non-series
PowerSeries
additionally supports:
g/exp
,g/cos
,g/sin
,g/asin
,g/tan
g/partial-derivative
, soPowerSeries
works well withD
Each of these acts as function composition for the single variable function that
the PowerSeries
represents. If s
is a PowerSeries
that applies as (s x)
,
(g/exp s)
returns a series that represents (g/exp (s x))
.
There are many more new methods (see the namespace for full documentation):
-
starting-with
renamed toseries
-
power-series
, analogousseries
but generates aPowerSeries
-
series*
andpower-series*
let you pass an explicit sequence -
series/take
removed in favor ofclojure.core/take
, since both series objects act as sequences -
generate
takes an additional optional argument to distinguish between series and power series -
Series
now implements more ofv/Value
- new
zero
,one
,identity
constants -
constant
returns a constant power series -
xpow
generates a series representing a bare power ofx
-
->function
turns aSeries
into aPowerSeries
-
value
,fmap
now handles bothSeries
andPowerSeries
-
(inflate s n)
expands each term$x^i$ ofs
to$x^{in}$ -
compose
returns the functional composition of twoPowerSeries
-
revert
returns the functional inverse of twoPowerSeries
-
integral
returns a series representing the definite integral of the suppliedPowerSeries
, 0 => infinity (optionally takes an integration constant)
The namespace also provides many built-in PowerSeries
instances:
exp-series
sin-series
cos-series
tan-series
sec-series
asin-series
acos-series
atan-series
acot-series
sinh-series
cosh-series
tanh-series
asinh-series
atanh-series
log1+x-series
log1-x-series
binomial-series
And two Series
(non-power):
fib-series
, the fibonacci sequencecatalan-series
, the Catalan numbers
::matrix
gained implementations for exp
, cos
, sin
, asin
, tan
,
acos
, asin
; these now return taylor series expansions of the operator, where
multiplication is composition as before.
Operator
gained implementations for cos
, sin
, asin
, tan
, acos
,
asin
; these now return taylor series expansions of the operator, where
multiplication is composition as before.
- Getting Github releases up to parity with the most recent release to Clojars.
- Did some refactoring and one breaking rename (Struct became Structure, since we don't abbreviate other deftypes). This also marks the point of departure for working with Functional Differential Geometry.
- This is the version that was current as of the talk @littleredcomputer gave at Clojure/west 2017, entitled "Physics in Clojure."