| infoA | Report mcp-abacus server availability, version, and environment information. toolsets lists the opt-in tool groups active in this build. It is EMPTY today —
every tool registers unconditionally — and is reserved for the future toolset
gating (SA.1); it is reported now so a client can read the active set once that
lands.
|
| helpA | Return mcp-abacus reference text for one section, to drive the evaluator. Sections: 'types' (the numeric types this build supports), 'language' (the
expression grammar — operators, precedence, literal forms), 'functions' (the
callable functions and their argument counts), and 'solver' (the solver tool —
solving / optimising one variable over a bracket). section is restricted to
these four names — advertised as a schema enum — so any other value is rejected
with the valid list. |
| calculateA | Evaluate an expression (or short program) in one numeric type; return value + precision. Use `calculate` when you want the VALUE of an expression. To instead see WHERE a
surprising answer rounded or overflowed — the per-node parse tree with each
sub-result — use `analyze`; to find the variable value(s) that drive an
expression to a root or extremum, use `solver`. All three share this expression
language and `mode`/`min_fixed_point_precision` arguments.
`mode` is the numeric type the WHOLE calculation runs in — every intermediate
result behaves exactly as that type would, so float rounding, fixed-point
scale, and rational exactness each show through. Modes:
fixed-point (default) exact scaled integer; money / ERC-20-safe; alias decimal
floating-point IEEE-754 double; ~15-17 sig. digits; aliases float64, double, float, ieee754
rational exact numerator/denominator; no irrationals; aliases fraction, frac
Grammar. Binary `+ - * / // %`; unary prefix `+ - ~`; `**` is POWER,
right-assoc, binds tighter than unary minus: -2**2 == -(2**2). Bitwise
`& | ^` (^ is XOR, NOT power) and `~` (NOT) work in EVERY type, on its own
stored bits (float's 64-bit IEEE pattern, fixed-point's mantissa, rational's
numerator/denominator). Both operands
of a binary op must share ONE type — there is no implicit promotion. Group
with `( )`. Functions: call as `name(arg, ...)` — e.g. `sqrt`, `sin`, `sum`;
each argument evaluates in the active type. For the full set and their
argument counts call `help('functions')`. Constants: `pi` and `e` are usable
bare (no parentheses), e.g. `2*pi`; assigning to them is an error. Literals: decimals
`12 3.14 .5 1e3 2.5e-4`; base integers
`0x1F 0b1010 0o17`; fixed-point `M@D == M x 10^-D`, where M MUST be
base-prefixed (0x/0o/0b) — a DECIMAL mantissa is INVALID: both `123@2` and
`123.45@2` error; write a decimal value as its digits (e.g. 123.45), never
with `@`. (`0x59682F00@9` = 1.5, `0xDE0B6B3A7640000@18` = 1 ETH.)
Variables & multi-line programs. Assign with `name = expr` (name is an
identifier `[A-Za-z_][A-Za-z0-9_]*`); a bare `name` reads it back, and reading
a name that was never assigned is an error. An assignment is itself an
expression — its value is the right-hand side — so `x = 2 + 3` returns 5 and
also binds `x`. Pass SEVERAL statements as one `expression` by separating them
with NEWLINES (`
): they run top to bottom sharing one variable scope, so a later line sees earlier bindings. EVERY bare-expression line is answered, in source order, in the valuesarray (below); assignment lines run silently for their bindings and are NOT echoed — except the final line, which is always the program's result and so is always echoed. When more than one line is answered, the top-level valueis a multi-line transcript, one = per answered line; a single answered line keepsvalueas just that result. E.g. "x = 10
y = x * 2
y + 1"returns 21 (onlyy + 1is a bare line), while "x = 10
(x - 1) / (x + 1)
(x + 1) / 100.0"answers both divisions. Scope lasts for the one call only — bindings do not carry over to the nextcalculate`.
Returns a dict: `value` is the result rendered as a string and ANNOTATED with
its precision verdict — "(exact)" when the result is the true value, else
"(inexact, rounded to N decimals)" — so e.g. `/` cannot silently mislead by
looking exact when it rounded. `value_hex_dump` is that same value in hex (the
bit-backed representation): fixed-point as M@D (mantissa in whole-byte hex,
`@scale` dropped at scale 0), floating-point as the raw 64-bit IEEE-754
pattern, and NULL in rational mode (a numerator/denominator pair has no single
integer to dump). `mode` is the RESOLVED numeric type the call ran in — always
the canonical name even when you passed an alias (e.g. "double" reads back as
"floating-point"), so the reply stands on its own. The exactness/scale facts
are also returned as separate fields: `exact` (bool — did the mode hold the
true value) and `precision` (the fixed-point decimal scale, or null when the
mode has none). NOTE: floating-point conservatively reports `exact: false` for
every result today, including ones a double holds exactly. `values` is the per-line
breakdown: an array with one object per answered line, in source order, each
`{source, value, value_hex_dump, exact, precision, offered_precision}` — `source` is
the re-rendered expression and the other fields mirror the top-level ones for that one
line. The LAST entry is the program's result, so its fields equal the top-level
`value_hex_dump`/`exact`/`precision`/`offered_precision` (and, for a single answered
line, the top-level `value`). On failure `value`/`value_hex_dump`/`mode`/`exact`/
`precision`/`offered_precision`/`values` are null and `error` carries a plain,
self-contained message — what went wrong (a malformed expression, a domain error,
or an unknown mode with the valid list). It reads as prose, not a log line; only
the inexact-abort diagnostic names its source line, and in prose. On success
`error` is null. For the full reference call `help`.
`min_fixed_point_precision` floors the fixed-point result at that many decimal
places: every operand is held at no fewer than that many fractional digits, so
a `/` that would otherwise round at scale 0 keeps more decimals. It is valid
ONLY in fixed-point mode (the other modes have no decimal scale) and must be a
non-negative integer; either violation is an `error`. Omit it (null) for no
floor.
`inexact_handling` chooses what happens when a result is INEXACT:
continue-and-report (default) evaluate normally and let the verdict surface
in `value`/`exact`; never reject. Aliases: continue, report.
abort-on-inexact stop and FAIL the moment any sub-result is inexact. The
call returns no value — `error` instead carries a
diagnostic naming the source line and laying out the
operation that went inexact in computed VALUES (e.g.
`1.00 / 3.00 = 0.33`), then how to enable inexact
calculations if you do want the rounded answer. Use it
when an approximate answer is unacceptable and you want to
be told precisely what and where, rather than silently
trusting a rounded figure. Aliases: abort, strict, exact-only.
An unknown value is an `error` listing the valid choices. Note floating-point
reports every result inexact, so abort-on-inexact there fails on the first value.
`offered_precision` is a what-if nudge, present (non-null) ONLY on an inexact
fixed-point result when you did NOT pass min_fixed_point_precision: it shows the
SAME expression at a few more decimals so you see the digits the rounding hid.
It is a nested object `{mode, min_fixed_point_precision, value, value_hex_dump,
exact}` mirroring the top-level reply — its `mode` is always "fixed-point", its
min_fixed_point_precision is the argument to pass to GET that fuller value, its
`value` is what you'd get back (annotated with its OWN precision verdict, since
the offered value may itself still be inexact — e.g. 10/3 never terminates),
and `value_hex_dump` is that offered value in hex. It is NOT the answer to the
call you made (that stays in the top-level `value`); it is null whenever there
is nothing to offer. Each `values` entry carries its OWN `offered_precision` under
the same gate, so every answered line steers independently.
|
| analyzeA | Evaluate an expression and return its AST as an indented tree of sub-results. Same arguments and evaluation as calculate — mode (fixed-point default,
floating-point, rational) and min_fixed_point_precision behave identically —
but instead of one final value this returns the WHOLE parse tree, each node
annotated with the Value it computed in that mode. Reach for it to see WHERE a
surprising answer comes from: which sub-expression rounded, overflowed, or lost
precision, rather than only the rounded result. For just the final value use
calculate; to find the variable value(s) that drive an expression to a root or
extremum, use solver. tree is a multi-line string, one node per line, indented by depth (root last-
applied operator at the top, literals at the leaves). Each line is
<OPCODE/LITERAL "lexeme"> Value = <value> (<type>[<scale>], <exact|inexact>)
followed by ·-separated per-mode details: the value in hex (fixed-point as
M@D with whole-byte digits, @<scale> dropped at scale 0; float as raw IEEE-754
bits), or a rational's decimal approximation. The <scale> is the fixed-point
decimal scale (omitted for modes without one). A fixed-point node that ROUNDED
its result also carries a final · rounding <residual> ≈ <approx> fragment: the
exact signed residual stored − true (a fraction, bounded by half a unit in the
last place) and its decimal approximation. It is named rounding, NOT error,
on purpose — the reply's top-level error field is the failure channel, so a
rounding label keeps "this node rounded" from being misread as "this node
failed". For example (1 + 1/2) * 3 in fixed-point:
BINARY_MUL Value = 3 (fixed-point[0], inexact) · hex 0x03
BINARY_ADD Value = 1 (fixed-point[0], inexact) · hex 0x01
LITERAL "1" Value = 1 (fixed-point[0], exact) · hex 0x01
BINARY_DIV Value = 0 (fixed-point[0], inexact) · hex 0x00 · rounding -1/2 ≈ -0.5
LITERAL "1" Value = 1 (fixed-point[0], exact) · hex 0x01
LITERAL "2" Value = 2 (fixed-point[0], exact) · hex 0x02
LITERAL "3" Value = 3 (fixed-point[0], exact) · hex 0x03
— the 1/2 = 0 leaf (inexact, scale 0) makes plain that fixed-point rounded the
half away (its rounding -1/2 is the exact half discarded), so the product is 3,
not 4.5, and every node above it inherits the inexactness. Those ancestors show no
rounding fragment: they introduced no rounding of their own, only carried the
leaf's. (Raise min_fixed_point_precision, or use a different mode, to keep those
digits.) On success tree is the rendering and error is null; on a bad mode, an invalid
min_fixed_point_precision, or a malformed/erroring expression, tree is null and
error carries the message (the same messages calculate returns). |
| solverA | Find the value(s) of one or more variables that drive an expression to a root or extremum. solver takes the SAME expression language as calculate — every operator,
function, literal form, and (crucially) multi-line programs with name = expr
assignments — but instead of evaluating the expression it SEARCHES for the value
of the unknown(s) that drive the expression to the chosen objective:
"find-root" (default): find where the expression equals zero. Write an
equation f = g as the expression f - g and find its root. "find-minimum" / "find-maximum": find where the expression reaches its
smallest / largest value within the bracket(s).
There are two input forms for the unknowns: SINGLE: variable + lower + upper — one unknown searched over the bracket
[lower, upper] (lower must be below upper). This is the default
golden-section engine. MULTIPLE: variables — a dict mapping each unknown name to its [lower, upper]
bracket, e.g. {"x": [0, 5], "y": [-4, 2]}. This needs the Nelder-Mead engine
(algorithm="nelder-mead"), which searches all the unknowns jointly.
Give exactly one of the two forms. Each unknown must OCCUR in the expression and
must NOT be assigned by it; every OTHER name is a constant the program sets via an
assignment line (e.g. "r = 0.05\np = 1000\np * (1 + r)**n - 2000" solving for
n with r, p fixed). A name that is neither an unknown nor assigned is an error.
objective (optional) names what to search for — "find-root", "find-minimum", or
"find-maximum"; omitted, it defaults to "find-root". (The older spellings solve,
minimise/maximise and their min/max and American forms are accepted too.)
algorithm (optional) names the search engine — "golden-section-search" (the
default, single-variable), "brent-parabolic" (single-variable too, parabolic
interpolation with a golden-section fallback — usually faster on smooth extrema),
or "nelder-mead" (multivariate, a bounds-clamped downhill simplex). The two
single-variable engines solve only the SINGLE form; the variables form requires
"nelder-mead". (golden, brent, simplex and a few other spellings are accepted
too.)
mode and min_fixed_point_precision behave as in calculate — the search runs
in that numeric type and the found value is reported in it — with ONE solver-only
rule: in fixed-point mode (the default) min_fixed_point_precision is REQUIRED.
Without it the search would run at scale 0, flooring the variable to whole numbers
and missing any non-integer solution, so the call is refused; pass it (e.g. 9), or
switch to floating-point / rational, which resolve sub-unit values natively and
need no floor. See calculate and help for the shared grammar and modes; if a
found value or objective looks off, analyze shows the per-node parse tree of the
expression (with the unknowns substituted) to reveal where it rounded or overflowed.
The search is bounded by a hard 2-second time limit. If it has not converged by
then it stops and reports the best value reached so far (a find-root that has not
got close enough to zero in that time is reported as no-solution, naming the limit). Returns a dict: solutions is a list of {variable, solution, solution_hex_dump},
one per unknown in input order — each solution rendered and marked "(approximate)"
(the search locates it to a tolerance, never exactly), with its bit-backed hex. For
the SINGLE form the scalar variable / solution / solution_hex_dump are also
set (the one unknown); for the MULTIPLE form those scalars are null and solutions
carries every value. value is the EXPRESSION evaluated at that solution, annotated
with its own precision verdict (near zero for find-root, the extremum otherwise),
and value_hex_dump its hex. mode is the resolved numeric type; exact and
precision describe value exactly as in calculate. objective echoes the
resolved objective, algorithm names the engine used, and iterations is how many
search steps it took. On any failure — a bad mode/precision, a malformed expression,
an invalid request (no/both input forms, empty bracket, unknown not in the
expression, golden-section asked for multiple unknowns, unknown objective or
algorithm), or no solution in the bracket — every data field is null and error
carries the message (a no-solution error reports the closest |expr| it reached); on
success error is null. |