Shelly provides a single module for convenient systems programming in Haskell.
run_
and other underscore variants that do not return stdout,runFoldLines
to run a fold operation over each line rather than loading all of stdout into memory,runHandle
and runHandles
for complete control over handles.The focus of this library on convenience combined with good error messages should make shelly approachable for newer users of Haskell.
The shelly-extra package has some additional functionality that requires additional dependencies, currently including a convenient concurrency/futures implementation.
String
and ByteString
rather than just use Text
.PATH
at compile-time.HSH, HsShellScript and shh (unlike Shelly currently) implement very efficient mechanisms for piping/redirecting in the system. turtle, like Shelly offers folding as a way to efficiently deal with a stream.
None of the alternatives to Shelly offer command tracing. For some this is an absolutely critical feature, particularly given that Haskell does not yet offer up stack traces.
Shelly's finders load all files into memory. This is simpler to use if you control the filesystem structure and know the system is bounded in size. However, if the filesystem structure is unbounded it consumes unbounded memory.
Shelly does not change the nature of shell scripting (text in, text out). If you want something more revolutionary you might try these:
Shelly's main goal is ease of use. There should be a primitive for every shell operation you need so you can easily build abstractions, so there are many of the usual file and environment operations.
There are 2 main entry points for running arbitrary commands: run
and cmd
.
They take a FilePath as their first argument. run
takes a [Text]
as its second argument.
cmd
takes a variadic number of arguments, and they can be either Text
or FilePath
.
Fun Example: shows an infectious script: it uploads itself to a server and runs itself over ssh
.
Of course, the development machine may need to be exactly the same OS as the server.
I recommend using the boilerplate at the top of this example in your projects. This includes setting line buffering if you are dealing with text and not binary data.
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE ExtendedDefaultRules #-}
{-# OPTIONS_GHC -fno-warn-type-defaults #-}
import Shelly
import System.IO
import Data.Text as T
default (T.Text)
main :: IO ()
main = do
hSetBuffering stdout LineBuffering
shelly $ verbosely $ do
host <- run "uname" ["-n"]
if T.stripEnd host == "local-machine"
then do d <- cmd "date"
c <- escaping False $ cmd "git" "log -1 | head -1 | awk '{print $2}'"
appendfile "log/deploy.log" $ T.intercalate " - " [T.stripEnd d, c]
uploads "my-server:/remote/path/" ["deploy"]
sshPairs_ "my-server" [("cd", ["/remote/path"]), ("./deploy", [])]
else do
cmd "./script/angel"
-- same path on remote host
-- will create directories
uploads :: Text -> [Text] -> Sh ()
uploads remote locals = rsync $ ["--relative"] ++ locals ++ [remote]
rsync :: [Text] -> Sh ()
rsync args = run_ "rsync" $ ["--delete", "-avz", "--no-g"] ++ args
Yes, as seen above you can write variadic functions in Haskell quite easily, you just can't compose them as easily.
I find cmd
to be more convenient, but I often use run
and command
variants when I am building up abstractions.
Building up abstractions with cmd will require type signatures.
-- easy signature, but only allows one argument
let cabal = cmd "cabal" :: Text -> Sh Text
-- more complex signature that allows partial application of cmd
let cabal = cmd "cabal" :: Shelly.ShellCmd result => result
By default, all commands are shell escaped.
If you want the shell to interpret special characters such as *
, just use escaping False $ do ...
.
Shelly's usage of Text
means you may need to convert between Text
and FilePath
sometimes.
This should be infrequent though because:
cmd
will convert FilePath
to Text
.</>
and <.>
combinators convert Text
into a FilePath
automatically.Manual conversion is done through toTextIgnore
or toTextWarn
.
Command cd
does not change the process working directory (essentially a global variable),
but instead changes the shelly state (which is thread safe).
All of the Shelly API takes this into account, internally shelly converts all paths to absolute paths. You can turn a relative path into an absolute with absPath
or canonic
or you can make a path relative to the Shelly working directory with relPath
.
Haskell's #1 weakness for IO code is a lack of stack traces.
Shelly gives you something different: detailed logging.
In most cases this should be more useful than a stack trace.
Shelly keeps a log of API usage and saves it to a .shelly
directory on failure.
If you use shellyNoDir
, the log will instead be printed to stderr
.
This is in addition to the verbosely
settings that will print out commands and their output as the program is running.
Shelly's own error messages are detailed and in some cases it will catch Haskell exceptions and re-throw them with better messages.
If you make your own primitive functions that do not use the existing Shelly API, you can create a wrapper in the Sh monad that use trace
or tag
to log what they are doing.
You can turn tracing off (not generally recommended) by setting tracing False
.