summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2008-04-28 09:32:42 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2008-04-28 09:32:42 (UTC)
commit939d32fda70ea66c9db51687beb3cea6da7b0599 (patch) (unidiff)
tree50915facf89b78e3856fe6b0564a26c3678c01ba
parent9ec5cd7944a7099515b7d41107007d6332a2540e (diff)
downloadcgit-939d32fda70ea66c9db51687beb3cea6da7b0599.zip
cgit-939d32fda70ea66c9db51687beb3cea6da7b0599.tar.gz
cgit-939d32fda70ea66c9db51687beb3cea6da7b0599.tar.bz2
Redesign the caching layer
The original caching layer in cgit has no upper bound on the number of concurrent cache entries, so when cgit is traversed by a spider (like the googlebot), the cache might end up filling your disk. Also, if any error occurs in the cache layer, no content is returned to the client. This patch redesigns the caching layer to avoid these flaws by * giving the cache a bound number of slots * disabling the cache for the current request when errors occur The cache size limit is implemented by hashing the querystring (the cache lookup key) and generating a cache filename based on this hash modulo the cache size. In order to detect hash collisions, the full lookup key (i.e. the querystring) is stored in the cache file (separated from its associated content by ascii 0). The cache filename is the reversed 8-digit hexadecimal representation of hash(key) % cache_size which should make the filesystem lookup pretty fast (if directory content is indexed/sorted); reversing the representation avoids the problem where all keys have equal prefix. There is a new config option, cache-size, which sets the upper bound for the cache. Default value for this option is 0, which has the same effect as setting nocache=1 (hence nocache is now deprecated). Included in this patch is also a new testfile which verifies that the new option works as intended. Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cache.c359
-rw-r--r--cache.h33
-rw-r--r--cgit.c166
-rw-r--r--cgit.h1
-rwxr-xr-xtests/setup.sh2
-rwxr-xr-xtests/t0020-validate-cache.sh67
6 files changed, 418 insertions, 210 deletions
diff --git a/cache.c b/cache.c
index 89f7ecd..e590d7b 100644
--- a/cache.c
+++ b/cache.c
@@ -1,120 +1,343 @@
1/* cache.c: cache management 1/* cache.c: cache management
2 * 2 *
3 * Copyright (C) 2006 Lars Hjemli 3 * Copyright (C) 2006 Lars Hjemli
4 * 4 *
5 * Licensed under GNU General Public License v2 5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text) 6 * (see COPYING for full license text)
7 *
8 *
9 * The cache is just a directory structure where each file is a cache slot,
10 * and each filename is based on the hash of some key (e.g. the cgit url).
11 * Each file contains the full key followed by the cached content for that
12 * key.
13 *
7 */ 14 */
8 15
9#include "cgit.h" 16#include "cgit.h"
10#include "cache.h" 17#include "cache.h"
11 18
12const int NOLOCK = -1; 19#define CACHE_BUFSIZE (1024 * 4)
20
21struct cache_slot {
22 const char *key;
23 int keylen;
24 int ttl;
25 cache_fill_fn fn;
26 void *cbdata;
27 int cache_fd;
28 int lock_fd;
29 const char *cache_name;
30 const char *lock_name;
31 int match;
32 struct stat cache_st;
33 struct stat lock_st;
34 int bufsize;
35 char buf[CACHE_BUFSIZE];
36};
13 37
14char *cache_safe_filename(const char *unsafe) 38/* Open an existing cache slot and fill the cache buffer with
39 * (part of) the content of the cache file. Return 0 on success
40 * and errno otherwise.
41 */
42static int open_slot(struct cache_slot *slot)
15{ 43{
16 static char buf[4][PATH_MAX]; 44 char *bufz;
17 static int bufidx; 45 int bufkeylen = -1;
18 char *s; 46
19 char c; 47 slot->cache_fd = open(slot->cache_name, O_RDONLY);
20 48 if (slot->cache_fd == -1)
21 bufidx++; 49 return errno;
22 bufidx &= 3; 50
23 s = buf[bufidx]; 51 if (fstat(slot->cache_fd, &slot->cache_st))
24 52 return errno;
25 while(unsafe && (c = *unsafe++) != 0) { 53
26 if (c == '/' || c == ' ' || c == '&' || c == '|' || 54 slot->bufsize = read(slot->cache_fd, slot->buf, sizeof(slot->buf));
27 c == '>' || c == '<' || c == '.') 55 if (slot->bufsize == 0)
28 c = '_'; 56 return errno;
29 *s++ = c; 57
30 } 58 bufz = memchr(slot->buf, 0, slot->bufsize);
31 *s = '\0'; 59 if (bufz)
32 return buf[bufidx]; 60 bufkeylen = bufz - slot->buf;
61
62 slot->match = bufkeylen == slot->keylen &&
63 !memcmp(slot->key, slot->buf, bufkeylen + 1);
64
65 return 0;
33} 66}
34 67
35int cache_exist(struct cacheitem *item) 68/* Close the active cache slot */
69static void close_slot(struct cache_slot *slot)
36{ 70{
37 if (stat(item->name, &item->st)) { 71 if (slot->cache_fd > 0) {
38 item->st.st_mtime = 0; 72 close(slot->cache_fd);
39 return 0; 73 slot->cache_fd = -1;
40 } 74 }
41 return 1;
42} 75}
43 76
44int cache_create_dirs() 77/* Print the content of the active cache slot (but skip the key). */
78static int print_slot(struct cache_slot *slot)
45{ 79{
46 char *path; 80 ssize_t i, j = 0;
81
82 i = lseek(slot->cache_fd, slot->keylen + 1, SEEK_SET);
83 if (i != slot->keylen + 1)
84 return errno;
47 85
48 path = fmt("%s", ctx.cfg.cache_root); 86 while((i=read(slot->cache_fd, slot->buf, sizeof(slot->buf))) > 0)
49 if (mkdir(path, S_IRWXU) && errno!=EEXIST) 87 j = write(STDOUT_FILENO, slot->buf, i);
88
89 if (j < 0)
90 return errno;
91 else
50 return 0; 92 return 0;
93}
51 94
52 if (!ctx.repo) 95/* Check if the slot has expired */
96static int is_expired(struct cache_slot *slot)
97{
98 if (slot->ttl < 0)
53 return 0; 99 return 0;
100 else
101 return slot->cache_st.st_mtime + slot->ttl*60 < time(NULL);
102}
54 103
55 path = fmt("%s/%s", ctx.cfg.cache_root, 104/* Check if the slot has been modified since we opened it.
56 cache_safe_filename(ctx.repo->url)); 105 * NB: If stat() fails, we pretend the file is modified.
106 */
107static int is_modified(struct cache_slot *slot)
108{
109 struct stat st;
57 110
58 if (mkdir(path, S_IRWXU) && errno!=EEXIST) 111 if (stat(slot->cache_name, &st))
59 return 0; 112 return 1;
113 return (st.st_ino != slot->cache_st.st_ino ||
114 st.st_mtime != slot->cache_st.st_mtime ||
115 st.st_size != slot->cache_st.st_size);
116}
60 117
61 if (ctx.qry.page) { 118/* Close an open lockfile */
62 path = fmt("%s/%s/%s", ctx.cfg.cache_root, 119static void close_lock(struct cache_slot *slot)
63 cache_safe_filename(ctx.repo->url), 120{
64 ctx.qry.page); 121 if (slot->lock_fd > 0) {
65 if (mkdir(path, S_IRWXU) && errno!=EEXIST) 122 close(slot->lock_fd);
66 return 0; 123 slot->lock_fd = -1;
67 } 124 }
68 return 1;
69} 125}
70 126
71int cache_refill_overdue(const char *lockfile) 127/* Create a lockfile used to store the generated content for a cache
128 * slot, and write the slot key + \0 into it.
129 * Returns 0 on success and errno otherwise.
130 */
131static int lock_slot(struct cache_slot *slot)
72{ 132{
73 struct stat st; 133 slot->lock_fd = open(slot->lock_name, O_RDWR|O_CREAT|O_EXCL,
134 S_IRUSR|S_IWUSR);
135 if (slot->lock_fd == -1)
136 return errno;
137 write(slot->lock_fd, slot->key, slot->keylen + 1);
138 return 0;
139}
74 140
75 if (stat(lockfile, &st)) 141/* Release the current lockfile. If `replace_old_slot` is set the
76 return 0; 142 * lockfile replaces the old cache slot, otherwise the lockfile is
143 * just deleted.
144 */
145static int unlock_slot(struct cache_slot *slot, int replace_old_slot)
146{
147 int err;
148
149 if (replace_old_slot)
150 err = rename(slot->lock_name, slot->cache_name);
77 else 151 else
78 return (time(NULL) - st.st_mtime > ctx.cfg.cache_max_create_time); 152 err = unlink(slot->lock_name);
153 return err;
79} 154}
80 155
81int cache_lock(struct cacheitem *item) 156/* Generate the content for the current cache slot by redirecting
157 * stdout to the lock-fd and invoking the callback function
158 */
159static int fill_slot(struct cache_slot *slot)
82{ 160{
83 int i = 0; 161 int tmp;
84 char *lockfile = xstrdup(fmt("%s.lock", item->name));
85 162
86 top: 163 /* Preserve stdout */
87 if (++i > ctx.cfg.max_lock_attempts) 164 tmp = dup(STDOUT_FILENO);
88 die("cache_lock: unable to lock %s: %s", 165 if (tmp == -1)
89 item->name, strerror(errno)); 166 return errno;
90 167
91 item->fd = open(lockfile, O_WRONLY|O_CREAT|O_EXCL, S_IRUSR|S_IWUSR); 168 /* Redirect stdout to lockfile */
169 if (dup2(slot->lock_fd, STDOUT_FILENO) == -1)
170 return errno;
92 171
93 if (item->fd == NOLOCK && errno == ENOENT && cache_create_dirs()) 172 /* Generate cache content */
94 goto top; 173 slot->fn(slot->cbdata);
95 174
96 if (item->fd == NOLOCK && errno == EEXIST && 175 /* Restore stdout */
97 cache_refill_overdue(lockfile) && !unlink(lockfile)) 176 if (dup2(tmp, STDOUT_FILENO) == -1)
98 goto top; 177 return errno;
99 178
100 free(lockfile); 179 /* Close the temporary filedescriptor */
101 return (item->fd > 0); 180 close(tmp);
181 return 0;
102} 182}
103 183
104int cache_unlock(struct cacheitem *item) 184/* Crude implementation of 32-bit FNV-1 hash algorithm,
185 * see http://www.isthe.com/chongo/tech/comp/fnv/ for details
186 * about the magic numbers.
187 */
188#define FNV_OFFSET 0x811c9dc5
189#define FNV_PRIME 0x01000193
190
191unsigned long hash_str(const char *str)
105{ 192{
106 close(item->fd); 193 unsigned long h = FNV_OFFSET;
107 return (rename(fmt("%s.lock", item->name), item->name) == 0); 194 unsigned char *s = (unsigned char *)str;
195
196 if (!s)
197 return h;
198
199 while(*s) {
200 h *= FNV_PRIME;
201 h ^= *s++;
202 }
203 return h;
108} 204}
109 205
110int cache_cancel_lock(struct cacheitem *item) 206static int process_slot(struct cache_slot *slot)
111{ 207{
112 return (unlink(fmt("%s.lock", item->name)) == 0); 208 int err;
209
210 err = open_slot(slot);
211 if (!err && slot->match) {
212 if (is_expired(slot)) {
213 if (!lock_slot(slot)) {
214 /* If the cachefile has been replaced between
215 * `open_slot` and `lock_slot`, we'll just
216 * serve the stale content from the original
217 * cachefile. This way we avoid pruning the
218 * newly generated slot. The same code-path
219 * is chosen if fill_slot() fails for some
220 * reason.
221 *
222 * TODO? check if the new slot contains the
223 * same key as the old one, since we would
224 * prefer to serve the newest content.
225 * This will require us to open yet another
226 * file-descriptor and read and compare the
227 * key from the new file, so for now we're
228 * lazy and just ignore the new file.
229 */
230 if (is_modified(slot) || fill_slot(slot)) {
231 unlock_slot(slot, 0);
232 close_lock(slot);
233 } else {
234 close_slot(slot);
235 unlock_slot(slot, 1);
236 slot->cache_fd = slot->lock_fd;
237 }
238 }
239 }
240 print_slot(slot);
241 close_slot(slot);
242 return 0;
243 }
244
245 /* If the cache slot does not exist (or its key doesn't match the
246 * current key), lets try to create a new cache slot for this
247 * request. If this fails (for whatever reason), lets just generate
248 * the content without caching it and fool the caller to belive
249 * everything worked out (but print a warning on stdout).
250 */
251
252 close_slot(slot);
253 if ((err = lock_slot(slot)) != 0) {
254 cache_log("[cgit] Unable to lock slot %s: %s (%d)\n",
255 slot->lock_name, strerror(err), err);
256 slot->fn(slot->cbdata);
257 return 0;
258 }
259
260 if ((err = fill_slot(slot)) != 0) {
261 cache_log("[cgit] Unable to fill slot %s: %s (%d)\n",
262 slot->lock_name, strerror(err), err);
263 unlock_slot(slot, 0);
264 close_lock(slot);
265 slot->fn(slot->cbdata);
266 return 0;
267 }
268 // We've got a valid cache slot in the lock file, which
269 // is about to replace the old cache slot. But if we
270 // release the lockfile and then try to open the new cache
271 // slot, we might get a race condition with a concurrent
272 // writer for the same cache slot (with a different key).
273 // Lets avoid such a race by just printing the content of
274 // the lock file.
275 slot->cache_fd = slot->lock_fd;
276 unlock_slot(slot, 1);
277 err = print_slot(slot);
278 close_slot(slot);
279 return err;
113} 280}
114 281
115int cache_expired(struct cacheitem *item) 282/* Print cached content to stdout, generate the content if necessary. */
283int cache_process(int size, const char *path, const char *key, int ttl,
284 cache_fill_fn fn, void *cbdata)
116{ 285{
117 if (item->ttl < 0) 286 unsigned long hash;
287 int len, i;
288 char filename[1024];
289 char lockname[1024 + 5]; /* 5 = ".lock" */
290 struct cache_slot slot;
291
292 /* If the cache is disabled, just generate the content */
293 if (size <= 0) {
294 fn(cbdata);
118 return 0; 295 return 0;
119 return item->st.st_mtime + item->ttl * 60 < time(NULL); 296 }
297
298 /* Verify input, calculate filenames */
299 if (!path) {
300 cache_log("[cgit] Cache path not specified, caching is disabled\n");
301 fn(cbdata);
302 return 0;
303 }
304 len = strlen(path);
305 if (len > sizeof(filename) - 10) { /* 10 = "/01234567\0" */
306 cache_log("[cgit] Cache path too long, caching is disabled: %s\n",
307 path);
308 fn(cbdata);
309 return 0;
310 }
311 if (!key)
312 key = "";
313 hash = hash_str(key) % size;
314 strcpy(filename, path);
315 if (filename[len - 1] != '/')
316 filename[len++] = '/';
317 for(i = 0; i < 8; i++) {
318 sprintf(filename + len++, "%x",
319 (unsigned char)(hash & 0xf));
320 hash >>= 4;
321 }
322 filename[len] = '\0';
323 strcpy(lockname, filename);
324 strcpy(lockname + len, ".lock");
325 slot.fn = fn;
326 slot.cbdata = cbdata;
327 slot.ttl = ttl;
328 slot.cache_name = filename;
329 slot.lock_name = lockname;
330 slot.key = key;
331 slot.keylen = strlen(key);
332 return process_slot(&slot);
120} 333}
334
335/* Print a message to stdout */
336void cache_log(const char *format, ...)
337{
338 va_list args;
339 va_start(args, format);
340 vfprintf(stderr, format, args);
341 va_end(args);
342}
343
diff --git a/cache.h b/cache.h
index 4dcbea3..5f2178d 100644
--- a/cache.h
+++ b/cache.h
@@ -1,23 +1,32 @@
1/* 1/*
2 * Since git has it's own cache.h which we include, 2 * Since git has it's own cache.h which we include,
3 * lets test on CGIT_CACHE_H to avoid confusion 3 * lets test on CGIT_CACHE_H to avoid confusion
4 */ 4 */
5 5
6#ifndef CGIT_CACHE_H 6#ifndef CGIT_CACHE_H
7#define CGIT_CACHE_H 7#define CGIT_CACHE_H
8 8
9struct cacheitem { 9typedef void (*cache_fill_fn)(void *cbdata);
10 char *name;
11 struct stat st;
12 int ttl;
13 int fd;
14};
15 10
16extern char *cache_safe_filename(const char *unsafe); 11
17extern int cache_lock(struct cacheitem *item); 12/* Print cached content to stdout, generate the content if necessary.
18extern int cache_unlock(struct cacheitem *item); 13 *
19extern int cache_cancel_lock(struct cacheitem *item); 14 * Parameters
20extern int cache_exist(struct cacheitem *item); 15 * size max number of cache files
21extern int cache_expired(struct cacheitem *item); 16 * path directory used to store cache files
17 * key the key used to lookup cache files
18 * ttl max cache time in seconds for this key
19 * fn content generator function for this key
20 * cbdata user-supplied data to the content generator function
21 *
22 * Return value
23 * 0 indicates success, everyting else is an error
24 */
25extern int cache_process(int size, const char *path, const char *key, int ttl,
26 cache_fill_fn fn, void *cbdata);
27
28
29/* Print a message to stdout */
30extern void cache_log(const char *format, ...);
22 31
23#endif /* CGIT_CACHE_H */ 32#endif /* CGIT_CACHE_H */
diff --git a/cgit.c b/cgit.c
index 38b0ba5..4dc8eec 100644
--- a/cgit.c
+++ b/cgit.c
@@ -1,95 +1,97 @@
1/* cgit.c: cgi for the git scm 1/* cgit.c: cgi for the git scm
2 * 2 *
3 * Copyright (C) 2006 Lars Hjemli 3 * Copyright (C) 2006 Lars Hjemli
4 * 4 *
5 * Licensed under GNU General Public License v2 5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text) 6 * (see COPYING for full license text)
7 */ 7 */
8 8
9#include "cgit.h" 9#include "cgit.h"
10#include "cache.h" 10#include "cache.h"
11#include "cmd.h" 11#include "cmd.h"
12#include "configfile.h" 12#include "configfile.h"
13#include "html.h" 13#include "html.h"
14#include "ui-shared.h" 14#include "ui-shared.h"
15 15
16const char *cgit_version = CGIT_VERSION; 16const char *cgit_version = CGIT_VERSION;
17 17
18void config_cb(const char *name, const char *value) 18void config_cb(const char *name, const char *value)
19{ 19{
20 if (!strcmp(name, "root-title")) 20 if (!strcmp(name, "root-title"))
21 ctx.cfg.root_title = xstrdup(value); 21 ctx.cfg.root_title = xstrdup(value);
22 else if (!strcmp(name, "css")) 22 else if (!strcmp(name, "css"))
23 ctx.cfg.css = xstrdup(value); 23 ctx.cfg.css = xstrdup(value);
24 else if (!strcmp(name, "logo")) 24 else if (!strcmp(name, "logo"))
25 ctx.cfg.logo = xstrdup(value); 25 ctx.cfg.logo = xstrdup(value);
26 else if (!strcmp(name, "index-header")) 26 else if (!strcmp(name, "index-header"))
27 ctx.cfg.index_header = xstrdup(value); 27 ctx.cfg.index_header = xstrdup(value);
28 else if (!strcmp(name, "index-info")) 28 else if (!strcmp(name, "index-info"))
29 ctx.cfg.index_info = xstrdup(value); 29 ctx.cfg.index_info = xstrdup(value);
30 else if (!strcmp(name, "logo-link")) 30 else if (!strcmp(name, "logo-link"))
31 ctx.cfg.logo_link = xstrdup(value); 31 ctx.cfg.logo_link = xstrdup(value);
32 else if (!strcmp(name, "module-link")) 32 else if (!strcmp(name, "module-link"))
33 ctx.cfg.module_link = xstrdup(value); 33 ctx.cfg.module_link = xstrdup(value);
34 else if (!strcmp(name, "virtual-root")) { 34 else if (!strcmp(name, "virtual-root")) {
35 ctx.cfg.virtual_root = trim_end(value, '/'); 35 ctx.cfg.virtual_root = trim_end(value, '/');
36 if (!ctx.cfg.virtual_root && (!strcmp(value, "/"))) 36 if (!ctx.cfg.virtual_root && (!strcmp(value, "/")))
37 ctx.cfg.virtual_root = ""; 37 ctx.cfg.virtual_root = "";
38 } else if (!strcmp(name, "nocache")) 38 } else if (!strcmp(name, "nocache"))
39 ctx.cfg.nocache = atoi(value); 39 ctx.cfg.nocache = atoi(value);
40 else if (!strcmp(name, "snapshots")) 40 else if (!strcmp(name, "snapshots"))
41 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value); 41 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value);
42 else if (!strcmp(name, "enable-index-links")) 42 else if (!strcmp(name, "enable-index-links"))
43 ctx.cfg.enable_index_links = atoi(value); 43 ctx.cfg.enable_index_links = atoi(value);
44 else if (!strcmp(name, "enable-log-filecount")) 44 else if (!strcmp(name, "enable-log-filecount"))
45 ctx.cfg.enable_log_filecount = atoi(value); 45 ctx.cfg.enable_log_filecount = atoi(value);
46 else if (!strcmp(name, "enable-log-linecount")) 46 else if (!strcmp(name, "enable-log-linecount"))
47 ctx.cfg.enable_log_linecount = atoi(value); 47 ctx.cfg.enable_log_linecount = atoi(value);
48 else if (!strcmp(name, "cache-size"))
49 ctx.cfg.cache_size = atoi(value);
48 else if (!strcmp(name, "cache-root")) 50 else if (!strcmp(name, "cache-root"))
49 ctx.cfg.cache_root = xstrdup(value); 51 ctx.cfg.cache_root = xstrdup(value);
50 else if (!strcmp(name, "cache-root-ttl")) 52 else if (!strcmp(name, "cache-root-ttl"))
51 ctx.cfg.cache_root_ttl = atoi(value); 53 ctx.cfg.cache_root_ttl = atoi(value);
52 else if (!strcmp(name, "cache-repo-ttl")) 54 else if (!strcmp(name, "cache-repo-ttl"))
53 ctx.cfg.cache_repo_ttl = atoi(value); 55 ctx.cfg.cache_repo_ttl = atoi(value);
54 else if (!strcmp(name, "cache-static-ttl")) 56 else if (!strcmp(name, "cache-static-ttl"))
55 ctx.cfg.cache_static_ttl = atoi(value); 57 ctx.cfg.cache_static_ttl = atoi(value);
56 else if (!strcmp(name, "cache-dynamic-ttl")) 58 else if (!strcmp(name, "cache-dynamic-ttl"))
57 ctx.cfg.cache_dynamic_ttl = atoi(value); 59 ctx.cfg.cache_dynamic_ttl = atoi(value);
58 else if (!strcmp(name, "max-message-length")) 60 else if (!strcmp(name, "max-message-length"))
59 ctx.cfg.max_msg_len = atoi(value); 61 ctx.cfg.max_msg_len = atoi(value);
60 else if (!strcmp(name, "max-repodesc-length")) 62 else if (!strcmp(name, "max-repodesc-length"))
61 ctx.cfg.max_repodesc_len = atoi(value); 63 ctx.cfg.max_repodesc_len = atoi(value);
62 else if (!strcmp(name, "max-commit-count")) 64 else if (!strcmp(name, "max-commit-count"))
63 ctx.cfg.max_commit_count = atoi(value); 65 ctx.cfg.max_commit_count = atoi(value);
64 else if (!strcmp(name, "summary-log")) 66 else if (!strcmp(name, "summary-log"))
65 ctx.cfg.summary_log = atoi(value); 67 ctx.cfg.summary_log = atoi(value);
66 else if (!strcmp(name, "summary-branches")) 68 else if (!strcmp(name, "summary-branches"))
67 ctx.cfg.summary_branches = atoi(value); 69 ctx.cfg.summary_branches = atoi(value);
68 else if (!strcmp(name, "summary-tags")) 70 else if (!strcmp(name, "summary-tags"))
69 ctx.cfg.summary_tags = atoi(value); 71 ctx.cfg.summary_tags = atoi(value);
70 else if (!strcmp(name, "agefile")) 72 else if (!strcmp(name, "agefile"))
71 ctx.cfg.agefile = xstrdup(value); 73 ctx.cfg.agefile = xstrdup(value);
72 else if (!strcmp(name, "renamelimit")) 74 else if (!strcmp(name, "renamelimit"))
73 ctx.cfg.renamelimit = atoi(value); 75 ctx.cfg.renamelimit = atoi(value);
74 else if (!strcmp(name, "robots")) 76 else if (!strcmp(name, "robots"))
75 ctx.cfg.robots = xstrdup(value); 77 ctx.cfg.robots = xstrdup(value);
76 else if (!strcmp(name, "clone-prefix")) 78 else if (!strcmp(name, "clone-prefix"))
77 ctx.cfg.clone_prefix = xstrdup(value); 79 ctx.cfg.clone_prefix = xstrdup(value);
78 else if (!strcmp(name, "repo.group")) 80 else if (!strcmp(name, "repo.group"))
79 ctx.cfg.repo_group = xstrdup(value); 81 ctx.cfg.repo_group = xstrdup(value);
80 else if (!strcmp(name, "repo.url")) 82 else if (!strcmp(name, "repo.url"))
81 ctx.repo = cgit_add_repo(value); 83 ctx.repo = cgit_add_repo(value);
82 else if (!strcmp(name, "repo.name")) 84 else if (!strcmp(name, "repo.name"))
83 ctx.repo->name = xstrdup(value); 85 ctx.repo->name = xstrdup(value);
84 else if (ctx.repo && !strcmp(name, "repo.path")) 86 else if (ctx.repo && !strcmp(name, "repo.path"))
85 ctx.repo->path = trim_end(value, '/'); 87 ctx.repo->path = trim_end(value, '/');
86 else if (ctx.repo && !strcmp(name, "repo.clone-url")) 88 else if (ctx.repo && !strcmp(name, "repo.clone-url"))
87 ctx.repo->clone_url = xstrdup(value); 89 ctx.repo->clone_url = xstrdup(value);
88 else if (ctx.repo && !strcmp(name, "repo.desc")) 90 else if (ctx.repo && !strcmp(name, "repo.desc"))
89 ctx.repo->desc = xstrdup(value); 91 ctx.repo->desc = xstrdup(value);
90 else if (ctx.repo && !strcmp(name, "repo.owner")) 92 else if (ctx.repo && !strcmp(name, "repo.owner"))
91 ctx.repo->owner = xstrdup(value); 93 ctx.repo->owner = xstrdup(value);
92 else if (ctx.repo && !strcmp(name, "repo.defbranch")) 94 else if (ctx.repo && !strcmp(name, "repo.defbranch"))
93 ctx.repo->defbranch = xstrdup(value); 95 ctx.repo->defbranch = xstrdup(value);
94 else if (ctx.repo && !strcmp(name, "repo.snapshots")) 96 else if (ctx.repo && !strcmp(name, "repo.snapshots"))
95 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */ 97 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */
@@ -98,364 +100,270 @@ void config_cb(const char *name, const char *value)
98 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount")) 100 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount"))
99 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value); 101 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value);
100 else if (ctx.repo && !strcmp(name, "repo.module-link")) 102 else if (ctx.repo && !strcmp(name, "repo.module-link"))
101 ctx.repo->module_link= xstrdup(value); 103 ctx.repo->module_link= xstrdup(value);
102 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) { 104 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) {
103 if (*value == '/') 105 if (*value == '/')
104 ctx.repo->readme = xstrdup(value); 106 ctx.repo->readme = xstrdup(value);
105 else 107 else
106 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value)); 108 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value));
107 } else if (!strcmp(name, "include")) 109 } else if (!strcmp(name, "include"))
108 parse_configfile(value, config_cb); 110 parse_configfile(value, config_cb);
109} 111}
110 112
111static void querystring_cb(const char *name, const char *value) 113static void querystring_cb(const char *name, const char *value)
112{ 114{
113 if (!strcmp(name,"r")) { 115 if (!strcmp(name,"r")) {
114 ctx.qry.repo = xstrdup(value); 116 ctx.qry.repo = xstrdup(value);
115 ctx.repo = cgit_get_repoinfo(value); 117 ctx.repo = cgit_get_repoinfo(value);
116 } else if (!strcmp(name, "p")) { 118 } else if (!strcmp(name, "p")) {
117 ctx.qry.page = xstrdup(value); 119 ctx.qry.page = xstrdup(value);
118 } else if (!strcmp(name, "url")) { 120 } else if (!strcmp(name, "url")) {
119 cgit_parse_url(value); 121 cgit_parse_url(value);
120 } else if (!strcmp(name, "qt")) { 122 } else if (!strcmp(name, "qt")) {
121 ctx.qry.grep = xstrdup(value); 123 ctx.qry.grep = xstrdup(value);
122 } else if (!strcmp(name, "q")) { 124 } else if (!strcmp(name, "q")) {
123 ctx.qry.search = xstrdup(value); 125 ctx.qry.search = xstrdup(value);
124 } else if (!strcmp(name, "h")) { 126 } else if (!strcmp(name, "h")) {
125 ctx.qry.head = xstrdup(value); 127 ctx.qry.head = xstrdup(value);
126 ctx.qry.has_symref = 1; 128 ctx.qry.has_symref = 1;
127 } else if (!strcmp(name, "id")) { 129 } else if (!strcmp(name, "id")) {
128 ctx.qry.sha1 = xstrdup(value); 130 ctx.qry.sha1 = xstrdup(value);
129 ctx.qry.has_sha1 = 1; 131 ctx.qry.has_sha1 = 1;
130 } else if (!strcmp(name, "id2")) { 132 } else if (!strcmp(name, "id2")) {
131 ctx.qry.sha2 = xstrdup(value); 133 ctx.qry.sha2 = xstrdup(value);
132 ctx.qry.has_sha1 = 1; 134 ctx.qry.has_sha1 = 1;
133 } else if (!strcmp(name, "ofs")) { 135 } else if (!strcmp(name, "ofs")) {
134 ctx.qry.ofs = atoi(value); 136 ctx.qry.ofs = atoi(value);
135 } else if (!strcmp(name, "path")) { 137 } else if (!strcmp(name, "path")) {
136 ctx.qry.path = trim_end(value, '/'); 138 ctx.qry.path = trim_end(value, '/');
137 } else if (!strcmp(name, "name")) { 139 } else if (!strcmp(name, "name")) {
138 ctx.qry.name = xstrdup(value); 140 ctx.qry.name = xstrdup(value);
139 } 141 }
140} 142}
141 143
142static void prepare_context(struct cgit_context *ctx) 144static void prepare_context(struct cgit_context *ctx)
143{ 145{
144 memset(ctx, 0, sizeof(ctx)); 146 memset(ctx, 0, sizeof(ctx));
145 ctx->cfg.agefile = "info/web/last-modified"; 147 ctx->cfg.agefile = "info/web/last-modified";
148 ctx->cfg.nocache = 0;
149 ctx->cfg.cache_size = 0;
146 ctx->cfg.cache_dynamic_ttl = 5; 150 ctx->cfg.cache_dynamic_ttl = 5;
147 ctx->cfg.cache_max_create_time = 5; 151 ctx->cfg.cache_max_create_time = 5;
148 ctx->cfg.cache_repo_ttl = 5; 152 ctx->cfg.cache_repo_ttl = 5;
149 ctx->cfg.cache_root = CGIT_CACHE_ROOT; 153 ctx->cfg.cache_root = CGIT_CACHE_ROOT;
150 ctx->cfg.cache_root_ttl = 5; 154 ctx->cfg.cache_root_ttl = 5;
151 ctx->cfg.cache_static_ttl = -1; 155 ctx->cfg.cache_static_ttl = -1;
152 ctx->cfg.css = "/cgit.css"; 156 ctx->cfg.css = "/cgit.css";
153 ctx->cfg.logo = "/git-logo.png"; 157 ctx->cfg.logo = "/git-logo.png";
154 ctx->cfg.max_commit_count = 50; 158 ctx->cfg.max_commit_count = 50;
155 ctx->cfg.max_lock_attempts = 5; 159 ctx->cfg.max_lock_attempts = 5;
156 ctx->cfg.max_msg_len = 60; 160 ctx->cfg.max_msg_len = 60;
157 ctx->cfg.max_repodesc_len = 60; 161 ctx->cfg.max_repodesc_len = 60;
158 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s"; 162 ctx->cfg.module_link = "./?repo=%s&page=commit&id=%s";
159 ctx->cfg.renamelimit = -1; 163 ctx->cfg.renamelimit = -1;
160 ctx->cfg.robots = "index, nofollow"; 164 ctx->cfg.robots = "index, nofollow";
161 ctx->cfg.root_title = "Git repository browser"; 165 ctx->cfg.root_title = "Git repository browser";
162 ctx->cfg.script_name = CGIT_SCRIPT_NAME; 166 ctx->cfg.script_name = CGIT_SCRIPT_NAME;
163 ctx->page.mimetype = "text/html"; 167 ctx->page.mimetype = "text/html";
164 ctx->page.charset = PAGE_ENCODING; 168 ctx->page.charset = PAGE_ENCODING;
165 ctx->page.filename = NULL; 169 ctx->page.filename = NULL;
166} 170 ctx->page.modified = time(NULL);
167 171 ctx->page.expires = ctx->page.modified;
168static int cgit_prepare_cache(struct cacheitem *item)
169{
170 if (!ctx.repo && ctx.qry.repo) {
171 ctx.page.title = fmt("%s - %s", ctx.cfg.root_title,
172 "Bad request");
173 cgit_print_http_headers(&ctx);
174 cgit_print_docstart(&ctx);
175 cgit_print_pageheader(&ctx);
176 cgit_print_error(fmt("Unknown repo: %s", ctx.qry.repo));
177 cgit_print_docend();
178 return 0;
179 }
180
181 if (!ctx.repo) {
182 item->name = xstrdup(fmt("%s/index.%s.html",
183 ctx.cfg.cache_root,
184 cache_safe_filename(ctx.qry.raw)));
185 item->ttl = ctx.cfg.cache_root_ttl;
186 return 1;
187 }
188
189 if (!ctx.qry.page) {
190 item->name = xstrdup(fmt("%s/%s/index.%s.html", ctx.cfg.cache_root,
191 cache_safe_filename(ctx.repo->url),
192 cache_safe_filename(ctx.qry.raw)));
193 item->ttl = ctx.cfg.cache_repo_ttl;
194 } else {
195 item->name = xstrdup(fmt("%s/%s/%s/%s.html", ctx.cfg.cache_root,
196 cache_safe_filename(ctx.repo->url),
197 ctx.qry.page,
198 cache_safe_filename(ctx.qry.raw)));
199 if (ctx.qry.has_symref)
200 item->ttl = ctx.cfg.cache_dynamic_ttl;
201 else if (ctx.qry.has_sha1)
202 item->ttl = ctx.cfg.cache_static_ttl;
203 else
204 item->ttl = ctx.cfg.cache_repo_ttl;
205 }
206 return 1;
207} 172}
208 173
209struct refmatch { 174struct refmatch {
210 char *req_ref; 175 char *req_ref;
211 char *first_ref; 176 char *first_ref;
212 int match; 177 int match;
213}; 178};
214 179
215int find_current_ref(const char *refname, const unsigned char *sha1, 180int find_current_ref(const char *refname, const unsigned char *sha1,
216 int flags, void *cb_data) 181 int flags, void *cb_data)
217{ 182{
218 struct refmatch *info; 183 struct refmatch *info;
219 184
220 info = (struct refmatch *)cb_data; 185 info = (struct refmatch *)cb_data;
221 if (!strcmp(refname, info->req_ref)) 186 if (!strcmp(refname, info->req_ref))
222 info->match = 1; 187 info->match = 1;
223 if (!info->first_ref) 188 if (!info->first_ref)
224 info->first_ref = xstrdup(refname); 189 info->first_ref = xstrdup(refname);
225 return info->match; 190 return info->match;
226} 191}
227 192
228char *find_default_branch(struct cgit_repo *repo) 193char *find_default_branch(struct cgit_repo *repo)
229{ 194{
230 struct refmatch info; 195 struct refmatch info;
231 196
232 info.req_ref = repo->defbranch; 197 info.req_ref = repo->defbranch;
233 info.first_ref = NULL; 198 info.first_ref = NULL;
234 info.match = 0; 199 info.match = 0;
235 for_each_branch_ref(find_current_ref, &info); 200 for_each_branch_ref(find_current_ref, &info);
236 if (info.match) 201 if (info.match)
237 return info.req_ref; 202 return info.req_ref;
238 else 203 else
239 return info.first_ref; 204 return info.first_ref;
240} 205}
241 206
242static int prepare_repo_cmd(struct cgit_context *ctx) 207static int prepare_repo_cmd(struct cgit_context *ctx)
243{ 208{
244 char *tmp; 209 char *tmp;
245 unsigned char sha1[20]; 210 unsigned char sha1[20];
246 int nongit = 0; 211 int nongit = 0;
247 212
248 setenv("GIT_DIR", ctx->repo->path, 1); 213 setenv("GIT_DIR", ctx->repo->path, 1);
249 setup_git_directory_gently(&nongit); 214 setup_git_directory_gently(&nongit);
250 if (nongit) { 215 if (nongit) {
251 ctx->page.title = fmt("%s - %s", ctx->cfg.root_title, 216 ctx->page.title = fmt("%s - %s", ctx->cfg.root_title,
252 "config error"); 217 "config error");
253 tmp = fmt("Not a git repository: '%s'", ctx->repo->path); 218 tmp = fmt("Not a git repository: '%s'", ctx->repo->path);
254 ctx->repo = NULL; 219 ctx->repo = NULL;
255 cgit_print_http_headers(ctx); 220 cgit_print_http_headers(ctx);
256 cgit_print_docstart(ctx); 221 cgit_print_docstart(ctx);
257 cgit_print_pageheader(ctx); 222 cgit_print_pageheader(ctx);
258 cgit_print_error(tmp); 223 cgit_print_error(tmp);
259 cgit_print_docend(); 224 cgit_print_docend();
260 return 1; 225 return 1;
261 } 226 }
262 ctx->page.title = fmt("%s - %s", ctx->repo->name, ctx->repo->desc); 227 ctx->page.title = fmt("%s - %s", ctx->repo->name, ctx->repo->desc);
263 228
264 if (!ctx->qry.head) { 229 if (!ctx->qry.head) {
265 ctx->qry.head = xstrdup(find_default_branch(ctx->repo)); 230 ctx->qry.head = xstrdup(find_default_branch(ctx->repo));
266 ctx->repo->defbranch = ctx->qry.head; 231 ctx->repo->defbranch = ctx->qry.head;
267 } 232 }
268 233
269 if (!ctx->qry.head) { 234 if (!ctx->qry.head) {
270 cgit_print_http_headers(ctx); 235 cgit_print_http_headers(ctx);
271 cgit_print_docstart(ctx); 236 cgit_print_docstart(ctx);
272 cgit_print_pageheader(ctx); 237 cgit_print_pageheader(ctx);
273 cgit_print_error("Repository seems to be empty"); 238 cgit_print_error("Repository seems to be empty");
274 cgit_print_docend(); 239 cgit_print_docend();
275 return 1; 240 return 1;
276 } 241 }
277 242
278 if (get_sha1(ctx->qry.head, sha1)) { 243 if (get_sha1(ctx->qry.head, sha1)) {
279 tmp = xstrdup(ctx->qry.head); 244 tmp = xstrdup(ctx->qry.head);
280 ctx->qry.head = ctx->repo->defbranch; 245 ctx->qry.head = ctx->repo->defbranch;
281 cgit_print_http_headers(ctx); 246 cgit_print_http_headers(ctx);
282 cgit_print_docstart(ctx); 247 cgit_print_docstart(ctx);
283 cgit_print_pageheader(ctx); 248 cgit_print_pageheader(ctx);
284 cgit_print_error(fmt("Invalid branch: %s", tmp)); 249 cgit_print_error(fmt("Invalid branch: %s", tmp));
285 cgit_print_docend(); 250 cgit_print_docend();
286 return 1; 251 return 1;
287 } 252 }
288 return 0; 253 return 0;
289} 254}
290 255
291static void process_request(struct cgit_context *ctx) 256static void process_request(void *cbdata)
292{ 257{
258 struct cgit_context *ctx = cbdata;
293 struct cgit_cmd *cmd; 259 struct cgit_cmd *cmd;
294 260
295 cmd = cgit_get_cmd(ctx); 261 cmd = cgit_get_cmd(ctx);
296 if (!cmd) { 262 if (!cmd) {
297 ctx->page.title = "cgit error"; 263 ctx->page.title = "cgit error";
298 ctx->repo = NULL; 264 ctx->repo = NULL;
299 cgit_print_http_headers(ctx); 265 cgit_print_http_headers(ctx);
300 cgit_print_docstart(ctx); 266 cgit_print_docstart(ctx);
301 cgit_print_pageheader(ctx); 267 cgit_print_pageheader(ctx);
302 cgit_print_error("Invalid request"); 268 cgit_print_error("Invalid request");
303 cgit_print_docend(); 269 cgit_print_docend();
304 return; 270 return;
305 } 271 }
306 272
307 if (cmd->want_repo && prepare_repo_cmd(ctx)) 273 if (cmd->want_repo && prepare_repo_cmd(ctx))
308 return; 274 return;
309 275
310 if (cmd->want_layout) { 276 if (cmd->want_layout) {
311 cgit_print_http_headers(ctx); 277 cgit_print_http_headers(ctx);
312 cgit_print_docstart(ctx); 278 cgit_print_docstart(ctx);
313 cgit_print_pageheader(ctx); 279 cgit_print_pageheader(ctx);
314 } 280 }
315 281
316 cmd->fn(ctx); 282 cmd->fn(ctx);
317 283
318 if (cmd->want_layout) 284 if (cmd->want_layout)
319 cgit_print_docend(); 285 cgit_print_docend();
320} 286}
321 287
322static long ttl_seconds(long ttl)
323{
324 if (ttl<0)
325 return 60 * 60 * 24 * 365;
326 else
327 return ttl * 60;
328}
329
330static void cgit_fill_cache(struct cacheitem *item, int use_cache)
331{
332 int stdout2;
333
334 if (use_cache) {
335 stdout2 = chk_positive(dup(STDOUT_FILENO),
336 "Preserving STDOUT");
337 chk_zero(close(STDOUT_FILENO), "Closing STDOUT");
338 chk_positive(dup2(item->fd, STDOUT_FILENO), "Dup2(cachefile)");
339 }
340
341 ctx.page.modified = time(NULL);
342 ctx.page.expires = ctx.page.modified + ttl_seconds(item->ttl);
343 process_request(&ctx);
344
345 if (use_cache) {
346 chk_zero(close(STDOUT_FILENO), "Close redirected STDOUT");
347 chk_positive(dup2(stdout2, STDOUT_FILENO),
348 "Restoring original STDOUT");
349 chk_zero(close(stdout2), "Closing temporary STDOUT");
350 }
351}
352
353static void cgit_check_cache(struct cacheitem *item)
354{
355 int i = 0;
356
357 top:
358 if (++i > ctx.cfg.max_lock_attempts) {
359 die("cgit_refresh_cache: unable to lock %s: %s",
360 item->name, strerror(errno));
361 }
362 if (!cache_exist(item)) {
363 if (!cache_lock(item)) {
364 sleep(1);
365 goto top;
366 }
367 if (!cache_exist(item)) {
368 cgit_fill_cache(item, 1);
369 cache_unlock(item);
370 } else {
371 cache_cancel_lock(item);
372 }
373 } else if (cache_expired(item) && cache_lock(item)) {
374 if (cache_expired(item)) {
375 cgit_fill_cache(item, 1);
376 cache_unlock(item);
377 } else {
378 cache_cancel_lock(item);
379 }
380 }
381}
382
383static void cgit_print_cache(struct cacheitem *item)
384{
385 static char buf[4096];
386 ssize_t i;
387
388 int fd = open(item->name, O_RDONLY);
389 if (fd<0)
390 die("Unable to open cached file %s", item->name);
391
392 while((i=read(fd, buf, sizeof(buf))) > 0)
393 write(STDOUT_FILENO, buf, i);
394
395 close(fd);
396}
397
398static void cgit_parse_args(int argc, const char **argv) 288static void cgit_parse_args(int argc, const char **argv)
399{ 289{
400 int i; 290 int i;
401 291
402 for (i = 1; i < argc; i++) { 292 for (i = 1; i < argc; i++) {
403 if (!strncmp(argv[i], "--cache=", 8)) { 293 if (!strncmp(argv[i], "--cache=", 8)) {
404 ctx.cfg.cache_root = xstrdup(argv[i]+8); 294 ctx.cfg.cache_root = xstrdup(argv[i]+8);
405 } 295 }
406 if (!strcmp(argv[i], "--nocache")) { 296 if (!strcmp(argv[i], "--nocache")) {
407 ctx.cfg.nocache = 1; 297 ctx.cfg.nocache = 1;
408 } 298 }
409 if (!strncmp(argv[i], "--query=", 8)) { 299 if (!strncmp(argv[i], "--query=", 8)) {
410 ctx.qry.raw = xstrdup(argv[i]+8); 300 ctx.qry.raw = xstrdup(argv[i]+8);
411 } 301 }
412 if (!strncmp(argv[i], "--repo=", 7)) { 302 if (!strncmp(argv[i], "--repo=", 7)) {
413 ctx.qry.repo = xstrdup(argv[i]+7); 303 ctx.qry.repo = xstrdup(argv[i]+7);
414 } 304 }
415 if (!strncmp(argv[i], "--page=", 7)) { 305 if (!strncmp(argv[i], "--page=", 7)) {
416 ctx.qry.page = xstrdup(argv[i]+7); 306 ctx.qry.page = xstrdup(argv[i]+7);
417 } 307 }
418 if (!strncmp(argv[i], "--head=", 7)) { 308 if (!strncmp(argv[i], "--head=", 7)) {
419 ctx.qry.head = xstrdup(argv[i]+7); 309 ctx.qry.head = xstrdup(argv[i]+7);
420 ctx.qry.has_symref = 1; 310 ctx.qry.has_symref = 1;
421 } 311 }
422 if (!strncmp(argv[i], "--sha1=", 7)) { 312 if (!strncmp(argv[i], "--sha1=", 7)) {
423 ctx.qry.sha1 = xstrdup(argv[i]+7); 313 ctx.qry.sha1 = xstrdup(argv[i]+7);
424 ctx.qry.has_sha1 = 1; 314 ctx.qry.has_sha1 = 1;
425 } 315 }
426 if (!strncmp(argv[i], "--ofs=", 6)) { 316 if (!strncmp(argv[i], "--ofs=", 6)) {
427 ctx.qry.ofs = atoi(argv[i]+6); 317 ctx.qry.ofs = atoi(argv[i]+6);
428 } 318 }
429 } 319 }
430} 320}
431 321
322static int calc_ttl()
323{
324 if (!ctx.repo)
325 return ctx.cfg.cache_root_ttl;
326
327 if (!ctx.qry.page)
328 return ctx.cfg.cache_repo_ttl;
329
330 if (ctx.qry.has_symref)
331 return ctx.cfg.cache_dynamic_ttl;
332
333 if (ctx.qry.has_sha1)
334 return ctx.cfg.cache_static_ttl;
335
336 return ctx.cfg.cache_repo_ttl;
337}
338
432int main(int argc, const char **argv) 339int main(int argc, const char **argv)
433{ 340{
434 struct cacheitem item;
435 const char *cgit_config_env = getenv("CGIT_CONFIG"); 341 const char *cgit_config_env = getenv("CGIT_CONFIG");
342 int err, ttl;
436 343
437 prepare_context(&ctx); 344 prepare_context(&ctx);
438 item.st.st_mtime = time(NULL);
439 cgit_repolist.length = 0; 345 cgit_repolist.length = 0;
440 cgit_repolist.count = 0; 346 cgit_repolist.count = 0;
441 cgit_repolist.repos = NULL; 347 cgit_repolist.repos = NULL;
442 348
443 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG, 349 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG,
444 config_cb); 350 config_cb);
445 ctx.repo = NULL; 351 ctx.repo = NULL;
446 if (getenv("SCRIPT_NAME")) 352 if (getenv("SCRIPT_NAME"))
447 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME")); 353 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME"));
448 if (getenv("QUERY_STRING")) 354 if (getenv("QUERY_STRING"))
449 ctx.qry.raw = xstrdup(getenv("QUERY_STRING")); 355 ctx.qry.raw = xstrdup(getenv("QUERY_STRING"));
450 cgit_parse_args(argc, argv); 356 cgit_parse_args(argc, argv);
451 http_parse_querystring(ctx.qry.raw, querystring_cb); 357 http_parse_querystring(ctx.qry.raw, querystring_cb);
452 if (!cgit_prepare_cache(&item)) 358
453 return 0; 359 ttl = calc_ttl();
454 if (ctx.cfg.nocache) { 360 ctx.page.expires += ttl*60;
455 cgit_fill_cache(&item, 0); 361 if (ctx.cfg.nocache)
456 } else { 362 ctx.cfg.cache_size = 0;
457 cgit_check_cache(&item); 363 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root,
458 cgit_print_cache(&item); 364 ctx.qry.raw, ttl, process_request, &ctx);
459 } 365 if (err)
460 return 0; 366 cache_log("[cgit] error %d - %s\n",
367 err, strerror(err));
368 return err;
461} 369}
diff --git a/cgit.h b/cgit.h
index a3b6535..f04d227 100644
--- a/cgit.h
+++ b/cgit.h
@@ -89,96 +89,97 @@ struct taginfo {
89}; 89};
90 90
91struct refinfo { 91struct refinfo {
92 const char *refname; 92 const char *refname;
93 struct object *object; 93 struct object *object;
94 union { 94 union {
95 struct taginfo *tag; 95 struct taginfo *tag;
96 struct commitinfo *commit; 96 struct commitinfo *commit;
97 }; 97 };
98}; 98};
99 99
100struct reflist { 100struct reflist {
101 struct refinfo **refs; 101 struct refinfo **refs;
102 int alloc; 102 int alloc;
103 int count; 103 int count;
104}; 104};
105 105
106struct cgit_query { 106struct cgit_query {
107 int has_symref; 107 int has_symref;
108 int has_sha1; 108 int has_sha1;
109 char *raw; 109 char *raw;
110 char *repo; 110 char *repo;
111 char *page; 111 char *page;
112 char *search; 112 char *search;
113 char *grep; 113 char *grep;
114 char *head; 114 char *head;
115 char *sha1; 115 char *sha1;
116 char *sha2; 116 char *sha2;
117 char *path; 117 char *path;
118 char *name; 118 char *name;
119 int ofs; 119 int ofs;
120}; 120};
121 121
122struct cgit_config { 122struct cgit_config {
123 char *agefile; 123 char *agefile;
124 char *cache_root; 124 char *cache_root;
125 char *clone_prefix; 125 char *clone_prefix;
126 char *css; 126 char *css;
127 char *index_header; 127 char *index_header;
128 char *index_info; 128 char *index_info;
129 char *logo; 129 char *logo;
130 char *logo_link; 130 char *logo_link;
131 char *module_link; 131 char *module_link;
132 char *repo_group; 132 char *repo_group;
133 char *robots; 133 char *robots;
134 char *root_title; 134 char *root_title;
135 char *script_name; 135 char *script_name;
136 char *virtual_root; 136 char *virtual_root;
137 int cache_size;
137 int cache_dynamic_ttl; 138 int cache_dynamic_ttl;
138 int cache_max_create_time; 139 int cache_max_create_time;
139 int cache_repo_ttl; 140 int cache_repo_ttl;
140 int cache_root_ttl; 141 int cache_root_ttl;
141 int cache_static_ttl; 142 int cache_static_ttl;
142 int enable_index_links; 143 int enable_index_links;
143 int enable_log_filecount; 144 int enable_log_filecount;
144 int enable_log_linecount; 145 int enable_log_linecount;
145 int max_commit_count; 146 int max_commit_count;
146 int max_lock_attempts; 147 int max_lock_attempts;
147 int max_msg_len; 148 int max_msg_len;
148 int max_repodesc_len; 149 int max_repodesc_len;
149 int nocache; 150 int nocache;
150 int renamelimit; 151 int renamelimit;
151 int snapshots; 152 int snapshots;
152 int summary_branches; 153 int summary_branches;
153 int summary_log; 154 int summary_log;
154 int summary_tags; 155 int summary_tags;
155}; 156};
156 157
157struct cgit_page { 158struct cgit_page {
158 time_t modified; 159 time_t modified;
159 time_t expires; 160 time_t expires;
160 char *mimetype; 161 char *mimetype;
161 char *charset; 162 char *charset;
162 char *filename; 163 char *filename;
163 char *title; 164 char *title;
164}; 165};
165 166
166struct cgit_context { 167struct cgit_context {
167 struct cgit_query qry; 168 struct cgit_query qry;
168 struct cgit_config cfg; 169 struct cgit_config cfg;
169 struct cgit_repo *repo; 170 struct cgit_repo *repo;
170 struct cgit_page page; 171 struct cgit_page page;
171}; 172};
172 173
173struct cgit_snapshot_format { 174struct cgit_snapshot_format {
174 const char *suffix; 175 const char *suffix;
175 const char *mimetype; 176 const char *mimetype;
176 write_archive_fn_t write_func; 177 write_archive_fn_t write_func;
177 int bit; 178 int bit;
178}; 179};
179 180
180extern const char *cgit_version; 181extern const char *cgit_version;
181 182
182extern struct cgit_repolist cgit_repolist; 183extern struct cgit_repolist cgit_repolist;
183extern struct cgit_context ctx; 184extern struct cgit_context ctx;
184extern const struct cgit_snapshot_format cgit_snapshot_formats[]; 185extern const struct cgit_snapshot_format cgit_snapshot_formats[];
diff --git a/tests/setup.sh b/tests/setup.sh
index 66bf406..e37306e 100755
--- a/tests/setup.sh
+++ b/tests/setup.sh
@@ -1,95 +1,95 @@
1# This file should be sourced by all test-scripts 1# This file should be sourced by all test-scripts
2# 2#
3# Main functions: 3# Main functions:
4# prepare_tests(description) - setup for testing, i.e. create repos+config 4# prepare_tests(description) - setup for testing, i.e. create repos+config
5# run_test(description, script) - run one test, i.e. eval script 5# run_test(description, script) - run one test, i.e. eval script
6# 6#
7# Helper functions 7# Helper functions
8# cgit_query(querystring) - call cgit with the specified querystring 8# cgit_query(querystring) - call cgit with the specified querystring
9# cgit_url(url) - call cgit with the specified virtual url 9# cgit_url(url) - call cgit with the specified virtual url
10# 10#
11# Example script: 11# Example script:
12# 12#
13# . setup.sh 13# . setup.sh
14# prepare_tests "html validation" 14# prepare_tests "html validation"
15# run_test 'repo index' 'cgit_url "/" | tidy -e' 15# run_test 'repo index' 'cgit_url "/" | tidy -e'
16# run_test 'repo summary' 'cgit_url "/foo" | tidy -e' 16# run_test 'repo summary' 'cgit_url "/foo" | tidy -e'
17 17
18 18
19mkrepo() { 19mkrepo() {
20 name=$1 20 name=$1
21 count=$2 21 count=$2
22 dir=$PWD 22 dir=$PWD
23 test -d $name && return 23 test -d $name && return
24 printf "Creating testrepo %s\n" $name 24 printf "Creating testrepo %s\n" $name
25 mkdir -p $name 25 mkdir -p $name
26 cd $name 26 cd $name
27 git init 27 git init
28 for ((n=1; n<=count; n++)) 28 for ((n=1; n<=count; n++))
29 do 29 do
30 echo $n >file-$n 30 echo $n >file-$n
31 git add file-$n 31 git add file-$n
32 git commit -m "commit $n" 32 git commit -m "commit $n"
33 done 33 done
34 cd $dir 34 cd $dir
35} 35}
36 36
37setup_repos() 37setup_repos()
38{ 38{
39 rm -rf trash/cache 39 rm -rf trash/cache
40 mkdir -p trash/cache 40 mkdir -p trash/cache
41 mkrepo trash/repos/foo 5 >/dev/null 41 mkrepo trash/repos/foo 5 >/dev/null
42 mkrepo trash/repos/bar 50 >/dev/null 42 mkrepo trash/repos/bar 50 >/dev/null
43 cat >trash/cgitrc <<EOF 43 cat >trash/cgitrc <<EOF
44virtual-root=/ 44virtual-root=/
45cache-root=$PWD/trash/cache 45cache-root=$PWD/trash/cache
46 46
47nocache=0 47cache-size=1021
48snapshots=tar.gz tar.bz zip 48snapshots=tar.gz tar.bz zip
49enable-log-filecount=1 49enable-log-filecount=1
50enable-log-linecount=1 50enable-log-linecount=1
51summary-log=5 51summary-log=5
52summary-branches=5 52summary-branches=5
53summary-tags=5 53summary-tags=5
54 54
55repo.url=foo 55repo.url=foo
56repo.path=$PWD/trash/repos/foo/.git 56repo.path=$PWD/trash/repos/foo/.git
57# Do not specify a description for this repo, as it then will be assigned 57# Do not specify a description for this repo, as it then will be assigned
58# the constant value "[no description]" (which actually used to cause a 58# the constant value "[no description]" (which actually used to cause a
59# segfault). 59# segfault).
60 60
61repo.url=bar 61repo.url=bar
62repo.path=$PWD/trash/repos/bar/.git 62repo.path=$PWD/trash/repos/bar/.git
63repo.desc=the bar repo 63repo.desc=the bar repo
64EOF 64EOF
65} 65}
66 66
67prepare_tests() 67prepare_tests()
68{ 68{
69 setup_repos 69 setup_repos
70 rm -f test-output.log 2>/dev/null 70 rm -f test-output.log 2>/dev/null
71 test_count=0 71 test_count=0
72 test_failed=0 72 test_failed=0
73 echo "[$0]" "$@" >test-output.log 73 echo "[$0]" "$@" >test-output.log
74 echo "$@" "($0)" 74 echo "$@" "($0)"
75} 75}
76 76
77tests_done() 77tests_done()
78{ 78{
79 printf "\n" 79 printf "\n"
80 if test $test_failed -gt 0 80 if test $test_failed -gt 0
81 then 81 then
82 printf "test: *** %s failure(s), logfile=%s\n" \ 82 printf "test: *** %s failure(s), logfile=%s\n" \
83 $test_failed "$(pwd)/test-output.log" 83 $test_failed "$(pwd)/test-output.log"
84 false 84 false
85 fi 85 fi
86} 86}
87 87
88run_test() 88run_test()
89{ 89{
90 desc=$1 90 desc=$1
91 script=$2 91 script=$2
92 ((test_count++)) 92 ((test_count++))
93 printf "\ntest %d: name='%s'\n" $test_count "$desc" >>test-output.log 93 printf "\ntest %d: name='%s'\n" $test_count "$desc" >>test-output.log
94 printf "test %d: eval='%s'\n" $test_count "$2" >>test-output.log 94 printf "test %d: eval='%s'\n" $test_count "$2" >>test-output.log
95 eval "$2" >>test-output.log 2>>test-output.log 95 eval "$2" >>test-output.log 2>>test-output.log
diff --git a/tests/t0020-validate-cache.sh b/tests/t0020-validate-cache.sh
new file mode 100755
index 0000000..53ec2eb
--- a/dev/null
+++ b/tests/t0020-validate-cache.sh
@@ -0,0 +1,67 @@
1#!/bin/sh
2
3. ./setup.sh
4
5prepare_tests 'Validate cache'
6
7run_test 'verify cache-size=0' '
8
9 rm -f trash/cache/* &&
10 sed -i -e "s/cache-size=1021$/cache-size=0/" trash/cgitrc &&
11 cgit_url "" &&
12 cgit_url "foo" &&
13 cgit_url "foo/refs" &&
14 cgit_url "foo/tree" &&
15 cgit_url "foo/log" &&
16 cgit_url "foo/diff" &&
17 cgit_url "foo/patch" &&
18 cgit_url "bar" &&
19 cgit_url "bar/refs" &&
20 cgit_url "bar/tree" &&
21 cgit_url "bar/log" &&
22 cgit_url "bar/diff" &&
23 cgit_url "bar/patch" &&
24 test 0 -eq $(ls trash/cache | wc -l)
25'
26
27run_test 'verify cache-size=1' '
28
29 rm -f trash/cache/* &&
30 sed -i -e "s/cache-size=0$/cache-size=1/" trash/cgitrc &&
31 cgit_url "" &&
32 cgit_url "foo" &&
33 cgit_url "foo/refs" &&
34 cgit_url "foo/tree" &&
35 cgit_url "foo/log" &&
36 cgit_url "foo/diff" &&
37 cgit_url "foo/patch" &&
38 cgit_url "bar" &&
39 cgit_url "bar/refs" &&
40 cgit_url "bar/tree" &&
41 cgit_url "bar/log" &&
42 cgit_url "bar/diff" &&
43 cgit_url "bar/patch" &&
44 test 1 -eq $(ls trash/cache | wc -l)
45'
46
47run_test 'verify cache-size=1021' '
48
49 rm -f trash/cache/* &&
50 sed -i -e "s/cache-size=1$/cache-size=1021/" trash/cgitrc &&
51 cgit_url "" &&
52 cgit_url "foo" &&
53 cgit_url "foo/refs" &&
54 cgit_url "foo/tree" &&
55 cgit_url "foo/log" &&
56 cgit_url "foo/diff" &&
57 cgit_url "foo/patch" &&
58 cgit_url "bar" &&
59 cgit_url "bar/refs" &&
60 cgit_url "bar/tree" &&
61 cgit_url "bar/log" &&
62 cgit_url "bar/diff" &&
63 cgit_url "bar/patch" &&
64 test 13 -eq $(ls trash/cache | wc -l)
65'
66
67tests_done