Description
(I am aware of issue #100 but I think it is about something else)
I think we need a better interface for rand
on e.g. polynomial rings and other of our datastructures. The problem with the current system is that it is kinda hard to discover how to use it, and even if you know, it requires a lot of guesswork. At least in my experience.
Here is an example of what we have right now:
julia> R, x = polynomial_ring(QQ, "x" => 1:3);
julia> rand(R, 1:3, 4:6, 7:9)
x[1]^6*x[2]^6*x[3]^4 + x[1]^6*x[2]^5*x[3]^6 + 9//8*x[1]^4*x[2]^4*x[3]^4
julia> rand(R, 1:3, 4:6, 7:9)
x[1]^5*x[2]^6*x[3]^5
julia> rand(R, 1:3, 4:6, 7:9)
x[1]^4*x[2]^4*x[3]^5 + 7//9*x[1]^4*x[2]^4*x[3]^4
So rand
has to be called with three ranges (or vectors): the first indicates the acceptable number of terms, the second indicates the allowed range of degree of each variable, the final one is used to determine which denominators and numerators the coefficients may have (I think).
The logic is clear in principle: the first two ranges control what is specific to polynomial rings; everything after controls the randomness of the coefficient ring. This can be nested then:
julia> S, y = R[:y1, :y2];
julia> rand(S, 1:3, 4:6, 7:9, 10:12, 13:15)
(x[1]^12*x[2]^12*x[3]^12 + 15//13*x[1]^12*x[2]^11*x[3]^10 + 14//15*x[1]^11*x[2]^12*x[3]^12 + 14//15*x[1]^11*x[2]^10*x[3]^11 + 405//182*x[1]^10*x[2]^12*x[3]^11 + 14//15*x[1]^10*x[2]^10*x[3]^12 + 421//210*x[1]^10*x[2]^10*x[3]^11)*y1^6*y2^6 + (x[1]^12*x[2]^11*x[3]^11 + 379//182*x[1]^12*x[2]^11*x[3]^10 + x[1]^11*x[2]^12*x[3]^12 + x[1]^11*x[2]^10*x[3]^12 + 14//15*x[1]^11*x[2]^10*x[3]^11 + x[1]^10*x[2]^11*x[3]^12 + 14//15*x[1]^10*x[2]^10*x[3]^11)*y1^5*y2^6 + (15//13*x[1]^12*x[2]^12*x[3]^12 + 391//210*x[1]^12*x[2]^12*x[3]^10 + 15//14*x[1]^12*x[2]^11*x[3]^11 + x[1]^11*x[2]^11*x[3]^12 + x[1]^11*x[2]^11*x[3]^11 + 13//15*x[1]^10*x[2]^11*x[3]^10 + 13//14*x[1]^10*x[2]^10*x[3]^12)*y1^4*y2^6
(In case you wonder about 405//182
: well if some terms are generated multiple times with different coefficients, they are of course merged...)
While this is "logical", I still find it difficult to use, and worse: to read in code using it without trying it out to verify.
It is also somewhat limited, e.g. there is no way to e.g. "get a random polynomial where all terms have total degree 4 to 6, and coefficients in the range -10:10
".
I propose that we design and switch to a new system that relies on kwargs to become more "self documenting". Here is a first example of how it might look like in the above examples; it's just a first draft to convey the idea and get the discussion going, so I am happy to hear if others have different and better ideas :-)
I'll give these in the form of examples, without explaining much what they mean, in the hopes that they are mostly self-documenting. These lists are not meant to be exhaustive nor do I insist all variants are needed (or even sensible), I am more concerned about the "flavor" here.
For QQ
we might offer:
rand(QQ; denominator=1:10, numerator=-5:5)
rand(QQ; denominator=10, value=-5:5)
rand(QQ; nbits=1:10)
rand(QQ; nbits=1:10)
rand(QQ, 7:9)
-- for backwards compatibility??rand(QQ)
-- either returns random values wrt some documented default values, or produces an error explaining how to use it / pointing to the docs
For GF(p)
we might offer
rand(GF(p))
rand(GF(p), non_zero=true)
rand(GF(p), value=1:p-1)
For a polynomial ring, specify using coefficients
the parameters to pass on to the rand
function on the coefficient ring. I envision this is either a named tuple (passed as kwargs), or a tuple (passed as regular args) or some other value (passes as a single arg).
So e.g. if the coefficient ring is QQ
then we might offer:
rand(R; total_degree=4:6, coefficients=(denominator=1:10, numerator=-5:5))
rand(R; nterms=1:3, degree=4:6, coefficients=7:9)
-- invokesrand(QQ, 7:9)
rand(R)
-- either uses some default settings, or raises a helpful error
If the coefficient ring is another polynomial ring over QQ
, one can recurse this, e.g.:
rand(R; total_degree=4:6, coefficients=(nterms=1:3, degree=4:6, coefficients=7:9))
Similar strategies should work for matrix spaces, matrix algebras, group rings, Lie algebras, matrix groups, power series, ...
Perhaps there are structures where this is not a good fit, in that case it would be good to know about them, so please list them.