aboutsummaryrefslogtreecommitdiffstats
path: root/history.html
blob: 4395107cb40e6ea77f933f84fa3abef26fc3219a (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
122
123
124
125
<?xml version="1.0" encoding="ISO-8859-1"?>
<!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN"
    "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd">
<html xmlns="http://www.w3.org/1999/xhtml">
<head>
<link rev="made" href="mailto:esr@snark.thyrsus.com" />
<meta name="description"
content="Fetchmail participation statistics" />
<meta name="keywords" content="fetchmail, growth, analysis" />
<title>Trends in the fetchmail project's growth</title>
<style type="text/css">
/*<![CDATA[*/
 span.c6 {color: brown}
 span.c5 {color: red}
 span.c4 {color: lime}
 span.c3 {color: blue}
 div.c2 {text-align: center}
 h1.c1 {text-align: center}
/*]]>*/
</style>
</head>
<body>
<table width="100%" cellpadding="0" summary="Canned page header">
<tr>
<td width="30%" align="right">$Date$</td>
</tr>
</table>

<hr />
<h1 class="c1">Trends in the fetchmail project's growth</h1>

<p>The scattergram below was made with Gnuplot 3.7 from data pulled
directly out of the project NEWS file using two custom
shellscripts, <a href="timeseries">timeseries</a> and <a
href="growthplot">growthplot</a>. If you see a broken-image icon,
upgrade to a <a
href="http://www.libpng.org/pub/png/pngapbr.html">browser that can
view PNGs</a>.</p>

<div class="c2"><img src="growth.png"
alt="Fetchmail trends graph" /></div>

<p>The graph shows the population growth of the fetchmail project.
The horizontal scale is days since baseline, which is when I
started collecting statistics in October 1996 at version 1.9.0.
Left vertical scale is number of participants. There is one data
point for each release; therefore, the changes in density of marks
indicate release frequency.</p>

<p>The peak in the earliest part of the graph (before the note "Bad
addresses dropped") seems to be an artifact; I was not regularly
dropping addresses that became invalid at the time. Turnover on the
list seems to be about 5% per month (but that's just my estimate, I
don't have numbers on this).</p>

<p>The <span class="c3">blue scatter of squares</span> is total
participants. The <span class="c4">green scatter of crosses</span>
is the count of people on fetchmail-friends after I split the list.
The <span class="c5">cyan scatter of diamonds</span> is the
population of fetchmail-announce after the split.</p>

<p>The <span class="c6">brown scatter of diamonds</span> tracks
project size in lines of code (right vertical axis). The scale
relationship between this scatter and the other three is
arbitrary.</p>

<p>This graph is quite revealing. Several trends stand out:</p>

<ul>
<li>
<p>Over time, the project population displays rather consistent
linear growth.</p>
</li>

<li>
<p>The key event in the project's lifetime was release 4.3.0 in
October 1997, when I declared the code to be out of development and
in maintainance mode, and split the fetchmail list.</p>
</li>

<li>
<p>The run-up to 4.3.0 saw the most intensive spate of releases in
the project's history (the gap in that run happened when I took a
two-week vacation). It was followed by a significant slowdown.</p>
</li>

<li>
<p>After 4.3.0, the developer population remained fairly stable
around an average of about 250 participants.</p>
</li>

<li>
<p>Essentially all population growth after 4.3.0 happened on the
announce list, among people using fetchmail but not active
co-developers.</p>
</li>

<li>
<p>The growth trend in code size looks sublinear, perhaps
logarithmic.</p>
</li>
</ul>

<p>The linear growth trend in population is particularly
interesting; a priori we might expect geometric or logistic growth,
given that the project spreads by word of mouth.</p>

<p>It has been suggested that the linear growth rate is the result
of a situation in which both number of projects and the population
of eligible programmers are rising on trend curves of the same
(probably exponential) rate.</p>

<hr />
<table width="100%" cellpadding="0" summary="Canned page header">
<tr>
<td width="30%" align="right">$Date$</td>
</tr>
</table>

<br clear="left" />
<address>Eric S. Raymond <a
href="mailto:esr@thyrsus.com">&lt;esr@thyrsus.com&gt;</a></address>
</body>
</html>
> |= SA_RESTART; #endif #ifdef SA_NOCLDSTOP /* SunOS 4.1 portability hack */ if (sig == SIGCHLD) sa_new.sa_flags |= SA_NOCLDSTOP; #endif sigaction(sig, &sa_new, &sa_old); rethandler = sa_old.sa_handler; #if defined(SIGPWR) if (sig == SIGCHLD) sigaction(SIGPWR, &sa_new, NULL); #endif #else /* HAVE_SIGACTION */ rethandler = signal(sig, handler); #if defined(SIGPWR) if (sig == SIGCHLD) signal(SIGPWR, handler); #endif /* system call should restart on all signals except SIGALRM */ siginterrupt(sig, sig == SIGALRM); #endif /* HAVE_SIGACTION */ return rethandler; } void deal_with_sigchld(void) { set_signal_handler(SIGCHLD, sigchld_handler); } int daemonize (const char *logfile) /* detach from control TTY, become process group leader, catch SIGCHLD */ { int fd, logfd; pid_t childpid; /* if we are started by init (process 1) via /etc/inittab we needn't bother to detach from our process group context */ if (getppid() == 1) goto nottyDetach; /* Ignore BSD terminal stop signals */ #ifdef SIGTTOU set_signal_handler(SIGTTOU, SIG_IGN); #endif #ifdef SIGTTIN set_signal_handler(SIGTTIN, SIG_IGN); #endif #ifdef SIGTSTP set_signal_handler(SIGTSTP, SIG_IGN); #endif /* In case we were not started in the background, fork and let the parent exit. Guarantees that the child is not a process group leader */ if ((childpid = fork()) < 0) { report(stderr, "fork (%s)\n", strerror(errno)); return(PS_IOERR); } else if (childpid > 0) exit(0); /* parent */ /* Make ourselves the leader of a new process group with no controlling terminal */ #if defined(HAVE_SETSID) /* POSIX */ /* POSIX makes this soooo easy to do */ if (setsid() < 0) { report(stderr, "setsid (%s)\n", strerror(errno)); return(PS_IOERR); } #elif defined(SIGTSTP) /* BSD */ /* change process group */ #ifndef __EMX__ setpgrp(0, getpid()); #endif /* lose controlling tty */ if ((fd = open("/dev/tty", O_RDWR)) >= 0) { ioctl(fd, TIOCNOTTY, (char *) 0); close(fd); /* not checking should be safe, there were no writes */ } #else /* SVR3 and older */ /* change process group */ #ifndef __EMX__ setpgrp(); #endif /* lose controlling tty */ set_signal_handler(SIGHUP, SIG_IGN); if ((childpid = fork()) < 0) { report(stderr, "fork (%s)\n", strerror(errno)); return(PS_IOERR); } else if (childpid > 0) { exit(0); /* parent */ } #endif nottyDetach: (void)close(0); /* Reopen stdin descriptor on /dev/null */ if ((fd = open("/dev/null", O_RDWR)) < 0) { /* stdin */ report(stderr, "cannot open /dev/null: %s\n", strerror(errno)); return(PS_IOERR); } if (logfile) { if ((logfd = open(logfile, O_CREAT|O_WRONLY|O_APPEND, 0666)) < 0) { /* stdout */ report(stderr, "cannot open %s: %s\n", logfile, strerror(errno)); return PS_IOERR; } } else logfd = 0; /* else use /dev/null */ /* Close any/all open file descriptors */ #if defined(HAVE_GETDTABLESIZE) fd = getdtablesize() - 1; #elif defined(NOFILE) fd = NOFILE - 1; #else /* make an educated guess */ fd = 1023; #endif while (fd >= 1) { if (fd != logfd) close(fd); /* not checking this should be safe, no writes */ -- fd; } if (dup(logfd) < 0 /* stdout */ || ((logfd == 0 || logfd >= 3) && dup(logfd) < 0)) { /* stderr */ report(stderr, "dup (%s)\n", strerror(errno)); return(PS_IOERR); } #ifdef HAVE_GETCWD /* move to root directory, so we don't prevent filesystem unmounts */ chdir("/"); #endif /* set our umask to something reasonable (we hope) */ #if defined(DEF_UMASK) umask(DEF_UMASK); #else umask(022); #endif deal_with_sigchld(); return(0); } flag is_a_file(int fd) /* is the given fd attached to a file? (used to control logging) */ { struct stat stbuf; /* * We'd like just to return 1 on (S_IFREG | S_IFBLK), * but weirdly enough, Linux ptys seem to have S_IFBLK * so this test would fail when run on an xterm. */ if (isatty(fd) || fstat(fd, &stbuf)) return(0); else if (stbuf.st_mode & (S_IFREG)) return(1); return(0); } /* daemon.c ends here */