{-# 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(
    -- * Type
    Rational
    -- * Construction
    , unsafeRatio
    , fromInteger
    , ratio
    -- * Other functionality
    , 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

-- | Represents an arbitrary-precision ratio.
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
    )

-- We maintain two invariants for Rational:
--
-- 1. The denominator is greater than zero.
-- 2. The numerator and denominator are coprime.

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)

-- These instances ensure that the following invariants don't break:
--
-- 1. The denominator is greater than 0; and
-- 2. The numerator and denominator are coprime.
--
-- For invariant 1, fromBuiltinData yields Nothing on violation, while
-- unsafeFromData calls error. Invariant 2 is kept maintained by use of
-- unsafeRatio.

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

-- | This mimics the behaviour of Aeson's instance for 'GHC.Rational'.
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)
      ]

-- | This mimics the behaviour of Aeson's instance for 'GHC.Rational'.
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

-- | Makes a 'Rational' from a numerator and a denominator.
--
-- = Important note
--
-- If given a zero denominator, this function will error. If you don't mind a
-- size increase, and care about safety, use 'ratio' instead.
{-# 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')

-- | Safely constructs a 'Rational' from a numerator and a denominator. Returns
-- 'Nothing' if given a zero denominator.
{-# 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'

-- | Converts a 'Rational' to a GHC 'Ratio.Rational', preserving value. Does not
-- work on-chain.
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

-- | Returns the numerator of its argument.
--
-- = Note
--
-- It is /not/ true in general that @'numerator' '<$>' 'ratio' x y = x@; this
-- will only hold if @x@ and @y@ are coprime. This is due to 'Rational'
-- normalizing the numerator and denominator.
{-# INLINABLE numerator #-}
numerator :: Rational -> Integer
numerator :: Rational -> Integer
numerator (Rational Integer
n Integer
_) = Integer
n

-- | Returns the denominator of its argument. This will always be greater than,
-- or equal to, 1, although the type does not describe this.
--
-- = Note
--
-- It is /not/ true in general that @'denominator' '<$>' 'ratio' x y = y@; this
-- will only hold if @x@ and @y@ are coprime. This is due to 'Rational'
-- normalizing the numerator and denominator.
{-# INLINABLE denominator #-}
denominator :: Rational -> Integer
denominator :: Rational -> Integer
denominator (Rational Integer
_ Integer
d) = Integer
d

-- | 0.5
{-# INLINABLE half #-}
half :: Rational
half :: Rational
half = Integer -> Integer -> Rational
Rational Integer
1 Integer
2

-- | Converts an 'Integer' into the equivalent 'Rational'.
{-# INLINABLE fromInteger #-}
fromInteger :: Integer -> Rational
fromInteger :: Integer -> Rational
fromInteger Integer
num = Integer -> Integer -> Rational
Rational Integer
num forall a. MultiplicativeMonoid a => a
P.one

-- | Converts a GHC 'Ratio.Rational', preserving value. Does not work on-chain.
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)

-- | Produces the additive inverse of its argument.
--
-- = Note
--
-- This is specialized for 'Rational'; use this instead of the generic version
-- of this function, as it is significantly smaller on-chain.
{-# 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

-- | Returns the absolute value of its argument.
--
-- = Note
--
-- This is specialized for 'Rational'; use this instead of the generic version
-- in @PlutusTx.Numeric@, as said generic version produces much larger on-chain
-- code than the specialized version here.
{-# 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

-- | @'properFraction' r@ returns the pair @(n, f)@, such that all of the
-- following hold:
--
-- * @'fromInteger' n 'P.+' f = r@;
-- * @n@ and @f@ both have the same sign as @r@; and
-- * @'abs' f 'P.<' 'P.one'@.
{-# 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)

-- | Gives the reciprocal of the argument; specifically, for @r 'P./='
-- 'P.zero'@, @r 'P.*' 'recip' r = 'P.one'@.
--
-- = Important note
--
-- The reciprocal of zero is mathematically undefined; thus, @'recip' 'P.zero'@
-- will error. Use with care.
{-# 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

-- | Returns the whole-number part of its argument, dropping any leftover
-- fractional part. More precisely, @'truncate' r = n@ where @(n, _) =
-- 'properFraction' r@, but is much more efficient.
{-# INLINABLE truncate #-}
truncate :: Rational -> Integer
truncate :: Rational -> Integer
truncate (Rational Integer
n Integer
d) = Integer
n Integer -> Integer -> Integer
`Builtins.quotientInteger` Integer
d

-- | @'round' r@ returns the nearest 'Integer' value to @r@. If @r@ is
-- equidistant between two values, the even value will be given.
{-# 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

-- From GHC.Real
-- | @'gcd' x y@ is the non-negative factor of both @x@ and @y@ of which
-- every common factor of @x@ and @y@ is also a factor; for example
-- @'gcd' 4 2 = 2@, @'gcd' (-4) 6 = 2@, @'gcd' 0 4@ = @4@. @'gcd' 0 0@ = @0@.
{-# 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')

-- From GHC.Real
-- | Given a numerator and denominator, produces a 'Rational' by dividing both
-- numerator and denominator by their greatest common divisor.
{-# 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)

-- Helpers

-- Euclid's algorithm
{-# 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

{- HLINT ignore -}

{- Note [Ratio]

An important invariant is that the denominator is always positive. This is
enforced by

* Construction of 'Rational' numbers with 'unsafeRational' (the constructor
  of 'Rational' is not exposed)
* Normalizing after every numeric operation.

The 'StdLib.Spec' module has some property tests that check the behaviour of
'round', 'truncate', '>=', etc. against that of their counterparts in
'GHC.Real'.

-}

{- NOTE [Integer division operations]

Plutus Core provides built-in functions 'divideInteger', 'modInteger',
'quotientInteger' and 'remainderInteger' which are implemented as the Haskell
functions 'div', 'mod', 'quot', and 'rem' respectively.

The operations 'div' and 'mod' go together, as do 'quot' and 'rem': * DO NOT use
'div' with 'rem' or 'quot' with 'mod' *.  For most purposes users shoud probably
use 'div' and 'mod': see below for details.

For any integers a and b with b nonzero we have

  a * (a  `div` b) + a `mod` b = a
  a * (a `quot` b) + a `rem` b = a

(all operations give a "divide by zero" error if b = 0).  The analagous
identities for div/rem and quot/mod don't hold in general, and this can
lead to problems if you use the wrong combination of operations.

For positive divisors b, div truncates downwards and mod always returns a
non-negative result (0 <= a `mod` b <= b-1), which is consistent with standard
mathematical practice.  The `quot` operation truncates towards zero and `rem`
will give a negative remainder if a<0 and b>0.  If a<0 and b<0 then `mod` willl
also yield a negative result.  Results for different combinations of signs are
shown below.

-------------------------------
|   n  d | div mod | quot rem |
|-----------------------------|
|  41  5 |  8   1  |   8   1  |
| -41  5 | -9   4  |  -8  -1  |
|  41 -5 | -9  -4  |  -8   1  |
| -41 -5 |  8  -1  |   8  -1  |
-------------------------------

For many purposes (in particular if you're doing modular arithmetic),
a positive remainder is what you want.  Using 'div' and 'mod' achieves
this for positive values of b (but not for b negative, although doing
artimetic modulo a negative number would be unusual).

There is another possibility (Euclidean division) which is arguably more
mathematically correct than both div/mod and quot/rem. Given integers a and b
with b != 0, this returns numbers q and r with a = q*b+r and 0 <= r < |b|.  For
the numbers above this gives

-------------------
|   n  d |  q   r |
|-----------------|
|  41  5 |  8   1 |
| -41  5 | -9   4 |
|  41 -5 | -8   1 |
| -41 -5 |  9   4 |
-------------------

We get a positive remainder in all cases, but note for instance that the pairs
(41,5) and (-41,-5) give different results, which might be unexpected.

For a discussion of the pros and cons of various versions of integer division,
see Raymond T. Boute, "The Euclidean definition of the functions div and mod",
ACM Transactions on Programming Languages and Systems, April 1992.  (PDF at
https://core.ac.uk/download/pdf/187613369.pdf)
-}