]>
git.cameronkatri.com Git - cgit.git/blob - scan-tree.c
3 * Copyright (C) 2006-2014 cgit Development Team <cgit@lists.zx2c4.com>
5 * Licensed under GNU General Public License v2
6 * (see COPYING for full license text)
10 #include "scan-tree.h"
11 #include "configfile.h"
14 /* return 1 if path contains a objects/ directory and a HEAD file */
15 static int is_git_dir(const char *path
)
18 struct strbuf pathbuf
= STRBUF_INIT
;
21 strbuf_addf(&pathbuf
, "%s/objects", path
);
22 if (stat(pathbuf
.buf
, &st
)) {
24 fprintf(stderr
, "Error checking path %s: %s (%d)\n",
25 path
, strerror(errno
), errno
);
28 if (!S_ISDIR(st
.st_mode
))
31 strbuf_reset(&pathbuf
);
32 strbuf_addf(&pathbuf
, "%s/HEAD", path
);
33 if (stat(pathbuf
.buf
, &st
)) {
35 fprintf(stderr
, "Error checking path %s: %s (%d)\n",
36 path
, strerror(errno
), errno
);
39 if (!S_ISREG(st
.st_mode
))
44 strbuf_release(&pathbuf
);
48 struct cgit_repo
*repo
;
49 repo_config_fn config_fn
;
51 static void repo_config(const char *name
, const char *value
)
53 config_fn(repo
, name
, value
);
56 static int gitconfig_config(const char *key
, const char *value
, void *cb
)
58 if (!strcmp(key
, "gitweb.owner"))
59 config_fn(repo
, "owner", value
);
60 else if (!strcmp(key
, "gitweb.description"))
61 config_fn(repo
, "desc", value
);
62 else if (!strcmp(key
, "gitweb.category"))
63 config_fn(repo
, "section", value
);
64 else if (starts_with(key
, "cgit."))
65 config_fn(repo
, key
+ 5, value
);
70 static char *xstrrchr(char *s
, char *from
, int c
)
72 while (from
>= s
&& *from
!= c
)
74 return from
< s
? NULL
: from
;
77 static void add_repo(const char *base
, struct strbuf
*path
, repo_config_fn fn
)
82 struct strbuf rel
= STRBUF_INIT
;
87 if (stat(path
->buf
, &st
)) {
88 fprintf(stderr
, "Error accessing %s: %s (%d)\n",
89 path
->buf
, strerror(errno
), errno
);
93 strbuf_addch(path
, '/');
96 if (ctx
.cfg
.strict_export
) {
97 strbuf_addstr(path
, ctx
.cfg
.strict_export
);
98 if(stat(path
->buf
, &st
))
100 strbuf_setlen(path
, pathlen
);
103 strbuf_addstr(path
, "noweb");
104 if (!stat(path
->buf
, &st
))
106 strbuf_setlen(path
, pathlen
);
108 if (!starts_with(path
->buf
, base
))
109 strbuf_addbuf(&rel
, path
);
111 strbuf_addstr(&rel
, path
->buf
+ strlen(base
) + 1);
113 if (!strcmp(rel
.buf
+ rel
.len
- 5, "/.git"))
114 strbuf_setlen(&rel
, rel
.len
- 5);
115 else if (rel
.len
&& rel
.buf
[rel
.len
- 1] == '/')
116 strbuf_setlen(&rel
, rel
.len
- 1);
118 fprintf(stderr
, "add_repo(): %s\n", rel
.buf
);
119 repo
= cgit_add_repo(rel
.buf
);
121 if (ctx
.cfg
.enable_git_config
) {
122 strbuf_addstr(path
, "config");
123 git_config_from_file(gitconfig_config
, path
->buf
, NULL
);
124 strbuf_setlen(path
, pathlen
);
127 if (ctx
.cfg
.remove_suffix
)
128 if ((p
= strrchr(repo
->url
, '.')) && !strcmp(p
, ".git"))
130 repo
->path
= xstrdup(path
->buf
);
131 while (!repo
->owner
) {
132 if ((pwd
= getpwuid(st
.st_uid
)) == NULL
) {
133 fprintf(stderr
, "Error reading owner-info for %s: %s (%d)\n",
134 path
->buf
, strerror(errno
), errno
);
138 if ((p
= strchr(pwd
->pw_gecos
, ',')))
140 repo
->owner
= xstrdup(pwd
->pw_gecos
? pwd
->pw_gecos
: pwd
->pw_name
);
143 if (repo
->desc
== cgit_default_repo_desc
|| !repo
->desc
) {
144 strbuf_addstr(path
, "description");
145 if (!stat(path
->buf
, &st
))
146 readfile(path
->buf
, &repo
->desc
, &size
);
147 strbuf_setlen(path
, pathlen
);
150 if (ctx
.cfg
.section_from_path
) {
151 n
= ctx
.cfg
.section_from_path
;
154 while (slash
&& n
&& (slash
= strchr(slash
+ 1, '/')))
157 slash
= rel
.buf
+ rel
.len
;
158 while (slash
&& n
&& (slash
= xstrrchr(rel
.buf
, slash
- 1, '/')))
163 repo
->section
= xstrdup(rel
.buf
);
165 fprintf(stderr
, "repo->name %s, repo->section %s\n", repo
->name
, repo
->section
);
166 if (starts_with(repo
->name
, repo
->section
)) {
167 repo
->name
+= strlen(repo
->section
);
168 if (*repo
->name
== '/')
174 strbuf_addstr(path
, "cgitrc");
175 if (!stat(path
->buf
, &st
))
176 parse_configfile(xstrdup(path
->buf
), &repo_config
);
178 strbuf_release(&rel
);
181 static void scan_path(const char *base
, const char *path
, repo_config_fn fn
)
183 DIR *dir
= opendir(path
);
185 struct strbuf pathbuf
= STRBUF_INIT
;
186 size_t pathlen
= strlen(path
);
189 fprintf(stderr
, "scan_path(): %s\n", path
);
191 fprintf(stderr
, "Error opening directory %s: %s (%d)\n",
192 path
, strerror(errno
), errno
);
196 strbuf_add(&pathbuf
, path
, strlen(path
));
197 if (is_git_dir(pathbuf
.buf
)) {
198 fprintf(stderr
, "scan_path() is_git_dir: %s\n", path
);
199 add_repo(base
, &pathbuf
, fn
);
202 strbuf_addstr(&pathbuf
, "/.git");
203 if (is_git_dir(pathbuf
.buf
)) {
204 add_repo(base
, &pathbuf
, fn
);
208 * Add one because we don't want to lose the trailing '/' when we
209 * reset the length of pathbuf in the loop below.
212 while ((ent
= readdir(dir
)) != NULL
) {
213 if (ent
->d_name
[0] == '.') {
214 if (ent
->d_name
[1] == '\0')
216 if (ent
->d_name
[1] == '.' && ent
->d_name
[2] == '\0')
218 if (!ctx
.cfg
.scan_hidden_path
)
221 strbuf_setlen(&pathbuf
, pathlen
);
222 strbuf_addstr(&pathbuf
, ent
->d_name
);
223 if (stat(pathbuf
.buf
, &st
)) {
224 fprintf(stderr
, "Error checking path %s: %s (%d)\n",
225 pathbuf
.buf
, strerror(errno
), errno
);
228 if (S_ISDIR(st
.st_mode
))
229 scan_path(base
, pathbuf
.buf
, fn
);
232 strbuf_release(&pathbuf
);
236 void scan_projects(const char *path
, const char *projectsfile
, repo_config_fn fn
)
238 struct strbuf line
= STRBUF_INIT
;
242 projects
= fopen(projectsfile
, "r");
244 fprintf(stderr
, "Error opening projectsfile %s: %s (%d)\n",
245 projectsfile
, strerror(errno
), errno
);
248 while (strbuf_getline(&line
, projects
, '\n') != EOF
) {
251 strbuf_insert(&line
, 0, "/", 1);
252 strbuf_insert(&line
, 0, path
, strlen(path
));
253 scan_path(path
, line
.buf
, fn
);
255 if ((err
= ferror(projects
))) {
256 fprintf(stderr
, "Error reading from projectsfile %s: %s (%d)\n",
257 projectsfile
, strerror(err
), err
);
260 strbuf_release(&line
);
263 void scan_tree(const char *path
, repo_config_fn fn
)
265 scan_path(path
, path
, fn
);