summaryrefslogtreecommitdiff
path: root/doc/haskell_newbie.mdwn
blob: ec42629ce09a2310127ccd2d94b3bbc6c94d7f49 (plain)
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
[[!meta title="Propellor configuration for the Haskell newbie"]]

Propellor's config file is written in Haskell, and
[Haskell](http://www.haskell.org/) is invaluable to extend Propellor with
your own custom properties. But you don't need to know about monads to
configure Propellor!

Let's take a quick tour of the `config.hs` file..

[[!format haskell """
-- This is the main configuration file for Propellor, and is used to build
-- the propellor program.
"""]]

So, `-- ` starts a comment in this file.

[[!format haskell """
import Propellor
import Propellor.CmdLine
import qualified Propellor.Property.File as File
import qualified Propellor.Property.Apt as Apt
import qualified Propellor.Property.User as User
import qualified Propellor.Property.Cron as Cron
"""]]

This loads up Propellor's modules. You'll almost certainly want these;
many more can be found in the [API documentation](http://hackage.haskell.org/package/propellor).

[[!format haskell """
main :: IO ()
main = defaultMain hosts
"""]]

This config file *is* the Propellor program, and so it needs a little
stub to go run itself. No need to ever change this part.
`hosts` is the list of hosts that you configure, and it comes next:

[[!format haskell """
-- The hosts propellor knows about.
-- Edit this to configure propellor!
hosts :: [Host]
hosts =
    [ host "mybox.example.com"
    	& os (System (Debian Unstable) "amd64")
        & Apt.stdSourcesList
    , host "server.example.com"
    	& os (System (Debian (Stable "jessie")) "amd64")
        & Apt.stdSourcesList
        & Apt.installed ["ssh"]
    ]
"""]]

This defines a list of hosts, with two hosts in it. 

The configuration for the mybox host first tells propellor what
OS it's running. Then the `stdSourcesList` line tells propellor to
configure its `/etc/apt/sources.list`, using its OS.
(Of course you might want to change that `Unstable` to `Stable`.)

Each property of the host is prefixed with an "&" operator. This just makes
a list of properties.

Some other properties you may find in your config.hs, or want to add:

[[!format haskell """
        & Apt.unattendedUpgrades
        & User.hasSomePassword (User "root")
        & "/etc/default/foodaemon" `File.containsLine` "ENABLED=yes"
        & Cron.runPropellor (Cron.Times "30 * * * *")
"""]]

Some of these properties can be reverted -- this makes Propellor undo whatever
effects they might have. For example, unattended upgrades can be scary, so
maybe you turned that on, but want to disable it now. To do so, just change
the "&" to a "!"

[[!format haskell """
        ! Apt.unattendedUpgrades
"""]]

Some properties cannot be reverted. Yet. It takes coding to implement
revertability. If you try to revert a property that does not support
reversion, propellor will **fail to compile**! This is a good thing..
it avoids you getting confused or bad things happening.

The error message when this happens might look a little scary. But if
you read through it, it's remarkably precise about what and where the problem
is.

<pre>
config.hs:30:19:
    Couldn't match expected type `RevertableProperty'
                with actual type `Property'
    In the return type of a call of `Apt.installed'
    In the second argument of `(!)', namely `Apt.installed ["ssh"]'
    In the first argument of `(&)', namely
      `host "mybox.example.com" & Apt.stdSourcesList Unstable
       & Apt.unattendedUpgrades
       ! Apt.installed ["ssh"]'
</pre>

Similarly, if you make a typo in the config file, you'll probably get a long
but informative error message.

<pre>
config.hs:27:19:
    Not in scope: `Apt.standardSourcesList'
    Perhaps you meant one of these:
      `Apt.stdSourcesList' (imported from Propellor.Property.Apt)
...
</pre>

That's really all there is to configuring Propellor. Once you
have a `config.hs` ready to try out, you can run `propellor --spin $host`
on one of the hosts configured in it.

See the [[README]] for a further quick start and [[Writing Properties]]
for guidance on extending propellor with your own custom properties.

(If you'd like to learn a little Haskell after all, check out
[Learn You a Haskell for Great Good](http://learnyouahaskell.com/).)