summaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorGravatar Joey Hess <joey@kitenet.net>2012-06-11 01:20:19 -0400
committerGravatar Joey Hess <joey@kitenet.net>2012-06-11 01:20:19 -0400
commit0b3e2bed783ade691baf60a4198aaa1034b28440 (patch)
treeb0709933e47c89dbb6518fc316584538874ccc57
parentd5884388b09347835df599d8a0dcea77e6795c10 (diff)
add a pid file
Writes pid to a file. Is supposed to take an exclusive lock, but that's not working, and it's too late for me to understand why.
-rw-r--r--Command/Watch.hs3
-rw-r--r--Locations.hs5
-rw-r--r--Utility/Daemon.hs45
3 files changed, 37 insertions, 16 deletions
diff --git a/Command/Watch.hs b/Command/Watch.hs
index 480bd3ede..8b8e5578d 100644
--- a/Command/Watch.hs
+++ b/Command/Watch.hs
@@ -67,7 +67,8 @@ start foreground = notBareRepo $ withStateMVar $ \st -> do
liftIO $ watch st
else do
logfd <- liftIO . openLog =<< fromRepo gitAnnexLogFile
- liftIO $ daemonize logfd False $ watch st
+ pidfile <- fromRepo gitAnnexPidFile
+ liftIO $ daemonize logfd (Just pidfile) False $ watch st
stop
watch :: MVar Annex.AnnexState -> IO ()
diff --git a/Locations.hs b/Locations.hs
index 1dcfdc0ff..9d27fbdae 100644
--- a/Locations.hs
+++ b/Locations.hs
@@ -23,6 +23,7 @@ module Locations (
gitAnnexIndex,
gitAnnexIndexLock,
gitAnnexIndexDirty,
+ gitAnnexPidFile,
gitAnnexLogFile,
gitAnnexSshDir,
gitAnnexRemotesDir,
@@ -146,6 +147,10 @@ gitAnnexIndexLock r = gitAnnexDir r </> "index.lck"
gitAnnexIndexDirty :: Git.Repo -> FilePath
gitAnnexIndexDirty r = gitAnnexDir r </> "index.dirty"
+{- Pid file for daemon mode. -}
+gitAnnexPidFile :: Git.Repo -> FilePath
+gitAnnexPidFile r = gitAnnexDir r </> "daemon.pid"
+
{- Log file for daemon mode. -}
gitAnnexLogFile :: Git.Repo -> FilePath
gitAnnexLogFile r = gitAnnexDir r </> "daemon.log"
diff --git a/Utility/Daemon.hs b/Utility/Daemon.hs
index be3df17b7..eeec0bfcc 100644
--- a/Utility/Daemon.hs
+++ b/Utility/Daemon.hs
@@ -1,4 +1,4 @@
-{- daemon functions
+{- daemon support
-
- Copyright 2012 Joey Hess <joey@kitenet.net>
-
@@ -7,24 +7,28 @@
module Utility.Daemon where
+import Common
+
import System.Posix
-import System.Directory
-import System.Exit
-import Control.Monad
{- Run an action as a daemon, with all output sent to a file descriptor.
-
- - Does not return. -}
-daemonize :: Fd -> Bool -> IO () -> IO ()
-daemonize logfd changedirectory a = do
- _ <- forkProcess child1
- end
+ - Can write its pid to a file, to guard against multiple instances
+ - running and allow easy termination.
+ -
+ - When successful, does not return. -}
+daemonize :: Fd -> Maybe FilePath -> Bool -> IO () -> IO ()
+daemonize logfd pidfile changedirectory a = do
+ pidfd <- lockpidfile
+ _ <- forkProcess $ child1 pidfd
+ out
where
- child1 = do
+ child1 pidfd = do
_ <- createSession
- _ <- forkProcess child2
- end
- child2 = do
+ _ <- forkProcess $ child2 pidfd
+ out
+ child2 pidfd = do
+ writepidfile pidfd
when changedirectory $
setCurrentDirectory "/"
nullfd <- openFd "/dev/null" ReadOnly Nothing defaultFileFlags
@@ -32,8 +36,19 @@ daemonize logfd changedirectory a = do
mapM_ (redir logfd) [stdOutput, stdError]
closeFd logfd
a
- end
+ out
+ lockpidfile = case pidfile of
+ Just file -> do
+ fd <- openFd file ReadWrite (Just stdFileMode) defaultFileFlags
+ setLock fd (WriteLock, AbsoluteSeek, 0, 0)
+ return $ Just fd
+ Nothing -> return Nothing
+ writepidfile pidfd =
+ case pidfd of
+ Just fd -> void $
+ fdWrite fd =<< show <$> getProcessID
+ Nothing -> return ()
redir newh h = do
closeFd h
dupTo newh h
- end = exitImmediately ExitSuccess
+ out = exitImmediately ExitSuccess