Make the hook config file path include the instance host

Before this patch, if you ran more than 1 instance as the same OS user, they'd
use the same config file path and overwrite it and cause post hooks to have
errors due to wrong config being used.
This commit is contained in:
fr33domlover 2019-10-20 09:19:49 +00:00
parent 65d730393a
commit a700dc4208
7 changed files with 63 additions and 41 deletions

View file

@ -34,15 +34,18 @@ import qualified Data.ByteString as B
import qualified Data.Text as T import qualified Data.Text as T
import qualified Data.Text.IO as TIO import qualified Data.Text.IO as TIO
writeDefaultsFile :: FilePath -> FilePath -> Text -> Text -> IO () writeDefaultsFile :: FilePath -> FilePath -> Text -> Text -> Text -> IO ()
writeDefaultsFile path cmd sharer repo = do writeDefaultsFile path cmd authority sharer repo = do
let file = path </> "_darcs" </> "prefs" </> "defaults" let file = path </> "_darcs" </> "prefs" </> "defaults"
TIO.writeFile file $ defaultsContent cmd sharer repo TIO.writeFile file $ defaultsContent cmd authority sharer repo
setFileMode file $ ownerReadMode .|. ownerWriteMode setFileMode file $ ownerReadMode .|. ownerWriteMode
where where
defaultsContent :: FilePath -> Text -> Text -> Text defaultsContent :: FilePath -> Text -> Text -> Text -> Text
defaultsContent hook sharer repo = defaultsContent hook authority sharer repo =
T.concat ["apply posthook ", T.pack hook, " ", sharer, " ", repo] T.concat
[ "apply posthook "
, T.pack hook, " ", authority, " ", sharer, " ", repo
]
{- {-
initialRepoTree :: FileName -> DirTree B.ByteString initialRepoTree :: FileName -> DirTree B.ByteString
@ -75,18 +78,20 @@ createRepo
-> FilePath -> FilePath
-- ^ Path of Vervis hook program -- ^ Path of Vervis hook program
-> Text -> Text
-- ^ Instance HTTP authority
-> Text
-- ^ Repo sharer textual ID -- ^ Repo sharer textual ID
-> Text -> Text
-- ^ Repo textual ID -- ^ Repo textual ID
-> IO () -> IO ()
createRepo parent name cmd sharer repo = do createRepo parent name cmd authority sharer repo = do
let path = parent </> name let path = parent </> name
createDirectory path createDirectory path
let settings = proc "darcs" ["init", "--no-working-dir", "--repodir", path] let settings = proc "darcs" ["init", "--no-working-dir", "--repodir", path]
(_, _, _, ph) <- createProcess settings (_, _, _, ph) <- createProcess settings
ec <- waitForProcess ph ec <- waitForProcess ph
case ec of case ec of
ExitSuccess -> writeDefaultsFile path cmd sharer repo ExitSuccess -> writeDefaultsFile path cmd authority sharer repo
ExitFailure n -> error $ "darcs init failed with exit code " ++ show n ExitFailure n -> error $ "darcs init failed with exit code " ++ show n
readPristineRoot :: FilePath -> IO (Maybe Int, Hash) readPristineRoot :: FilePath -> IO (Maybe Int, Hash)

View file

@ -54,18 +54,21 @@ instance SpecToEventTime GitTime where
specToEventTime = specToEventTime . gitTimeUTC specToEventTime = specToEventTime . gitTimeUTC
specsToEventTimes = specsToEventTimes . fmap gitTimeUTC specsToEventTimes = specsToEventTimes . fmap gitTimeUTC
hookContent :: FilePath -> Text -> Text -> Text hookContent :: FilePath -> Text -> Text -> Text -> Text
hookContent hook sharer repo = hookContent hook authority sharer repo =
T.concat ["#!/bin/sh\nexec ", T.pack hook, " ", sharer, " ", repo] T.concat
[ "#!/bin/sh\nexec ", T.pack hook
, " ", authority, " ", sharer, " ", repo
]
writeHookFile :: FilePath -> FilePath -> Text -> Text -> IO () writeHookFile :: FilePath -> FilePath -> Text -> Text -> Text -> IO ()
writeHookFile path cmd sharer repo = do writeHookFile path cmd authority sharer repo = do
let file = path </> "hooks" </> "post-receive" let file = path </> "hooks" </> "post-receive"
TIO.writeFile file $ hookContent cmd sharer repo TIO.writeFile file $ hookContent cmd authority sharer repo
setFileMode file ownerModes setFileMode file ownerModes
initialRepoTree :: FilePath -> Text -> Text -> FileName -> DirTree Text initialRepoTree :: FilePath -> Text -> Text -> Text -> FileName -> DirTree Text
initialRepoTree hook sharer repo dir = initialRepoTree hook authority sharer repo dir =
Dir dir Dir dir
[ Dir "branches" [] [ Dir "branches" []
, File "config" , File "config"
@ -77,7 +80,7 @@ initialRepoTree hook sharer repo dir =
"Unnamed repository; edit this file to name the repository." "Unnamed repository; edit this file to name the repository."
, File "HEAD" "ref: refs/heads/master" , File "HEAD" "ref: refs/heads/master"
, Dir "hooks" , Dir "hooks"
[ File "post-receive" $ hookContent hook sharer repo [ File "post-receive" $ hookContent hook authority sharer repo
] ]
, Dir "info" , Dir "info"
[ File "exclude" "" [ File "exclude" ""
@ -105,12 +108,14 @@ createRepo
-> FilePath -> FilePath
-- ^ Path of Vervis hook program -- ^ Path of Vervis hook program
-> Text -> Text
-- ^ Instance HTTP authority
-> Text
-- ^ Repo sharer textual ID -- ^ Repo sharer textual ID
-> Text -> Text
-- ^ Repo textual ID -- ^ Repo textual ID
-> IO () -> IO ()
createRepo path name cmd sharer repo = do createRepo path name cmd authority sharer repo = do
let tree = path :/ initialRepoTree cmd sharer repo name let tree = path :/ initialRepoTree cmd authority sharer repo name
result <- writeDirectoryWith TIO.writeFile tree result <- writeDirectoryWith TIO.writeFile tree
let errs = failures $ dirTree result let errs = failures $ dirTree result
when (not . null $ errs) $ when (not . null $ errs) $

View file

@ -61,13 +61,13 @@ import Yesod.Persist.Core
import Yesod.Static import Yesod.Static
import qualified Data.Text as T (unpack) import qualified Data.Text as T (unpack)
import qualified Data.HashMap.Strict as M (empty)
import Database.Persist.Schema.PostgreSQL (schemaBackend) import Database.Persist.Schema.PostgreSQL (schemaBackend)
import Yesod.Mail.Send (runMailer) import Yesod.Mail.Send (runMailer)
import Control.Concurrent.ResultShare import Control.Concurrent.ResultShare
import Data.KeyFile import Data.KeyFile
import Network.FedURI
import Yesod.MonadSite import Yesod.MonadSite
import Control.Concurrent.Local import Control.Concurrent.Local
@ -187,8 +187,8 @@ makeFoundation appSettings = do
-- Perform database migration using our application's logging settings. -- Perform database migration using our application's logging settings.
--runLoggingT (runSqlPool (runMigration migrateAll) pool) logFunc --runLoggingT (runSqlPool (runMigration migrateAll) pool) logFunc
let hLocal = appInstanceHost appSettings
flip runWorker app $ runSiteDB $ do flip runWorker app $ runSiteDB $ do
let hLocal = appInstanceHost appSettings
r <- migrateDB hLocal hashidsCtx r <- migrateDB hLocal hashidsCtx
case r of case r of
Left err -> do Left err -> do
@ -202,7 +202,8 @@ makeFoundation appSettings = do
writePostReceiveHooks writePostReceiveHooks
writePostApplyHooks writePostApplyHooks
writeHookConfig Config let hostString = T.unpack $ renderAuthority hLocal
writeHookConfig hostString Config
{ configSecret = hookSecretText appHookSecret { configSecret = hookSecretText appHookSecret
, configPort = fromIntegral $ appPort appSettings , configPort = fromIntegral $ appPort appSettings
, configMaxCommits = 20 , configMaxCommits = 20

View file

@ -63,6 +63,8 @@ import qualified Database.Esqueleto as E
import qualified Development.Darcs.Internal.Patch.Parser as P import qualified Development.Darcs.Internal.Patch.Parser as P
import Network.FedURI
import Yesod.ActivityPub
import Yesod.MonadSite import Yesod.MonadSite
import Darcs.Local.Repository import Darcs.Local.Repository
@ -334,6 +336,8 @@ writePostApplyHooks = do
E.where_ $ r E.^. RepoVcs E.==. E.val VCSDarcs E.where_ $ r E.^. RepoVcs E.==. E.val VCSDarcs
return (s E.^. SharerIdent, r E.^. RepoIdent) return (s E.^. SharerIdent, r E.^. RepoIdent)
hook <- asksSite $ appPostApplyHookFile . appSettings hook <- asksSite $ appPostApplyHookFile . appSettings
authority <- asksSite $ renderAuthority . siteInstanceHost
for_ repos $ \ (E.Value shr, E.Value rp) -> do for_ repos $ \ (E.Value shr, E.Value rp) -> do
path <- askRepoDir shr rp path <- askRepoDir shr rp
liftIO $ writeDefaultsFile path hook (shr2text shr) (rp2text rp) liftIO $
writeDefaultsFile path hook authority (shr2text shr) (rp2text rp)

View file

@ -67,6 +67,8 @@ import qualified Data.Text.Encoding.Error as TE (lenientDecode)
import qualified Data.Vector as V (fromList) import qualified Data.Vector as V (fromList)
import qualified Database.Esqueleto as E import qualified Database.Esqueleto as E
import Network.FedURI
import Yesod.ActivityPub
import Yesod.MonadSite import Yesod.MonadSite
import Data.ByteString.Char8.Local (takeLine) import Data.ByteString.Char8.Local (takeLine)
@ -339,6 +341,7 @@ writePostReceiveHooks = do
E.where_ $ r E.^. RepoVcs E.==. E.val VCSGit E.where_ $ r E.^. RepoVcs E.==. E.val VCSGit
return (s E.^. SharerIdent, r E.^. RepoIdent) return (s E.^. SharerIdent, r E.^. RepoIdent)
hook <- asksSite $ appPostReceiveHookFile . appSettings hook <- asksSite $ appPostReceiveHookFile . appSettings
authority <- asksSite $ renderAuthority . siteInstanceHost
for_ repos $ \ (E.Value shr, E.Value rp) -> do for_ repos $ \ (E.Value shr, E.Value rp) -> do
path <- askRepoDir shr rp path <- askRepoDir shr rp
liftIO $ writeHookFile path hook (shr2text shr) (rp2text rp) liftIO $ writeHookFile path hook authority (shr2text shr) (rp2text rp)

View file

@ -86,6 +86,7 @@ import qualified Data.Text.Lazy.Encoding as L (decodeUtf8With)
import qualified Database.Esqueleto as E import qualified Database.Esqueleto as E
import Data.MediaType import Data.MediaType
import Network.FedURI
import Web.ActivityPub hiding (Repo) import Web.ActivityPub hiding (Repo)
import Yesod.ActivityPub import Yesod.ActivityPub
import Yesod.FedURI import Yesod.FedURI
@ -145,6 +146,7 @@ postReposR user = do
liftIO $ createDirectoryIfMissing True parent liftIO $ createDirectoryIfMissing True parent
let repoName = let repoName =
unpack $ CI.foldedCase $ unRpIdent $ nrpIdent nrp unpack $ CI.foldedCase $ unRpIdent $ nrpIdent nrp
host <- asksSite siteInstanceHost
case nrpVcs nrp of case nrpVcs nrp of
VCSDarcs -> do VCSDarcs -> do
hook <- getsYesod $ appPostApplyHookFile . appSettings hook <- getsYesod $ appPostApplyHookFile . appSettings
@ -153,6 +155,7 @@ postReposR user = do
parent parent
repoName repoName
hook hook
(renderAuthority host)
(shr2text user) (shr2text user)
(rp2text $ nrpIdent nrp) (rp2text $ nrpIdent nrp)
VCSGit -> do VCSGit -> do
@ -162,6 +165,7 @@ postReposR user = do
parent parent
repoName repoName
hook hook
(renderAuthority host)
(shr2text user) (shr2text user)
(rp2text $ nrpIdent nrp) (rp2text $ nrpIdent nrp)
pid <- requireAuthId pid <- requireAuthId

View file

@ -149,15 +149,15 @@ instance FromJSON Push
instance ToJSON Push instance ToJSON Push
getVervisCachePath :: IO FilePath getVervisCachePath :: String -> IO FilePath
getVervisCachePath = getXdgDirectory XdgCache "vervis" getVervisCachePath host = (</> host) <$> getXdgDirectory XdgCache "vervis"
hookConfigFileName :: String hookConfigFileName :: String
hookConfigFileName = "hook-config.json" hookConfigFileName = "hook-config.json"
writeHookConfig :: Config -> IO () writeHookConfig :: String -> Config -> IO ()
writeHookConfig config = do writeHookConfig host config = do
cachePath <- getVervisCachePath cachePath <- getVervisCachePath host
createDirectoryIfMissing True cachePath createDirectoryIfMissing True cachePath
encodeFile (cachePath </> hookConfigFileName) config encodeFile (cachePath </> hookConfigFileName) config
@ -306,17 +306,17 @@ reportNewCommits config sharer repo = do
postReceive :: IO () postReceive :: IO ()
postReceive = do postReceive = do
cachePath <- getVervisCachePath (host, sharer, repo) <- do
args <- getArgs
case args of
[h, s, r] -> return (h, T.pack s, T.pack r)
_ -> die "Unexpected number of arguments"
cachePath <- getVervisCachePath host
config <- do config <- do
mc <- decodeFileStrict' $ cachePath </> hookConfigFileName mc <- decodeFileStrict' $ cachePath </> hookConfigFileName
case mc of case mc of
Nothing -> die "Parsing hook config failed" Nothing -> die "Parsing hook config failed"
Just c -> return c Just c -> return c
args <- getArgs
(sharer, repo) <-
case args of
[s, r] -> return (T.pack s, T.pack r)
_ -> die "Unexpected number of arguments"
reportNewCommits config sharer repo reportNewCommits config sharer repo
reportNewPatches :: Config -> Text -> Text -> IO () reportNewPatches :: Config -> Text -> Text -> IO ()
@ -416,15 +416,15 @@ reportNewPatches config sharer repo = do
postApply :: IO () postApply :: IO ()
postApply = do postApply = do
cachePath <- getVervisCachePath (host, sharer, repo) <- do
args <- getArgs
case args of
[h, s, r] -> return (h, T.pack s, T.pack r)
_ -> die "Unexpected number of arguments"
cachePath <- getVervisCachePath host
config <- do config <- do
mc <- decodeFileStrict' $ cachePath </> hookConfigFileName mc <- decodeFileStrict' $ cachePath </> hookConfigFileName
case mc of case mc of
Nothing -> die "Parsing hook config failed" Nothing -> die "Parsing hook config failed"
Just c -> return c Just c -> return c
args <- getArgs
(sharer, repo) <-
case args of
[s, r] -> return (T.pack s, T.pack r)
_ -> die "Unexpected number of arguments"
reportNewPatches config sharer repo reportNewPatches config sharer repo