Custom cradle support
This commit is contained in:
		
							parent
							
								
									247e4e0e76
								
							
						
					
					
						commit
						5d9d6f5630
					
				| @ -95,20 +95,26 @@ cabalHelperCache = Cached { | |||||||
|                 , a == a' |                 , a == a' | ||||||
|                 ] |                 ] | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| withCabal :: (MonadIO m, GmEnv m) => m a -> m a | withCabal :: (MonadIO m, GmEnv m) => m a -> m a | ||||||
| withCabal action = do | withCabal action = do | ||||||
|     crdl <- cradle |     crdl <- cradle | ||||||
|     opts <- options |     opts <- options | ||||||
|     liftIO $ whenM (isSetupConfigOutOfDate <$> getCurrentWorld crdl) $ |     liftIO $ whenM (isSetupConfigOutOfDate <$> getCurrentWorld crdl) $ | ||||||
|         withDirectory_ (cradleRootDir crdl) $ do |         withDirectory_ (cradleRootDir crdl) $ do | ||||||
|             let progOpts = |             let pkgDbArgs = "--package-db=clear" : map pkgDbArg (cradlePkgDbStack crdl) | ||||||
|  |                 progOpts = | ||||||
|                     [ "--with-ghc=" ++ T.ghcProgram opts ] |                     [ "--with-ghc=" ++ T.ghcProgram opts ] | ||||||
|                     -- Only pass ghc-pkg if it was actually set otherwise we |                     -- Only pass ghc-pkg if it was actually set otherwise we | ||||||
|                     -- might break cabal's guessing logic |                     -- might break cabal's guessing logic | ||||||
|                     ++ if T.ghcPkgProgram opts /= T.ghcPkgProgram defaultOptions |                     ++ if T.ghcPkgProgram opts /= T.ghcPkgProgram defaultOptions | ||||||
|                          then [ "--with-ghc-pkg=" ++ T.ghcPkgProgram opts ] |                          then [ "--with-ghc-pkg=" ++ T.ghcPkgProgram opts ] | ||||||
|                          else [] |                          else [] | ||||||
|  |                     ++ pkgDbArgs | ||||||
|             void $ readProcess (T.cabalProgram opts) ("configure":progOpts) "" |             void $ readProcess (T.cabalProgram opts) ("configure":progOpts) "" | ||||||
|             writeAutogenFiles $ cradleRootDir crdl </> "dist" |             writeAutogenFiles $ cradleRootDir crdl </> "dist" | ||||||
|     action |     action | ||||||
|  | 
 | ||||||
|  | pkgDbArg :: GhcPkgDb -> String | ||||||
|  | pkgDbArg GlobalDb      = "--package-db=global" | ||||||
|  | pkgDbArg UserDb        = "--package-db=user" | ||||||
|  | pkgDbArg (PackageDb p) = "--package-db=" ++ p | ||||||
|  | |||||||
| @ -28,7 +28,7 @@ findCradle = findCradle' =<< getCurrentDirectory | |||||||
| 
 | 
 | ||||||
| findCradle' :: FilePath -> IO Cradle | findCradle' :: FilePath -> IO Cradle | ||||||
| findCradle' dir = run $ do | findCradle' dir = run $ do | ||||||
|     (cabalCradle dir `mplus`  sandboxCradle dir `mplus` plainCradle dir) |     (customCradle dir `mplus` cabalCradle dir `mplus`  sandboxCradle dir `mplus` plainCradle dir) | ||||||
|  where run a = fillTempDir =<< (fromJust <$> runMaybeT a) |  where run a = fillTempDir =<< (fromJust <$> runMaybeT a) | ||||||
| 
 | 
 | ||||||
| findSpecCradle :: FilePath -> IO Cradle | findSpecCradle :: FilePath -> IO Cradle | ||||||
| @ -52,6 +52,21 @@ fillTempDir crdl = do | |||||||
|   tmpDir <- liftIO $ newTempDir (cradleRootDir crdl) |   tmpDir <- liftIO $ newTempDir (cradleRootDir crdl) | ||||||
|   return crdl { cradleTempDir = tmpDir } |   return crdl { cradleTempDir = tmpDir } | ||||||
| 
 | 
 | ||||||
