module Main ( main ) where import Cabal2Nix.Hackage ( hashPackage, readCabalFile ) import Cabal2Nix.Generate ( cabal2nix ) import Cabal2Nix.Normalize ( normalize ) import Distribution.NixOS.Derivation.Cabal import Control.Exception ( bracket ) import Control.Monad ( when ) import Distribution.PackageDescription ( package, packageDescription ) import Distribution.PackageDescription.Parse ( parsePackageDescription, ParseResult(..) ) import Distribution.Text ( disp ) import System.Console.GetOpt ( OptDescr(..), ArgDescr(..), ArgOrder(..), usageInfo, getOpt ) import System.Environment ( getArgs ) import System.Exit ( exitFailure, exitSuccess ) import System.IO ( hPutStrLn, hFlush, stdout, stderr ) data Configuration = Configuration { optPrintHelp :: Bool , optPrintVersion :: Bool , optSha256 :: Maybe String , optMaintainer :: [String] , optPlatform :: [String] , optHaddock :: Bool , optDoCheck :: Bool , optJailbreak :: Bool , optHyperlinkSource :: Bool , optHackageDb :: Maybe FilePath } deriving (Show) defaultConfiguration :: Configuration defaultConfiguration = Configuration { optPrintHelp = False , optPrintVersion = False , optSha256 = Nothing , optMaintainer = [] , optPlatform = [] , optHaddock = True , optDoCheck = True , optJailbreak = False , optHyperlinkSource = True , optHackageDb = Nothing } options :: [OptDescr (Configuration -> Configuration)] options = [ Option "h" ["help"] (NoArg (\o -> o { optPrintHelp = True })) "show this help text" , Option "" ["hackage-db"] (ReqArg (\x o -> o { optHackageDb = Just x }) "FILEPATH") "path to the local hackage db in tar format" , Option "" ["sha256"] (ReqArg (\x o -> o { optSha256 = Just x }) "HASH") "sha256 hash of source tarball" , Option "m" ["maintainer"] (ReqArg (\x o -> o { optMaintainer = x : optMaintainer o }) "MAINTAINER") "maintainer of this package (may be specified multiple times)" , Option "p" ["platform"] (ReqArg (\x o -> o { optPlatform = x : optPlatform o }) "PLATFORM") "supported build platforms (may be specified multiple times)" , Option "" ["jailbreak"] (NoArg (\o -> o { optJailbreak = True })) "don't honor version restrictions on build inputs" , Option "" ["no-haddock"] (NoArg (\o -> o { optHaddock = False })) "don't run Haddock when building this package" , Option "" ["no-check"] (NoArg (\o -> o { optDoCheck = False })) "don't run regression test suites of this package" , Option "" ["no-hyperlink-source"] (NoArg (\o -> o { optHyperlinkSource = False })) "don't add pretty-printed source code to the documentation" ] usage :: String usage = usageInfo "Usage: cabal2nix [options] url-to-cabal-file" options ++ unlines [ "" , "Recognized URI schemes:" , "" , " cabal://pkgname-pkgversion download the specified package from Hackage" , " cabal://pkgname download latest version of the specified package from Hackage" , " http://host/path fetch the Cabal file via HTTP" , " file:///local/path load the Cabal file from the local disk" , " /local/path abbreviated version of file URI" ] cmdlineError :: String -> IO a cmdlineError "" = hPutStrLn stderr usage >> exitFailure cmdlineError errMsg = hPutStrLn stderr errMsg >> cmdlineError "" main :: IO () main = bracket (return ()) (\() -> hFlush stdout >> hFlush stderr) $ \() -> do args' <- getArgs (cfg,args) <- case getOpt Permute options args' of (o,n,[] ) -> return (foldl (flip ($)) defaultConfiguration o,n) (_,_,errs) -> cmdlineError (concatMap ("*** "++) errs) when (optPrintHelp cfg) (putStr usage >> exitSuccess) when (length args /= 1) (cmdlineError "*** exactly one url-to-cabal-file must be specified\n") cabal' <- fmap parsePackageDescription (readCabalFile (optHackageDb cfg) (head args)) cabal <- case cabal' of ParseOk _ a -> return a ParseFailed err -> do hPutStrLn stderr ("*** cannot parse cabal file: " ++ show err) exitFailure let packageId = package (packageDescription cabal) sha <- case optSha256 cfg of Just hash -> return hash Nothing -> hashPackage packageId let deriv = (cabal2nix cabal) { sha256 = sha, runHaddock = optHaddock cfg, jailbreak = optJailbreak cfg } deriv' = deriv { metaSection = (metaSection deriv) { maintainers = optMaintainer cfg , platforms = optPlatform cfg } , doCheck = doCheck deriv && optDoCheck cfg } putStr (show (disp (normalize deriv')))