1 /* cache.c: cache management
3 * Copyright (C) 2006 Lars Hjemli
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
11 const int NOLOCK
= -1;
13 char *cache_safe_filename(const char *unsafe
)
15 static char buf
[PATH_MAX
];
19 while(unsafe
&& (c
= *unsafe
++) != 0) {
20 if (c
== '/' || c
== ' ' || c
== '&' || c
== '|' ||
21 c
== '>' || c
== '<' || c
== '.')
29 int cache_exist(struct cacheitem
*item
)
31 if (stat(item
->name
, &item
->st
)) {
32 item
->st
.st_mtime
= 0;
38 int cache_create_dirs()
42 path
= fmt("%s", cgit_cache_root
);
43 if (mkdir(path
, S_IRWXU
) && errno
!=EEXIST
)
49 path
= fmt("%s/%s", cgit_cache_root
,
50 cache_safe_filename(cgit_repo
->url
));
52 if (mkdir(path
, S_IRWXU
) && errno
!=EEXIST
)
55 if (cgit_query_page
) {
56 path
= fmt("%s/%s/%s", cgit_cache_root
,
57 cache_safe_filename(cgit_repo
->url
),
59 if (mkdir(path
, S_IRWXU
) && errno
!=EEXIST
)
65 int cache_refill_overdue(const char *lockfile
)
69 if (stat(lockfile
, &st
))
72 return (time(NULL
) - st
.st_mtime
> cgit_cache_max_create_time
);
75 int cache_lock(struct cacheitem
*item
)
78 char *lockfile
= xstrdup(fmt("%s.lock", item
->name
));
81 if (++i
> cgit_max_lock_attempts
)
82 die("cache_lock: unable to lock %s: %s",
83 item
->name
, strerror(errno
));
85 item
->fd
= open(lockfile
, O_WRONLY
|O_CREAT
|O_EXCL
, S_IRUSR
|S_IWUSR
);
87 if (item
->fd
== NOLOCK
&& errno
== ENOENT
&& cache_create_dirs())
90 if (item
->fd
== NOLOCK
&& errno
== EEXIST
&&
91 cache_refill_overdue(lockfile
) && !unlink(lockfile
))
95 return (item
->fd
> 0);
98 int cache_unlock(struct cacheitem
*item
)
101 return (rename(fmt("%s.lock", item
->name
), item
->name
) == 0);
104 int cache_cancel_lock(struct cacheitem
*item
)
106 return (unlink(fmt("%s.lock", item
->name
)) == 0);
109 int cache_expired(struct cacheitem
*item
)
113 return item
->st
.st_mtime
+ item
->ttl
* 60 < time(NULL
);