{-# LANGUAGE DeriveAnyClass #-}
{-# LANGUAGE DeriveGeneric #-}
{-# LANGUAGE DerivingStrategies #-}
{-# LANGUAGE FlexibleContexts #-}
{-# LANGUAGE FlexibleInstances #-}
{-# LANGUAGE MultiParamTypeClasses #-}
{-# LANGUAGE MultiWayIf #-}
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TemplateHaskell #-}
{-# OPTIONS_GHC -fno-omit-interface-pragmas #-}
{-# OPTIONS_GHC -fplugin-opt PlutusTx.Plugin:debug-context #-}
{-# OPTIONS_GHC -fno-ignore-interface-pragmas #-}
module PlutusTx.Ratio(
Rational
, unsafeRatio
, fromInteger
, ratio
, numerator
, denominator
, round
, truncate
, properFraction
, recip
, abs
, negate
, half
, fromGHC
, toGHC
, reduce
, gcd
) where
import PlutusTx.Applicative qualified as P
import PlutusTx.Base qualified as P
import PlutusTx.Bool qualified as P
import PlutusTx.Eq qualified as P
import PlutusTx.ErrorCodes qualified as P
import PlutusTx.Integer (Integer)
import PlutusTx.IsData qualified as P
import PlutusTx.Lift qualified as P
import PlutusTx.Maybe qualified as P
import PlutusTx.Numeric qualified as P
import PlutusTx.Ord qualified as P
import PlutusTx.Trace qualified as P
import PlutusTx.Builtins qualified as Builtins
import Control.Monad (guard)
import Data.Aeson (FromJSON (parseJSON), ToJSON (toJSON), object, withObject, (.:))
import GHC.Real qualified as Ratio
import Prelude (Ord (..), Show, (*))
import Prelude qualified as Haskell
data Rational = Rational Integer Integer
deriving stock (
Rational -> Rational -> Bool
forall a. (a -> a -> Bool) -> (a -> a -> Bool) -> Eq a
/= :: Rational -> Rational -> Bool
$c/= :: Rational -> Rational -> Bool
== :: Rational -> Rational -> Bool
$c== :: Rational -> Rational -> Bool
Haskell.Eq,
Int -> Rational -> ShowS
[Rational] -> ShowS
Rational -> String
forall a.
(Int -> a -> ShowS) -> (a -> String) -> ([a] -> ShowS) -> Show a
showList :: [Rational] -> ShowS
$cshowList :: [Rational] -> ShowS
show :: Rational -> String
$cshow :: Rational -> String
showsPrec :: Int -> Rational -> ShowS
$cshowsPrec :: Int -> Rational -> ShowS
Show
)
instance P.Eq Rational where
{-# INLINABLE (==) #-}
Rational Integer
n Integer
d == :: Rational -> Rational -> Bool
== Rational Integer
n' Integer
d' = Integer
n forall a. Eq a => a -> a -> Bool
P.== Integer
n' Bool -> Bool -> Bool
P.&& Integer
d forall a. Eq a => a -> a -> Bool
P.== Integer
d'
instance P.Ord Rational where
{-# INLINABLE compare #-}
compare :: Rational -> Rational -> Ordering
compare (Rational Integer
n Integer
d) (Rational Integer
n' Integer
d') = forall a. Ord a => a -> a -> Ordering
P.compare (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
{-# INLINABLE (<=) #-}
Rational Integer
n Integer
d <= :: Rational -> Rational -> Bool
<= Rational Integer
n' Integer
d' = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. Ord a => a -> a -> Bool
P.<= (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
{-# INLINABLE (>=) #-}
Rational Integer
n Integer
d >= :: Rational -> Rational -> Bool
>= Rational Integer
n' Integer
d' = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. Ord a => a -> a -> Bool
P.>= (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
{-# INLINABLE (<) #-}
Rational Integer
n Integer
d < :: Rational -> Rational -> Bool
< Rational Integer
n' Integer
d' = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. Ord a => a -> a -> Bool
P.< (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
{-# INLINABLE (>) #-}
Rational Integer
n Integer
d > :: Rational -> Rational -> Bool
> Rational Integer
n' Integer
d' = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. Ord a => a -> a -> Bool
P.> (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
instance Ord Rational where
compare :: Rational -> Rational -> Ordering
compare (Rational Integer
n Integer
d) (Rational Integer
n' Integer
d') = forall a. Ord a => a -> a -> Ordering
compare (Integer
n forall a. Num a => a -> a -> a
* Integer
d') (Integer
n' forall a. Num a => a -> a -> a
* Integer
d)
Rational Integer
n Integer
d <= :: Rational -> Rational -> Bool
<= Rational Integer
n' Integer
d' = (Integer
n forall a. Num a => a -> a -> a
* Integer
d') forall a. Ord a => a -> a -> Bool
<= (Integer
n' forall a. Num a => a -> a -> a
* Integer
d)
Rational Integer
n Integer
d >= :: Rational -> Rational -> Bool
>= Rational Integer
n' Integer
d' = (Integer
n forall a. Num a => a -> a -> a
* Integer
d') forall a. Ord a => a -> a -> Bool
>= (Integer
n' forall a. Num a => a -> a -> a
* Integer
d)
Rational Integer
n Integer
d < :: Rational -> Rational -> Bool
< Rational Integer
n' Integer
d' = (Integer
n forall a. Num a => a -> a -> a
* Integer
d') forall a. Ord a => a -> a -> Bool
< (Integer
n' forall a. Num a => a -> a -> a
* Integer
d)
Rational Integer
n Integer
d > :: Rational -> Rational -> Bool
> Rational Integer
n' Integer
d' = (Integer
n forall a. Num a => a -> a -> a
* Integer
d') forall a. Ord a => a -> a -> Bool
> (Integer
n' forall a. Num a => a -> a -> a
* Integer
d)
instance P.AdditiveSemigroup Rational where
{-# INLINABLE (+) #-}
Rational Integer
n Integer
d + :: Rational -> Rational -> Rational
+ Rational Integer
n' Integer
d' =
let newNum :: Integer
newNum = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. AdditiveSemigroup a => a -> a -> a
P.+ (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
newDen :: Integer
newDen = Integer
d forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d'
gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
newNum Integer
newDen
in Integer -> Integer -> Rational
Rational (Integer
newNum Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
(Integer
newDen Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
instance P.AdditiveMonoid Rational where
{-# INLINABLE zero #-}
zero :: Rational
zero = Integer -> Integer -> Rational
Rational forall a. AdditiveMonoid a => a
P.zero forall a. MultiplicativeMonoid a => a
P.one
instance P.AdditiveGroup Rational where
{-# INLINABLE (-) #-}
Rational Integer
n Integer
d - :: Rational -> Rational -> Rational
- Rational Integer
n' Integer
d' =
let newNum :: Integer
newNum = (Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d') forall a. AdditiveGroup a => a -> a -> a
P.- (Integer
n' forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d)
newDen :: Integer
newDen = Integer
d forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d'
gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
newNum Integer
newDen
in Integer -> Integer -> Rational
Rational (Integer
newNum Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
(Integer
newDen Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
instance P.MultiplicativeSemigroup Rational where
{-# INLINABLE (*) #-}
Rational Integer
n Integer
d * :: Rational -> Rational -> Rational
* Rational Integer
n' Integer
d' =
let newNum :: Integer
newNum = Integer
n forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
n'
newDen :: Integer
newDen = Integer
d forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
d'
gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
newNum Integer
newDen
in Integer -> Integer -> Rational
Rational (Integer
newNum Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
(Integer
newDen Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
instance P.MultiplicativeMonoid Rational where
{-# INLINABLE one #-}
one :: Rational
one = Integer -> Integer -> Rational
Rational forall a. MultiplicativeMonoid a => a
P.one forall a. MultiplicativeMonoid a => a
P.one
instance P.Module Integer Rational where
{-# INLINABLE scale #-}
scale :: Integer -> Rational -> Rational
scale Integer
i (Rational Integer
n Integer
d) = let newNum :: Integer
newNum = Integer
i forall a. MultiplicativeSemigroup a => a -> a -> a
P.* Integer
n
gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
newNum Integer
d in
Integer -> Integer -> Rational
Rational (Integer
newNum Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
(Integer
d Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
instance P.ToData Rational where
{-# INLINABLE toBuiltinData #-}
toBuiltinData :: Rational -> BuiltinData
toBuiltinData (Rational Integer
n Integer
d) = forall a. ToData a => a -> BuiltinData
P.toBuiltinData (Integer
n, Integer
d)
instance P.FromData Rational where
{-# INLINABLE fromBuiltinData #-}
fromBuiltinData :: BuiltinData -> Maybe Rational
fromBuiltinData BuiltinData
dat = do
(Integer
n, Integer
d) <- forall a. FromData a => BuiltinData -> Maybe a
P.fromBuiltinData BuiltinData
dat
forall (f :: * -> *). Alternative f => Bool -> f ()
guard (Integer
d forall a. Eq a => a -> a -> Bool
P./= forall a. AdditiveMonoid a => a
P.zero)
forall (f :: * -> *) a. Applicative f => a -> f a
P.pure forall b c a. (b -> c) -> (a -> b) -> a -> c
P.. Integer -> Integer -> Rational
unsafeRatio Integer
n forall a b. (a -> b) -> a -> b
P.$ Integer
d
instance P.UnsafeFromData Rational where
{-# INLINABLE unsafeFromBuiltinData #-}
unsafeFromBuiltinData :: BuiltinData -> Rational
unsafeFromBuiltinData = forall a b c. (a -> b -> c) -> (a, b) -> c
P.uncurry Integer -> Integer -> Rational
unsafeRatio forall b c a. (b -> c) -> (a -> b) -> a -> c
P.. forall a. UnsafeFromData a => BuiltinData -> a
P.unsafeFromBuiltinData
instance ToJSON Rational where
toJSON :: Rational -> Value
toJSON (Rational Integer
n Integer
d) =
[Pair] -> Value
object
[ (Key
"numerator", forall a. ToJSON a => a -> Value
toJSON Integer
n)
, (Key
"denominator", forall a. ToJSON a => a -> Value
toJSON Integer
d)
]
instance FromJSON Rational where
parseJSON :: Value -> Parser Rational
parseJSON = forall a. String -> (Object -> Parser a) -> Value -> Parser a
withObject String
"Rational" forall a b. (a -> b) -> a -> b
Haskell.$ \Object
obj -> do
Integer
n <- Object
obj forall a. FromJSON a => Object -> Key -> Parser a
.: Key
"numerator"
Integer
d <- Object
obj forall a. FromJSON a => Object -> Key -> Parser a
.: Key
"denominator"
case Integer -> Integer -> Maybe Rational
ratio Integer
n Integer
d of
Maybe Rational
Haskell.Nothing -> forall (m :: * -> *) a. MonadFail m => String -> m a
Haskell.fail String
"Zero denominator is invalid."
Haskell.Just Rational
r -> forall (f :: * -> *) a. Applicative f => a -> f a
Haskell.pure Rational
r
{-# INLINABLE unsafeRatio #-}
unsafeRatio :: Integer -> Integer -> Rational
unsafeRatio :: Integer -> Integer -> Rational
unsafeRatio Integer
n Integer
d
| Integer
d forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero = forall a. () -> a
Builtins.error ()
| Integer
d forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero = Integer -> Integer -> Rational
unsafeRatio (forall a. AdditiveGroup a => a -> a
P.negate Integer
n) (forall a. AdditiveGroup a => a -> a
P.negate Integer
d)
| Bool
P.True =
let gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
n Integer
d
in Integer -> Integer -> Rational
Rational (Integer
n Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
(Integer
d Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd')
{-# INLINABLE ratio #-}
ratio :: Integer -> Integer -> P.Maybe Rational
ratio :: Integer -> Integer -> Maybe Rational
ratio Integer
n Integer
d
| Integer
d forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero = forall a. Maybe a
P.Nothing
| Integer
d forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero = forall a. a -> Maybe a
P.Just (Integer -> Integer -> Rational
unsafeRatio (forall a. AdditiveGroup a => a -> a
P.negate Integer
n) (forall a. AdditiveGroup a => a -> a
P.negate Integer
d))
| Bool
P.True =
let gcd' :: Integer
gcd' = Integer -> Integer -> Integer
euclid Integer
n Integer
d
in forall a. a -> Maybe a
P.Just forall b c a. (b -> c) -> (a -> b) -> a -> c
P..
Integer -> Integer -> Rational
Rational (Integer
n Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd') forall a b. (a -> b) -> a -> b
P.$
Integer
d Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
gcd'
toGHC :: Rational -> Ratio.Rational
toGHC :: Rational -> Rational
toGHC (Rational Integer
n Integer
d) = Integer
n forall a. Integral a => a -> a -> Ratio a
Ratio.% Integer
d
{-# INLINABLE numerator #-}
numerator :: Rational -> Integer
numerator :: Rational -> Integer
numerator (Rational Integer
n Integer
_) = Integer
n
{-# INLINABLE denominator #-}
denominator :: Rational -> Integer
denominator :: Rational -> Integer
denominator (Rational Integer
_ Integer
d) = Integer
d
{-# INLINABLE half #-}
half :: Rational
half :: Rational
half = Integer -> Integer -> Rational
Rational Integer
1 Integer
2
{-# INLINABLE fromInteger #-}
fromInteger :: Integer -> Rational
fromInteger :: Integer -> Rational
fromInteger Integer
num = Integer -> Integer -> Rational
Rational Integer
num forall a. MultiplicativeMonoid a => a
P.one
fromGHC :: Ratio.Rational -> Rational
fromGHC :: Rational -> Rational
fromGHC Rational
r = Integer -> Integer -> Rational
unsafeRatio (forall a. Ratio a -> a
Ratio.numerator Rational
r) (forall a. Ratio a -> a
Ratio.denominator Rational
r)
{-# INLINABLE negate #-}
negate :: Rational -> Rational
negate :: Rational -> Rational
negate (Rational Integer
n Integer
d) = Integer -> Integer -> Rational
Rational (forall a. AdditiveGroup a => a -> a
P.negate Integer
n) Integer
d
{-# INLINABLE abs #-}
abs :: Rational -> Rational
abs :: Rational -> Rational
abs rat :: Rational
rat@(Rational Integer
n Integer
d)
| Integer
n forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero = Integer -> Integer -> Rational
Rational (forall a. AdditiveGroup a => a -> a
P.negate Integer
n) Integer
d
| Bool
P.True = Rational
rat
{-# INLINABLE properFraction #-}
properFraction :: Rational -> (Integer, Rational)
properFraction :: Rational -> (Integer, Rational)
properFraction (Rational Integer
n Integer
d) =
(Integer
n Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
d,
Integer -> Integer -> Rational
Rational (Integer
n Integer -> Integer -> Integer
`Builtins.remainderInteger` Integer
d) Integer
d)
{-# INLINABLE recip #-}
recip :: Rational -> Rational
recip :: Rational -> Rational
recip (Rational Integer
n Integer
d)
| Integer
n forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero = forall a. () -> a
Builtins.error ()
| Integer
n forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero = Integer -> Integer -> Rational
Rational (forall a. AdditiveGroup a => a -> a
P.negate Integer
d) (forall a. AdditiveGroup a => a -> a
P.negate Integer
n)
| Bool
P.True = Integer -> Integer -> Rational
Rational Integer
d Integer
n
{-# INLINABLE truncate #-}
truncate :: Rational -> Integer
truncate :: Rational -> Integer
truncate (Rational Integer
n Integer
d) = Integer
n Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
d
{-# INLINABLE round #-}
round :: Rational -> Integer
round :: Rational -> Integer
round Rational
x =
let (Integer
n, Rational
r) = Rational -> (Integer, Rational)
properFraction Rational
x
m :: Integer
m = if Rational
r forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero then Integer
n forall a. AdditiveGroup a => a -> a -> a
P.- forall a. MultiplicativeMonoid a => a
P.one else Integer
n forall a. AdditiveSemigroup a => a -> a -> a
P.+ forall a. MultiplicativeMonoid a => a
P.one
flag :: Rational
flag = Rational -> Rational
abs Rational
r forall a. AdditiveGroup a => a -> a -> a
P.- Rational
half
in if
| Rational
flag forall a. Ord a => a -> a -> Bool
P.< forall a. AdditiveMonoid a => a
P.zero -> Integer
n
| Rational
flag forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero -> if Integer -> Integer -> Integer
Builtins.modInteger Integer
n Integer
2 forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero
then Integer
n
else Integer
m
| Bool
P.True -> Integer
m
{-# INLINABLE gcd #-}
gcd :: Integer -> Integer -> Integer
gcd :: Integer -> Integer -> Integer
gcd Integer
a Integer
b = Integer -> Integer -> Integer
gcd' (forall n. (Ord n, AdditiveGroup n) => n -> n
P.abs Integer
a) (forall n. (Ord n, AdditiveGroup n) => n -> n
P.abs Integer
b) where
gcd' :: Integer -> Integer -> Integer
gcd' Integer
a' Integer
b'
| Integer
b' forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero = Integer
a'
| Bool
P.True = Integer -> Integer -> Integer
gcd' Integer
b' (Integer
a' Integer -> Integer -> Integer
`Builtins.remainderInteger` Integer
b')
{-# INLINABLE reduce #-}
reduce :: Integer -> Integer -> Rational
reduce :: Integer -> Integer -> Rational
reduce Integer
x Integer
y
| Integer
y forall a. Eq a => a -> a -> Bool
P.== Integer
0 = forall a. BuiltinString -> a
P.traceError BuiltinString
P.ratioHasZeroDenominatorError
| Bool
P.True =
let d :: Integer
d = Integer -> Integer -> Integer
gcd Integer
x Integer
y in
Integer -> Integer -> Rational
Rational (Integer
x Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
d)
(Integer
y Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
d)
{-# INLINABLE euclid #-}
euclid :: Integer -> Integer -> Integer
euclid :: Integer -> Integer -> Integer
euclid Integer
x Integer
y
| Integer
y forall a. Eq a => a -> a -> Bool
P.== forall a. AdditiveMonoid a => a
P.zero = Integer
x
| Bool
P.True = Integer -> Integer -> Integer
euclid Integer
y (Integer
x Integer -> Integer -> Integer
`Builtins.modInteger` Integer
y)
P.makeLift ''Rational