-
Notifications
You must be signed in to change notification settings - Fork 2
/
Copy pathBootstrap.hs
506 lines (490 loc) · 22.1 KB
/
Bootstrap.hs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
{-# LANGUAGE ScopedTypeVariables #-}
{-# LANGUAGE TypeApplications #-}
-- | Copyright : (c) Crown Copyright GCHQ
module Bootstrap (nixBootstrap) where
import Bootstrap.Cli
( Command (CommandHelp, CommandRun, CommandVersion),
RunConfig (RunConfig, rcAllowDirty, rcFromScratch, rcNonInteractive, rcWithDevContainer),
allowDirtyFlagName,
fromScratchFlagName,
parseCommand,
showHelp,
)
import Bootstrap.Data.Bootstrappable
( Bootstrappable (bootstrapName),
)
import Bootstrap.Data.Bootstrappable.BuildNix (buildNixFor)
import Bootstrap.Data.Bootstrappable.DevContainer
( devContainerDockerComposeFor,
devContainerDockerfileFor,
devContainerJsonFor,
)
import Bootstrap.Data.Bootstrappable.Elm.ElmJson (elmJsonFor)
import Bootstrap.Data.Bootstrappable.Elm.IndexHtml (elmIndexHtmlFor)
import Bootstrap.Data.Bootstrappable.Elm.IndexJs (elmIndexJsFor)
import Bootstrap.Data.Bootstrappable.Elm.MainElm (mainElmFor)
import Bootstrap.Data.Bootstrappable.Elm.PackageJson (elmPackageJsonFor)
import Bootstrap.Data.Bootstrappable.Elm.Review.Config
( elmReviewConfigFor,
)
import Bootstrap.Data.Bootstrappable.Elm.Review.ElmJson
( elmReviewElmJsonFor,
)
import Bootstrap.Data.Bootstrappable.Envrc (Envrc (Envrc))
import Bootstrap.Data.Bootstrappable.FlakeNix (flakeNixFor)
import Bootstrap.Data.Bootstrappable.GitPodYml (gitPodYmlFor)
import Bootstrap.Data.Bootstrappable.Gitignore (gitignoreFor)
import Bootstrap.Data.Bootstrappable.GitlabCIConfig (gitlabCIConfigFor)
import Bootstrap.Data.Bootstrappable.Go.Modfile (goModfileFor)
import Bootstrap.Data.Bootstrappable.Haskell.LibHs (libHsFor)
import Bootstrap.Data.Bootstrappable.Haskell.MainHs (mainHsFor)
import Bootstrap.Data.Bootstrappable.Haskell.PackageYaml (packageYamlFor)
import Bootstrap.Data.Bootstrappable.Haskell.PreludeHs (preludeHsFor)
import Bootstrap.Data.Bootstrappable.HaskellPackagesNix (haskellPackagesNixFor)
import Bootstrap.Data.Bootstrappable.NixPreCommitHookConfig (nixPreCommitHookConfigFor)
import Bootstrap.Data.Bootstrappable.NixShellCompat (NixShellCompat (NixShellCompat))
import Bootstrap.Data.Bootstrappable.Python.Requirements (Requirements (Requirements))
import Bootstrap.Data.Bootstrappable.Readme
( Readme
( Readme,
readmeHasDevContainer,
readmeMBuildPlan,
readmeProjectName,
readmeProjectType
),
)
import Bootstrap.Data.Bootstrappable.Rust.CargoLock (cargoLockFor)
import Bootstrap.Data.Bootstrappable.Rust.CargoToml (cargoTomlFor)
import Bootstrap.Data.Bootstrappable.Rust.MainRs (mainRsFor)
import Bootstrap.Data.Bootstrappable.VSCodeExtensions (vsCodeExtensionsFileFor)
import Bootstrap.Data.Bootstrappable.VSCodeSettings (vsCodeSettingsFor)
import Bootstrap.Data.BuildPlan
( BuildPlan (BuildPlan),
BuildPlanFile,
bootstrap,
confirmBuildPlan,
toBuildPlanFile,
toBuildPlanFiles,
)
import Bootstrap.Data.Config
( LoadConfigResult
( LoadConfigResultError,
LoadConfigResultFound,
LoadConfigResultNotFound
),
NonFlakeConfigException,
configFor,
configProjectName,
configProjectType,
configSetUpContinuousIntegration,
configSetUpPreCommitHooks,
configSetUpVSCodeDevContainer,
loadConfig,
)
import Bootstrap.Data.ContinuousIntegration
( ContinuousIntegrationConfig (ContinuousIntegrationConfig),
)
import Bootstrap.Data.DevContainer (DevContainerConfig (DevContainerConfig, unDevContainerConfig))
import Bootstrap.Data.GHCVersion (printGHCVersion)
import Bootstrap.Data.HList (HList (HNil), (~:))
import Bootstrap.Data.PreCommitHook (PreCommitHooksConfig (PreCommitHooksConfig, unPreCommitHooksConfig))
import Bootstrap.Data.ProjectName (ProjectName, mkProjectName)
import Bootstrap.Data.ProjectType
( ArtefactId (ArtefactId),
ElmMode (ElmModeBare, ElmModeNode),
ElmModeSimple (ElmModeSimpleBare, ElmModeSimpleNode),
ElmOptions (ElmOptions, elmOptionElmMode, elmOptionProvideElmReview),
HasProjectSuperType (projectSuperType),
HaskellOptions (HaskellOptions),
InstallLombok (InstallLombok),
InstallMinishift (InstallMinishift),
JavaOptions (JavaOptions),
ProjectSuperType
( PSTElm,
PSTGo,
PSTHaskell,
PSTJava,
PSTMinimal,
PSTNode,
PSTPython,
PSTRust
),
ProjectType (Elm, Go, Haskell, Java, Minimal, Node, Python, Rust),
PythonVersion (Python39),
SetUpGoBuild (SetUpGoBuild),
SetUpJavaBuild (NoJavaBuild, SetUpJavaBuild),
jdkName,
nodePackageManagerName,
projectSuperTypeName,
promptHaskellProjectType,
)
import Bootstrap.Data.Version (MajorVersion (MajorVersion), displayMajorVersion)
import Bootstrap.Error (CanDieOnError (dieOnError', dieOnErrorWithPrefix))
import Bootstrap.GitPod (resetPermissionsInGitPod)
import Bootstrap.Monad (MonadBootstrap)
import Bootstrap.Nix.Evaluate (NixBinaryPaths, getAvailableGHCVersions, getNixBinaryPaths, getNixConfig, getNixVersion)
import Bootstrap.Nix.Flake (generateIntermediateFlake)
import Bootstrap.Terminal
( askIfReproducibleBuildRequired,
promptChoice,
promptNonemptyText,
promptYesNo,
promptYesNoWithDefault,
putErrorLn,
withAttribute,
withAttributes,
)
import Bootstrap.Terminal.Icon (icon)
import Bootstrap.Unix (git)
import Control.Lens ((^.))
import Control.Monad.Catch (catchAll)
import Data.Char (isSpace)
import qualified Data.Text as T
import Data.Version (showVersion)
import Paths_nix_bootstrap (version)
import Relude.Extra.Map (alter, toPairs)
import qualified Relude.Extra.Map as M
import qualified Relude.Unsafe as Unsafe
import System.Directory (doesFileExist, doesPathExist, getCurrentDirectory, removeFile)
import System.FilePath (takeFileName)
import System.Terminal
( MonadColorPrinter (blue, foreground, green, yellow),
MonadFormattingPrinter (bold, underlined),
MonadPrinter (putLn, putText, putTextLn),
runTerminalT,
withTerminal,
)
-- | The name of the current directory (excluding the rest of its path)
newtype CurrentDirectoryName = CurrentDirectoryName Text
nixBootstrap :: IO ()
nixBootstrap = withTerminal $ runTerminalT do
parseCommand >>= \case
CommandHelp errs -> showHelp errs
CommandRun initialRunConfig ->
do
mConfig <-
loadConfig >>= \case
LoadConfigResultFound c -> pure $ Just c
LoadConfigResultNotFound -> pure Nothing
LoadConfigResultError e -> case fromException @NonFlakeConfigException e of
Just _nonFlakeConfigException -> do
putErrorLn "Cannot upgrade config; nix-bootstrap now requires all projects to use flakes."
putErrorLn " Please bootstrap from scratch by deleting your nix-bootstrap config file and running nix-bootstrap again."
putErrorLn " Note that this will result in some old infrastructure remaining for you to clean up."
exitFailure
Nothing -> do
putErrorLn . toText $ displayException e
putErrorLn "Invalid config file found; please delete the file and try again."
exitFailure
showWelcomeMessage
resetPermissionsInGitPod
let nonInteractive = maybe False (const $ rcNonInteractive initialRunConfig) mConfig
runConfig =
initialRunConfig
{ rcNonInteractive = nonInteractive
}
(currentDirectoryName, nixBinaryPaths) <- performInitialChecks runConfig
buildPlan <-
if rcFromScratch runConfig
then do
projectName <- promptProjectName currentDirectoryName
preCommitHooksConfig <- promptPreCommitHooksConfig
generateIntermediateFlake nixBinaryPaths runConfig projectName
promptBuildConfig nixBinaryPaths runConfig projectName preCommitHooksConfig
else case mConfig of
Just cfg -> do
withAttributes [bold, foreground yellow] . putTextLn $
"Using configuration from state file; re-run with --"
<> fromScratchFlagName
<> " to ignore the old configuration and bootstrap from scratch."
makeBuildPlan
MakeBuildPlanArgs
{ mbpNixBinaryPaths = nixBinaryPaths,
mbpProjectName = cfg ^. configProjectName,
mbpProjectType = cfg ^. configProjectType,
mbpPreCommitHooksConfig = cfg ^. configSetUpPreCommitHooks,
mbpContinuousIntegrationConfig = cfg ^. configSetUpContinuousIntegration,
mbpDevContainerConfig = cfg ^. configSetUpVSCodeDevContainer,
mbpRunConfig = runConfig
}
Nothing -> do
projectName <- promptProjectName currentDirectoryName
preCommitHooksConfig <- promptPreCommitHooksConfig
generateIntermediateFlake nixBinaryPaths runConfig projectName
promptBuildConfig nixBinaryPaths runConfig projectName preCommitHooksConfig
confirmBuildPlan buildPlan >>= \case
Just confirmedBuildPlan -> do
bootstrap confirmedBuildPlan
liftIO (removeFile ".nix-bootstrap.toml" `catchAll` const pass)
resetPermissionsInGitPod
showCompletionMessage
trackAllFilesInGit
Nothing -> do
putTextLn "Okay, exiting."
exitFailure
CommandVersion -> do
putTextLn $ "nix-bootstrap version " <> toText (showVersion version)
performInitialChecks :: forall m. (MonadBootstrap m) => RunConfig -> m (CurrentDirectoryName, NixBinaryPaths)
performInitialChecks rc = do
currentDirectoryName <- toText . takeFileName <$> liftIO getCurrentDirectory
when (currentDirectoryName == "nix-bootstrap" || currentDirectoryName == "nix-bootstrap-hs") $
putErrorLn "In nix-bootstrap directory; exiting. (Test in another directory.)" >> exitFailure
runMaybeT getNixBinaryPaths >>= \case
Nothing -> do
putErrorLn "Could not get the nix binary path. If it's not installed, please install it by running the following command:"
putLn
putText " "
withAttribute underlined $ putTextLn "sh <(curl -L https://nixos.org/nix/install) --daemon"
exitFailure
Just nixBinaryPaths -> do
inGitRepo <- liftIO $ doesPathExist ".git"
if inGitRepo
then checkWorkingStateIsClean rc
else offerToInitialiseGitRepo
checkNixFlakesAreConfigured nixBinaryPaths
pure (CurrentDirectoryName currentDirectoryName, nixBinaryPaths)
where
checkWorkingStateIsClean :: RunConfig -> m ()
checkWorkingStateIsClean RunConfig {rcAllowDirty} = do
statusOutput <-
fmap toText
. dieOnErrorWithPrefix "Could not check current git working tree state"
. ExceptT
$ git ["status", "-s"]
unless (T.null (T.strip statusOutput)) do
withAttributes [bold, foreground yellow] $ putTextLn "Git working tree is not clean."
unless rcAllowDirty do
putErrorLn $ "Refusing to proceed without the --" <> allowDirtyFlagName <> " flag enabled."
exitFailure
checkNixFlakesAreConfigured :: NixBinaryPaths -> m ()
checkNixFlakesAreConfigured nixBinaryPaths = checkNixVersion *> checkExperimentalFeatures
where
checkNixVersion :: m ()
checkNixVersion = do
putTextLn "Checking nix flakes are configured"
nixVersion <- dieOnError' $ getNixVersion nixBinaryPaths
let requiredNixVersion = MajorVersion 2 4
when (nixVersion < requiredNixVersion) do
putErrorLn $
"Your nix version ("
<> displayMajorVersion nixVersion
<> ") doesn't support flakes. Please upgrade to nix >= "
<> displayMajorVersion requiredNixVersion
<> "."
exitFailure
checkExperimentalFeatures :: m ()
checkExperimentalFeatures = do
nixConfig <- dieOnError' $ getNixConfig nixBinaryPaths
case M.lookup "experimental-features" nixConfig of
Just experimentalFeaturesConfig -> do
let enabledFeatures = T.strip <$> T.split isSpace experimentalFeaturesConfig
unless
("flakes" `elem` enabledFeatures && "nix-command" `elem` enabledFeatures)
showFlakeConfigError
_ -> showFlakeConfigError
showFlakeConfigError :: m a
showFlakeConfigError = do
putErrorLn "Nix flakes are not properly configured on this system."
putText "Visit "
withAttribute underlined $ putText "https://nixos.wiki/wiki/flakes#Installing_flakes"
putTextLn " to find out how to configure them."
exitFailure
offerToInitialiseGitRepo :: forall m. (MonadBootstrap m) => m ()
offerToInitialiseGitRepo = do
putErrorLn "No `.git` found relative to the current path."
promptYesNo "Would you like to intialise git in the current directory?" >>= \case
True ->
void
. dieOnErrorWithPrefix "Something went wrong initialising the git repository; exiting"
. ExceptT
$ git ["init"]
False -> do
putTextLn "Okay, exiting."
exitFailure
showWelcomeMessage :: (MonadBootstrap m) => m ()
showWelcomeMessage = do
withAttributes [bold, foreground blue] $ putTextLn `mapM_` icon
putLn
putTextLn "This program will take you through the process of setting up a new project with nix-based infrastructure."
putTextLn "It is expected to be run in an empty git repository."
-- | Asks the user to enter a project name
promptProjectName :: (MonadBootstrap m) => CurrentDirectoryName -> m ProjectName
promptProjectName cdn@(CurrentDirectoryName currentDirectoryName) =
promptNonemptyText (Just currentDirectoryName) "Please enter a project name"
>>= ( \case
Just projectName -> pure projectName
Nothing -> do
putErrorLn "Invalid project name. Project names must begin with a letter and contain only alphanumerical characters, spaces, dashes (-), and underscores(_)."
promptProjectName cdn
)
. mkProjectName
-- | Asks the user to decide whether they'd like to have pre-commit hooks set up
promptPreCommitHooksConfig :: (MonadBootstrap m) => m PreCommitHooksConfig
promptPreCommitHooksConfig =
PreCommitHooksConfig
<$> promptYesNo "Would you like pre-commit hooks to be included in what nix-bootstrap sets up?"
promptBuildConfig ::
(MonadBootstrap m) =>
NixBinaryPaths ->
RunConfig ->
ProjectName ->
PreCommitHooksConfig ->
m BuildPlan
promptBuildConfig
mbpNixBinaryPaths
mbpRunConfig@RunConfig {rcWithDevContainer}
mbpProjectName
mbpPreCommitHooksConfig = do
mbpDevContainerConfig <-
DevContainerConfig
<$> promptYesNoWithDefault
(unDevContainerConfig <$> rcWithDevContainer)
"Would you like to set up a VSCode DevContainer?"
mbpProjectType <- promptProjectType mbpNixBinaryPaths mbpDevContainerConfig
mbpContinuousIntegrationConfig <-
ContinuousIntegrationConfig
<$> promptYesNo "Would you like to set up CI with GitLab CI?"
makeBuildPlan MakeBuildPlanArgs {..}
promptProjectType :: forall m. (MonadBootstrap m) => NixBinaryPaths -> DevContainerConfig -> m ProjectType
promptProjectType nixBinaryPaths devContainerConfig = do
superType <- promptChoice "Select a project type:" universeNonEmpty projectSuperTypeName
case superType of
PSTMinimal -> pure Minimal
PSTElm -> do
elmModeSimple <- promptChoice "How would you like to use Elm?" universeNonEmpty \case
ElmModeSimpleBare -> "On its own"
ElmModeSimpleNode -> "As part of a Node application"
elmOptionElmMode <- case elmModeSimple of
ElmModeSimpleBare -> pure ElmModeBare
ElmModeSimpleNode ->
ElmModeNode
<$> promptChoice "Select a node package manager:" universeNonEmpty nodePackageManagerName
elmOptionProvideElmReview <-
promptYesNo
"Would you like to set up elm-review with a default configuration for code quality?"
pure $ Elm ElmOptions {..}
PSTHaskell -> do
availableGHCVersions <- getAvailableGHCVersions nixBinaryPaths
case nonEmpty $ toList availableGHCVersions of
Just availableGHCVersions' -> do
haskellProjectType <- promptHaskellProjectType
ghcVersion <- promptChoice "Select a version of the Glasgow Haskell Compiler:" availableGHCVersions' printGHCVersion
pure . Haskell $ HaskellOptions ghcVersion haskellProjectType
Nothing -> putErrorLn "Could not find any versions of GHC in nixpkgs" *> exitFailure
PSTNode -> do
packageManager <- promptChoice "Select a package manager:" universeNonEmpty nodePackageManagerName
pure $ Node packageManager
PSTGo -> do
setUpGoBuild <- SetUpGoBuild <$> askIfReproducibleBuildRequired
pure $ Go setUpGoBuild
PSTJava -> do
jdk <- promptChoice "Select a Java Development Kit:" universeNonEmpty jdkName
installMinishift <- InstallMinishift <$> promptYesNo "Would you like to install Minishift?"
installLombok <-
InstallLombok
<$> promptYesNoWithDefault
(if unDevContainerConfig devContainerConfig then Nothing else Just False)
"Would you like to install the Lombok VSCode extension?"
setUpJavaBuild <-
askIfReproducibleBuildRequired >>= \case
False -> pure NoJavaBuild
True ->
SetUpJavaBuild . ArtefactId
<$> promptNonemptyText Nothing "Enter your Maven ArtefactId (e.g. the 'demo' in 'com.example.demo'): "
pure . Java $ JavaOptions installMinishift installLombok setUpJavaBuild jdk
PSTPython -> pure $ Python Python39
PSTRust -> pure Rust
data MakeBuildPlanArgs = MakeBuildPlanArgs
{ mbpNixBinaryPaths :: NixBinaryPaths,
mbpProjectName :: ProjectName,
mbpProjectType :: ProjectType,
mbpPreCommitHooksConfig :: PreCommitHooksConfig,
mbpContinuousIntegrationConfig :: ContinuousIntegrationConfig,
mbpDevContainerConfig :: DevContainerConfig,
mbpRunConfig :: RunConfig
}
makeBuildPlan :: (MonadBootstrap m) => MakeBuildPlanArgs -> m BuildPlan
makeBuildPlan mbp = case projectSuperType $ mbpProjectType mbp of
PSTPython -> do
putErrorLn "nix-bootstrap no longer supports Python. Please see https://github.com/gchq/nix-bootstrap/issues/6 for details."
exitFailure
_ -> makeNonPythonBuildPlan mbp
makeNonPythonBuildPlan :: forall m. (MonadBootstrap m) => MakeBuildPlanArgs -> m BuildPlan
makeNonPythonBuildPlan MakeBuildPlanArgs {..} = do
initialBuildPlanMap <- mkInitialBuildPlanMap
readme <- readmeWithBuildPlan . BuildPlan $ toPairs initialBuildPlanMap
pure . BuildPlan . toPairs $ alter (const $ pure readme) (bootstrapName initialReadme) initialBuildPlanMap
where
initialReadme :: Readme
initialReadme =
Readme
{ readmeProjectName = mbpProjectName,
readmeProjectType = mbpProjectType,
readmeHasDevContainer = mbpDevContainerConfig,
readmeMBuildPlan = Nothing
}
mkInitialBuildPlanMap :: m (Map FilePath BuildPlanFile)
mkInitialBuildPlanMap = do
let nixPreCommitHookConfig =
if unPreCommitHooksConfig mbpPreCommitHooksConfig
then Just $ nixPreCommitHookConfigFor mbpProjectType
else Nothing
buildNix = buildNixFor mbpProjectName mbpProjectType
goModfile <-
case mbpProjectType of
Go (SetUpGoBuild True) ->
Just <$> goModfileFor mbpNixBinaryPaths mbpProjectName
_ -> pure Nothing
pythonRequirementsFile <- case mbpProjectType of
Python _ -> do
requirementsExist <- liftIO $ doesFileExist (bootstrapName Requirements)
if requirementsExist then pure Nothing else pure $ Just Requirements
_ -> pure Nothing
fromList
<$> toBuildPlanFiles
( configFor mbpProjectName mbpProjectType mbpPreCommitHooksConfig mbpContinuousIntegrationConfig mbpDevContainerConfig
~: Envrc mbpPreCommitHooksConfig
~: gitignoreFor mbpProjectType mbpPreCommitHooksConfig
~: initialReadme
~: buildNix
~: flakeNixFor mbpProjectName mbpProjectType mbpPreCommitHooksConfig nixPreCommitHookConfig buildNix
~: NixShellCompat
~: nixPreCommitHookConfig
~: gitlabCIConfigFor mbpContinuousIntegrationConfig mbpProjectType nixPreCommitHookConfig
~: devContainerDockerComposeFor mbpDevContainerConfig mbpProjectName
~: devContainerDockerfileFor mbpDevContainerConfig
~: devContainerJsonFor mbpDevContainerConfig mbpProjectName mbpProjectType
~: vsCodeExtensionsFileFor mbpProjectType
~: vsCodeSettingsFor mbpDevContainerConfig
~: goModfile
~: pythonRequirementsFile
~: mainElmFor mbpProjectType
~: elmJsonFor mbpProjectType
~: elmReviewElmJsonFor mbpProjectType
~: elmReviewConfigFor mbpProjectType
~: elmPackageJsonFor mbpProjectType
~: elmIndexHtmlFor mbpProjectName mbpProjectType
~: elmIndexJsFor mbpProjectType
~: packageYamlFor mbpNixBinaryPaths mbpProjectName mbpProjectType
~: preludeHsFor mbpProjectType
~: libHsFor mbpProjectType
~: mainHsFor mbpProjectType
~: haskellPackagesNixFor mbpProjectType
~: cargoLockFor mbpProjectType mbpProjectName
~: cargoTomlFor mbpProjectType mbpProjectName
~: mainRsFor mbpProjectType
~: gitPodYmlFor mbpProjectType
~: HNil
)
readmeWithBuildPlan :: BuildPlan -> m BuildPlanFile
readmeWithBuildPlan initialPlan =
snd . Unsafe.fromJust
<$> toBuildPlanFile initialReadme {readmeMBuildPlan = Just initialPlan}
showCompletionMessage :: (MonadBootstrap m) => m ()
showCompletionMessage = do
putLn
putTextLn "All steps have finished successfully."
putTextLn "Once complete, you will be advised to run direnv allow if you're using direnv."
withAttributes [bold, foreground green] $ putTextLn "nix-bootstrap complete!"
trackAllFilesInGit :: (MonadBootstrap m) => m ()
trackAllFilesInGit = void $ git ["add", "-N", "."]