summaryrefslogtreecommitdiffabout
Unidiff
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cache.c16
-rw-r--r--cgit.c4
2 files changed, 15 insertions, 5 deletions
diff --git a/cache.c b/cache.c
index a996109..aa97ae1 100644
--- a/cache.c
+++ b/cache.c
@@ -159,230 +159,240 @@ static int unlock_slot(struct cache_slot *slot, int replace_old_slot)
159 err = rename(slot->lock_name, slot->cache_name); 159 err = rename(slot->lock_name, slot->cache_name);
160 else 160 else
161 err = unlink(slot->lock_name); 161 err = unlink(slot->lock_name);
162 162
163 if (err) 163 if (err)
164 return errno; 164 return errno;
165 165
166 return 0; 166 return 0;
167} 167}
168 168
169/* Generate the content for the current cache slot by redirecting 169/* Generate the content for the current cache slot by redirecting
170 * stdout to the lock-fd and invoking the callback function 170 * stdout to the lock-fd and invoking the callback function
171 */ 171 */
172static int fill_slot(struct cache_slot *slot) 172static int fill_slot(struct cache_slot *slot)
173{ 173{
174 int tmp; 174 int tmp;
175 175
176 /* Preserve stdout */ 176 /* Preserve stdout */
177 tmp = dup(STDOUT_FILENO); 177 tmp = dup(STDOUT_FILENO);
178 if (tmp == -1) 178 if (tmp == -1)
179 return errno; 179 return errno;
180 180
181 /* Redirect stdout to lockfile */ 181 /* Redirect stdout to lockfile */
182 if (dup2(slot->lock_fd, STDOUT_FILENO) == -1) 182 if (dup2(slot->lock_fd, STDOUT_FILENO) == -1)
183 return errno; 183 return errno;
184 184
185 /* Generate cache content */ 185 /* Generate cache content */
186 slot->fn(slot->cbdata); 186 slot->fn(slot->cbdata);
187 187
188 /* Restore stdout */ 188 /* Restore stdout */
189 if (dup2(tmp, STDOUT_FILENO) == -1) 189 if (dup2(tmp, STDOUT_FILENO) == -1)
190 return errno; 190 return errno;
191 191
192 /* Close the temporary filedescriptor */ 192 /* Close the temporary filedescriptor */
193 if (close(tmp)) 193 if (close(tmp))
194 return errno; 194 return errno;
195 195
196 return 0; 196 return 0;
197} 197}
198 198
199/* Crude implementation of 32-bit FNV-1 hash algorithm, 199/* Crude implementation of 32-bit FNV-1 hash algorithm,
200 * see http://www.isthe.com/chongo/tech/comp/fnv/ for details 200 * see http://www.isthe.com/chongo/tech/comp/fnv/ for details
201 * about the magic numbers. 201 * about the magic numbers.
202 */ 202 */
203#define FNV_OFFSET 0x811c9dc5 203#define FNV_OFFSET 0x811c9dc5
204#define FNV_PRIME 0x01000193 204#define FNV_PRIME 0x01000193
205 205
206unsigned long hash_str(const char *str) 206unsigned long hash_str(const char *str)
207{ 207{
208 unsigned long h = FNV_OFFSET; 208 unsigned long h = FNV_OFFSET;
209 unsigned char *s = (unsigned char *)str; 209 unsigned char *s = (unsigned char *)str;
210 210
211 if (!s) 211 if (!s)
212 return h; 212 return h;
213 213
214 while(*s) { 214 while(*s) {
215 h *= FNV_PRIME; 215 h *= FNV_PRIME;
216 h ^= *s++; 216 h ^= *s++;
217 } 217 }
218 return h; 218 return h;
219} 219}
220 220
221static int process_slot(struct cache_slot *slot) 221static int process_slot(struct cache_slot *slot)
222{ 222{
223 int err; 223 int err;
224 224
225 err = open_slot(slot); 225 err = open_slot(slot);
226 if (!err && slot->match) { 226 if (!err && slot->match) {
227 if (is_expired(slot)) { 227 if (is_expired(slot)) {
228 if (!lock_slot(slot)) { 228 if (!lock_slot(slot)) {
229 /* If the cachefile has been replaced between 229 /* If the cachefile has been replaced between
230 * `open_slot` and `lock_slot`, we'll just 230 * `open_slot` and `lock_slot`, we'll just
231 * serve the stale content from the original 231 * serve the stale content from the original
232 * cachefile. This way we avoid pruning the 232 * cachefile. This way we avoid pruning the
233 * newly generated slot. The same code-path 233 * newly generated slot. The same code-path
234 * is chosen if fill_slot() fails for some 234 * is chosen if fill_slot() fails for some
235 * reason. 235 * reason.
236 * 236 *
237 * TODO? check if the new slot contains the 237 * TODO? check if the new slot contains the
238 * same key as the old one, since we would 238 * same key as the old one, since we would
239 * prefer to serve the newest content. 239 * prefer to serve the newest content.
240 * This will require us to open yet another 240 * This will require us to open yet another
241 * file-descriptor and read and compare the 241 * file-descriptor and read and compare the
242 * key from the new file, so for now we're 242 * key from the new file, so for now we're
243 * lazy and just ignore the new file. 243 * lazy and just ignore the new file.
244 */ 244 */
245 if (is_modified(slot) || fill_slot(slot)) { 245 if (is_modified(slot) || fill_slot(slot)) {
246 unlock_slot(slot, 0); 246 unlock_slot(slot, 0);
247 close_lock(slot); 247 close_lock(slot);
248 } else { 248 } else {
249 close_slot(slot); 249 close_slot(slot);
250 unlock_slot(slot, 1); 250 unlock_slot(slot, 1);
251 slot->cache_fd = slot->lock_fd; 251 slot->cache_fd = slot->lock_fd;
252 } 252 }
253 } 253 }
254 } 254 }
255 print_slot(slot); 255 if ((err = print_slot(slot)) != 0) {
256 cache_log("[cgit] error printing cache %s: %s (%d)\n",
257 slot->cache_name,
258 strerror(err),
259 err);
260 }
256 close_slot(slot); 261 close_slot(slot);
257 return 0; 262 return err;
258 } 263 }
259 264
260 /* If the cache slot does not exist (or its key doesn't match the 265 /* If the cache slot does not exist (or its key doesn't match the
261 * current key), lets try to create a new cache slot for this 266 * current key), lets try to create a new cache slot for this
262 * request. If this fails (for whatever reason), lets just generate 267 * request. If this fails (for whatever reason), lets just generate
263 * the content without caching it and fool the caller to belive 268 * the content without caching it and fool the caller to belive
264 * everything worked out (but print a warning on stdout). 269 * everything worked out (but print a warning on stdout).
265 */ 270 */
266 271
267 close_slot(slot); 272 close_slot(slot);
268 if ((err = lock_slot(slot)) != 0) { 273 if ((err = lock_slot(slot)) != 0) {
269 cache_log("[cgit] Unable to lock slot %s: %s (%d)\n", 274 cache_log("[cgit] Unable to lock slot %s: %s (%d)\n",
270 slot->lock_name, strerror(err), err); 275 slot->lock_name, strerror(err), err);
271 slot->fn(slot->cbdata); 276 slot->fn(slot->cbdata);
272 return 0; 277 return 0;
273 } 278 }
274 279
275 if ((err = fill_slot(slot)) != 0) { 280 if ((err = fill_slot(slot)) != 0) {
276 cache_log("[cgit] Unable to fill slot %s: %s (%d)\n", 281 cache_log("[cgit] Unable to fill slot %s: %s (%d)\n",
277 slot->lock_name, strerror(err), err); 282 slot->lock_name, strerror(err), err);
278 unlock_slot(slot, 0); 283 unlock_slot(slot, 0);
279 close_lock(slot); 284 close_lock(slot);
280 slot->fn(slot->cbdata); 285 slot->fn(slot->cbdata);
281 return 0; 286 return 0;
282 } 287 }
283 // We've got a valid cache slot in the lock file, which 288 // We've got a valid cache slot in the lock file, which
284 // is about to replace the old cache slot. But if we 289 // is about to replace the old cache slot. But if we
285 // release the lockfile and then try to open the new cache 290 // release the lockfile and then try to open the new cache
286 // slot, we might get a race condition with a concurrent 291 // slot, we might get a race condition with a concurrent
287 // writer for the same cache slot (with a different key). 292 // writer for the same cache slot (with a different key).
288 // Lets avoid such a race by just printing the content of 293 // Lets avoid such a race by just printing the content of
289 // the lock file. 294 // the lock file.
290 slot->cache_fd = slot->lock_fd; 295 slot->cache_fd = slot->lock_fd;
291 unlock_slot(slot, 1); 296 unlock_slot(slot, 1);
292 err = print_slot(slot); 297 if ((err = print_slot(slot)) != 0) {
298 cache_log("[cgit] error printing cache %s: %s (%d)\n",
299 slot->cache_name,
300 strerror(err),
301 err);
302 }
293 close_slot(slot); 303 close_slot(slot);
294 return err; 304 return err;
295} 305}
296 306
297/* Print cached content to stdout, generate the content if necessary. */ 307/* Print cached content to stdout, generate the content if necessary. */
298int cache_process(int size, const char *path, const char *key, int ttl, 308int cache_process(int size, const char *path, const char *key, int ttl,
299 cache_fill_fn fn, void *cbdata) 309 cache_fill_fn fn, void *cbdata)
300{ 310{
301 unsigned long hash; 311 unsigned long hash;
302 int len, i; 312 int len, i;
303 char filename[1024]; 313 char filename[1024];
304 char lockname[1024 + 5]; /* 5 = ".lock" */ 314 char lockname[1024 + 5]; /* 5 = ".lock" */
305 struct cache_slot slot; 315 struct cache_slot slot;
306 316
307 /* If the cache is disabled, just generate the content */ 317 /* If the cache is disabled, just generate the content */
308 if (size <= 0) { 318 if (size <= 0) {
309 fn(cbdata); 319 fn(cbdata);
310 return 0; 320 return 0;
311 } 321 }
312 322
313 /* Verify input, calculate filenames */ 323 /* Verify input, calculate filenames */
314 if (!path) { 324 if (!path) {
315 cache_log("[cgit] Cache path not specified, caching is disabled\n"); 325 cache_log("[cgit] Cache path not specified, caching is disabled\n");
316 fn(cbdata); 326 fn(cbdata);
317 return 0; 327 return 0;
318 } 328 }
319 len = strlen(path); 329 len = strlen(path);
320 if (len > sizeof(filename) - 10) { /* 10 = "/01234567\0" */ 330 if (len > sizeof(filename) - 10) { /* 10 = "/01234567\0" */
321 cache_log("[cgit] Cache path too long, caching is disabled: %s\n", 331 cache_log("[cgit] Cache path too long, caching is disabled: %s\n",
322 path); 332 path);
323 fn(cbdata); 333 fn(cbdata);
324 return 0; 334 return 0;
325 } 335 }
326 if (!key) 336 if (!key)
327 key = ""; 337 key = "";
328 hash = hash_str(key) % size; 338 hash = hash_str(key) % size;
329 strcpy(filename, path); 339 strcpy(filename, path);
330 if (filename[len - 1] != '/') 340 if (filename[len - 1] != '/')
331 filename[len++] = '/'; 341 filename[len++] = '/';
332 for(i = 0; i < 8; i++) { 342 for(i = 0; i < 8; i++) {
333 sprintf(filename + len++, "%x", 343 sprintf(filename + len++, "%x",
334 (unsigned char)(hash & 0xf)); 344 (unsigned char)(hash & 0xf));
335 hash >>= 4; 345 hash >>= 4;
336 } 346 }
337 filename[len] = '\0'; 347 filename[len] = '\0';
338 strcpy(lockname, filename); 348 strcpy(lockname, filename);
339 strcpy(lockname + len, ".lock"); 349 strcpy(lockname + len, ".lock");
340 slot.fn = fn; 350 slot.fn = fn;
341 slot.cbdata = cbdata; 351 slot.cbdata = cbdata;
342 slot.ttl = ttl; 352 slot.ttl = ttl;
343 slot.cache_name = filename; 353 slot.cache_name = filename;
344 slot.lock_name = lockname; 354 slot.lock_name = lockname;
345 slot.key = key; 355 slot.key = key;
346 slot.keylen = strlen(key); 356 slot.keylen = strlen(key);
347 return process_slot(&slot); 357 return process_slot(&slot);
348} 358}
349 359
350/* Return a strftime formatted date/time 360/* Return a strftime formatted date/time
351 * NB: the result from this function is to shared memory 361 * NB: the result from this function is to shared memory
352 */ 362 */
353char *sprintftime(const char *format, time_t time) 363char *sprintftime(const char *format, time_t time)
354{ 364{
355 static char buf[64]; 365 static char buf[64];
356 struct tm *tm; 366 struct tm *tm;
357 367
358 if (!time) 368 if (!time)
359 return NULL; 369 return NULL;
360 tm = gmtime(&time); 370 tm = gmtime(&time);
361 strftime(buf, sizeof(buf)-1, format, tm); 371 strftime(buf, sizeof(buf)-1, format, tm);
362 return buf; 372 return buf;
363} 373}
364 374
365int cache_ls(const char *path) 375int cache_ls(const char *path)
366{ 376{
367 DIR *dir; 377 DIR *dir;
368 struct dirent *ent; 378 struct dirent *ent;
369 int err = 0; 379 int err = 0;
370 struct cache_slot slot; 380 struct cache_slot slot;
371 char fullname[1024]; 381 char fullname[1024];
372 char *name; 382 char *name;
373 383
374 if (!path) { 384 if (!path) {
375 cache_log("[cgit] cache path not specified\n"); 385 cache_log("[cgit] cache path not specified\n");
376 return -1; 386 return -1;
377 } 387 }
378 if (strlen(path) > 1024 - 10) { 388 if (strlen(path) > 1024 - 10) {
379 cache_log("[cgit] cache path too long: %s\n", 389 cache_log("[cgit] cache path too long: %s\n",
380 path); 390 path);
381 return -1; 391 return -1;
382 } 392 }
383 dir = opendir(path); 393 dir = opendir(path);
384 if (!dir) { 394 if (!dir) {
385 err = errno; 395 err = errno;
386 cache_log("[cgit] unable to open path %s: %s (%d)\n", 396 cache_log("[cgit] unable to open path %s: %s (%d)\n",
387 path, strerror(err), err); 397 path, strerror(err), err);
388 return err; 398 return err;
diff --git a/cgit.c b/cgit.c
index 2036ceb..ac882c3 100644
--- a/cgit.c
+++ b/cgit.c
@@ -287,100 +287,100 @@ static void process_request(void *cbdata)
287 return; 287 return;
288 } 288 }
289 289
290 if (ctx->repo && prepare_repo_cmd(ctx)) 290 if (ctx->repo && prepare_repo_cmd(ctx))
291 return; 291 return;
292 292
293 if (cmd->want_layout) { 293 if (cmd->want_layout) {
294 cgit_print_http_headers(ctx); 294 cgit_print_http_headers(ctx);
295 cgit_print_docstart(ctx); 295 cgit_print_docstart(ctx);
296 cgit_print_pageheader(ctx); 296 cgit_print_pageheader(ctx);
297 } 297 }
298 298
299 cmd->fn(ctx); 299 cmd->fn(ctx);
300 300
301 if (cmd->want_layout) 301 if (cmd->want_layout)
302 cgit_print_docend(); 302 cgit_print_docend();
303} 303}
304 304
305static void cgit_parse_args(int argc, const char **argv) 305static void cgit_parse_args(int argc, const char **argv)
306{ 306{
307 int i; 307 int i;
308 308
309 for (i = 1; i < argc; i++) { 309 for (i = 1; i < argc; i++) {
310 if (!strncmp(argv[i], "--cache=", 8)) { 310 if (!strncmp(argv[i], "--cache=", 8)) {
311 ctx.cfg.cache_root = xstrdup(argv[i]+8); 311 ctx.cfg.cache_root = xstrdup(argv[i]+8);
312 } 312 }
313 if (!strcmp(argv[i], "--nocache")) { 313 if (!strcmp(argv[i], "--nocache")) {
314 ctx.cfg.nocache = 1; 314 ctx.cfg.nocache = 1;
315 } 315 }
316 if (!strncmp(argv[i], "--query=", 8)) { 316 if (!strncmp(argv[i], "--query=", 8)) {
317 ctx.qry.raw = xstrdup(argv[i]+8); 317 ctx.qry.raw = xstrdup(argv[i]+8);
318 } 318 }
319 if (!strncmp(argv[i], "--repo=", 7)) { 319 if (!strncmp(argv[i], "--repo=", 7)) {
320 ctx.qry.repo = xstrdup(argv[i]+7); 320 ctx.qry.repo = xstrdup(argv[i]+7);
321 } 321 }
322 if (!strncmp(argv[i], "--page=", 7)) { 322 if (!strncmp(argv[i], "--page=", 7)) {
323 ctx.qry.page = xstrdup(argv[i]+7); 323 ctx.qry.page = xstrdup(argv[i]+7);
324 } 324 }
325 if (!strncmp(argv[i], "--head=", 7)) { 325 if (!strncmp(argv[i], "--head=", 7)) {
326 ctx.qry.head = xstrdup(argv[i]+7); 326 ctx.qry.head = xstrdup(argv[i]+7);
327 ctx.qry.has_symref = 1; 327 ctx.qry.has_symref = 1;
328 } 328 }
329 if (!strncmp(argv[i], "--sha1=", 7)) { 329 if (!strncmp(argv[i], "--sha1=", 7)) {
330 ctx.qry.sha1 = xstrdup(argv[i]+7); 330 ctx.qry.sha1 = xstrdup(argv[i]+7);
331 ctx.qry.has_sha1 = 1; 331 ctx.qry.has_sha1 = 1;
332 } 332 }
333 if (!strncmp(argv[i], "--ofs=", 6)) { 333 if (!strncmp(argv[i], "--ofs=", 6)) {
334 ctx.qry.ofs = atoi(argv[i]+6); 334 ctx.qry.ofs = atoi(argv[i]+6);
335 } 335 }
336 } 336 }
337} 337}
338 338
339static int calc_ttl() 339static int calc_ttl()
340{ 340{
341 if (!ctx.repo) 341 if (!ctx.repo)
342 return ctx.cfg.cache_root_ttl; 342 return ctx.cfg.cache_root_ttl;
343 343
344 if (!ctx.qry.page) 344 if (!ctx.qry.page)
345 return ctx.cfg.cache_repo_ttl; 345 return ctx.cfg.cache_repo_ttl;
346 346
347 if (ctx.qry.has_symref) 347 if (ctx.qry.has_symref)
348 return ctx.cfg.cache_dynamic_ttl; 348 return ctx.cfg.cache_dynamic_ttl;
349 349
350 if (ctx.qry.has_sha1) 350 if (ctx.qry.has_sha1)
351 return ctx.cfg.cache_static_ttl; 351 return ctx.cfg.cache_static_ttl;
352 352
353 return ctx.cfg.cache_repo_ttl; 353 return ctx.cfg.cache_repo_ttl;
354} 354}
355 355
356int main(int argc, const char **argv) 356int main(int argc, const char **argv)
357{ 357{
358 const char *cgit_config_env = getenv("CGIT_CONFIG"); 358 const char *cgit_config_env = getenv("CGIT_CONFIG");
359 int err, ttl; 359 int err, ttl;
360 360
361 prepare_context(&ctx); 361 prepare_context(&ctx);
362 cgit_repolist.length = 0; 362 cgit_repolist.length = 0;
363 cgit_repolist.count = 0; 363 cgit_repolist.count = 0;
364 cgit_repolist.repos = NULL; 364 cgit_repolist.repos = NULL;
365 365
366 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG, 366 parse_configfile(cgit_config_env ? cgit_config_env : CGIT_CONFIG,
367 config_cb); 367 config_cb);
368 ctx.repo = NULL; 368 ctx.repo = NULL;
369 if (getenv("SCRIPT_NAME")) 369 if (getenv("SCRIPT_NAME"))
370 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME")); 370 ctx.cfg.script_name = xstrdup(getenv("SCRIPT_NAME"));
371 if (getenv("QUERY_STRING")) 371 if (getenv("QUERY_STRING"))
372 ctx.qry.raw = xstrdup(getenv("QUERY_STRING")); 372 ctx.qry.raw = xstrdup(getenv("QUERY_STRING"));
373 cgit_parse_args(argc, argv); 373 cgit_parse_args(argc, argv);
374 http_parse_querystring(ctx.qry.raw, querystring_cb); 374 http_parse_querystring(ctx.qry.raw, querystring_cb);
375 375
376 ttl = calc_ttl(); 376 ttl = calc_ttl();
377 ctx.page.expires += ttl*60; 377 ctx.page.expires += ttl*60;
378 if (ctx.cfg.nocache) 378 if (ctx.cfg.nocache)
379 ctx.cfg.cache_size = 0; 379 ctx.cfg.cache_size = 0;
380 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root, 380 err = cache_process(ctx.cfg.cache_size, ctx.cfg.cache_root,
381 ctx.qry.raw, ttl, process_request, &ctx); 381 ctx.qry.raw, ttl, process_request, &ctx);
382 if (err) 382 if (err)
383 cache_log("[cgit] error %d - %s\n", 383 cgit_print_error(fmt("Error processing page: %s (%d)",
384 err, strerror(err)); 384 strerror(err), err));
385 return err; 385 return err;
386} 386}