summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2009-03-15 07:57:33 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2009-03-15 07:57:33 (UTC)
commit6fddad7251021b307c8a3f70fdd2aa04c3f74eaa (patch) (unidiff)
tree942bd640a13ac3290bc2b8b94279cf68cf404abd
parent112973615a78ce61fd6e767128df03b075be72ca (diff)
downloadcgit-6fddad7251021b307c8a3f70fdd2aa04c3f74eaa.zip
cgit-6fddad7251021b307c8a3f70fdd2aa04c3f74eaa.tar.gz
cgit-6fddad7251021b307c8a3f70fdd2aa04c3f74eaa.tar.bz2
ui-snapshot: avoid segfault when no filename is specified
Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--ui-snapshot.c23
1 files changed, 17 insertions, 6 deletions
diff --git a/ui-snapshot.c b/ui-snapshot.c
index f25613e..5372f5d 100644
--- a/ui-snapshot.c
+++ b/ui-snapshot.c
@@ -95,100 +95,111 @@ static int make_snapshot(const struct cgit_snapshot_format *format,
95 commit = lookup_commit_reference(sha1); 95 commit = lookup_commit_reference(sha1);
96 if(!commit) { 96 if(!commit) {
97 cgit_print_error(fmt("Not a commit reference: %s", hex)); 97 cgit_print_error(fmt("Not a commit reference: %s", hex));
98 return 1; 98 return 1;
99 } 99 }
100 memset(&args, 0, sizeof(args)); 100 memset(&args, 0, sizeof(args));
101 if (prefix) { 101 if (prefix) {
102 args.base = fmt("%s/", prefix); 102 args.base = fmt("%s/", prefix);
103 args.baselen = strlen(prefix) + 1; 103 args.baselen = strlen(prefix) + 1;
104 } else { 104 } else {
105 args.base = ""; 105 args.base = "";
106 args.baselen = 0; 106 args.baselen = 0;
107 } 107 }
108 args.tree = commit->tree; 108 args.tree = commit->tree;
109 args.time = commit->date; 109 args.time = commit->date;
110 ctx.page.mimetype = xstrdup(format->mimetype); 110 ctx.page.mimetype = xstrdup(format->mimetype);
111 ctx.page.filename = xstrdup(filename); 111 ctx.page.filename = xstrdup(filename);
112 cgit_print_http_headers(&ctx); 112 cgit_print_http_headers(&ctx);
113 format->write_func(&args); 113 format->write_func(&args);
114 return 0; 114 return 0;
115} 115}
116 116
117/* Try to guess the requested revision from the requested snapshot name. 117/* Try to guess the requested revision from the requested snapshot name.
118 * First the format extension is stripped, e.g. "cgit-0.7.2.tar.gz" become 118 * First the format extension is stripped, e.g. "cgit-0.7.2.tar.gz" become
119 * "cgit-0.7.2". If this is a valid commit object name we've got a winner. 119 * "cgit-0.7.2". If this is a valid commit object name we've got a winner.
120 * Otherwise, if the snapshot name has a prefix matching the result from 120 * Otherwise, if the snapshot name has a prefix matching the result from
121 * repo_basename(), we strip the basename and any following '-' and '_' 121 * repo_basename(), we strip the basename and any following '-' and '_'
122 * characters ("cgit-0.7.2" -> "0.7.2") and check the resulting name once 122 * characters ("cgit-0.7.2" -> "0.7.2") and check the resulting name once
123 * more. If this still isn't a valid commit object name, we check if pre- 123 * more. If this still isn't a valid commit object name, we check if pre-
124 * pending a 'v' to the remaining snapshot name ("0.7.2" -> "v0.7.2") gives 124 * pending a 'v' to the remaining snapshot name ("0.7.2" -> "v0.7.2") gives
125 * us something valid. 125 * us something valid.
126 */ 126 */
127static const char *get_ref_from_filename(const char *url, const char *filename, 127static const char *get_ref_from_filename(const char *url, const char *filename,
128 const struct cgit_snapshot_format *format) 128 const struct cgit_snapshot_format *format)
129{ 129{
130 const char *reponame; 130 const char *reponame;
131 unsigned char sha1[20]; 131 unsigned char sha1[20];
132 char *snapshot; 132 char *snapshot;
133 133
134 snapshot = xstrdup(filename); 134 snapshot = xstrdup(filename);
135 snapshot[strlen(snapshot) - strlen(format->suffix)] = '\0'; 135 snapshot[strlen(snapshot) - strlen(format->suffix)] = '\0';
136 fprintf(stderr, "snapshot=%s\n", snapshot); 136 fprintf(stderr, "snapshot=%s\n", snapshot);
137 137
138 if (get_sha1(snapshot, sha1) == 0) 138 if (get_sha1(snapshot, sha1) == 0)
139 return snapshot; 139 return snapshot;
140 140
141 reponame = cgit_repobasename(url); 141 reponame = cgit_repobasename(url);
142 fprintf(stderr, "reponame=%s\n", reponame); 142 fprintf(stderr, "reponame=%s\n", reponame);
143 if (prefixcmp(snapshot, reponame) == 0) { 143 if (prefixcmp(snapshot, reponame) == 0) {
144 snapshot += strlen(reponame); 144 snapshot += strlen(reponame);
145 while (snapshot && (*snapshot == '-' || *snapshot == '_')) 145 while (snapshot && (*snapshot == '-' || *snapshot == '_'))
146 snapshot++; 146 snapshot++;
147 } 147 }
148 148
149 if (get_sha1(snapshot, sha1) == 0) 149 if (get_sha1(snapshot, sha1) == 0)
150 return snapshot; 150 return snapshot;
151 151
152 snapshot = fmt("v%s", snapshot); 152 snapshot = fmt("v%s", snapshot);
153 if (get_sha1(snapshot, sha1) == 0) 153 if (get_sha1(snapshot, sha1) == 0)
154 return snapshot; 154 return snapshot;
155 155
156 return NULL; 156 return NULL;
157} 157}
158 158
159void show_error(char *msg)
160{
161 ctx.page.mimetype = "text/html";
162 cgit_print_http_headers(&ctx);
163 cgit_print_docstart(&ctx);
164 cgit_print_pageheader(&ctx);
165 cgit_print_error(msg);
166 cgit_print_docend();
167}
168
159void cgit_print_snapshot(const char *head, const char *hex, 169void cgit_print_snapshot(const char *head, const char *hex,
160 const char *filename, int snapshots, int dwim) 170 const char *filename, int snapshots, int dwim)
161{ 171{
162 const struct cgit_snapshot_format* f; 172 const struct cgit_snapshot_format* f;
163 char *prefix = NULL; 173 char *prefix = NULL;
164 174
175 if (!filename) {
176 show_error("No snapshot name specified");
177 return;
178 }
179
165 f = get_format(filename); 180 f = get_format(filename);
166 if (!f) { 181 if (!f) {
167 ctx.page.mimetype = "text/html"; 182 show_error(xstrdup(fmt("Unsupported snapshot format: %s",
168 cgit_print_http_headers(&ctx); 183 filename)));
169 cgit_print_docstart(&ctx);
170 cgit_print_pageheader(&ctx);
171 cgit_print_error(fmt("Unsupported snapshot format: %s", filename));
172 cgit_print_docend();
173 return; 184 return;
174 } 185 }
175 186
176 if (!hex && dwim) { 187 if (!hex && dwim) {
177 hex = get_ref_from_filename(ctx.repo->url, filename, f); 188 hex = get_ref_from_filename(ctx.repo->url, filename, f);
178 if (hex == NULL) { 189 if (hex == NULL) {
179 html_status(404, "Not found", 0); 190 html_status(404, "Not found", 0);
180 return; 191 return;
181 } 192 }
182 prefix = xstrdup(filename); 193 prefix = xstrdup(filename);
183 prefix[strlen(filename) - strlen(f->suffix)] = '\0'; 194 prefix[strlen(filename) - strlen(f->suffix)] = '\0';
184 } 195 }
185 196
186 if (!hex) 197 if (!hex)
187 hex = head; 198 hex = head;
188 199
189 if (!prefix) 200 if (!prefix)
190 prefix = xstrdup(cgit_repobasename(ctx.repo->url)); 201 prefix = xstrdup(cgit_repobasename(ctx.repo->url));
191 202
192 make_snapshot(f, hex, prefix, filename); 203 make_snapshot(f, hex, prefix, filename);
193 free(prefix); 204 free(prefix);
194} 205}