File: cfarticle.latex

package info (click to toggle)
cfengine 1.4.9-3
  • links: PTS
  • area: main
  • in suites: hamm
  • size: 3,540 kB
  • ctags: 1,861
  • sloc: ansic: 25,408; sh: 1,708; perl: 1,088; lex: 690; makefile: 435; lisp: 182; yacc: 101; csh: 24
file content (1404 lines) | stat: -rw-r--r-- 50,414 bytes parent folder | download | duplicates (2)
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
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
726
727
728
729
730
731
732
733
734
735
736
737
738
739
740
741
742
743
744
745
746
747
748
749
750
751
752
753
754
755
756
757
758
759
760
761
762
763
764
765
766
767
768
769
770
771
772
773
774
775
776
777
778
779
780
781
782
783
784
785
786
787
788
789
790
791
792
793
794
795
796
797
798
799
800
801
802
803
804
805
806
807
808
809
810
811
812
813
814
815
816
817
818
819
820
821
822
823
824
825
826
827
828
829
830
831
832
833
834
835
836
837
838
839
840
841
842
843
844
845
846
847
848
849
850
851
852
853
854
855
856
857
858
859
860
861
862
863
864
865
866
867
868
869
870
871
872
873
874
875
876
877
878
879
880
881
882
883
884
885
886
887
888
889
890
891
892
893
894
895
896
897
898
899
900
901
902
903
904
905
906
907
908
909
910
911
912
913
914
915
916
917
918
919
920
921
922
923
924
925
926
927
928
929
930
931
932
933
934
935
936
937
938
939
940
941
942
943
944
945
946
947
948
949
950
951
952
953
954
955
956
957
958
959
960
961
962
963
964
965
966
967
968
969
970
971
972
973
974
975
976
977
978
979
980
981
982
983
984
985
986
987
988
989
990
991
992
993
994
995
996
997
998
999
1000
1001
1002
1003
1004
1005
1006
1007
1008
1009
1010
1011
1012
1013
1014
1015
1016
1017
1018
1019
1020
1021
1022
1023
1024
1025
1026
1027
1028
1029
1030
1031
1032
1033
1034
1035
1036
1037
1038
1039
1040
1041
1042
1043
1044
1045
1046
1047
1048
1049
1050
1051
1052
1053
1054
1055
1056
1057
1058
1059
1060
1061
1062
1063
1064
1065
1066
1067
1068
1069
1070
1071
1072
1073
1074
1075
1076
1077
1078
1079
1080
1081
1082
1083
1084
1085
1086
1087
1088
1089
1090
1091
1092
1093
1094
1095
1096
1097
1098
1099
1100
1101
1102
1103
1104
1105
1106
1107
1108
1109
1110
1111
1112
1113
1114
1115
1116
1117
1118
1119
1120
1121
1122
1123
1124
1125
1126
1127
1128
1129
1130
1131
1132
1133
1134
1135
1136
1137
1138
1139
1140
1141
1142
1143
1144
1145
1146
1147
1148
1149
1150
1151
1152
1153
1154
1155
1156
1157
1158
1159
1160
1161
1162
1163
1164
1165
1166
1167
1168
1169
1170
1171
1172
1173
1174
1175
1176
1177
1178
1179
1180
1181
1182
1183
1184
1185
1186
1187
1188
1189
1190
1191
1192
1193
1194
1195
1196
1197
1198
1199
1200
1201
1202
1203
1204
1205
1206
1207
1208
1209
1210
1211
1212
1213
1214
1215
1216
1217
1218
1219
1220
1221
1222
1223
1224
1225
1226
1227
1228
1229
1230
1231
1232
1233
1234
1235
1236
1237
1238
1239
1240
1241
1242
1243
1244
1245
1246
1247
1248
1249
1250
1251
1252
1253
1254
1255
1256
1257
1258
1259
1260
1261
1262
1263
1264
1265
1266
1267
1268
1269
1270
1271
1272
1273
1274
1275
1276
1277
1278
1279
1280
1281
1282
1283
1284
1285
1286
1287
1288
1289
1290
1291
1292
1293
1294
1295
1296
1297
1298
1299
1300
1301
1302
1303
1304
1305
1306
1307
1308
1309
1310
1311
1312
1313
1314
1315
1316
1317
1318
1319
1320
1321
1322
1323
1324
1325
1326
1327
1328
1329
1330
1331
1332
1333
1334
1335
1336
1337
1338
1339
1340
1341
1342
1343
1344
1345
1346
1347
1348
1349
1350
1351
1352
1353
1354
1355
1356
1357
1358
1359
1360
1361
1362
1363
1364
1365
1366
1367
1368
1369
1370
1371
1372
1373
1374
1375
1376
1377
1378
1379
1380
1381
1382
1383
1384
1385
1386
1387
1388
1389
1390
1391
1392
1393
1394
1395
1396
1397
1398
1399
1400
1401
1402
1403
1404
\documentstyle[11pt]{article}

\title{\em The GNU Configuration Engine}
\author{Mark Burgess}
\date{}

    \textheight 40\baselineskip
    \oddsidemargin 0.1 in      %   Left margin on odd-numbered pages.
    \evensidemargin 0.15 in    %   Left margin on even-numbered pages.
    \marginparwidth 1 in       %   Width of marginal notes.
    \oddsidemargin 0.125 in    %   Note that \oddsidemargin = \evensidemargin
    \evensidemargin 0.125 in
    \marginparwidth 0.75 in
    \textwidth 5.125 in % Width of text line.


\begin{document}

\raggedright
\parindent 0.75cm
\begin{quotation}
\setlength{\textwidth}{3.5 in}
\begin{center}
\vspace{1cm}
{\Large\bf\em A Site Configuration Engine~~~~~~~~~~~~~~}
\end{center}

\vspace{1cm}

\raggedleft Mark Burgess, Computing Department, Oslo College~~~~~~~~~~~~~~

Physics Department,University of Oslo~~~~~~~~~~~~~~

\vspace{1cm}
\raggedright

\rule{3.5 in}{1mm}

\vspace{0.5cm} 

