1
0
mirror of https://github.com/bennofs/nix-script synced 2025-01-10 04:44:21 +01:00
nix-script/nix-script.hs

132 lines
4.3 KiB
Haskell
Raw Normal View History

2015-08-29 20:35:53 +02:00
#!/usr/bin/env nix-script
2014-08-31 16:59:47 +02:00
#!> haskell
#! haskell | text lens
#! shell | nix
{-# LANGUAGE OverloadedStrings #-}
{-# LANGUAGE TupleSections #-}
import Control.Monad
import Control.Applicative
import System.Environment
import Data.List
import Data.Text (Text)
import Data.Text.Lens (_Text)
import Control.Lens
import Control.Exception.Lens
import System.IO.Error.Lens
import System.Exit
import System.Posix.Process
import System.Posix.IO
import System.IO
import Data.Char
import Data.Monoid
import qualified Data.Text as Text
2015-09-10 19:50:14 +02:00
-- | Information about a language
2014-08-31 16:59:47 +02:00
data LangDef = LangDef
2015-08-29 23:11:08 +02:00
{ name :: String -- ^ Name of this language
2014-08-31 16:59:47 +02:00
, deps :: [Text] -> [Text] -- ^ Convert langunage-specific dependencies to nix packages
, run :: FilePath -> (String, [String]) -- ^ Command to run the given file as script
, repl :: FilePath -> (String, [String]) -- ^ Command to load the given file in an interpreter
}
2015-09-10 19:50:14 +02:00
basePackages :: [Text]
basePackages = ["coreutils", "utillinux"]
2014-08-31 16:59:47 +02:00
languages :: [LangDef]
2015-08-29 23:11:08 +02:00
languages = [haskell, python, javascript, perl, shell]
2014-08-31 16:59:47 +02:00
haskell :: LangDef
haskell = LangDef "haskell" d r i where
2015-09-10 19:50:14 +02:00
d pkgs = pure $
2015-08-29 20:35:53 +02:00
"haskellPackages.ghcWithPackages (hs: with hs; [" <> Text.unwords pkgs <> "])"
2014-08-31 16:59:47 +02:00
r script = ("runhaskell" , [script])
i script = ("ghci" , [script])
2015-08-29 23:11:08 +02:00
python :: LangDef
python = LangDef "python" d r i where
2015-09-10 19:50:14 +02:00
d pkgs = "python" : map ("pythonPackages." <>) pkgs
2015-08-29 23:11:08 +02:00
r script = ("python" , [script])
i script = ("python" , ["-i", script])
javascript :: LangDef
javascript = LangDef "javascript" d r i where
2015-09-10 19:50:14 +02:00
d pkgs = "node" : map ("nodePackages." <>) pkgs
2015-08-29 23:11:08 +02:00
r script = ("node" , [script])
i script = ("node" , [])
perl :: LangDef
perl = LangDef "perl" d r i where
2015-09-10 19:50:14 +02:00
d pkgs = "perl" : map ("perlPackages." <>) pkgs
2015-08-29 23:11:08 +02:00
r script = ("perl" , [script])
i script = ("perl" , ["-d", script])
2014-08-31 16:59:47 +02:00
shell :: LangDef
2015-09-10 19:50:14 +02:00
shell = LangDef "shell" d r i where
d = mappend ("bash" : basePackages)
2014-08-31 16:59:47 +02:00
r script = ("bash", [script])
i _ = ("bash", [])
2015-09-10 19:50:14 +02:00
passthrough :: String -> LangDef
passthrough name = LangDef name d r i where
d = mappend basePackages
r script = (name, [script])
i _ = (name, [])
2014-08-31 16:59:47 +02:00
lookupLangDef :: String -> IO LangDef
lookupLangDef n
| Just def <- find ((n ==) . name) languages = return def
2015-09-10 19:50:14 +02:00
| otherwise = return (passthrough n)
2014-08-31 16:59:47 +02:00
makeDeps :: String -> [String] -> IO [String]
makeDeps lang ds = lookupLangDef lang <&> \def ->
map (view _Text) $ deps def (map (review _Text) ds)
parseDepLine :: [String] -> IO (String, [String])
parseDepLine (lang:"|":deps) = return (lang, deps)
parseDepLine x = fail $ "Invalid dependency specification: " ++ unwords x
makeCommand :: String -> Bool -> String -> IO (String, [String])
makeCommand lang interactive file = lookupLangDef lang <&> \def ->
(if interactive then repl else run) def file
makeEnvArg :: String -> IO String
makeEnvArg env = f $ getEnv env <&> \val -> env ++ "=" ++ val where
f = handling_ (_IOException.errorType._NoSuchThing) $ return ""
makeXargsCommand :: String -> Int -> IO String
makeXargsCommand cmd fd = do
let xargsFile = "/proc/self/fd/" ++ show fd
envStr <- unwords <$> traverse makeEnvArg
["LOCALE_ARCHIVE", "LANG", "TERMINFO", "TERM"]
return $ "env " ++ envStr ++ " xargs -a " ++ xargsFile ++ " -d '\\n' " ++ cmd ++ ""
main :: IO ()
main = do
progName <- getProgName
args <- getArgs
let interactive = "i" `isSuffixOf` progName
case args ^? _Cons of
Nothing -> fail $ "usage: " ++ progName ++ " <file>" ++ " [missing file name]"
Just (file, args') -> do
header <- drop 1 . map (drop 2) . takeWhile ("#!" `isPrefixOf`) . lines <$> readFile file
case header ^? _Cons of
Just ('>':lang, depHeader) -> do
deps <- concat <$> traverse (uncurry makeDeps <=< parseDepLine . words) depHeader
let deps' = "findutils" : deps
let depArgs = concatMap (\x -> ["-p", x]) deps'
(cmd,cmdArgs) <- makeCommand (under _Text Text.strip lang) interactive file
(readFd, writeFd) <- createPipe
writeH <- fdToHandle writeFd
hPutStrLn writeH (unlines cmdArgs) >> hFlush writeH
hClose writeH
xargsCmd <- makeXargsCommand cmd (fromIntegral readFd)
let finalArgs = "--pure" : "--command" : xargsCmd : depArgs
executeFile "nix-shell" True finalArgs Nothing
_ -> fail "missing language to run as"