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

Allow users to extend the environment

This commit is contained in:
rnhmjoj 2015-09-21 05:19:16 +02:00
parent 5cff3ec2c5
commit 5af095e97c

View File

@ -3,7 +3,8 @@ module Main where
import Control.Monad (when) import Control.Monad (when)
import Data.Maybe (fromMaybe) import Data.Maybe (fromMaybe)
import Data.List (isPrefixOf, find) import Data.Char (isSpace)
import Data.List (isPrefixOf, find, (\\))
import System.Environment (lookupEnv, getProgName, getArgs) import System.Environment (lookupEnv, getProgName, getArgs)
import System.Process (callProcess) import System.Process (callProcess)
import System.Posix.Escape.Unicode (escapeMany) import System.Posix.Escape.Unicode (escapeMany)
@ -85,6 +86,14 @@ lookupLang n =
fromMaybe (passthrough n) (find ((n ==) . name) languages) fromMaybe (passthrough n) (find ((n ==) . name) languages)
-- | Extract environment declaration from the header
filterEnv :: [String] -> (Env, [String])
filterEnv header = (vars env, header \\ env)
where
vars = concatMap (drop 2 . words)
env = filter (isPrefixOf "env" . dropWhile isSpace) header
-- | Parse dependencies declaration line -- | Parse dependencies declaration line
parseHeader :: String -> [String] parseHeader :: String -> [String]
parseHeader = uncurry trans . split . words parseHeader = uncurry trans . split . words
@ -110,8 +119,8 @@ makeCmd (program, args) args' defs =
-- | Create environment variable to run the script with -- | Create environment variable to run the script with
makeEnv :: IO Env makeEnv :: Env -> IO Env
makeEnv = mapM format baseEnv where makeEnv extra = mapM format (baseEnv ++ extra) where
format var = maybe "" (\x -> var ++ "=" ++ x) <$> lookupEnv var format var = maybe "" (\x -> var ++ "=" ++ x) <$> lookupEnv var
@ -130,12 +139,13 @@ main = do
script <- readFile file script <- readFile file
case header script of case header script of
(('>':identifier) : lines) -> do (('>':identifier) : lines) -> do
let pkgs = concatMap parseHeader lines let (env, deps) = filterEnv lines
language = dropWhile (==' ') identifier pkgs = concatMap parseHeader deps
language = dropWhile isSpace identifier
interactive = last progName == 'i' interactive = last progName == 'i'
interpreter = makeInter language interactive file interpreter = makeInter language interactive file
cmd <- makeCmd interpreter args <$> makeEnv cmd <- makeCmd interpreter args <$> makeEnv env
callProcess "nix-shell" ("--pure" : "--command" : cmd : "-p" : pkgs) callProcess "nix-shell" ("--pure" : "--command" : cmd : "-p" : pkgs)
_ -> fail "missing or invalid header" _ -> fail "missing or invalid header"