ABSTRACT: A language-based system administration\\
tool is presented.  System maintenance tasks are\\
automated and the configuration of all networked\\
hosts are defined in a central file. Host configur-\\
ation may be tested and repaired any number of times\\
without the need for human intervention. Cfengine\\
uses a decision making process based on class memb-\\
ership and is therefore optimized for dealing large\\
numbers of related hosts as well as individually\\
pin-pointed systems.

\vspace{0.5cm}
\rule{3.5 in}{1mm}
\end{quotation}

\eject

\section{\em Introduction}
          
The proliferation of TCP/IP networks, combined with the increased availability of
cheap UNIX-like solutions, continues to make machine-parks grow at
a rate which keeps system administrators on their toes.  This presents
a practical difficulty to administrators: how does one keep track of
hundreds or perhaps thousands of systems and be sure that they are
configured according to the network standard? In spite of the efforts
of standardizing organizations, the various operating system
alternatives from major software developers are all substantially
different from a system adminstrators perspective and, on a large
heterogeneous network, one is forced to undergo an often tiresome
process of adapting each type of system in order to make the
alternatives cooperate harmoniously.  Traditionally, such fixes have
been made by hand or with the help of shell scripts---a procedure
which becomes increasingly cumbersome and haphazard as the size of a
network expands beyond a handful of systems. A viable tool for
efficiently systemizing the administration of such a network has been
lacking for some time.  

The purpose of GNU cfengine\cite{cfdoc,cfrap,cftalk} is to provide a high
level, language-based interface for the task of system administration.
Using cfengine, administrators can create a single file which defines
the configuration of all hosts on an arbitrarily large
network. Changes made in this single file can cause system-wide
changes to take place, or can pin-point actions to be taken on a
single host. The configuration language hides the differences between
different operating systems and automates frequently performed tasks,
thereby creating a very high level description. This can be used to
both document and enforce the characteristics, interrelationships and
dependencies of all hosts from a single, easily readable file. A
cfengine configuration program can be used to automatically set up a
new host from scratch, making all the changes necessary to blend it
into the local network; it can also be run an unrestricted number of
times to check or maintain the state of that configuration. By
defining system configuration in a central file, accidents which
destroy the changes made on a special host are no longer a problem,
since a single run of the systemwide program will restore the
configuration to the defined standard.
          
The functionality of cfengine can be summarized by the following
list:
          
\begin{itemize}
\item Testing and configuration of the network interface
\item Simple automated text file editing
\item Symbolic link management
\item Testing and setting the permissions and ownership of files
\item Systematic deletion of garbage files
\item Systematic automated mounting of NFS filesystems.
\item Other sanity checks.
\end{itemize}

This article is a short conceptual presentation of cfengine.
Tutorials and more information can be obtained from the distributed
documentation\cite{cfdoc}.

\section{\em Why a new language?}

Cfengine's main contribution to system administration is to provide
relevant tools for a limited number of frequently-used-operations.
Cfengine supplements the functionality of peer languages,
such as Perl and lower level scripting languages, by providing `free' checks
which are built in to the engine itself. This frees programs from the
clutter of `irrelevant' checking code and admits a more
conceptually user-friendly interface.
For example, the command to create a symbolic link in cfengine is
\begin{verbatim}
   file1 -> file2
\end{verbatim}
The corresponding command in shell is
\begin{verbatim}
   ln -s file2 file1
\end{verbatim}
The functionality which cfengine adds here is the following
algorithm, which is executed for every single link defined in
a cfengine program:
\begin{itemize}
\item Does link exist? If not, create it.
\item Is the name a plain file or directory, not a link? If so signal a warning.
\item If link exists, does it point to the location specified?
\item If yes, do nothing, say nothing.
\item If not, signal a warning.
\end{itemize}
The above algorithm is designed in such a way that it can be run an
unlimited number of times without generating spurious and
uninteresting output. In shell, the execution of the command \verb+ln
-s+ twice results in
\begin{verbatim}
borg% ln -s  cv.tex bla
borg% ln -s cv.tex bla
ln: bla: File exists
\end{verbatim}
an irrelevant and unhelpful error message.  In cfengine, the execution
of the link script $n$-times results in no output unless verbose mode
is selected.  Additional spices exist, such as the ability to link all
of the children in a particular directory to corresponding files in
another. Again, extensive checking both of new files and previously
existing files is made.

In this example, cfengine has not provided anything which could not be
reproduced in a shell script---what it has done is to simplify the
code required to perform the appropriate actions considerably, by
hiding the irrelevant details in the language definition. This is the
function of high level languages. Similar features are true of the
other operations performed by cfengine. Some of these will be
mentioned in the remainder of the text.

\section{\em Classes}

One of the aims of cfengine is to make configuration programs as
transparent as possible. A key design feature which makes this
possible is the introduction of a class-based decision structure. A
system-wide configuration program must make a considerable number of
decisions in order to match statements to hosts. In a traditional
scripting language this would mean coding a large number of {\tt
if}\ldots{\tt then}\ldots{\tt else} statements, perhaps nested many
times. Since a test is required not only to determine which hosts a
particular command applies to, but also to determine whether or not
the present state of configuration is correct, the number of tests
very easily accounts for the bulk of coding in any program. To avoid
this scenario, cfengine uses a procedure of whittling away irrelevant
statements by classifying them according to certain properties of the
host executing the program.

A class based decision structure is possible because a cfengine
configuration program is run by every host on the network
individually. Each host knows its own name, the type of operating
system it is running and can determine whether it belongs to certain
groups or not. Each host which runs a cfengine program therefore
builds up a list of its own attributes (called classes).  A class may,
in fact, consist of the following:

\begin{itemize}
\item The hostname of a machine.
\item The operating system and architecture of the host.
\item A user-defined group to which the host belongs.
\item A day of the week.
\item The logical AND of any of the above.
\end{itemize}

Given that a host knows its own class attributes, it can now pick out
what it needs from a list of commands provided the commands are also
labelled with the classes to which they apply.  A command is only
executed if a given host is in the same class as the command it finds
in the configuration program---a host can pick out only the commands
which it knows apply to itself and ignore the others. There is no need
for formal decision structures, it is enough to label each statement
with classes.  At the simplest level, one has commands belonging only
to a single class, say the operating system type of the hosts:


