aboutsummaryrefslogtreecommitdiffhomepage
diff options
context:
space:
mode:
authorGravatar Kurtis Rader <krader@skepticism.us>2016-05-02 20:53:48 -0700
committerGravatar Kurtis Rader <krader@skepticism.us>2016-05-03 19:29:04 -0700
commitd97c22df2daaeca6c2cec7f513babb667618f1f0 (patch)
tree736ca60214fe8745b4cb01e23921b1cc5e542479
parent3ad5c7c2890dca8e75bd116c17d7d9fe5c9ec608 (diff)
add floating point output to `math` command
This makes it easy for the user to request floating point output with the desired number of digits after the decimal point (not to be confused with significant digits). Note that this is just a thin wrapper so someone can say `math -s3 10 / 3` rather than `math "scale=3; 10 /3"`. Resolves #1643
-rw-r--r--doc_src/math.txt20
-rw-r--r--share/functions/math.fish57
-rw-r--r--tests/math.err0
-rw-r--r--tests/math.in9
-rw-r--r--tests/math.out8
-rw-r--r--tests/math.status1
6 files changed, 74 insertions, 21 deletions
diff --git a/doc_src/math.txt b/doc_src/math.txt
index 994f1515..a22b0e6e 100644
--- a/doc_src/math.txt
+++ b/doc_src/math.txt
@@ -1,9 +1,8 @@
-
\section math math - Perform mathematics calculations
\subsection math-synopsis Synopsis
\fish{synopsis}
-math EXPRESSION
+math [-sN] EXPRESSION
\endfish
\subsection math-description Description
@@ -12,9 +11,26 @@ math EXPRESSION
For a description of the syntax supported by math, see the manual for the bc program. Keep in mind that parameter expansion takes place on any expressions before they are evaluated. This can be very useful in order to perform calculations involving shell variables or the output of command substitutions, but it also means that parenthesis have to be escaped.
+The following options are available:
+
+- `-sN` Sets the scale of the result. `N` must be an integer and defaults to zero. This simply sets bc's `scale` variable to the provided value. Note that you cannot put a space between `-s` and `N`.
+
+\subsection return-values Return Values
+
+If invalid options or no expression is provided the return `status` is two. If the expression is invalid the return `status` is three. If bc returns a result of `0` (literally, not `0.0` or similar variants) the return `status` is one otherwise it's zero.
\subsection math-example Examples
`math 1+1` outputs 2.
`math $status-128` outputs the numerical exit status of the last command minus 128.
+
+`math 10 / 6` outputs `1`.
+
+`math -s0 10.0 / 6.0` outputs `1`.
+
+`math -s3 10 / 6` outputs `1.666`.
+
+\subsection math-cautions Cautions
+
+Note that the modulo operator (`x % y`) is not well defined for floating point arithmetic. The `bc` command produces a nonsensical result rather than emit an error and fail in that case. It doesn't matter if the arguments are integers; e.g., `10 % 4`. You'll still get an incorrect result. Do not use the `-sN` flag with N greater than zero if you want sensible answers when using the modulo operator.
diff --git a/share/functions/math.fish b/share/functions/math.fish
index c190ec63..ed638c69 100644
--- a/share/functions/math.fish
+++ b/share/functions/math.fish
@@ -1,23 +1,42 @@
-
function math --description "Perform math calculations in bc"
- if count $argv >/dev/null
- switch $argv[1]
- case -h --h --he --hel --help
- __fish_print_help math
- return 0
- end
+ set -l scale 0 # default is integer arithmetic
+
+ if set -q argv[1]
+ switch $argv[1]
+ case '-s*' # user wants to specify the scale of the output
+ set scale (string replace -- '-s' '' $argv[1])
+ if not string match -q -r '^\d+$' "$scale"
+ echo 'Expected an integer to follow -s' >&2
+ return 2 # missing argument is an error
+ end
+ set -e argv[1]
+
+ case -h --h --he --hel --help
+ __fish_print_help math
+ return 0
+ end
+ end
+
+ if not set -q argv[1]
+ return 2 # no arguments is an error
+ end
+
+ # Stitch lines together manually. We can't rely on BC_LINE_LENGTH because some systems don't
+ # have a new enough version of bc.
+ set -l out (echo "scale=$scale; $argv" | bc | string replace -r '\\\\$' '' | string join '')
+ switch "$out"
+ case ''
+ # No output indicates an error occurred.
+ return 3
- # Stitch lines together manually
- # we can't rely on BC_LINE_LENGTH because some systems don't have a bc version "new" enough
- set -l out (echo $argv | bc | string replace -r '\\\\$' '' | string join '')
- test -z "$out"; and return 1
- echo $out
- switch $out
- case 0
- return 1
- end
- return 0
- end
- return 2
+ case 0
+ # For historical reasons a zero result translates to a failure status.
+ echo 0
+ return 1
+ case '*'
+ # For historical reasons a non-zero result translates to a success status.
+ echo $out
+ return 0
+ end
end
diff --git a/tests/math.err b/tests/math.err
new file mode 100644
index 00000000..e69de29b
--- /dev/null
+++ b/tests/math.err
diff --git a/tests/math.in b/tests/math.in
new file mode 100644
index 00000000..d8cb5092
--- /dev/null
+++ b/tests/math.in
@@ -0,0 +1,9 @@
+math 3 / 2
+math 10/6
+math -s0 10 / 6
+math -s3 10/6
+math '10 % 6'
+math -s0 '10 % 6'
+math '23 % 7'
+math -s6 '5 / 3 * 0.3'
+true
diff --git a/tests/math.out b/tests/math.out
new file mode 100644
index 00000000..0a46b6a5
--- /dev/null
+++ b/tests/math.out
@@ -0,0 +1,8 @@
+1
+1
+1
+1.666
+4
+4
+2
+.499999
diff --git a/tests/math.status b/tests/math.status
new file mode 100644
index 00000000..573541ac
--- /dev/null
+++ b/tests/math.status
@@ -0,0 +1 @@
+0