-
Notifications
You must be signed in to change notification settings - Fork 4
/
Copy pathCoreUtils.hs
109 lines (93 loc) · 3.55 KB
/
CoreUtils.hs
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
{-# LANGUAGE QuasiQuotes #-}
-- Implement some simple coreutils commands
import Control.Monad (void)
import Data.Function ((&))
import Data.Word (Word8)
import System.Environment (getArgs)
import Streamly.Data.Array (Array)
import Streamly.Data.Fold (Fold)
import Streamly.FileSystem.Path (path)
import qualified Streamly.Console.Stdio as Stdio
import qualified Streamly.Data.Fold as Fold
import qualified Streamly.Data.Stream as Stream
import qualified Streamly.FileSystem.Path as Path
import qualified Streamly.Internal.FileSystem.DirIO as Dir (readFiles)
import qualified Streamly.Internal.FileSystem.File as File
-- | > cat input.txt
cat :: IO ()
cat =
File.readChunks "input.txt" -- Stream IO (Array Word8)
& Stream.fold Stdio.writeChunks -- IO ()
-- | Read all files from a directory and write to a single output file.
-- > cat dir/* > output.txt
catDirTo :: IO ()
catDirTo =
Dir.readFiles [path|dir|] -- Stream IO Path
-- NOTE: This is a redundant step that we are forced to do as we've still
-- not introduced FileIO module that uses Path yet.
& fmap Path.toString -- Stream IO String
& Stream.unfoldEach File.chunkReader -- Stream IO (Array Word8)
& File.fromChunks "output.txt" -- IO ()
-- | > cp input.txt output.txt
cp :: IO ()
cp =
File.readChunks "input.txt" -- Stream IO (Array Word8)
& File.fromChunks "output.txt" -- IO ()
-- | > cat input.txt >> output.txt
append :: IO ()
append =
File.readChunks "input.txt" -- Stream IO (Array Word8)
& File.writeAppendChunks "output.txt" -- IO ()
-- | > cat input.txt | tee output1.txt > output.txt
tap :: IO ()
tap =
File.readChunks "input.txt" -- Stream IO (Array Word8)
& Stream.tap (File.writeChunks "output1.txt") -- Stream IO (Array Word8)
& File.fromChunks "output.txt" -- IO ()
{-
-- | > cat input.txt | tee output1.txt > output.txt
-- output1.txt is processed/written to in a separate thread.
tapAsync :: IO ()
tapAsync =
Stdio.readChunks () -- Stream IO (Array Word8)
& Stream.tapAsyncK
(File.fromChunks "output1.txt") -- Stream IO (Array Word8)
& File.fromChunks "output.txt" -- IO ()
-}
-- | > cat input.txt | tee output1.txt > output.txt
tee :: IO ()
tee =
File.readChunks "input.txt" -- Stream IO (Array Word8)
& Stream.fold t -- IO ((),())
& void -- IO ()
where
-- Combine two folds into a single fold
t :: Fold IO (Array Word8) ((),())
t = Fold.tee
(File.writeChunks "output1.txt") -- Fold IO (Array Word8) ()
(File.writeChunks "output.txt") -- Fold IO (Array Word8) ()
{-
-- | > grep -c "the" input.txt
grepc :: IO ()
grepc = do
File.toBytes "input.txt" -- Stream IO Word8
& Stream.splitOnSeq (Array.fromList pat) Fold.drain -- Stream IO ()
& Stream.length -- IO Int
>>= print . subtract 1 -- IO ()
where
pat :: [Word8]
pat = map (fromIntegral . ord) "the"
-}
main :: IO ()
main = do
cmd <- fmap head getArgs
case cmd of
"cat" -> putStrLn "cat" >> cat
"catDirTo" -> putStrLn "catDirTo" >> catDirTo
"cp" -> putStrLn "cp" >> cp
"append" -> putStrLn "append" >> append
"tap" -> putStrLn "tap" >> tap
-- "tapAsync" -> putStrLn "tapAsync" >> tapAsync
"tee" -> putStrLn "tee" >> tee
-- "grepc" -> putStrLn "grepc" >> grepc
_ -> putStrLn $ "Unknown command: " ++ cmd