\begin{quote}

{\tt ultrix}::

     ~~~~~~~~~{\em statements}

\end{quote}


Here the statements which follow the class
{\tt ultrix} are executed only if the host
is an Ultrix system. To combine classes, signifying multiple
membership dots are used:

\begin{quote}

{\tt ultrix.Monday.mygroup}::

     ~~~~~~~~~~{\em statements}\\

\end{quote}


In this example, the statements which follow are only
executed if the host is of type Ultrix, the day is
Monday and the host is a member of the user-defined
group {\tt mygroup}.

User-defined classes can be defined and undefined on the command line
and in the action sequence in order to switch certain statements on
and off for special purposes. This makes it easy to isolate parts of a
global configuration for partial execution.  It is, for example,
useful to mark very time-consuming operations with a class `heavy'
which can then be undefined in order to execute a `quick' version of
the program.

\section{\em Syntax}

The syntax employed by cfengine resembles in some ways a
Makefile\cite{make}, where instead of targets one has classes. Each
cfengine program is a free format file composed of a number of
sections. Each section deals with a particular task, such as symbolic
links or file editing.  Each section defines actions for various
classes (see figure 1).
\begin{figure}[ht]
\begin{quote}
\tt
\#\\
\# Cfengine program\\
\#\\
groups:

~~~myclass = ( host1 host2 host3 +@NIS-netgroup )\\

control:

~~~  actionsequence = ( links files )

links:

~~~ {\em class}::

~~~~~~ \verb+/tmp/x  -> /usr/tmp/x+\\

~~~ {\em class1.class2.class3}::

~~~~~~ \verb+/bin/tcsh -> /usr/local/bin/tcsh+\\

~~~files:

~~~  myclass::

~~~~~~ \verb+/usr/local owner=root mode=o-w action=fixall+

\end{quote}
\caption{The form of a simple cfengine program. The format is free, use of space is arbitrary.}
\end{figure}
Class membership of statements is, in fact, optional: if
no special membership is specified, a statement is
assumed to belong to all classes and is executed
on the host running the program. 

