mirror of
https://github.com/anoma/juvix.git
synced 2024-12-03 09:41:10 +03:00
Create separate polysemy
and effectful
based prelude (#2662)
This commit is contained in:
parent
999f5391c2
commit
dfd664e184
@ -1,614 +1,8 @@
|
||||
module Juvix.Prelude.Base
|
||||
( module Juvix.Prelude.Base,
|
||||
module Control.Applicative,
|
||||
module Data.Graph,
|
||||
module Text.Show.Unicode,
|
||||
module Data.Map.Strict,
|
||||
module Data.Set,
|
||||
module Data.IntMap.Strict,
|
||||
module Data.IntSet,
|
||||
module Control.Monad.Extra,
|
||||
module Control.Monad.Fix,
|
||||
module Data.Bitraversable,
|
||||
module Data.Bool,
|
||||
module Data.Char,
|
||||
module Data.Either.Extra,
|
||||
module Data.Bifunctor,
|
||||
module Data.Eq,
|
||||
module Data.Foldable,
|
||||
module Data.Function,
|
||||
module Data.Functor,
|
||||
module Safe.Exact,
|
||||
module Safe.Foldable,
|
||||
module Data.Hashable,
|
||||
module Data.Int,
|
||||
module Data.List.Extra,
|
||||
module Data.List.NonEmpty.Extra,
|
||||
module Data.Maybe,
|
||||
module Data.Monoid,
|
||||
module Data.Ord,
|
||||
module Data.Semigroup,
|
||||
module Prelude,
|
||||
module Data.Singletons,
|
||||
module Data.Singletons.Sigma,
|
||||
module Data.Singletons.TH,
|
||||
module Data.Stream,
|
||||
module Data.String,
|
||||
module Data.Text.Encoding,
|
||||
module Data.Text.IO,
|
||||
module Data.Text.IO.Utf8,
|
||||
module Data.Traversable,
|
||||
module Data.Tuple.Extra,
|
||||
module Data.Typeable,
|
||||
module Data.Void,
|
||||
module Data.Word,
|
||||
module GHC.Enum,
|
||||
module GHC.Generics,
|
||||
module GHC.Num,
|
||||
module GHC.Real,
|
||||
module Lens.Micro.Platform,
|
||||
module Polysemy,
|
||||
module Polysemy.Error,
|
||||
module Polysemy.Input,
|
||||
module Polysemy.Fixpoint,
|
||||
module Polysemy.Output,
|
||||
module Polysemy.Reader,
|
||||
module Polysemy.Tagged,
|
||||
module Polysemy.Resource,
|
||||
module Polysemy.Scoped,
|
||||
module Polysemy.State,
|
||||
module Language.Haskell.TH.Syntax,
|
||||
module Prettyprinter,
|
||||
module System.Exit,
|
||||
module System.FilePath,
|
||||
module System.IO,
|
||||
module Text.Show,
|
||||
module Control.Monad.Catch,
|
||||
module Control.Monad.Zip,
|
||||
Data,
|
||||
Text,
|
||||
pack,
|
||||
unpack,
|
||||
strip,
|
||||
assert,
|
||||
HashMap,
|
||||
ByteString,
|
||||
HashSet,
|
||||
IsString (..),
|
||||
Alternative (..),
|
||||
MonadIO (..),
|
||||
type (~),
|
||||
( module Juvix.Prelude.Base.Foundation,
|
||||
module Juvix.Prelude.Base.Polysemy,
|
||||
)
|
||||
where
|
||||
|
||||
import Control.Applicative
|
||||
import Control.Monad.Catch (MonadMask, MonadThrow, throwM)
|
||||
import Control.Monad.Extra hiding (fail, forM, mconcatMapM, whileJustM)
|
||||
import Control.Monad.Extra qualified as Monad
|
||||
import Control.Monad.Fix
|
||||
import Control.Monad.IO.Class (MonadIO (..))
|
||||
import Control.Monad.Zip
|
||||
import Data.Bifunctor hiding (first, second)
|
||||
import Data.Bitraversable
|
||||
import Data.Bool
|
||||
import Data.ByteString (ByteString)
|
||||
import Data.Char
|
||||
import Data.Char qualified as Char
|
||||
import Data.Data
|
||||
import Data.Either.Extra
|
||||
import Data.Eq
|
||||
import Data.Foldable hiding (foldr1, minimum, minimumBy)
|
||||
import Data.Function
|
||||
import Data.Functor
|
||||
import Data.Graph (Graph, SCC (..), Vertex, stronglyConnComp)
|
||||
import Data.HashMap.Strict (HashMap)
|
||||
import Data.HashMap.Strict qualified as HashMap
|
||||
import Data.HashSet (HashSet)
|
||||
import Data.HashSet qualified as HashSet
|
||||
import Data.Hashable
|
||||
import Data.Int
|
||||
import Data.IntMap.Strict (IntMap)
|
||||
import Data.IntMap.Strict qualified as IntMap
|
||||
import Data.IntSet (IntSet)
|
||||
import Data.List.Extra hiding (allSame, foldr1, groupSortOn, head, last, mconcatMap, replicate, unzip)
|
||||
import Data.List.Extra qualified as List
|
||||
import Data.List.NonEmpty qualified as NonEmpty
|
||||
import Data.List.NonEmpty.Extra
|
||||
( NonEmpty (..),
|
||||
appendList,
|
||||
head,
|
||||
last,
|
||||
maximum1,
|
||||
maximumOn1,
|
||||
minimum1,
|
||||
minimumOn1,
|
||||
prependList,
|
||||
some1,
|
||||
(|:),
|
||||
)
|
||||
import Data.Map.Strict (Map)
|
||||
import Data.Maybe
|
||||
import Data.Monoid
|
||||
import Data.Ord
|
||||
import Data.Semigroup (Semigroup, sconcat, (<>))
|
||||
import Data.Set (Set)
|
||||
import Data.Singletons hiding ((@@))
|
||||
import Data.Singletons.Sigma
|
||||
import Data.Singletons.TH (genSingletons, promoteOrdInstances, singOrdInstances)
|
||||
import Data.Stream (Stream)
|
||||
import Data.Stream qualified as Stream
|
||||
import Data.String
|
||||
import Data.Text (Text, pack, strip, unpack)
|
||||
import Data.Text qualified as Text
|
||||
import Data.Text.Encoding
|
||||
import Data.Text.IO hiding (appendFile, getContents, getLine, hGetContents, hGetLine, hPutStr, hPutStrLn, interact, putStr, putStrLn, readFile, writeFile)
|
||||
import Data.Text.IO qualified as Text
|
||||
import Data.Text.IO.Utf8 hiding (putStr, putStrLn, readFile, writeFile)
|
||||
import Data.Text.IO.Utf8 qualified as Utf8
|
||||
import Data.Traversable
|
||||
import Data.Tuple.Extra hiding (both)
|
||||
import Data.Type.Equality (type (~))
|
||||
import Data.Typeable hiding (TyCon)
|
||||
import Data.Void
|
||||
import Data.Word
|
||||
import GHC.Base (assert)
|
||||
import GHC.Enum
|
||||
import GHC.Err qualified as Err
|
||||
import GHC.Generics (Generic)
|
||||
import GHC.Num
|
||||
import GHC.Real
|
||||
import GHC.Stack.Types
|
||||
import Language.Haskell.TH.Syntax (Exp, Lift, Q)
|
||||
import Lens.Micro.Platform
|
||||
import Path
|
||||
import Path.IO qualified as Path hiding (getCurrentDir, setCurrentDir, withCurrentDir)
|
||||
import Polysemy hiding (embed)
|
||||
import Polysemy.Embed qualified as Embed
|
||||
import Polysemy.Error hiding (fromEither)
|
||||
import Polysemy.Fixpoint
|
||||
import Polysemy.Input
|
||||
import Polysemy.Output
|
||||
import Polysemy.Reader
|
||||
import Polysemy.Resource
|
||||
import Polysemy.Scoped
|
||||
import Polysemy.State
|
||||
import Polysemy.Tagged hiding (tag)
|
||||
import Prettyprinter (Doc, (<+>))
|
||||
import Safe.Exact
|
||||
import Safe.Foldable
|
||||
import System.Exit
|
||||
import System.FilePath (FilePath, dropTrailingPathSeparator, normalise, (<.>), (</>))
|
||||
import System.IO hiding
|
||||
( appendFile,
|
||||
getContents,
|
||||
getLine,
|
||||
hGetContents,
|
||||
hGetLine,
|
||||
hPutStr,
|
||||
hPutStrLn,
|
||||
interact,
|
||||
openBinaryTempFile,
|
||||
openTempFile,
|
||||
putStr,
|
||||
putStrLn,
|
||||
readFile,
|
||||
readFile',
|
||||
writeFile,
|
||||
)
|
||||
import System.IO.Error
|
||||
import Text.Read qualified as Text
|
||||
import Text.Show (Show)
|
||||
import Text.Show qualified as Show
|
||||
import Text.Show.Unicode (urecover, ushow)
|
||||
import Prelude (Double)
|
||||
|
||||
type EmbedIO = Embed.Embed IO
|
||||
|
||||
embed :: (Member EmbedIO r) => IO a -> Sem r a
|
||||
embed = Embed.embed
|
||||
|
||||
traverseM ::
|
||||
(Monad m, Traversable m, Applicative f) =>
|
||||
(a1 -> f (m a2)) ->
|
||||
m a1 ->
|
||||
f (m a2)
|
||||
traverseM f = fmap join . traverse f
|
||||
|
||||
composeM :: (Monad m) => Int -> (a -> m a) -> a -> m a
|
||||
composeM 0 _ a = return a
|
||||
composeM n f a = composeM (n - 1) f a >>= f
|
||||
|
||||
compose :: Int -> (a -> a) -> a -> a
|
||||
compose 0 _ a = a
|
||||
compose n f a = f (compose (n - 1) f a)
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
mapReader :: (Member (Reader e1) r) => (e1 -> e2) -> Sem (Reader e2 ': r) a -> Sem r a
|
||||
mapReader f s = do
|
||||
e <- ask
|
||||
runReader (f e) s
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- String related util functions.
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
show :: (Show a, IsString str) => a -> str
|
||||
show = fromString . Show.show
|
||||
|
||||
toUpperFirst :: String -> String
|
||||
toUpperFirst [] = []
|
||||
toUpperFirst (x : xs) = Char.toUpper x : xs
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Text
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
prime :: Text -> Text
|
||||
prime "" = "_X"
|
||||
prime "?" = "_X"
|
||||
prime name = case Text.splitOn "'" name of
|
||||
[name', ""] -> name' <> "'0"
|
||||
[name', num] -> name' <> "'" <> maybe (num <> "'") (show . (+ 1)) (Text.readMaybe (unpack num) :: Maybe Word)
|
||||
_ -> name <> "'"
|
||||
|
||||
freshName :: HashSet Text -> Text -> Text
|
||||
freshName names name | HashSet.member name names = freshName names (prime name)
|
||||
freshName _ name = name
|
||||
|
||||
isValidIdentChar :: Char -> Bool
|
||||
isValidIdentChar c = c == '_' || ((isLetter c || isDigit c) && isAscii c)
|
||||
|
||||
isFirstLetter :: String -> Bool
|
||||
isFirstLetter = \case
|
||||
h : _ -> isLetter h
|
||||
_ -> False
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Foldable
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
-- | Returns the repeated elements
|
||||
findRepeated :: forall a. (Ord a) => [a] -> [a]
|
||||
findRepeated = mapMaybe rep . groupSortOn' id
|
||||
where
|
||||
rep :: [a] -> Maybe a
|
||||
rep = \case
|
||||
a : _ : _ -> Just a
|
||||
_ -> Nothing
|
||||
|
||||
allDifferent :: forall a. (Ord a) => [a] -> Bool
|
||||
allDifferent = null . findRepeated
|
||||
|
||||
allSame :: forall t a. (Eq a, Foldable t) => t a -> Bool
|
||||
allSame t = case nonEmpty t of
|
||||
Nothing -> True
|
||||
Just (a :| as) -> all (== a) as
|
||||
|
||||
nonEmpty :: (Foldable l) => l a -> Maybe (NonEmpty a)
|
||||
nonEmpty = NonEmpty.nonEmpty . toList
|
||||
|
||||
foldr1 :: (a -> a -> a) -> NonEmpty a -> a
|
||||
foldr1 = List.foldr1
|
||||
|
||||
sconcatMap :: (Semigroup c) => (a -> c) -> NonEmpty a -> c
|
||||
sconcatMap f = sconcat . fmap f
|
||||
|
||||
mconcatMap :: (Monoid c, Foldable t) => (a -> c) -> t a -> c
|
||||
mconcatMap f = List.mconcatMap f . toList
|
||||
|
||||
mconcatMapM :: (Monad m, Monoid c, Foldable t) => (a -> m c) -> t a -> m c
|
||||
mconcatMapM f = Monad.mconcatMapM f . toList
|
||||
|
||||
concatWith :: (Foldable t, Monoid a) => (a -> a -> a) -> t a -> a
|
||||
concatWith f ds = case nonEmpty ds of
|
||||
Nothing -> mempty
|
||||
Just ds' -> foldr1 f ds'
|
||||
{-# INLINE concatWith #-}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- HashMap
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
tableInsert ::
|
||||
(Hashable k) =>
|
||||
(a -> v) ->
|
||||
(a -> v -> v) ->
|
||||
k ->
|
||||
a ->
|
||||
HashMap k v ->
|
||||
HashMap k v
|
||||
tableInsert s f k a = over (at k) (Just . aux)
|
||||
where
|
||||
aux = \case
|
||||
Just v -> f a v
|
||||
Nothing -> s a
|
||||
|
||||
tableNestedInsert ::
|
||||
(Hashable k1, Hashable k2) =>
|
||||
k1 ->
|
||||
k2 ->
|
||||
a ->
|
||||
HashMap k1 (HashMap k2 a) ->
|
||||
HashMap k1 (HashMap k2 a)
|
||||
tableNestedInsert k1 k2 = tableInsert (HashMap.singleton k2) (HashMap.insert k2) k1
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- List
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
revAppend :: [a] -> [a] -> [a]
|
||||
revAppend [] !ys = ys
|
||||
revAppend (x : xs) !ys = revAppend xs (x : ys)
|
||||
|
||||
-- | strict map
|
||||
map' :: (a -> b) -> [a] -> [b]
|
||||
map' _ [] = []
|
||||
map' f (h : t) =
|
||||
-- keeping the lets separate ensures that `v` is evaluated before `vs`
|
||||
let !v = f h
|
||||
in let !vs = map' f t
|
||||
in v : vs
|
||||
|
||||
-- | longest common prefix
|
||||
commonPrefix :: forall a. (Eq a) => [a] -> [a] -> [a]
|
||||
commonPrefix a b = reverse (go [] a b)
|
||||
where
|
||||
go :: [a] -> [a] -> [a] -> [a]
|
||||
go ac x y = case (x, y) of
|
||||
(x' : xs, y' : ys)
|
||||
| x' == y' -> go (x' : ac) xs ys
|
||||
_ -> ac
|
||||
|
||||
zip4Exact :: [a] -> [b] -> [c] -> [d] -> [(a, b, c, d)]
|
||||
zip4Exact [] [] [] [] = []
|
||||
zip4Exact (x1 : t1) (x2 : t2) (x3 : t3) (x4 : t4) = (x1, x2, x3, x4) : zip4Exact t1 t2 t3 t4
|
||||
zip4Exact _ _ _ _ = error "zip4Exact"
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- NonEmpty
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
nonEmptyUnsnoc :: NonEmpty a -> (Maybe (NonEmpty a), a)
|
||||
nonEmptyUnsnoc e = (NonEmpty.nonEmpty (NonEmpty.init e), NonEmpty.last e)
|
||||
|
||||
tail' :: (HasCallStack) => [a] -> [a]
|
||||
tail' = \case
|
||||
[] -> impossible
|
||||
_ : xs -> xs
|
||||
|
||||
head' :: (HasCallStack) => [a] -> a
|
||||
head' = \case
|
||||
[] -> impossible
|
||||
x : _ -> x
|
||||
|
||||
nonEmpty' :: (HasCallStack) => [a] -> NonEmpty a
|
||||
nonEmpty' = fromJust . nonEmpty
|
||||
|
||||
_nonEmpty :: Lens' [a] (Maybe (NonEmpty a))
|
||||
_nonEmpty f x = maybe [] toList <$> f (nonEmpty x)
|
||||
|
||||
groupSortOn :: (Ord b) => (a -> b) -> [a] -> [NonEmpty a]
|
||||
groupSortOn f = map nonEmpty' . List.groupSortOn f
|
||||
|
||||
groupSortOn' :: (Ord b) => (a -> b) -> [a] -> [[a]]
|
||||
groupSortOn' = List.groupSortOn
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Errors
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
error :: (HasCallStack) => Text -> a
|
||||
error = Err.error . unpack
|
||||
|
||||
{-# DEPRECATED undefined "undefined" #-}
|
||||
undefined :: (HasCallStack) => a
|
||||
undefined = Err.error "undefined"
|
||||
|
||||
-- | Used to indicate impossible corner cases.
|
||||
impossible :: (HasCallStack) => a
|
||||
impossible = Err.error "impossible"
|
||||
|
||||
eassert :: Bool -> Sem r ()
|
||||
eassert b = assert b $ return ()
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
infixl 7 <+?>
|
||||
|
||||
(<+?>) :: Doc ann -> Maybe (Doc ann) -> Doc ann
|
||||
(<+?>) a = maybe a (a <+>)
|
||||
|
||||
infixr 7 <?+>
|
||||
|
||||
(<?+>) :: Maybe (Doc ann) -> Doc ann -> Doc ann
|
||||
(<?+>) = \case
|
||||
Nothing -> id
|
||||
Just a -> (a <+>)
|
||||
|
||||
infixr 7 ?<>
|
||||
|
||||
(?<>) :: (Semigroup m) => Maybe m -> m -> m
|
||||
(?<>) = maybe id (<>)
|
||||
|
||||
infixl 7 <>?
|
||||
|
||||
(<>?) :: (Semigroup m) => m -> Maybe m -> m
|
||||
(<>?) a = maybe a (a <>)
|
||||
|
||||
data Indexed a = Indexed
|
||||
{ _indexedIx :: Int,
|
||||
_indexedThing :: a
|
||||
}
|
||||
deriving stock (Generic, Show, Eq, Ord, Foldable, Traversable)
|
||||
|
||||
instance Functor Indexed where
|
||||
fmap f (Indexed i a) = Indexed i (f a)
|
||||
|
||||
instance (Hashable a) => Hashable (Indexed a)
|
||||
|
||||
indexFrom :: Int -> [a] -> [Indexed a]
|
||||
indexFrom i = zipWith Indexed [i ..]
|
||||
|
||||
makeLenses ''Indexed
|
||||
|
||||
toTuple :: Indexed a -> (Int, a)
|
||||
toTuple i = (i ^. indexedIx, i ^. indexedThing)
|
||||
|
||||
filterIndexed :: (a -> Bool) -> [Indexed a] -> [Indexed a]
|
||||
filterIndexed f = filter (f . (^. indexedThing))
|
||||
|
||||
fromText :: (IsString a) => Text -> a
|
||||
fromText = fromString . unpack
|
||||
|
||||
fromRightIO' :: (MonadIO m) => (e -> m ()) -> m (Either e r) -> m r
|
||||
fromRightIO' pp = do
|
||||
eitherM ifLeft return
|
||||
where
|
||||
ifLeft e = pp e >> liftIO exitFailure
|
||||
|
||||
fromRightIO :: (MonadIO m) => (e -> Text) -> m (Either e r) -> m r
|
||||
fromRightIO pp = fromRightIO' (putStrLn . pp)
|
||||
|
||||
putStr :: (MonadIO m) => Text -> m ()
|
||||
putStr = liftIO . Text.putStr
|
||||
|
||||
putStrLn :: (MonadIO m) => Text -> m ()
|
||||
putStrLn = liftIO . Text.putStrLn
|
||||
|
||||
optional_ :: (Alternative m) => m a -> m ()
|
||||
optional_ = void . optional
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Misc
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
-- | applies a function n times
|
||||
iterateN :: Int -> (a -> a) -> a -> a
|
||||
iterateN n f = (!! n) . iterate f
|
||||
|
||||
nubHashable :: (Hashable a) => [a] -> [a]
|
||||
nubHashable = HashSet.toList . HashSet.fromList
|
||||
|
||||
allElements :: (Bounded a, Enum a) => [a]
|
||||
allElements = [minBound .. maxBound]
|
||||
|
||||
replicate :: (Integral n) => n -> a -> [a]
|
||||
replicate n a = List.replicate (fromIntegral n) a
|
||||
|
||||
infixr 3 .&&.
|
||||
|
||||
(.&&.) :: (a -> Bool) -> (a -> Bool) -> a -> Bool
|
||||
(f .&&. g) a = f a && g a
|
||||
|
||||
infixr 3 ..&&..
|
||||
|
||||
(..&&..) :: (a -> b -> Bool) -> (a -> b -> Bool) -> (a -> b -> Bool)
|
||||
(f ..&&.. g) a = f a .&&. g a
|
||||
|
||||
infixr 2 .||.
|
||||
|
||||
(.||.) :: (a -> Bool) -> (a -> Bool) -> a -> Bool
|
||||
(a .||. b) c = a c || b c
|
||||
|
||||
eqOn :: (Eq b) => (a -> b) -> a -> a -> Bool
|
||||
eqOn = ((==) `on`)
|
||||
|
||||
class CanonicalProjection a b where
|
||||
project :: a -> b
|
||||
|
||||
instance CanonicalProjection a a where
|
||||
project = id
|
||||
|
||||
instance CanonicalProjection Void a where
|
||||
project = absurd
|
||||
|
||||
instance CanonicalProjection a () where
|
||||
project = const ()
|
||||
|
||||
-- | 'project' with type arguments swapped. Useful for type application
|
||||
project' :: forall b a. (CanonicalProjection a b) => a -> b
|
||||
project' = project
|
||||
|
||||
ensureFile :: (MonadIO m, MonadThrow m) => Path Abs File -> m ()
|
||||
ensureFile f =
|
||||
unlessM
|
||||
(Path.doesFileExist f)
|
||||
(throwM (mkIOError doesNotExistErrorType "" Nothing (Just (toFilePath f))))
|
||||
|
||||
flattenSCC :: SCC a -> NonEmpty a
|
||||
flattenSCC = \case
|
||||
AcyclicSCC a -> pure a
|
||||
-- Ideally `CyclicSCC`'s argument would have type `NonEmpty a` instead of `[a]`
|
||||
CyclicSCC as -> nonEmpty' as
|
||||
|
||||
execOutputList :: Sem (Output o ': r) a -> Sem r [o]
|
||||
execOutputList = fmap fst . runOutputList
|
||||
|
||||
whileJustM :: forall m a. (Monad m) => m (Maybe a) -> m [a]
|
||||
whileJustM m = go []
|
||||
where
|
||||
go :: [a] -> m [a]
|
||||
go acc = do
|
||||
r <- m
|
||||
case r of
|
||||
Nothing -> return (reverse acc)
|
||||
Just r' -> go (r' : acc)
|
||||
|
||||
sequenceEndWith :: (Monad m, Foldable l) => m () -> l (m ()) -> m ()
|
||||
sequenceEndWith sep l = sequenceWith sep l >> sep
|
||||
|
||||
sequenceWith :: forall m l. (Monad m, Foldable l) => m () -> l (m ()) -> m ()
|
||||
sequenceWith sep = go . toList
|
||||
where
|
||||
go :: [m ()] -> m ()
|
||||
go = \case
|
||||
[] -> return ()
|
||||
[x] -> x
|
||||
(x : xs) -> x >> sep >> go xs
|
||||
|
||||
-- | Removes the first element that satisfies a predicate and returns it
|
||||
popFirstJust :: (a -> Maybe b) -> [a] -> (Maybe b, [a])
|
||||
popFirstJust f = \case
|
||||
[] -> (Nothing, [])
|
||||
(h : hs) -> case f h of
|
||||
Nothing -> (h :) <$> popFirstJust f hs
|
||||
Just x -> (Just x, hs)
|
||||
|
||||
uncurryF :: (Functor f) => (a -> b -> c) -> f (a, b) -> f c
|
||||
uncurryF g input_ = uncurry g <$> input_
|
||||
|
||||
indexedByInt :: (Foldable f) => (a -> Int) -> f a -> IntMap a
|
||||
indexedByInt getIx l = IntMap.fromList [(getIx i, i) | i <- toList l]
|
||||
|
||||
indexedByHash :: (Foldable f, Hashable k) => (a -> k) -> f a -> HashMap k a
|
||||
indexedByHash getIx l = HashMap.fromList [(getIx i, i) | i <- toList l]
|
||||
|
||||
hashMap :: (Foldable f, Hashable k) => f (k, v) -> HashMap k v
|
||||
hashMap = HashMap.fromList . toList
|
||||
|
||||
runInputInfinite :: Stream i -> Sem (Input i ': r) a -> Sem r a
|
||||
runInputInfinite s =
|
||||
evalState s
|
||||
. reinterpret
|
||||
( \case
|
||||
Input -> do
|
||||
Stream.Cons i is <- get
|
||||
put is
|
||||
return i
|
||||
)
|
||||
|
||||
ensureLn :: Text -> Text
|
||||
ensureLn t =
|
||||
case Text.unsnoc t of
|
||||
Nothing -> t
|
||||
Just (_, y) -> case y of
|
||||
'\n' -> t
|
||||
_ -> Text.snoc t '\n'
|
||||
|
||||
readFile :: (MonadIO m) => Path Abs File -> m Text
|
||||
readFile = liftIO . Utf8.readFile . toFilePath
|
||||
|
||||
writeFileEnsureLn :: (MonadIO m) => Path Abs File -> Text -> m ()
|
||||
writeFileEnsureLn p = liftIO . Utf8.writeFile (toFilePath p)
|
||||
{-# INLINE writeFileEnsureLn #-}
|
||||
import Juvix.Prelude.Base.Foundation
|
||||
import Juvix.Prelude.Base.Polysemy
|
||||
|
563
src/Juvix/Prelude/Base/Foundation.hs
Normal file
563
src/Juvix/Prelude/Base/Foundation.hs
Normal file
@ -0,0 +1,563 @@
|
||||
module Juvix.Prelude.Base.Foundation
|
||||
( module Juvix.Prelude.Base.Foundation,
|
||||
module Control.Applicative,
|
||||
module Data.Graph,
|
||||
module Text.Show.Unicode,
|
||||
module Data.Map.Strict,
|
||||
module Data.Set,
|
||||
module Data.IntMap.Strict,
|
||||
module Data.IntSet,
|
||||
module Control.Monad.Extra,
|
||||
module Control.Monad.Fix,
|
||||
module Data.Bitraversable,
|
||||
module Data.Bool,
|
||||
module Data.Char,
|
||||
module Data.Either.Extra,
|
||||
module Data.Bifunctor,
|
||||
module Data.Eq,
|
||||
module Data.Foldable,
|
||||
module Data.Function,
|
||||
module Data.Functor,
|
||||
module Safe.Exact,
|
||||
module Safe.Foldable,
|
||||
module Data.Hashable,
|
||||
module Data.Int,
|
||||
module Data.List.Extra,
|
||||
module Data.List.NonEmpty.Extra,
|
||||
module Data.Maybe,
|
||||
module Data.Monoid,
|
||||
module Data.Ord,
|
||||
module Data.Semigroup,
|
||||
module Prelude,
|
||||
module Data.Singletons,
|
||||
module Data.Singletons.Sigma,
|
||||
module Data.Singletons.TH,
|
||||
module Data.Stream,
|
||||
module Data.String,
|
||||
module Data.Text.Encoding,
|
||||
module Data.Text.IO,
|
||||
module Data.Text.IO.Utf8,
|
||||
module Data.Traversable,
|
||||
module Data.Tuple.Extra,
|
||||
module Data.Typeable,
|
||||
module Data.Void,
|
||||
module Data.Word,
|
||||
module GHC.Enum,
|
||||
module GHC.Generics,
|
||||
module GHC.Num,
|
||||
module GHC.Real,
|
||||
module Lens.Micro.Platform,
|
||||
module Language.Haskell.TH.Syntax,
|
||||
module Prettyprinter,
|
||||
module System.Exit,
|
||||
module System.FilePath,
|
||||
module System.IO,
|
||||
module Text.Show,
|
||||
module Control.Monad.Catch,
|
||||
module Control.Monad.Zip,
|
||||
Data,
|
||||
Text,
|
||||
pack,
|
||||
unpack,
|
||||
strip,
|
||||
assert,
|
||||
HashMap,
|
||||
ByteString,
|
||||
HashSet,
|
||||
IsString (..),
|
||||
Alternative (..),
|
||||
MonadIO (..),
|
||||
type (~),
|
||||
)
|
||||
where
|
||||
|
||||
import Control.Applicative
|
||||
import Control.Monad.Catch (MonadMask, MonadThrow, throwM)
|
||||
import Control.Monad.Extra hiding (fail, forM, mconcatMapM, whileJustM)
|
||||
import Control.Monad.Extra qualified as Monad
|
||||
import Control.Monad.Fix
|
||||
import Control.Monad.IO.Class (MonadIO (..))
|
||||
import Control.Monad.Zip
|
||||
import Data.Bifunctor hiding (first, second)
|
||||
import Data.Bitraversable
|
||||
import Data.Bool
|
||||
import Data.ByteString (ByteString)
|
||||
import Data.Char
|
||||
import Data.Char qualified as Char
|
||||
import Data.Data
|
||||
import Data.Either.Extra
|
||||
import Data.Eq
|
||||
import Data.Foldable hiding (foldr1, minimum, minimumBy)
|
||||
import Data.Function
|
||||
import Data.Functor
|
||||
import Data.Graph (Graph, SCC (..), Vertex, stronglyConnComp)
|
||||
import Data.HashMap.Strict (HashMap)
|
||||
import Data.HashMap.Strict qualified as HashMap
|
||||
import Data.HashSet (HashSet)
|
||||
import Data.HashSet qualified as HashSet
|
||||
import Data.Hashable
|
||||
import Data.Int
|
||||
import Data.IntMap.Strict (IntMap)
|
||||
import Data.IntMap.Strict qualified as IntMap
|
||||
import Data.IntSet (IntSet)
|
||||
import Data.List.Extra hiding (allSame, foldr1, groupSortOn, head, last, mconcatMap, replicate, unzip)
|
||||
import Data.List.Extra qualified as List
|
||||
import Data.List.NonEmpty qualified as NonEmpty
|
||||
import Data.List.NonEmpty.Extra
|
||||
( NonEmpty (..),
|
||||
appendList,
|
||||
head,
|
||||
last,
|
||||
maximum1,
|
||||
maximumOn1,
|
||||
minimum1,
|
||||
minimumOn1,
|
||||
prependList,
|
||||
some1,
|
||||
(|:),
|
||||
)
|
||||
import Data.Map.Strict (Map)
|
||||
import Data.Maybe
|
||||
import Data.Monoid
|
||||
import Data.Ord
|
||||
import Data.Semigroup (Semigroup, sconcat, (<>))
|
||||
import Data.Set (Set)
|
||||
import Data.Singletons hiding ((@@))
|
||||
import Data.Singletons.Sigma
|
||||
import Data.Singletons.TH (genSingletons, promoteOrdInstances, singOrdInstances)
|
||||
import Data.Stream (Stream)
|
||||
import Data.String
|
||||
import Data.Text (Text, pack, strip, unpack)
|
||||
import Data.Text qualified as Text
|
||||
import Data.Text.Encoding
|
||||
import Data.Text.IO hiding (appendFile, getContents, getLine, hGetContents, hGetLine, hPutStr, hPutStrLn, interact, putStr, putStrLn, readFile, writeFile)
|
||||
import Data.Text.IO qualified as Text
|
||||
import Data.Text.IO.Utf8 hiding (putStr, putStrLn, readFile, writeFile)
|
||||
import Data.Text.IO.Utf8 qualified as Utf8
|
||||
import Data.Traversable
|
||||
import Data.Tuple.Extra hiding (both)
|
||||
import Data.Type.Equality (type (~))
|
||||
import Data.Typeable hiding (TyCon)
|
||||
import Data.Void
|
||||
import Data.Word
|
||||
import GHC.Base (assert)
|
||||
import GHC.Enum
|
||||
import GHC.Err qualified as Err
|
||||
import GHC.Generics (Generic)
|
||||
import GHC.Num
|
||||
import GHC.Real
|
||||
import GHC.Stack.Types
|
||||
import Language.Haskell.TH.Syntax (Exp, Lift, Q)
|
||||
import Lens.Micro.Platform
|
||||
import Path
|
||||
import Path.IO qualified as Path hiding (getCurrentDir, setCurrentDir, withCurrentDir)
|
||||
import Prettyprinter (Doc, (<+>))
|
||||
import Safe.Exact
|
||||
import Safe.Foldable
|
||||
import System.Exit
|
||||
import System.FilePath (FilePath, dropTrailingPathSeparator, normalise, (<.>), (</>))
|
||||
import System.IO hiding
|
||||
( appendFile,
|
||||
getContents,
|
||||
getLine,
|
||||
hGetContents,
|
||||
hGetLine,
|
||||
hPutStr,
|
||||
hPutStrLn,
|
||||
interact,
|
||||
openBinaryTempFile,
|
||||
openTempFile,
|
||||
putStr,
|
||||
putStrLn,
|
||||
readFile,
|
||||
readFile',
|
||||
writeFile,
|
||||
)
|
||||
import System.IO.Error
|
||||
import Text.Read qualified as Text
|
||||
import Text.Show (Show)
|
||||
import Text.Show qualified as Show
|
||||
import Text.Show.Unicode (urecover, ushow)
|
||||
import Prelude (Double)
|
||||
|
||||
traverseM ::
|
||||
(Monad m, Traversable m, Applicative f) =>
|
||||
(a1 -> f (m a2)) ->
|
||||
m a1 ->
|
||||
f (m a2)
|
||||
traverseM f = fmap join . traverse f
|
||||
|
||||
composeM :: (Monad m) => Int -> (a -> m a) -> a -> m a
|
||||
composeM 0 _ a = return a
|
||||
composeM n f a = composeM (n - 1) f a >>= f
|
||||
|
||||
compose :: Int -> (a -> a) -> a -> a
|
||||
compose 0 _ a = a
|
||||
compose n f a = f (compose (n - 1) f a)
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- String related util functions.
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
show :: (Show a, IsString str) => a -> str
|
||||
show = fromString . Show.show
|
||||
|
||||
toUpperFirst :: String -> String
|
||||
toUpperFirst [] = []
|
||||
toUpperFirst (x : xs) = Char.toUpper x : xs
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Text
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
prime :: Text -> Text
|
||||
prime "" = "_X"
|
||||
prime "?" = "_X"
|
||||
prime name = case Text.splitOn "'" name of
|
||||
[name', ""] -> name' <> "'0"
|
||||
[name', num] -> name' <> "'" <> maybe (num <> "'") (show . (+ 1)) (Text.readMaybe (unpack num) :: Maybe Word)
|
||||
_ -> name <> "'"
|
||||
|
||||
freshName :: HashSet Text -> Text -> Text
|
||||
freshName names name | HashSet.member name names = freshName names (prime name)
|
||||
freshName _ name = name
|
||||
|
||||
isValidIdentChar :: Char -> Bool
|
||||
isValidIdentChar c = c == '_' || ((isLetter c || isDigit c) && isAscii c)
|
||||
|
||||
isFirstLetter :: String -> Bool
|
||||
isFirstLetter = \case
|
||||
h : _ -> isLetter h
|
||||
_ -> False
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Foldable
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
-- | Returns the repeated elements
|
||||
findRepeated :: forall a. (Ord a) => [a] -> [a]
|
||||
findRepeated = mapMaybe rep . groupSortOn' id
|
||||
where
|
||||
rep :: [a] -> Maybe a
|
||||
rep = \case
|
||||
a : _ : _ -> Just a
|
||||
_ -> Nothing
|
||||
|
||||
allDifferent :: forall a. (Ord a) => [a] -> Bool
|
||||
allDifferent = null . findRepeated
|
||||
|
||||
allSame :: forall t a. (Eq a, Foldable t) => t a -> Bool
|
||||
allSame t = case nonEmpty t of
|
||||
Nothing -> True
|
||||
Just (a :| as) -> all (== a) as
|
||||
|
||||
nonEmpty :: (Foldable l) => l a -> Maybe (NonEmpty a)
|
||||
nonEmpty = NonEmpty.nonEmpty . toList
|
||||
|
||||
foldr1 :: (a -> a -> a) -> NonEmpty a -> a
|
||||
foldr1 = List.foldr1
|
||||
|
||||
sconcatMap :: (Semigroup c) => (a -> c) -> NonEmpty a -> c
|
||||
sconcatMap f = sconcat . fmap f
|
||||
|
||||
mconcatMap :: (Monoid c, Foldable t) => (a -> c) -> t a -> c
|
||||
mconcatMap f = List.mconcatMap f . toList
|
||||
|
||||
mconcatMapM :: (Monad m, Monoid c, Foldable t) => (a -> m c) -> t a -> m c
|
||||
mconcatMapM f = Monad.mconcatMapM f . toList
|
||||
|
||||
concatWith :: (Foldable t, Monoid a) => (a -> a -> a) -> t a -> a
|
||||
concatWith f ds = case nonEmpty ds of
|
||||
Nothing -> mempty
|
||||
Just ds' -> foldr1 f ds'
|
||||
{-# INLINE concatWith #-}
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- HashMap
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
tableInsert ::
|
||||
(Hashable k) =>
|
||||
(a -> v) ->
|
||||
(a -> v -> v) ->
|
||||
k ->
|
||||
a ->
|
||||
HashMap k v ->
|
||||
HashMap k v
|
||||
tableInsert s f k a = over (at k) (Just . aux)
|
||||
where
|
||||
aux = \case
|
||||
Just v -> f a v
|
||||
Nothing -> s a
|
||||
|
||||
tableNestedInsert ::
|
||||
(Hashable k1, Hashable k2) =>
|
||||
k1 ->
|
||||
k2 ->
|
||||
a ->
|
||||
HashMap k1 (HashMap k2 a) ->
|
||||
HashMap k1 (HashMap k2 a)
|
||||
tableNestedInsert k1 k2 = tableInsert (HashMap.singleton k2) (HashMap.insert k2) k1
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- List
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
revAppend :: [a] -> [a] -> [a]
|
||||
revAppend [] !ys = ys
|
||||
revAppend (x : xs) !ys = revAppend xs (x : ys)
|
||||
|
||||
-- | strict map
|
||||
map' :: (a -> b) -> [a] -> [b]
|
||||
map' _ [] = []
|
||||
map' f (h : t) =
|
||||
-- keeping the lets separate ensures that `v` is evaluated before `vs`
|
||||
let !v = f h
|
||||
in let !vs = map' f t
|
||||
in v : vs
|
||||
|
||||
-- | longest common prefix
|
||||
commonPrefix :: forall a. (Eq a) => [a] -> [a] -> [a]
|
||||
commonPrefix a b = reverse (go [] a b)
|
||||
where
|
||||
go :: [a] -> [a] -> [a] -> [a]
|
||||
go ac x y = case (x, y) of
|
||||
(x' : xs, y' : ys)
|
||||
| x' == y' -> go (x' : ac) xs ys
|
||||
_ -> ac
|
||||
|
||||
zip4Exact :: [a] -> [b] -> [c] -> [d] -> [(a, b, c, d)]
|
||||
zip4Exact [] [] [] [] = []
|
||||
zip4Exact (x1 : t1) (x2 : t2) (x3 : t3) (x4 : t4) = (x1, x2, x3, x4) : zip4Exact t1 t2 t3 t4
|
||||
zip4Exact _ _ _ _ = error "zip4Exact"
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- NonEmpty
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
nonEmptyUnsnoc :: NonEmpty a -> (Maybe (NonEmpty a), a)
|
||||
nonEmptyUnsnoc e = (NonEmpty.nonEmpty (NonEmpty.init e), NonEmpty.last e)
|
||||
|
||||
tail' :: (HasCallStack) => [a] -> [a]
|
||||
tail' = \case
|
||||
[] -> impossible
|
||||
_ : xs -> xs
|
||||
|
||||
head' :: (HasCallStack) => [a] -> a
|
||||
head' = \case
|
||||
[] -> impossible
|
||||
x : _ -> x
|
||||
|
||||
nonEmpty' :: (HasCallStack) => [a] -> NonEmpty a
|
||||
nonEmpty' = fromJust . nonEmpty
|
||||
|
||||
_nonEmpty :: Lens' [a] (Maybe (NonEmpty a))
|
||||
_nonEmpty f x = maybe [] toList <$> f (nonEmpty x)
|
||||
|
||||
groupSortOn :: (Ord b) => (a -> b) -> [a] -> [NonEmpty a]
|
||||
groupSortOn f = map nonEmpty' . List.groupSortOn f
|
||||
|
||||
groupSortOn' :: (Ord b) => (a -> b) -> [a] -> [[a]]
|
||||
groupSortOn' = List.groupSortOn
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Errors
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
error :: (HasCallStack) => Text -> a
|
||||
error = Err.error . unpack
|
||||
|
||||
{-# DEPRECATED undefined "undefined" #-}
|
||||
undefined :: (HasCallStack) => a
|
||||
undefined = Err.error "undefined"
|
||||
|
||||
-- | Used to indicate impossible corner cases.
|
||||
impossible :: (HasCallStack) => a
|
||||
impossible = Err.error "impossible"
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
infixl 7 <+?>
|
||||
|
||||
(<+?>) :: Doc ann -> Maybe (Doc ann) -> Doc ann
|
||||
(<+?>) a = maybe a (a <+>)
|
||||
|
||||
infixr 7 <?+>
|
||||
|
||||
(<?+>) :: Maybe (Doc ann) -> Doc ann -> Doc ann
|
||||
(<?+>) = \case
|
||||
Nothing -> id
|
||||
Just a -> (a <+>)
|
||||
|
||||
infixr 7 ?<>
|
||||
|
||||
(?<>) :: (Semigroup m) => Maybe m -> m -> m
|
||||
(?<>) = maybe id (<>)
|
||||
|
||||
infixl 7 <>?
|
||||
|
||||
(<>?) :: (Semigroup m) => m -> Maybe m -> m
|
||||
(<>?) a = maybe a (a <>)
|
||||
|
||||
data Indexed a = Indexed
|
||||
{ _indexedIx :: Int,
|
||||
_indexedThing :: a
|
||||
}
|
||||
deriving stock (Generic, Show, Eq, Ord, Foldable, Traversable)
|
||||
|
||||
instance Functor Indexed where
|
||||
fmap f (Indexed i a) = Indexed i (f a)
|
||||
|
||||
instance (Hashable a) => Hashable (Indexed a)
|
||||
|
||||
indexFrom :: Int -> [a] -> [Indexed a]
|
||||
indexFrom i = zipWith Indexed [i ..]
|
||||
|
||||
makeLenses ''Indexed
|
||||
|
||||
toTuple :: Indexed a -> (Int, a)
|
||||
toTuple i = (i ^. indexedIx, i ^. indexedThing)
|
||||
|
||||
filterIndexed :: (a -> Bool) -> [Indexed a] -> [Indexed a]
|
||||
filterIndexed f = filter (f . (^. indexedThing))
|
||||
|
||||
fromText :: (IsString a) => Text -> a
|
||||
fromText = fromString . unpack
|
||||
|
||||
fromRightIO' :: (MonadIO m) => (e -> m ()) -> m (Either e r) -> m r
|
||||
fromRightIO' pp = do
|
||||
eitherM ifLeft return
|
||||
where
|
||||
ifLeft e = pp e >> liftIO exitFailure
|
||||
|
||||
fromRightIO :: (MonadIO m) => (e -> Text) -> m (Either e r) -> m r
|
||||
fromRightIO pp = fromRightIO' (putStrLn . pp)
|
||||
|
||||
putStr :: (MonadIO m) => Text -> m ()
|
||||
putStr = liftIO . Text.putStr
|
||||
|
||||
putStrLn :: (MonadIO m) => Text -> m ()
|
||||
putStrLn = liftIO . Text.putStrLn
|
||||
|
||||
optional_ :: (Alternative m) => m a -> m ()
|
||||
optional_ = void . optional
|
||||
|
||||
--------------------------------------------------------------------------------
|
||||
-- Misc
|
||||
--------------------------------------------------------------------------------
|
||||
|
||||
-- | applies a function n times
|
||||
iterateN :: Int -> (a -> a) -> a -> a
|
||||
iterateN n f = (!! n) . iterate f
|
||||
|
||||
nubHashable :: (Hashable a) => [a] -> [a]
|
||||
nubHashable = HashSet.toList . HashSet.fromList
|
||||
|
||||
allElements :: (Bounded a, Enum a) => [a]
|
||||
allElements = [minBound .. maxBound]
|
||||
|
||||
replicate :: (Integral n) => n -> a -> [a]
|
||||
replicate n a = List.replicate (fromIntegral n) a
|
||||
|
||||
infixr 3 .&&.
|
||||
|
||||
(.&&.) :: (a -> Bool) -> (a -> Bool) -> a -> Bool
|
||||
(f .&&. g) a = f a && g a
|
||||
|
||||
infixr 3 ..&&..
|
||||
|
||||
(..&&..) :: (a -> b -> Bool) -> (a -> b -> Bool) -> (a -> b -> Bool)
|
||||
(f ..&&.. g) a = f a .&&. g a
|
||||
|
||||
infixr 2 .||.
|
||||
|
||||
(.||.) :: (a -> Bool) -> (a -> Bool) -> a -> Bool
|
||||
(a .||. b) c = a c || b c
|
||||
|
||||
eqOn :: (Eq b) => (a -> b) -> a -> a -> Bool
|
||||
eqOn = ((==) `on`)
|
||||
|
||||
class CanonicalProjection a b where
|
||||
project :: a -> b
|
||||
|
||||
instance CanonicalProjection a a where
|
||||
project = id
|
||||
|
||||
instance CanonicalProjection Void a where
|
||||
project = absurd
|
||||
|
||||
instance CanonicalProjection a () where
|
||||
project = const ()
|
||||
|
||||
-- | 'project' with type arguments swapped. Useful for type application
|
||||
project' :: forall b a. (CanonicalProjection a b) => a -> b
|
||||
project' = project
|
||||
|
||||
ensureFile :: (MonadIO m, MonadThrow m) => Path Abs File -> m ()
|
||||
ensureFile f =
|
||||
unlessM
|
||||
(Path.doesFileExist f)
|
||||
(throwM (mkIOError doesNotExistErrorType "" Nothing (Just (toFilePath f))))
|
||||
|
||||
flattenSCC :: SCC a -> NonEmpty a
|
||||
flattenSCC = \case
|
||||
AcyclicSCC a -> pure a
|
||||
-- Ideally `CyclicSCC`'s argument would have type `NonEmpty a` instead of `[a]`
|
||||
CyclicSCC as -> nonEmpty' as
|
||||
|
||||
whileJustM :: forall m a. (Monad m) => m (Maybe a) -> m [a]
|
||||
whileJustM m = go []
|
||||
where
|
||||
go :: [a] -> m [a]
|
||||
go acc = do
|
||||
r <- m
|
||||
case r of
|
||||
Nothing -> return (reverse acc)
|
||||
Just r' -> go (r' : acc)
|
||||
|
||||
sequenceEndWith :: (Monad m, Foldable l) => m () -> l (m ()) -> m ()
|
||||
sequenceEndWith sep l = sequenceWith sep l >> sep
|
||||
|
||||
sequenceWith :: forall m l. (Monad m, Foldable l) => m () -> l (m ()) -> m ()
|
||||
sequenceWith sep = go . toList
|
||||
where
|
||||
go :: [m ()] -> m ()
|
||||
go = \case
|
||||
[] -> return ()
|
||||
[x] -> x
|
||||
(x : xs) -> x >> sep >> go xs
|
||||
|
||||
-- | Removes the first element that satisfies a predicate and returns it
|
||||
popFirstJust :: (a -> Maybe b) -> [a] -> (Maybe b, [a])
|
||||
popFirstJust f = \case
|
||||
[] -> (Nothing, [])
|
||||
(h : hs) -> case f h of
|
||||
Nothing -> (h :) <$> popFirstJust f hs
|
||||
Just x -> (Just x, hs)
|
||||
|
||||
uncurryF :: (Functor f) => (a -> b -> c) -> f (a, b) -> f c
|
||||
uncurryF g input_ = uncurry g <$> input_
|
||||
|
||||
indexedByInt :: (Foldable f) => (a -> Int) -> f a -> IntMap a
|
||||
indexedByInt getIx l = IntMap.fromList [(getIx i, i) | i <- toList l]
|
||||
|
||||
indexedByHash :: (Foldable f, Hashable k) => (a -> k) -> f a -> HashMap k a
|
||||
indexedByHash getIx l = HashMap.fromList [(getIx i, i) | i <- toList l]
|
||||
|
||||
hashMap :: (Foldable f, Hashable k) => f (k, v) -> HashMap k v
|
||||
hashMap = HashMap.fromList . toList
|
||||
|
||||
ensureLn :: Text -> Text
|
||||
ensureLn t =
|
||||
case Text.unsnoc t of
|
||||
Nothing -> t
|
||||
Just (_, y) -> case y of
|
||||
'\n' -> t
|
||||
_ -> Text.snoc t '\n'
|
||||
|
||||
readFile :: (MonadIO m) => Path Abs File -> m Text
|
||||
readFile = liftIO . Utf8.readFile . toFilePath
|
||||
|
||||
writeFileEnsureLn :: (MonadIO m) => Path Abs File -> Text -> m ()
|
||||
writeFileEnsureLn p = liftIO . Utf8.writeFile (toFilePath p)
|
||||
{-# INLINE writeFileEnsureLn #-}
|
55
src/Juvix/Prelude/Base/Polysemy.hs
Normal file
55
src/Juvix/Prelude/Base/Polysemy.hs
Normal file
@ -0,0 +1,55 @@
|
||||
module Juvix.Prelude.Base.Polysemy
|
||||
( module Juvix.Prelude.Base.Polysemy,
|
||||
module Polysemy,
|
||||
module Polysemy.Error,
|
||||
module Polysemy.Input,
|
||||
module Polysemy.Fixpoint,
|
||||
module Polysemy.Output,
|
||||
module Polysemy.Reader,
|
||||
module Polysemy.Tagged,
|
||||
module Polysemy.Resource,
|
||||
module Polysemy.Scoped,
|
||||
module Polysemy.State,
|
||||
)
|
||||
where
|
||||
|
||||
import Data.Stream qualified as Stream
|
||||
import Juvix.Prelude.Base.Foundation
|
||||
import Polysemy hiding (embed)
|
||||
import Polysemy.Embed qualified as Embed
|
||||
import Polysemy.Error hiding (fromEither)
|
||||
import Polysemy.Fixpoint
|
||||
import Polysemy.Input
|
||||
import Polysemy.Output
|
||||
import Polysemy.Reader
|
||||
import Polysemy.Resource
|
||||
import Polysemy.Scoped
|
||||
import Polysemy.State
|
||||
import Polysemy.Tagged hiding (tag)
|
||||
|
||||
type EmbedIO = Embed.Embed IO
|
||||
|
||||
embed :: (Member EmbedIO r) => IO a -> Sem r a
|
||||
embed = Embed.embed
|
||||
|
||||
mapReader :: (Member (Reader e1) r) => (e1 -> e2) -> Sem (Reader e2 ': r) a -> Sem r a
|
||||
mapReader f s = do
|
||||
e <- ask
|
||||
runReader (f e) s
|
||||
|
||||
eassert :: Bool -> Sem r ()
|
||||
eassert b = assert b $ return ()
|
||||
|
||||
execOutputList :: Sem (Output o ': r) a -> Sem r [o]
|
||||
execOutputList = fmap fst . runOutputList
|
||||
|
||||
runInputInfinite :: Stream i -> Sem (Input i ': r) a -> Sem r a
|
||||
runInputInfinite s =
|
||||
evalState s
|
||||
. reinterpret
|
||||
( \case
|
||||
Input -> do
|
||||
Stream.Cons i is <- get
|
||||
put is
|
||||
return i
|
||||
)
|
8
src/Juvix/Prelude/BaseEffectful.hs
Normal file
8
src/Juvix/Prelude/BaseEffectful.hs
Normal file
@ -0,0 +1,8 @@
|
||||
module Juvix.Prelude.BaseEffectful
|
||||
( module Juvix.Prelude.Base.Foundation,
|
||||
module Juvix.Prelude.Effects,
|
||||
)
|
||||
where
|
||||
|
||||
import Juvix.Prelude.Base.Foundation
|
||||
import Juvix.Prelude.Effects
|
@ -1,7 +1,7 @@
|
||||
module Juvix.Prelude.Effects.Accum where
|
||||
|
||||
import Data.Kind qualified as GHC
|
||||
import Juvix.Prelude.Base hiding (Effect, Output, output, runOutputList)
|
||||
import Juvix.Prelude.Base.Foundation
|
||||
import Juvix.Prelude.Effects.Base
|
||||
|
||||
data Accum (o :: GHC.Type) :: Effect
|
||||
|
@ -16,7 +16,7 @@ import Effectful.Internal.Env (getEnv, putEnv)
|
||||
import Effectful.Reader.Static
|
||||
import Effectful.State.Static.Local
|
||||
import Effectful.TH
|
||||
import Juvix.Prelude.Base (($), (<$>), type (~))
|
||||
import Juvix.Prelude.Base.Foundation
|
||||
|
||||
overStaticRep :: (DispatchOf e ~ 'Static sideEffects, e :> r) => (StaticRep e -> StaticRep e) -> Eff r ()
|
||||
overStaticRep f = unsafeEff $ \r -> do
|
||||
|
Loading…
Reference in New Issue
Block a user