{-# LANGUAGE CPP #-}
{-# LANGUAGE OverloadedStrings #-}

-- | Functions for initializing self-contained local postgreSQL
-- database clusters (useful in development more than production).
module Database.PostgreSQL.Devel (
      createLocalDB, configLocalDB, startLocalDB
    , initLocalDB, stopLocalDB, setLocalDB
    , withTempDB
    , resetConnection
  ) where

import Control.Exception
import Control.Monad
#if __GLASGOW_HASKELL__ < 710
import Data.Functor
#endif
import Data.List
import Database.PostgreSQL.Simple
import System.Directory
import System.Exit
import System.FilePath
import System.IO
import System.IO.Error
import System.Posix.Env
import System.Posix.Temp
import System.Process

isNonEmptyDir :: FilePath -> IO Bool
isNonEmptyDir dir =
  catchJust (\e -> if isDoesNotExistError e then Just () else Nothing)
  ((> 2) . length <$> getDirectoryContents dir)
  (const $ return False)

addDirectives :: [(String, String)] -> [String] -> [String]
addDirectives directives [] = map snd directives
addDirectives directives (cl:cls)
  | Just l <- lookup directive directives =
      (if comment then [l, cl] else [l]) ++
      addDirectives (directives \\ [(directive,l)]) cls
  | otherwise = cl : addDirectives directives cls
  where (comment, directive)
          | '#':clr <- cl, [(d,_)] <- lex clr = (True, d)
          | [(d,_)] <- lex cl                 = (False, d)
          | otherwise                         = (False, "")

-- | Set configuration parameters on a database by editing the
-- @postgresql.conf@ file.  Takes the database directory and a list of
-- @(@/parameter/@,@ /full-line/@)@ pairs.  For example, when creating
-- a throw-away database cluster you later intend to discard, you
-- might say:
--
-- > configLocalDB dbpath [("fsync", "fsync = off")]
--
-- Note that the second element of each pair is the complete
-- configuration line.  It is not correct to say:
--
-- > configLocalDB dbpath [("fsync", "off")]   -- INCORRECT
--
configLocalDB :: FilePath -> [(String, String)] -> IO ()
configLocalDB dir directives = do
  let confpath = dir </> "postgresql.conf"
  oldconf <- lines <$> readFile confpath
  let conf = unlines $ addDirectives directives oldconf
  length conf `seq` writeFile confpath conf

singleQuote :: String -> String
singleQuote ('\'':t) = "''" ++ singleQuote t
singleQuote (h:t)    = h : singleQuote t
singleQuote []       = ""

pgDirectives :: FilePath -> [(String, String)]
pgDirectives dir = [
    ("unix_socket_directories"
    , "unix_socket_directories = '" ++ singleQuote dir ++ "'")
  , ("logging_collector",  "logging_collector = yes")
  , ("listen_addresses", "listen_addresses = ''")]

pgDirectives92 :: FilePath -> [(String, String)]
pgDirectives92 dir = map depluralize $ pgDirectives dir
  where depluralize ("unix_socket_directories", _) =
          ("unix_socket_directory"
          , "unix_socket_directory = '" ++ singleQuote dir ++ "'")
        depluralize kv = kv

-- | Create a directory for a local database cluster entirely
-- self-contained within one directory.  This is accomplished by
-- creating a new PostgreSQL database cluster in the directory and
-- setting the following configuration options in @postgresql.conf@:
--
-- * @listen_address@ is set to empty (i.e., @\'\'@), so that no TCP
-- socket is bound, avoiding conflicts with any other running instaces
-- of PostgreSQL.
--
-- * @logging_collector@ is set to @yes@, so that all message logs are
--   kept in the @pg_log@ subdirectory of the directory you specified.
--
-- Note this function does /not/ start a postgres server after
-- creating the directory.  You will seperately need to start the
-- server using 'startLocalDB' or 'initLocalDB'.  (And note that
-- 'initLocalDB' already calls @createLocalDB@ if the directory does
-- not exist or is empty.  Hence the primary use of this function is
-- if you want to call 'configLocalDB' between 'createLocalDB' and
-- 'startLocalDB'.)
createLocalDB :: FilePath -> IO ()
createLocalDB dir = do
  (exit, _, err) <- readProcessWithExitCode "pg_ctl"
                      ["-D", dir, "-o", "--no-locale", "init"] ""
  when (exit /= ExitSuccess) $ fail err
  dir' <- canonicalizePath dir
  writeFile (dir </> "README_BEFORE_DELETING") $
    "## IMPORTANT:  Run the following command before deleting this " ++
    "directory ##\n\n" ++
    "pg_ctl -D " ++ showCommandForUser dir' [] ++ " stop -m immediate\n\n"
  version <- readFile (dir </> "PG_VERSION")
  case reads version of
    [(v, _)] | v < (9.3 :: Double) -> configLocalDB dir $ pgDirectives92 dir'
    _                              -> configLocalDB dir $ pgDirectives dir'

systemNoStdout :: String -> [String] -> IO ExitCode
systemNoStdout prog args =
  bracket (openFile "/dev/null" ReadWriteMode) hClose $ \devnull -> do
    let cp = (proc prog args) { std_in = UseHandle devnull
                              , std_out = UseHandle devnull }
    (_,_,_,pid) <- createProcess cp
    waitForProcess pid

-- | Start a local database if the server is not already running.
-- Otherwise, does nothing, but returns a 'ConnectInfo' in either
-- case.  The database server will continue running after the current
-- process exits (but see 'stopLocalDB').
startLocalDB :: FilePath -> IO ConnectInfo
startLocalDB dir0 = do
  dir <- canonicalizePath dir0
  (e0, _, _) <- readProcessWithExitCode "pg_ctl" ["status", "-D", dir] ""
  when (e0 /= ExitSuccess) $ do
    e1 <- systemNoStdout "pg_ctl" [ "start", "-w", "-D", dir ]
    when (e1 /= ExitSuccess) $ fail "could not start postgres"
  return defaultConnectInfo { connectHost = dir
                            , connectUser = ""
                            , connectDatabase = "postgres"
                            }

-- | A combination of 'createLocalDB' and 'startLocalDB'.
--
-- The parameter is a PostgreSQL data directory.  If the directory is
-- empty or does not exist, this function creates a new database
-- cluster (via 'createLocalDB').  Then, if a database server is not
-- already running for the directory, starts a server.  No matter
-- what, returns a 'ConnectInfo' that will connect to the server
-- running on this local database.
--
-- Note that if @initLocalDB@ starts a postgres server, the server
-- process will continue running after the process that called
-- @initLocalDB@ exits.  This is normally fine.  Since multiple client
-- processes may access the same PostgreSQL database, it makes sense
-- for the first client to start the database and no one to stop it.
-- See 'stopLocalDB' if you wish to stop the server process (which you
-- should always do before deleting a test cluster).  See also
-- 'withTempDB' to create a temporary cluster for the purposes of
-- running a test suite.
initLocalDB :: FilePath -> IO ConnectInfo
initLocalDB dir = do
  exists <- isNonEmptyDir dir
  unless exists $ createLocalDB dir
  startLocalDB dir

-- | Stop the server for a local database cluster entirely
-- self-contained within one directory.  You must call this before
-- deleting the directory, or else stray postgres processes will
-- linger forever.  If the argument is the empty string, looks for the
-- database directory in the @PGDATA@ environment variable.
stopLocalDB :: FilePath -> IO ()
stopLocalDB dir0 = do
  dir <- if not (null dir0) then return dir0 else do
    mpgd <- getEnv "PGDATA"
    case mpgd of Just pgd -> return pgd
                 _        -> fail "stopLocalDB: must specify database"
  e <- systemNoStdout "pg_ctl" ["stop", "-D", dir, "-m", "fast"]
  when (e /= ExitSuccess) $ fail "could not stop postgres"

-- | Set environment variables to make a local database cluster the
-- default.  Also returns shell commands you can eval or cut-and-paste
-- into your shell to make @pg_ctl@ and @psql@ access a local database
-- cluster.
setLocalDB :: FilePath -> IO String
setLocalDB dir0 = do
  dir1 <- canonicalizePath dir0
  setEnv "PGHOST" dir1 True
  setEnv "PGDATA" dir1 True
  setEnv "PGDATABASE" "postgres" True
  let dir = showCommandForUser dir1 []
  msh <- getEnv "SHELL"
  return $ case msh of Just sh | isSuffixOf "csh" sh ->
                         "setenv PGDATA " ++ dir ++ "; setenv PGHOST " ++ dir
                       _ -> "export PGDATA=" ++ dir ++ " PGHOST=" ++ dir

-- | Run a function with a completely fresh database cluster that gets
-- deleted on return.  Since the entire database is blown away when
-- the function returns, @withTempDB@ is obviously only useful for
-- test suites.
withTempDB :: (ConnectInfo -> IO a) -> IO a
withTempDB f = bracket createdir removeDirectoryRecursive $ \d ->
  flip finally (stopLocalDB d) $ do
    createLocalDB d
    configLocalDB d [("fsync", "fsync = off")
                    , ("synchronous_commit", "synchronous_commit = off")
                    , ("full_page_writes", "full_page_writes = off")]
    initLocalDB d >>= f
  where createdir = do
          tmp <- getTemporaryDirectory
          mkdtemp $ tmp </> "db."

-- | Reset a connection to its default state before re-cycling it for
-- another thread or request.
resetConnection :: Connection -> IO ()
resetConnection c = (void $ execute_ c "DISCARD ALL") `catch` \SqlError{} ->
  void $ execute_ c "ROLLBACK" >> execute_ c "DISCARD ALL"