Figure 1 illustrates, with a trivial example, the basic points of
syntax in a cfengine program. Each program is a free format textfile
containing symbol declarations and `actions' to be performed. More
formally, it is a list composed of the elements of the form

\begin{quote}
\em 
section{\em \tt :}\\

~~~class{\tt::}\\

~~~~~~statements
\end{quote}
Omitting a {\em class} specifier is equivalent to using
a wildcard class {\tt any::} which means that the following
statements are to be executed on all hosts. Statements
have a syntax which depends on the section of the program: 
some of these will define symbolic links, others
specify editing actions for files etc. 

The meaning of the example program in figure 1 is the
following.  The first few lines are comments and are identified by
lines beginning with the hash symbol `{\tt \#}'.  The {\tt groups}
section of a program defines a new class called {\tt myclass}. This
class has as its members host1, host2, host3 and all of the hosts in
the netgroup NIS-netgroup. If the host which executes the cfengine
program is one of those hosts, it inherits the class {\tt myclass} and
statements which also belong to {\tt myclass} can then be executed.

In contrast to a makefile, the dependencies in a cfengine program are
not files which must exist but `host attributes' which must be
present. In a makefile, actions are performed if the target does not
exist; in a cfengine program, actions are performed if the classes
{\em do} match the present state of the system.  In other words, a
cfengine program is not an instruction of how to build a system, but a
statement of what many different classes of system should look like.
Some users have focused on the similarities with make and have
suggested that the make program combined with shell scripts
would do the same job. While it is certainly true that any problem
can be solved in a variety of ways, the criticism is somewhat
misguided since the real gains in using cfengine are that one avoids
having to write long and complicated scripts employing repetetive
checking procedures. Cfengine is a classic meta-language: it
eliminates the need for tiresome repetetive coding by absorbing
frequently used code back into to language.

The control part of a cfengine program is used to set certain
internal variables and to define macros. The most important system
variable is a list called the {\tt actionsequence}. Without an action
sequence, a cfengine program does nothing. It is a way of switching on
and off certain statements. For example, if one adds the item `{\tt
links}' to the action sequence, cfengine will process all of the link
commands which belong to classes the current host belongs to.  The
action seqeuence determines the order and number of times in which
these bulk actions are carried out (the actual ordering of the
declarations in a cfengine program is irrelevant and should be used to
achieve conceptual clarity rather than to indicate the sequence of
events). If this bulk handling of commands is too coarse, finer
control is achieved by using the notation

\begin{quote}
\tt ~~~actionsequence =

~~~~~~   (

~~~~~~   links.{\em class1}.{\em class2}

~~~~~~   ...

~~~~~~   links.{\em class3}

~~~~~~   )

\end{quote}
which means: execute links commands---but, on the first pass,
define the additional classes {\em class1} and {\em class2}
for the duration of this pass only;
on the second pass define the additonal symbol {\em class3}
for the duration of the pass. The result is that, in the first
case, only links labelled by {\em class1} and {\em class2}
will be executed and in the second case only links labelled
with {\em class3} will be executed. Classes defined in the
action sequence have no lasting effect. They are `local'
to a given action and are only used to achieve a finer
control over the sequence of execution. They are
attributes of the current task rather than of the host.

The keywords or actions in the actionsequence are internally
defined and are taken from the following list, which
is printed incidentally in the order in which the
actions might typically be called:
\begin{verbatim}
      mountall            # mount filesystems in fstab
      mountinfo           # scan mounted filesystems
      checktimezone       # check timezone
      netconfig           # check net interface config
      resolve             # check resolver setup
      unmount             # unmount any filesystems
      shellcommands       # execute shell commands
      editfiles           # edit files
      addmounts           # add new filesystems to system
      directories         # make any directories
      links               # check and maintain links
      mailcheck           # check mailserver
      required            # check required filesystems
      tidy                # tidy files
      disable             # disable files
      files               # check file permissions 

\end{verbatim}
For a full explanation of these functions, the reader
is referred to the cfengine documentation.

\section{\em Functions}

In this section, a cursory overview of the functionality
of cfengine is presented.

\subsection{\em Network}

The configuration of the ethernet interface is one of
the prerequisites for getting a host up and running.
It includes informing the ethernet interface of the
subnetmask, broadcast address, default route of the host.
In addition, the Domain
Name Service has to be configured. These tasks are
handled by cfengine at a high level. It is
sufficient to define

\begin{itemize}
\item the value of the internal variable {\tt netmask},
\item the bit-convention for determining the broadcast
address (either all ones or all zeros),
\item the default route for packets (normally the 
address of the local gateway),
\item the system domain name,
\item an ordered list of nameservers.
\end{itemize}
These can naturally be specified either once for
all hosts or individually by special classes,
depending on the physical organization of the net.

If the appropriate directives are added to the
action sequence, cfengine uses this information to check the 
present state of the ethernet
device and, if necessary, configure it to the standard
defined in the configuration program. The default
route is added to the static routing table
if necessary. Cfengine then loads
the file `\verb+/etc/resolv.conf+', ensures that
the DNS domain name is correct and that the correct
nameservers are present with the defined priority.

\subsection{\em File editing}

One of the characteristics of BSD/System 5 systems
is that they are configured primarily by human-readable
textfiles. This makes it easy for humans to configure
the system and it also simplifies the automation of the procedure.
Most configuration files are line-based text files,
a fact which explains the popularity of, for example, the Perl programming
language\cite{perl}. 
Cfengine does not attempt to compete with Perl or
its peers. Its
internal editing functions operate at a higher level
which are designed for transparency rather than flexibilty.
Fortunately most editing operations involve appending a 
few lines to a file, commenting out certain lines or deleting
lines. Files are edited with commands
from the following list:

\begin{quote}
\tt
   DeleteLinesStarting "{\em text...}"\\
   DeleteLinesContaining "{\em text...}"\\
   AppendIfNoSuchLine "{\em text...}"\\
   PrependIfNoSuchLine "{\em text...}"\\
   WarnIfNoSuchLine "{\em text...}"\\
   WarnIfLineMatching "{\em text...}"\\
   WarnIfLineStarting "{\em text...}"\\
   WarnIfLineContaining "{\em text...}"\\
   WarnIfNoLineStarting "{\em text...}"\\
   WarnIfNoLineContaining "{\em text...}"\\
   HashCommentLinesContaining "{\em text...}"\\
   HashCommentLinesStarting "{\em text...}"\\
   HashCommentLinesMatching "{\em text...}"\\
   SlashCommentLinesContaining "{\em text...}"\\
   SlashCommentLinesStarting "{\em text...}"\\
   SlashCommentLinesMatching "{\em text...}"\\
   PercentCommentLinesContaining "{\em text...}"\\
   PercentCommentLinesStarting "{\em text...}"\\
   PercentCommentLinesMatching "{\em text...}"\\
\end{quote}
Commands containing the word `comment' are used to `comment out'
certain lines from a textfile---i.e. render a line impotent
without actually deleting it. Three types of comment are
supported: shell style (hash) `\verb+#+', `\verb+%+' as used
in TeX and on AIX systems, and \verb-C++--style `\verb+//+'.

An example of the use of this might be the following. Each new GNU/Linux
installation contains a line in the start-up scripts which
deletes the contents of the `message of the day' file each time
the system boots. 
On a system which boots often this would be irritating.
This line could be commented out for every
GNU/Linux system on the network with a simple command:

\begin{verbatim}

editfiles:

   linux::

      { /etc/rc.d/rc.S

      HashCommentLinesContaining "motd"
      }

\end{verbatim} Other applications for these editing commands include
monitoring and controlling root-access to hosts by editing files such as
`\verb+.rhosts+' and setting up standard environment variables in
global shell resource files--- for example, to set the timezone.

Files are loaded into cfengine and edited in memory. They are
only saved again if modifications to the file are carried out,
in which case the old file is preserved by adding a suffix
to the filename. When files are edited, cfengine generates a
warning for the administrator's inspection so that the reason
for the change can be investigated.

The behaviour of cfengine should not be confused with that
of {\tt sed} or {\tt perl}. Again, it is true that nothing
really new is introduced, but that a considerable saving
of user-programming is involved---moreover a common interface
is used, taking full advantage of the class selectors.
Some functionality is reproduced
for convenience, but the specific functions have been chosen
on the basis of (i) their readability and (ii) the fact that
they are `frequently-required-functions'. A typical file
editing session involves the following points:
\begin{itemize}
\item Load file into memory.
\item Is the size of the file within sensible user-definable limits? 
If not, file could be binary, refuse to edit.
\item Check each editing command and count the number of edits made.
\item If number of edits is greater than zero, rename the old file
and save the edited version in its place. Inform about the edit.
\item If no edits are made, do nothing, say nothing.
\end{itemize}
Equivalent one-line sed operations involve editing the same file
perhaps many times to achieve the same results---without the
safety checks in addition.
          
\subsection{\em Mount model}

Cfengine regards NFS filesystems as resources. Resources,
like actions, also belong to classes and are mounted on the basis 
of class decisions. Cfengine
automates the mount procedure as far as possible;
administrators have only to specify a number of servers
for a class of hosts and cfengine will edit the appropriate
filesystem tables and attempt to mount the resources
automatically. 

Cfengine distinguishes between two types of mountable
resources which it refers to as {\em binary filesystems}
and {\em home filesystems}. Binary filesystems
contain architecture-specific data---i.e. compiled software
which only applies to the operating system under which
it was compiled. Home filesystems contain users' login
areas and can be mounted meaningfully on any type of host.
The way information is structured in cfengine programs
makes mounting of binary and home resources quite transparent.
For each class of hosts one defines a number of binary
servers and a number of home servers. Cfengine mounts
automatically all the declared resources from all a host's
servers by referring to a list which contains every filesytem
resource available on the network. Network resources are
defined like this:

\begin{verbatim}

mountables:

     server:/site/server/home1
     server:/site/server/home2
     server2:/site/serv2/local

\end{verbatim}
The name of the server (preceding the colon) and the remote directory 
name (following the colon) 
are declared in this list so that cfengine can search for
resources of different types.
Employing a user-definable pattern, cfengine can
distinguish between home and binary resources and mount the
appropriate resources on directories with the same names as the source
filesystems. Note that the key to the success of this model
is that remote filesystems are mounted on directories with
the same name on the local host. This is not a restriction provided
one uses a rational naming scheme and any anomalies can be handled
by the `miscellaneous mount' command (which is more akward
syntactically but lifts the naming restriction).

To make the scheme work then, it is necessary to introduce a strict naming
convention for filesystem mount-points\cite{borge}.  While this is
user-configurable, the recommended convention is to mount all
filesystems according according to a three component directory name:

\begin{quote} 
\em {\tt /}site{\tt /}hostname{\tt /}file-system-name
\end{quote} 
in which the site name is the name of your local
department or section (separate subnet), the hostname is the name of
the host which is the server for the filesystem and the final link is
the name of the directory itself. Strict adherence to this system
means that no two filesystems will ever collide. Symbolic links can
then be used to make cosmetic changes to the system, for example to
create an alias from {\tt server2:/site/serv2/local} to {\tt
/usr/local}.

The issue of editing the exports files on the servers is not addressed
directly by cfengine since there is no unique way of handling this
issue. If necessary it could be dealt with using the editfiles
facility. In practice it easier to deal with exports by hand---if only
for security reasons.

The model cfengine uses for mounting filesystems
around the network is simple and effective. The amount
of writing required to add a large number of filesystems
to either a single host 
or a class of hosts is simply equal to the number of servers
on which the resources reside.

Although most filesystems fall into the categories binary and home,
some---like information databases and sharable resources---do not.
These remaining resources can be dealt with using a
miscellaneous mount command which makes no reference to
a special model. A small amount of extra writing is required
in this case. For example:

\begin{verbatim}
miscmounts:

   myhost::

      otherhost:/site/otherhost/info /library/database rw
\end{verbatim}

Cfengine hard-mounts filesystems by default. In contrast to
the NFS auto-mounter\cite{automount} the filesystems are mounted by editing the
filesystem table so that all filesystems are avilable from
boot time. Hence the functionality does not compete
with the automounter but augments it.
          
\subsection{\em Files and links}

File and link management takes several forms.
Actions are divided into three categories called
{\tt files}, {\tt tidy} and {\tt links}. The first of
these is used to check the existence of, the ownership
and permissions of files. The second concerns the systematic
deletion of garbage files. The third is a link manager
which tests, makes and destroys links. The monitoring
of file access bits and ownership can be set up for
individual files and for directory trees, with controlled
recursion. Files which do not meet the specified criterea
can be `fixed' ---i.e. automatically set to the correct
permissions, or can simply be brought to the attention of
the system admnistrator by a warning.
The syntax of such a command is as follows:

\begin{quote}
\tt
files:

~~~   {\em class}::

~~~~~~/{\em path} mode={\em mode} owner={\em owner} group={\em group}\\

~~~~~~~~~~~~~~~  recurse={\em no-of-levels} action={\em action}

\end{quote}
The directory or file name is the point at which cfengine
begins looking for files. From this point the search for files
proceeds recursively into subdirectories with a maximum limit set by
the {\tt recurse} directive, and various options for dealing with
symbolic links and device boundaries. The mode-string defines the
allowed filemode (by analogy with {\tt chmod}) and the owner and group
may specify lists of acceptable user-ids and group-ids. The action
taken in response to a file which does not meet acceptable criterea is
specified in the action directive. It includes warning about or
directly fixing all files, or plain files or directories only. Safe
defaults exist for these directives so that in practice they may be
treated as options.

For example,
\begin{verbatim}
files:

  any::
       /usr/*/bin mode=a+rx,o-w own=root r=inf act=fixall
\end{verbatim}
which (in abbreviated form) would check recursively all files and
directories starting from directories matching the wildcard (e.g. {\tt
/usr/local/bin}, {\tt /usr/ucb/bin}).  By default, \verb+fixall+
causes the permissions and ownership of the files to be fixed without
further warning.

The creation of symbolic links is illustrated in figure 1 and
the checking algorithm was discussion in section 2. In addition to
the creation of single links, one may also specify the creation of
multiple links with a single command. The command
\begin{verbatim}
links:

   binaryhost::

      /local/elm/bin +> /local/bin
\end{verbatim}
links all of the files in \verb+/local/elm/bin+ to corresponding
files in \verb+/local/bin+. This provides, amongst other things, one
simple way of installing software packages in regular `bin'
directories without controlling users' PATH variable. A further
facility makes use of cfengine's knowledge of available (mounted) binary
resources to search for matches to specific links. Readers are
referred to the full documentation concerning this feature.

The need to tidy junk files has become increasingly evident during the
history of cfengine. Files build up quickly in areas like {\tt /tmp/},
{\tt /var/tmp}. Many users use these areas for receiving large
ftp-files so that their disk usage will not be noticed!  To give
another example, just in the last few months the arrival of
netscape\cite{netscape} World Wide Web client, with its caching
facilities, has flooded harddisks at Oslo with hundreds of megabytes of
WWW files. In addition the regular appearence of `\verb+core+' files\footnote{On
some systems, core dumps cannot be switched off!}
and compilation by-products (\verb+.o+ files and \verb+.log+ files
etc.) fills disks with large files which many users do not understand.
The problem is easily remedied by a few lines in the cfengine
configuration. Files can be deleted if they have not been accessed for
$n$-days. Recursive searches are both possible and highly practical
here. In following example:
\begin{verbatim}

tidy:

   AllHomeServers::

      home                 pattern=core       r=inf age=0
      home/.wastebacket    pattern=*          r=inf age=14
      home/.netscape-cache pattern=cache????* r=inf age=2
      home/.MCOM-cache     pattern=cache????* r=inf age=2

\end{verbatim}
all hosts in the group {\tt AllHomeServers} are instructed to
iterate over all users' home directories (using the wildcard
{\tt home}) and look for files matching special patterns.
Cfengine tests the {\em access time} of files and deletes
only files older than the specified limits. Hence all core
files, in this example, are deleted immediately, whereas files in the
subdirectory `\verb+.wastebasket+' are deleted
only after they have lain there untouched for 14 days, and so on.

\subsection{\em Calling scripts}

Above all, the aim of cfengine is to present a simple interface to
system administrators. The actions which are built into the engine are
aimed at solving the most pressing problems, not at solving every
problem.  In many cases administrators will still need to write
scripts to carry out more specific tasks. These scripts can still be
profitably run from cfengine. Variables and macros defined in cfengine
can be passed to scripts so that scripts can make maximal advantage of
the class based decisions. Also note that, since the days of the week
are also classes in cfengine, it is straightforward to run weekly
scripts from the cfengine environment (assuming that the configuration
program is executed daily). An obvious use for this is to update
databases, like the fast-find database one day of the week, or to run
quota checks on disks.

\begin{verbatim}
shellcommands:

   myhost.Sunday::

      "/usr/bin/find/updatedb"

\end{verbatim}
          
\section{\em How cfengine is run}

Cfengine was designed to be run as a batch job, ideally at night when
system disk load is low.  Because its policy is to check and then
correct, it can also be run manually any number of times without ill
effects.  Cfengine runs silently by default, producing a message only
if something is wrong. It is therefore convenient to have error
messages mailed to the system administrator. This is accomplished by
running cfengine from a wrapper script which reports the name of the
host and forwards the text from cfengine. Suitable wrapper scripts are
included with the cfengine distribution.

Since cfengine only acts when action needs to be taken, a cfengine
program can be run any number of times without harmful side
effects. A typical scenario is the following.  On the arrival of a new
machine, a single NFS directory is then mounted by hand to gain access
to a compiled version of cfengine and the global configuration
file. Cfengine is run and the machine is instantly configured---all
symbolic links, NFS filesystems and textfiles are in place. The host
is now installed.  This should be sufficient. A reboot of the host
should now have no effect on the configuration.  Cfengine can itself
be programmed to add itself to the cron file so that it is run each
night so as to monitor the host on a regular basis.

The global cfengine program can also profitably be called up in the
system boot scripts \verb+/etc/rc.local+ or its equivalent, perhaps
with certain actions excluded to save time. It can be used to set the
netmask, broadcast address and default route as well as checking the
ordering of nameservers in \verb+/etc/resolv.conf+ each time the
system boots.
          
\section{\em Security}

Cfengine has built in features which are designed for system
security. The ability to monitor file permissions and ownership is the
first step.  A common problem is that files obtained by an ftp session
get transferred with a user-id which belongs to a completely random
user on the local system. This can either cause access problems or
compromise the security of the files. A busy administrator could
easily overlook this or simply forget to change the ownership of the
files.  A routine check of all files would discover this fact very
quickly.

A by-product of the file checking is that cfengine maintains a list of
all known setuid-root programs and setgid-root programs which it finds
in the course of checking the system. When a new setuid-root program
appears on the system, a warning is always issued so that any
potentially `dangerous' software is brought to the administrator's
attention. In most cases it will be the administrator who has
installed the software, but on other occaisions this could help to
reveal surreptitiously installed programs.

Using cfengine as a scripting langauge is also made safer. If a cfengine
script is made setuid-root (on a system which allows you to do this), 
it is still possible to restrict the users
who can run that script as a secondary check. For example:

\begin{verbatim}
   access = ( mark root )

\end{verbatim}
An access control list defines the usernames who may run a
program. This makes it easy to write a program which can be run by
others to fix a particular problem on the system.  Responsibility can
thus be disseminated quite safely to system helpers.

Cfengine does not have to be run setuid root, nor do any of its
features demand the availability of this feature.  However, on systems
which do support this option, it is presumed that this will be a
helpful additional feature.  Caution should always be exercised when
opening privileged access to non-privileged users.
          
\section{\em Scripting language}

Although the focus of attention has always been the construction of
systemwide configuration files, cfengine can also be used to
write smaller scripts. For example, the following script provides
a useful way for users to manage their own files, opening files for
collaboration with other users
and closing others which are private.

\begin{verbatim}
#!/local/gnu/bin/cfengine -f
#
# Open my shared directory for others in my group
#

control:

  actionsequence = ( files )

files:

  $(HOME)         mode=a+rx r=0 action=fixdirs
  $(HOME)/share   mode=ug+rw,o-rwx r=inf group=share act=fixall
  $(HOME)/private mode=0600 r=inf action=fixall

\end{verbatim}
The first line ensures that the user running the script has
a home directory which is open to other users. The second
line opens the subdirectory `{\tt share}' to the group {\tt share}
and tells cfengine to fix the files recursively. Note that,
in recursive searches, cfengine will automatically set the
`x' flag on directories if the corresponding `r' flag is defined.
          
\section{\em Experience}

Cfengine has been on test, in prototype form, for three years
during its development. In addition the recent GNU release is now in used at around
twenty sites around the world.
The number of features has grown
in accordance with experience in using it and for its GNU release
the syntax has been altered radically from earlier versions.
New features are incorporated as feedback is received through
the offical mail point \verb+bug-cfengine@gnu.ai.mit.edu+.

The philosophy employed in writing the configuration scripts has been
to define as many general rules as possible.  Special exceptions are
to be avoided since they increase the size of the configuration any
make programs harder to understand.  This might give the impression of
a loss of flexibilty, but systematic administration procedures on a
large scale are by necessity simple minded and general.  More
difficult, specific issues can be dealt with locally, using local
scripts (written in cfengine or some other utility) and controlled by
individuals who are closer to the individual host concerned. In most
cases, special configuration requirements are a result of specially
licensed software which runs only on a single host, or perhaps a small
cluster of hosts. These can nearly always be integrated into the
global configuration by using symbolic links. Cfengine has two
powerful features for building and managing large number of symbolic
links automatically. Indeed, experience shows that cfengine would be a
useful tool if the only thing it did was to manage symbolic links. The
use and maintainance of links (whose names can be based on systemwide
variables) opens up a new way of making easily understandable and {\em
maintainable} patches to systems.

Certain habitual practices must naturally be relearned in order to
make effective use of cfengine: administrators, used to configuring
systems by hand, have to discipline themselves to make changes only in the
configuration file and then run cfengine to make a change. Initially
this introduces an extra step, and therefore a certain
amount of resistance, but on networks supporting hundreds of
hosts this minor overhead is worth the potential rewards.

\section{Example program}

Here is a more substantial example program to illustrate the
uses for cfengine. Follow the comments for the details. It
is difficult to represent all of the useful features here;
hopefully there is enough in this example to whet the appetite
for more. 

\small
\begin{verbatim}
#######################################################
# 
#  CFENGINE CONFIGURATION FOR site = iu.hioslo.no
#
#######################################################

groups:
 
   science = ( nexus ferengi regula borg dax lore axis )
   diskless = ( regula ferengi lore )

   AllHomeServers   = ( nexus )
   AllBinaryServers = ( nexus borg )

   OIH_servers = ( nexus borg )
   OIH_clients = ( ferengi regula dax lore )

   XTerminalServer = ( nexus )
   WWWServers = ( nexus )
   FTPserver = ( nexus )

   LPD_clients = ( ferengi regula borg dax lore axis )

#######################################################

control: 

   access    = ( root )     # only root gets to start this

   site      = ( iu )
   domain    = ( iu.hioslo.no )
   sysadm    = ( sysadm@iu.hioslo.no ) # errors to ..

   netmask   = ( 255.255.255.0 )
   timezone  = ( MET )
   nfstype   = ( nfs )

   sensiblesize  = ( 1000 )  # missing filesystem if total bytes
                             # in fs less than 1000 (arbitrary)
   sensiblecount = ( 2 )     # missing filesystem if total files
                             # in fs less than 2 (arbitrary)
   editfilesize  = ( 6000 )  # Safety: don't edit files bigger than
                             # 6000 bytes - could be a mistake!

   actionsequence =          # Checking order...
      (
      mountall
      mountinfo
      checktimezone
      netconfig
      resolve
      unmount
      shellcommands
      editfiles
      addmounts
      directories
      links
      mailcheck
      mountall
      required
      tidy
      disable
      files
      )

   mountpattern = ( /$(site)/$(host) ) 

     # user dirs are u1, u2 etc

   homepattern = ( u? )   

   addclasses = ( exclude )

 #
 # Macros & constants
 #

   main_server = ( nexus )
   gnu_path = ( /local/bin/gnu )
   ftp = ( /local/ftp )

#######################################################

 # Nexus is the only host holding users' home dirs, so we
 # have to mount these on all systems listed in science

homeservers:

   science:: nexus

 # nexus and borg hold the binaries for /local for their
 # respective OS types...  so any machines of these types
 # in science should mount all non-home dirs from the
 # list of mountables. In this case there is only
 # .../local to mount, but there could be any number
 # handled by this one command.

binservers:

   science.solaris::     nexus
   science.linux::       borg

  # The mail intray is on nexus and (on nexus) is called
  # /var/mail. This will be mounted where the local OS
  # expects to find it e.g. /usr/spool/mail on BSD.

mailserver:

   any::
          nexus:/var/mail

  # This is a list of all mountable partitions
  # available by NFS. (Used by binservers/homesevers)

mountables:

   any::

         nexus:/iu/nexus/u1
         nexus:/iu/nexus/u2
         nexus:/iu/nexus/local
         borg:/iu/linux/local

  # An exception to a general rule - here it proves
  # convenient to mount a solaris binary fs onto a
  # linux machine because it contains some config
  # files which are useful.

miscmounts:

   borg::   nexus:/iu/nexus/local /iu/nexus/local ro


#######################################################

import:                        

  # Some rules can be made so general that they can be
  # collected into a separate file to make this file
  # less cluttered.

   any::      cf.global_classes
   linux::    cf.linux_classes

#######################################################

broadcast:

 # All our networks use the newer `ones' convention
 # for broadcasting, but some still use zeroes.

  ones

 # Set a default route to the local gateway for all
 # hosts

defaultroute:

  oih-gw

#######################################################

resolve:

  # Our nameservers (applies to all hosts)

      128.39.89.10 
      158.36.85.10 
      129.241.1.99

#######################################################

links:

  # Everyone needs a local dir. $(binserver) expands to
  # hostname if that dir exists -- if not it expands to the

      /local -> /$(site)/$(binserver)/local

  # Make sure we dispose of silly sendmail and replace it
  # with Berkeley V8 in /local/mail

   solaris::

      /usr/lib/sendmail     ->! /local/mail/bin/sendmail
      /etc/mail/sendmail.cf ->! /local/mail/etc/sendmail.cf

  # Link some packages into /local/bin so we don't have
  # to have a 10 mile long PATH variable...

   nexus::

     /local/bin       +> /local/perl/bin
     /local/bin       +> /local/elm/bin

#######################################################

tidy: 

  # List some files we want *deleted* once and for all...
  # The age refers to the access time of the files...
  # First tidy the users' home dirs, then the tmp areas.

   AllHomeServers.exclude::

       home                 pat=core      r=inf age=0
       home                 pat=a.out     r=inf age=2
       home                 p=*%          r=inf age=2
       home                 p=*~          r=inf age=2
       home                 p=#*          r=inf age=1    
       home                 p=*.dvi       r=inf age=14
       home/.wastebacket    p=*           r=inf age=14
       home/.netscape-cache p=cache????*  r=inf age=2
       home/.MCOM-cache     p=cache????*  r=inf age=2

   any::

      /tmp/                 pat=*         r=inf A=1
      /var/tmp              pat=*         r=inf A=1
      /                     pat=core      r=1   A=0

#######################################################

files:

   # All the local binaries should be owned by root
   # and nothing should be writable to the world!

   AllBinaryServers.exclude::

      /local mode=-0002 r=inf owner=root group=0,1,2,3,4,5,6

   # Make sure that none of the users' files are unwittingly
   # writable by others and delete any links which point
   # nowhere and confuse everyone.
   # Note ``ignore'' exception for www directory below, since
   # some users want to user nobody to be able to edit a
   # guestbook file...

   AllHomeServers.exclude::

      home m=o-w R=inf act=fixall links=tidy

   # Make sure the local ftp dirs have the right
   # permissions...

   FTPserver.solaris::

      $(ftp)/pub  mode=755 o=ftp g=ftp r=inf act=fixall
      $(ftp)/Obin mode=111 o=root g=other act=fixall
      $(ftp)/etc  mode=111 o=root g=other act=fixdirs
      $(ftp)/usr/bin/ls mode=111 o=root g=other act=fixall
      $(ftp)/dev mode=555 o=root g=other act=fixall
      $(ftp)/usr mode=555 o=root g=other act=fixdirs

#######################################################

directories:

   solaris::

      /usr/lib/X11/nls  # for httpd

   borg::

      /local/tmp  mode=1777 o=root g=0

#######################################################

ignore:    

   # Don't enter these directories in recursive descents

   any::
      
      .X11
      !*
      /local/lib/gnu/emacs/lock/
      /local/tmp
      /local/ftp
      /local/bin/top
      /local/lib/tex/fonts
      /local/etc
      /local/www
      /local/httpd_1.4/conf
      /local/mutils/etc/finger.log

  # For users' home dirs, so ``nobody'' can edit the guestbook

      www

#######################################################

required:

   # All hosts should have access to the /local dir. Warn if
   # they don't, or it looks funny (sensiblesize, sensiblecount)

      /${faculty}/${binserver}/local

#######################################################

editfiles:

  # Some basic files to edit.

   solaris::

      { /etc/netmasks

      DeleteLinesContaining "255.255.254.0"
      AppendIfNoSuchLine "128.39  255.255.255.0"
      }

   # cfengine installs itself as a cron job.

      { /var/spool/cron/crontabs/root

      AppendIfNoSuchLine "0 0 * * * \
      /local/gnu/lib/cfengine/bin/cfwrap \
      /local/gnu/lib/cfengine/bin/cfdaily"
      }

   nexus::

      { /etc/services

      WarnIfNoLineContaining "http"
      WarnIfNoLineContaining "pop"
      WarnIfNoLineContaining "bootpc          68/udp"
      WarnIfNoLineContaining "bootp           67/udp"
      }

      { /etc/inetd.conf

      AppendIfNoSuchLine "bootp dgram udp wait root \
      /local/bin/bootpd bootpd -i -d"
      }

   any::

      { /etc/shells

      AppendIfNoSuchLine "/local/bin/tcsh"
      }

#######################################################

shellcommands: 

  # Update the find/locate databases and the
  # manual key on sundays...

   AllBinaryServers.solaris.exclude::
 
      "/local/gnu/lib/locate/updatedb"

   AllBinaryServers.sun4.Saturday.exclude.Sunday::

      "/usr/bin/catman -w -M /local/man"
      "/usr/bin/catman -w -M /local/X11R5/man"
      "/usr/bin/catman -w -M /usr/man"
      "/usr/bin/catman -w -M /local/gnu/man"


#######################################################

disable:

 # Good to disable log files periodically so they don't
 # grow too big!

 WWWServers.Sunday::

   /local/httpd_1.4/logs/access_log
   /local/httpd_1.4/logs/agent_log
   /local/httpd_1.4/logs/error_log
   /local/httpd_1.4/logs/referer_log

  # Disable sendmail if it's a file. If it's the link
  # we made further up, leave it!
  # Also delete standard .login file which tcsh can't
  # understand.

 solaris::

    /usr/lib/sendmail type=file
    /etc/.login type=file

\end{verbatim}
\normalsize

\section{\em Summary}

Cfengine is a language based interface for automating key areas of
system administration on potentially large TCP/IP networks. The
configuration of all hosts on a local area network may be steered from
a single, central program, whose primary aim is to be as simple as
possible to understand. Cfengine enhances the functionality of shell
programs and provides an integrated environment for system
configuration which avoids excessive CPU usage (pipes) and minimizes
disk accesses.  The full functionality of the engine has not been
discussed in this article: readers are referred to the GNU package
itself for comprehensive documentation and examples.
          
Future enhancements include the further development of the text
editing facilities and the possibility of interfacing to companion
tools for process monitoring in real time. Cfengine could also
be enhanced by the introduction of a daemon which ensured that
it was run (allbeit silently) on every host. Ideally,
cfengine configuration files would be available in a distributed
database such as NIS.

Cfengine can be obtained by anonymous ftp from any GNU site. A list
of GNU sites can be obtained by connecting to \verb+prep.ai.mit.edu+
by anonymous ftp. The current version at the time of writing
if 1.1.0 and it runs on SunOS/Solaris, HPUX, ULTRIX, IRIX, OSF1,
LINUX and AIX.

I am grateful to Richard Stallman, Ola Borreb\ae k and 
Morten Hanshaugen for their constructive criticisms.

\bibliographystyle{unsrt}
\begin{thebibliography}{9}
\bibitem{cfdoc} M. Burgess, GNU cfengine, Free Software Foundation, 1995.
\bibitem{cfrap} M. Burgess, Cfengine, University of Oslo report 1993.
\bibitem{cftalk} Cfengine was first presented publicly at the CERN HEPIX meeting,
October 1994, France.
\bibitem{make} See, for example, A. Oram and S. Talbott, Managing projects with
make, O'Reilly \& Assoc. (1991)
\bibitem{perl} L. Wall and R. Schwarz, Programming perl, O'Reilly \& Assoc. (1990)
\bibitem{borge} This naming convention was first suggested to me by Knut Borge
of USIT, University of Oslo.
\bibitem{netscape} The Netscape program, Netscape Communications Corporation: http:/home.netscape.com.
\bibitem{automount} The NFS automounter, Sun Microsystems, SunOS/Solaris manual pages.
\end{thebibliography}

\end{document}