-
Notifications
You must be signed in to change notification settings - Fork 8
/
decimal.ex
116 lines (80 loc) · 2.79 KB
/
decimal.ex
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
defmodule RDF.Decimal do
@moduledoc """
`RDF.Datatype` for XSD decimal.
"""
use RDF.Datatype, id: RDF.Datatype.NS.XSD.decimal
import RDF.Literal.Guards
alias Elixir.Decimal, as: D
@impl RDF.Datatype
def convert(value, opts)
def convert(%D{coef: coef} = value, opts) when coef in ~w[qNaN sNaN inf]a,
do: super(value, opts)
def convert(%D{} = decimal, _),
do: canonical_decimal(decimal)
def convert(value, opts) when is_integer(value),
do: value |> D.new() |> convert(opts)
def convert(value, opts) when is_float(value),
do: value |> D.from_float() |> convert(opts)
def convert(value, opts) when is_binary(value) do
if String.contains?(value, ~w[e E]) do
super(value, opts)
else
case D.parse(value) do
{:ok, decimal} -> convert(decimal, opts)
:error -> super(value, opts)
end
end
end
def convert(value, opts), do: super(value, opts)
@impl RDF.Datatype
def canonical_lexical(value)
def canonical_lexical(%D{sign: sign, coef: :qNaN}),
do: if sign == 1, do: "NaN", else: "-NaN"
def canonical_lexical(%D{sign: sign, coef: :sNaN}),
do: if sign == 1, do: "sNaN", else: "-sNaN"
def canonical_lexical(%D{sign: sign, coef: :inf}),
do: if sign == 1, do: "Infinity", else: "-Infinity"
def canonical_lexical(%D{} = decimal),
do: D.to_string(decimal, :normal)
def canonical_decimal(%D{coef: 0} = decimal),
do: %{decimal | exp: -1}
def canonical_decimal(%D{coef: coef, exp: 0} = decimal),
do: %{decimal | coef: coef * 10, exp: -1}
def canonical_decimal(%D{coef: coef, exp: exp} = decimal)
when exp > 0,
do: canonical_decimal(%{decimal | coef: coef * 10, exp: exp - 1})
def canonical_decimal(%D{coef: coef} = decimal)
when Kernel.rem(coef, 10) != 0,
do: decimal
def canonical_decimal(%D{coef: coef, exp: exp} = decimal),
do: canonical_decimal(%{decimal | coef: Kernel.div(coef, 10), exp: exp + 1})
@impl RDF.Datatype
def cast(literal)
def cast(%RDF.Literal{datatype: datatype} = literal) do
cond do
not RDF.Literal.valid?(literal) ->
nil
is_xsd_decimal(datatype) ->
literal
literal == RDF.false ->
new(0.0)
literal == RDF.true ->
new(1.0)
is_xsd_string(datatype) ->
literal.value
|> new()
|> canonical()
|> validate_cast()
is_number(literal.value) and (is_xsd_integer(datatype) or
is_xsd_double(datatype) or is_xsd_float(datatype)) ->
new(literal.value)
true ->
nil
end
end
def cast(_), do: nil
@impl RDF.Datatype
def equal_value?(left, right), do: RDF.Numeric.equal_value?(left, right)
@impl RDF.Datatype
def compare(left, right), do: RDF.Numeric.compare(left, right)
end