{- Copyright 2013,2014 Marcelo Millani This file is part of boomange. boomange is free software: you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation, either version 3 of the License, or (at your option) any later version. boomange is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with boomange. If not, see -} import Control.Exception import Control.Monad import System.IO import System.Directory import System.Environment import System.FilePath import Data.List import Text.Printf import Paths_boomange -- automatically generated import Data.DescriLo import Data.Builder import Data.Entities hiding (name) import Data.Loader appName = "boomange" appVersion = "0.1.2.1" -- header to be used on the sample config file configHeader = "# This is the configuration file for " ++ appName ++ "\n" ++ "# For a full description of its syntax, see the haddock documentation of DescriLo\n" ++ "# In the config description:\n" ++ "# output - where should the resulting html file be placed\n" ++ "# header - file to be prepended to output\n" ++ "# footer - file to be appended to output\n" ++ "# there are no other values for config\n\n" ++ "# In the watch description, there may be an unlimited amount of values and the left part is always ignored by the program and may be used for organization.\n" ++ "# The right part indicates which file will be read to create the output. It will behave as if all of the files were concatenated.\n\n" -- header to be used on the sample bookmarks file bookmarksHeader = "#\n" ++ "# This is a sample bookmarks file for " ++ appName ++ "\n" ++ "# For a full description of its syntax, see the haddock documentation for SimtreeLo\n" ++ "# The first line indicates the comment pattern" ++ "# Leaves represent the URI of their direct parents\n" generateBookmarks config = do header <- readFile $ headerFile config footer <- readFile $ footerFile config final <- openFile (outputFile config) WriteMode bookmarks <- loadBookmarks $ watch config let body = htmlBookmarks bookmarks hPutStr final header hPutStr final body hPutStr final footer hClose final printf "Output written to %s\n" (outputFile config) -- | gets the directory where configuration files should be placed -- -- First, checks if XDG_CONFIG_HOME exists, producing $XDG_CONFIG_HOME/appName if it does -- if it does not, the checks if HOME does, producing $HOME/.config/appName if it does -- if it still fails, returns getAppUserDataDirectory appName getConfigDirectory appName = let failXDG e = do dir <- getEnv "HOME" return $ dir ++ [pathSeparator] ++ ".config" ++ [pathSeparator] ++ appName failHOME e = getAppUserDataDirectory appName in do handle (failHOME::SomeException -> IO FilePath) $ handle (failXDG::SomeException -> IO FilePath) $ do dir <- getEnv "XDG_CONFIG_HOME" return $ dir ++ [pathSeparator] ++ appName -- | installs a basic configuration installConfig cDir = let htmlDir = cDir ++ [pathSeparator] ++ "html" bookFile = cDir ++ [pathSeparator] ++ "bookmarks" outFile = cDir ++ [pathSeparator] ++ "bookmarks.html" config = Description { name = "config" , values = [ ("output",outFile) , ("header",htmlDir ++ [pathSeparator] ++ "header.html") , ("footer", htmlDir ++ [pathSeparator] ++ "footer.html") ] } watch = Description { name = "watch" , values = [ ("default",bookFile) ] } sampleBookmarks = "Boomange\n\tDocumentation\n" ++ "\t\tDescriLo\n\t\t\thttp://hackage.haskell.org/package/descrilo-0.1.0.0/docs/Data-DescriLo.html\n" ++ "\t\tSimtreeLo\n\t\t\thttp://hackage.haskell.org/package/simtreelo-0.1.0.0/docs/Data-Simtreelo.html\n" cFile = cDir ++ [pathSeparator] ++ "config" in do -- creates the base config file hcFile <- openFile cFile WriteMode hPutStr hcFile configHeader hPutStr hcFile $ show config hPutStr hcFile $ show watch hClose hcFile -- copies the default html files headerFile <- getDataFileName "html/header.html" footerFile <- getDataFileName "html/footer.html" cssFile <- getDataFileName "style.css" -- creates the html folder createDirectoryIfMissing True htmlDir -- copies html files copyFile headerFile $ htmlDir ++ [pathSeparator] ++ "header.html" copyFile footerFile $ htmlDir ++ [pathSeparator] ++ "footer.html" -- copies css file copyFile cssFile $ cDir ++ [pathSeparator] ++ "style.css" -- creates a sample bookmarks file hBookmarks <- openFile bookFile WriteMode hPutStr hBookmarks bookmarksHeader hPutStr hBookmarks $ sampleBookmarks hClose hBookmarks data Action = Help | ConfigFile String | Version | Status | Invalid String | Generate deriving Eq parseArgs args = case args of "-h":r -> Help : parseArgs r "--help":r -> Help : parseArgs r "-c":file:r -> ConfigFile file : parseArgs r "--config":file:r -> ConfigFile file : parseArgs r "-v":r -> Version : parseArgs r "--version":r -> Version : parseArgs r "-s":r -> Status : parseArgs r "--status":r -> Status : parseArgs r [] -> [] other:r -> Invalid other : Help : parseArgs r execute [] _ = return () execute (h:r) configFiles = case h of Help -> do printf "usage: %s [OPTION...]\n" appName putStr $ "Options:\n" ++ "\t-h, --help shows this help text\n" ++ "\t-c, --config uses as config instead of the default\n" ++ "\t-v, --version outputs version and exits\n"++ "\t-s, --status outputs configuration file info and exits\n" Version -> do printf "%s %s\n" appName appVersion putStr $ "Copyright (C) 2013,2014 Marcelo Garlet Millani\n" ++ "License GPLv3+: GNU GPL version 3 or later .\n" ++ "This is free software: you are free to change and redistribute it.\n" ++ "There is NO WARRANTY, to the extent permitted by law.\n" Status -> mapM showStatus configFiles >>= mapM_ (mapM_ putStrLn) Invalid opt -> do printf "'%s' is not a valid option or has an incorrect number of arguments\n" opt Generate -> do configs <- mapM loadConfig configFiles mapM_ generateBookmarks configs >> execute r configFiles showStatus :: String -> IO [String] showStatus configFile = do config <- loadConfig configFile return $ printf "%s:" configFile : (printf "\tOutput file:\n\t\t%s" (outputFile config)) : (printf "\tHeader file:\n\t\t%s" (headerFile config)) : (printf "\tFooter file:\n\t\t%s" (footerFile config)) : "\tBookmarks files:" : (map (\x -> "\t\t" ++ x) $ watch config) separateConfigs [] = ([],[]) separateConfigs (h:r) = let (actions, configs) = separateConfigs r in case h of ConfigFile x -> (actions, x : configs) other -> (other : actions, configs) main = do args <- getArgs let (argActions, argConfigs) = separateConfigs $ parseArgs args -- if help or version were asked, does not generate bookmarks actions = if elem Help argActions || elem Version argActions || elem Status argActions then argActions else (Generate : argActions) -- if no configuration file was given, uses the default one configs <- if argConfigs == [] then do cDir <- getConfigDirectory appName -- if the configuration directory does not exists, sets it up confExists <- doesDirectoryExist cDir when (not confExists) $ do createDirectoryIfMissing True cDir installConfig cDir return $ [cDir ++ "/config"] else (return argConfigs) execute actions configs {- cDir <- getConfigDirectory appName -- if the configuration directory does not exists, sets it up confExists <- doesDirectoryExist cDir when (not confExists) $ do createDirectoryIfMissing True cDir installConfig cDir let cFile = cDir ++ "/config" config <- loadConfig cFile generateBookmarks config -}