|  | customCradle :: FilePath -> MaybeT IO Cradle | ||||||
|  | customCradle wdir = do | ||||||
|  |     cabalFile <- MaybeT $ findCabalFile wdir | ||||||
|  |     let cabalDir = takeDirectory cabalFile | ||||||
|  |     cradleFile <- MaybeT $ findCradleFile cabalDir | ||||||
|  |     tmpDir <- liftIO $ newTempDir cabalDir | ||||||
|  |     pkgDbStack <- liftIO $ parseCradle cradleFile | ||||||
|  |     return Cradle { | ||||||
|  |         cradleCurrentDir = wdir | ||||||
|  |       , cradleRootDir    = cabalDir | ||||||
|  |       , cradleTempDir    = tmpDir | ||||||
|  |       , cradleCabalFile  = Just cabalFile | ||||||
|  |       , cradlePkgDbStack = pkgDbStack | ||||||
|  |       } | ||||||
|  | 
 | ||||||
| cabalCradle :: FilePath -> MaybeT IO Cradle | cabalCradle :: FilePath -> MaybeT IO Cradle | ||||||
| cabalCradle wdir = do | cabalCradle wdir = do | ||||||
|     cabalFile <- MaybeT $ findCabalFile wdir |     cabalFile <- MaybeT $ findCabalFile wdir | ||||||
| @ -95,3 +110,21 @@ getPackageDbStack :: FilePath -- ^ Project Directory (where the | |||||||
|                   -> IO [GhcPkgDb] |                   -> IO [GhcPkgDb] | ||||||
| getPackageDbStack cdir = | getPackageDbStack cdir = | ||||||
|     ([GlobalDb] ++) . maybe [UserDb] return <$> getSandboxDb cdir |     ([GlobalDb] ++) . maybe [UserDb] return <$> getSandboxDb cdir | ||||||
|  | 
 | ||||||
|  | -- Just for testing | ||||||
|  | findCradleWithoutSandbox :: IO Cradle | ||||||
|  | findCradleWithoutSandbox = do | ||||||
|  |     cradle <- findCradle | ||||||
|  |     return cradle { cradlePkgDbStack = [GlobalDb]} -- FIXME | ||||||
|  | 
 | ||||||
|  | 
 | ||||||
|  | parseCradle :: FilePath -> IO [GhcPkgDb] | ||||||
|  | parseCradle path = do | ||||||
|  |     source <- readFile path | ||||||
|  |     return $ parseCradle' source | ||||||
|  |   where | ||||||
|  |     parseCradle' source = map parsePkgDb $ filter (not . null) $ lines source | ||||||
|  | 
 | ||||||
|  |     parsePkgDb "global" = GlobalDb | ||||||
|  |     parsePkgDb "user" = UserDb | ||||||
|  |     parsePkgDb s = PackageDb s | ||||||
|  | |||||||
| @ -193,7 +193,6 @@ cabalBuildPlatform = unsafePerformIO $ buildPlatform | |||||||
| packageCache :: String | packageCache :: String | ||||||
| packageCache = "package.cache" | packageCache = "package.cache" | ||||||
| 
 | 
 | ||||||
| 
 |  | ||||||
| -- | Filename of the symbol table cache file. | -- | Filename of the symbol table cache file. | ||||||
| symbolCache :: Cradle -> FilePath | symbolCache :: Cradle -> FilePath | ||||||
| symbolCache crdl = cradleTempDir crdl </> symbolCacheFile | symbolCache crdl = cradleTempDir crdl </> symbolCacheFile | ||||||
| @ -206,3 +205,13 @@ resolvedComponentsCacheFile = setupConfigPath <.> "ghc-mod.resolved-components" | |||||||
| 
 | 
 | ||||||
| cabalHelperCacheFile :: String | cabalHelperCacheFile :: String | ||||||
| cabalHelperCacheFile = setupConfigPath <.> "ghc-mod.cabal-helper" | cabalHelperCacheFile = setupConfigPath <.> "ghc-mod.cabal-helper" | ||||||
|  | 
 | ||||||
|  | -- | @findCradleFile dir@. Searches for a @.ghc-mod.cradle@ file in @dir@. | ||||||
|  | -- If it exists in the given directory it is returned otherwise @findCradleFile@ returns @Nothing@ | ||||||
|  | findCradleFile :: FilePath -> IO (Maybe FilePath) | ||||||
|  | findCradleFile directory = do | ||||||
|  |     let path = directory </> "ghc-mod.cradle" | ||||||
|  |     exists <- doesFileExist $ path | ||||||
|  |     case exists of | ||||||
|  |         True  -> return $ Just path | ||||||
|  |         False -> return Nothing | ||||||
|  | |||||||
| @ -272,7 +272,7 @@ data GhcModError | |||||||
| 
 | 
 | ||||||
|     | GMECabalStateFile GMConfigStateFileError |     | GMECabalStateFile GMConfigStateFileError | ||||||
|       -- ^ Reading Cabal's state configuration file falied somehow. |       -- ^ Reading Cabal's state configuration file falied somehow. | ||||||
|       deriving (Eq,Show,Typeable) |     deriving (Eq,Show,Typeable) | ||||||
| 
 | 
 | ||||||
| instance Error GhcModError where | instance Error GhcModError where | ||||||
|     noMsg = GMENoMsg |     noMsg = GMENoMsg | ||||||
|  | |||||||
							
								
								
									
										13
									
								
								README.md
									
									
									
									
									
								
							
							
						
						
									
										13
									
								
								README.md
									
									
									
									
									
								
							| @ -44,6 +44,19 @@ Make sure you're not using the MELPA version of `ghc.el` otherwise you might get | |||||||
| all sorts of nasty conflicts. | all sorts of nasty conflicts. | ||||||
| 
 | 
 | ||||||
| 
 | 
 | ||||||
|  | ## Custom ghc-mod cradle | ||||||
|  | 
 | ||||||
|  | To customize the package databases used by `ghc-mod`, put a file called `.ghc-mod.cradle` beside the `.cabal` file with the following syntax: | ||||||
|  | 
 | ||||||
|  | ``` | ||||||
|  | temp directory root | ||||||
|  | package db 1 | ||||||
|  | ... | ||||||
|  | package db n | ||||||
|  | ``` | ||||||
|  | 
 | ||||||
|  | each package database line is either a *path* to a package database, or `global` or `user`. | ||||||
|  | 
 | ||||||
| ## IRC | ## IRC | ||||||
| 
 | 
 | ||||||
| If you have any problems, suggestions, comments swing by | If you have any problems, suggestions, comments swing by | ||||||
|  | |||||||
| @ -71,4 +71,11 @@ spec = do | |||||||
|                 cradleCabalFile  res `shouldBe` |                 cradleCabalFile  res `shouldBe` | ||||||
|                   Just ("test" </> "data" </> "broken-sandbox" </> "dummy.cabal") |                   Just ("test" </> "data" </> "broken-sandbox" </> "dummy.cabal") | ||||||
| 
 | 
 | ||||||
|  |         it "uses the custom cradle file if present" $ do | ||||||
|  |             withDirectory "test/data/custom-cradle" $ \dir -> do | ||||||
|  |                 res <- relativeCradle dir <$> findCradle | ||||||
|  |                 cradleCurrentDir res `shouldBe` "test" </> "data" </> "custom-cradle" | ||||||
|  |                 cradleRootDir res    `shouldBe` "test" </> "data" </> "custom-cradle" | ||||||
|  |                 cradleCabalFile res  `shouldBe` Just ("test" </> "data" </> "custom-cradle" </> "dummy.cabal") | ||||||
|  |                 cradlePkgDbStack res `shouldBe` [PackageDb "a/packages", GlobalDb, PackageDb "b/packages", UserDb, PackageDb "c/packages"] | ||||||
|                 cradlePkgDbStack res `shouldBe` [GlobalDb, UserDb] |                 cradlePkgDbStack res `shouldBe` [GlobalDb, UserDb] | ||||||
|  | |||||||
							
								
								
									
										5
									
								
								test/data/custom-cradle/.ghc-mod.cradle
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										5
									
								
								test/data/custom-cradle/.ghc-mod.cradle
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1,5 @@ | |||||||
|  | a/packages | ||||||
|  | global | ||||||
|  | b/packages | ||||||
|  | user | ||||||
|  | c/packages | ||||||
							
								
								
									
										1
									
								
								test/data/custom-cradle/dummy.cabal
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										1
									
								
								test/data/custom-cradle/dummy.cabal
									
									
									
									
									
										Normal file
									
								
							| @ -0,0 +1 @@ | |||||||
|  | dummy | ||||||
		Loading…
	
		Reference in New Issue
	
	Block a user
	 Daniel Vigovszky
						Daniel Vigovszky