summaryrefslogtreecommitdiffabout
authorLars Hjemli <hjemli@gmail.com>2009-07-31 15:38:38 (UTC)
committer Lars Hjemli <hjemli@gmail.com>2009-07-31 15:39:51 (UTC)
commitd6f6072560c963065b13c704fa1fa6f8950e4bac (patch) (unidiff)
tree096a542cd404d4a174f2f4a1da44ab0df99aa7c3
parent286a905842dc0bec6d21a614ec4a97c5f19d5bc4 (diff)
downloadcgit-d6f6072560c963065b13c704fa1fa6f8950e4bac.zip
cgit-d6f6072560c963065b13c704fa1fa6f8950e4bac.tar.gz
cgit-d6f6072560c963065b13c704fa1fa6f8950e4bac.tar.bz2
Add generic filter/plugin infrastructure
The functions cgit_open_filter() and cgit_close_filter() can be used to execute filters on the output stream from cgit. Signed-off-by: Lars Hjemli <hjemli@gmail.com>
Diffstat (more/less context) (ignore whitespace changes)
-rw-r--r--cgit.c15
-rw-r--r--cgit.h12
-rw-r--r--shared.c35
3 files changed, 62 insertions, 0 deletions
diff --git a/cgit.c b/cgit.c
index 2039ab1..779a464 100644
--- a/cgit.c
+++ b/cgit.c
@@ -1,147 +1,162 @@
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#include "ui-stats.h" 15#include "ui-stats.h"
16#include "scan-tree.h" 16#include "scan-tree.h"
17 17
18const char *cgit_version = CGIT_VERSION; 18const char *cgit_version = CGIT_VERSION;
19 19
20struct cgit_filter *new_filter(const char *cmd, int extra_args)
21{
22 struct cgit_filter *f;
23
24 if (!cmd)
25 return NULL;
26
27 f = xmalloc(sizeof(struct cgit_filter));
28 f->cmd = xstrdup(cmd);
29 f->argv = xmalloc((2 + extra_args) * sizeof(char *));
30 f->argv[0] = f->cmd;
31 f->argv[1] = NULL;
32 return f;
33}
34
20void config_cb(const char *name, const char *value) 35void config_cb(const char *name, const char *value)
21{ 36{
22 if (!strcmp(name, "root-title")) 37 if (!strcmp(name, "root-title"))
23 ctx.cfg.root_title = xstrdup(value); 38 ctx.cfg.root_title = xstrdup(value);
24 else if (!strcmp(name, "root-desc")) 39 else if (!strcmp(name, "root-desc"))
25 ctx.cfg.root_desc = xstrdup(value); 40 ctx.cfg.root_desc = xstrdup(value);
26 else if (!strcmp(name, "root-readme")) 41 else if (!strcmp(name, "root-readme"))
27 ctx.cfg.root_readme = xstrdup(value); 42 ctx.cfg.root_readme = xstrdup(value);
28 else if (!strcmp(name, "css")) 43 else if (!strcmp(name, "css"))
29 ctx.cfg.css = xstrdup(value); 44 ctx.cfg.css = xstrdup(value);
30 else if (!strcmp(name, "favicon")) 45 else if (!strcmp(name, "favicon"))
31 ctx.cfg.favicon = xstrdup(value); 46 ctx.cfg.favicon = xstrdup(value);
32 else if (!strcmp(name, "footer")) 47 else if (!strcmp(name, "footer"))
33 ctx.cfg.footer = xstrdup(value); 48 ctx.cfg.footer = xstrdup(value);
34 else if (!strcmp(name, "head-include")) 49 else if (!strcmp(name, "head-include"))
35 ctx.cfg.head_include = xstrdup(value); 50 ctx.cfg.head_include = xstrdup(value);
36 else if (!strcmp(name, "header")) 51 else if (!strcmp(name, "header"))
37 ctx.cfg.header = xstrdup(value); 52 ctx.cfg.header = xstrdup(value);
38 else if (!strcmp(name, "logo")) 53 else if (!strcmp(name, "logo"))
39 ctx.cfg.logo = xstrdup(value); 54 ctx.cfg.logo = xstrdup(value);
40 else if (!strcmp(name, "index-header")) 55 else if (!strcmp(name, "index-header"))
41 ctx.cfg.index_header = xstrdup(value); 56 ctx.cfg.index_header = xstrdup(value);
42 else if (!strcmp(name, "index-info")) 57 else if (!strcmp(name, "index-info"))
43 ctx.cfg.index_info = xstrdup(value); 58 ctx.cfg.index_info = xstrdup(value);
44 else if (!strcmp(name, "logo-link")) 59 else if (!strcmp(name, "logo-link"))
45 ctx.cfg.logo_link = xstrdup(value); 60 ctx.cfg.logo_link = xstrdup(value);
46 else if (!strcmp(name, "module-link")) 61 else if (!strcmp(name, "module-link"))
47 ctx.cfg.module_link = xstrdup(value); 62 ctx.cfg.module_link = xstrdup(value);
48 else if (!strcmp(name, "virtual-root")) { 63 else if (!strcmp(name, "virtual-root")) {
49 ctx.cfg.virtual_root = trim_end(value, '/'); 64 ctx.cfg.virtual_root = trim_end(value, '/');
50 if (!ctx.cfg.virtual_root && (!strcmp(value, "/"))) 65 if (!ctx.cfg.virtual_root && (!strcmp(value, "/")))
51 ctx.cfg.virtual_root = ""; 66 ctx.cfg.virtual_root = "";
52 } else if (!strcmp(name, "nocache")) 67 } else if (!strcmp(name, "nocache"))
53 ctx.cfg.nocache = atoi(value); 68 ctx.cfg.nocache = atoi(value);
54 else if (!strcmp(name, "noheader")) 69 else if (!strcmp(name, "noheader"))
55 ctx.cfg.noheader = atoi(value); 70 ctx.cfg.noheader = atoi(value);
56 else if (!strcmp(name, "snapshots")) 71 else if (!strcmp(name, "snapshots"))
57 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value); 72 ctx.cfg.snapshots = cgit_parse_snapshots_mask(value);
58 else if (!strcmp(name, "enable-index-links")) 73 else if (!strcmp(name, "enable-index-links"))
59 ctx.cfg.enable_index_links = atoi(value); 74 ctx.cfg.enable_index_links = atoi(value);
60 else if (!strcmp(name, "enable-log-filecount")) 75 else if (!strcmp(name, "enable-log-filecount"))
61 ctx.cfg.enable_log_filecount = atoi(value); 76 ctx.cfg.enable_log_filecount = atoi(value);
62 else if (!strcmp(name, "enable-log-linecount")) 77 else if (!strcmp(name, "enable-log-linecount"))
63 ctx.cfg.enable_log_linecount = atoi(value); 78 ctx.cfg.enable_log_linecount = atoi(value);
64 else if (!strcmp(name, "max-stats")) 79 else if (!strcmp(name, "max-stats"))
65 ctx.cfg.max_stats = cgit_find_stats_period(value, NULL); 80 ctx.cfg.max_stats = cgit_find_stats_period(value, NULL);
66 else if (!strcmp(name, "cache-size")) 81 else if (!strcmp(name, "cache-size"))
67 ctx.cfg.cache_size = atoi(value); 82 ctx.cfg.cache_size = atoi(value);
68 else if (!strcmp(name, "cache-root")) 83 else if (!strcmp(name, "cache-root"))
69 ctx.cfg.cache_root = xstrdup(value); 84 ctx.cfg.cache_root = xstrdup(value);
70 else if (!strcmp(name, "cache-root-ttl")) 85 else if (!strcmp(name, "cache-root-ttl"))
71 ctx.cfg.cache_root_ttl = atoi(value); 86 ctx.cfg.cache_root_ttl = atoi(value);
72 else if (!strcmp(name, "cache-repo-ttl")) 87 else if (!strcmp(name, "cache-repo-ttl"))
73 ctx.cfg.cache_repo_ttl = atoi(value); 88 ctx.cfg.cache_repo_ttl = atoi(value);
74 else if (!strcmp(name, "cache-static-ttl")) 89 else if (!strcmp(name, "cache-static-ttl"))
75 ctx.cfg.cache_static_ttl = atoi(value); 90 ctx.cfg.cache_static_ttl = atoi(value);
76 else if (!strcmp(name, "cache-dynamic-ttl")) 91 else if (!strcmp(name, "cache-dynamic-ttl"))
77 ctx.cfg.cache_dynamic_ttl = atoi(value); 92 ctx.cfg.cache_dynamic_ttl = atoi(value);
78 else if (!strcmp(name, "embedded")) 93 else if (!strcmp(name, "embedded"))
79 ctx.cfg.embedded = atoi(value); 94 ctx.cfg.embedded = atoi(value);
80 else if (!strcmp(name, "max-message-length")) 95 else if (!strcmp(name, "max-message-length"))
81 ctx.cfg.max_msg_len = atoi(value); 96 ctx.cfg.max_msg_len = atoi(value);
82 else if (!strcmp(name, "max-repodesc-length")) 97 else if (!strcmp(name, "max-repodesc-length"))
83 ctx.cfg.max_repodesc_len = atoi(value); 98 ctx.cfg.max_repodesc_len = atoi(value);
84 else if (!strcmp(name, "max-repo-count")) 99 else if (!strcmp(name, "max-repo-count"))
85 ctx.cfg.max_repo_count = atoi(value); 100 ctx.cfg.max_repo_count = atoi(value);
86 else if (!strcmp(name, "max-commit-count")) 101 else if (!strcmp(name, "max-commit-count"))
87 ctx.cfg.max_commit_count = atoi(value); 102 ctx.cfg.max_commit_count = atoi(value);
88 else if (!strcmp(name, "summary-log")) 103 else if (!strcmp(name, "summary-log"))
89 ctx.cfg.summary_log = atoi(value); 104 ctx.cfg.summary_log = atoi(value);
90 else if (!strcmp(name, "summary-branches")) 105 else if (!strcmp(name, "summary-branches"))
91 ctx.cfg.summary_branches = atoi(value); 106 ctx.cfg.summary_branches = atoi(value);
92 else if (!strcmp(name, "summary-tags")) 107 else if (!strcmp(name, "summary-tags"))
93 ctx.cfg.summary_tags = atoi(value); 108 ctx.cfg.summary_tags = atoi(value);
94 else if (!strcmp(name, "agefile")) 109 else if (!strcmp(name, "agefile"))
95 ctx.cfg.agefile = xstrdup(value); 110 ctx.cfg.agefile = xstrdup(value);
96 else if (!strcmp(name, "renamelimit")) 111 else if (!strcmp(name, "renamelimit"))
97 ctx.cfg.renamelimit = atoi(value); 112 ctx.cfg.renamelimit = atoi(value);
98 else if (!strcmp(name, "robots")) 113 else if (!strcmp(name, "robots"))
99 ctx.cfg.robots = xstrdup(value); 114 ctx.cfg.robots = xstrdup(value);
100 else if (!strcmp(name, "clone-prefix")) 115 else if (!strcmp(name, "clone-prefix"))
101 ctx.cfg.clone_prefix = xstrdup(value); 116 ctx.cfg.clone_prefix = xstrdup(value);
102 else if (!strcmp(name, "local-time")) 117 else if (!strcmp(name, "local-time"))
103 ctx.cfg.local_time = atoi(value); 118 ctx.cfg.local_time = atoi(value);
104 else if (!strcmp(name, "repo.group")) 119 else if (!strcmp(name, "repo.group"))
105 ctx.cfg.repo_group = xstrdup(value); 120 ctx.cfg.repo_group = xstrdup(value);
106 else if (!strcmp(name, "repo.url")) 121 else if (!strcmp(name, "repo.url"))
107 ctx.repo = cgit_add_repo(value); 122 ctx.repo = cgit_add_repo(value);
108 else if (!strcmp(name, "repo.name")) 123 else if (!strcmp(name, "repo.name"))
109 ctx.repo->name = xstrdup(value); 124 ctx.repo->name = xstrdup(value);
110 else if (ctx.repo && !strcmp(name, "repo.path")) 125 else if (ctx.repo && !strcmp(name, "repo.path"))
111 ctx.repo->path = trim_end(value, '/'); 126 ctx.repo->path = trim_end(value, '/');
112 else if (ctx.repo && !strcmp(name, "repo.clone-url")) 127 else if (ctx.repo && !strcmp(name, "repo.clone-url"))
113 ctx.repo->clone_url = xstrdup(value); 128 ctx.repo->clone_url = xstrdup(value);
114 else if (ctx.repo && !strcmp(name, "repo.desc")) 129 else if (ctx.repo && !strcmp(name, "repo.desc"))
115 ctx.repo->desc = xstrdup(value); 130 ctx.repo->desc = xstrdup(value);
116 else if (ctx.repo && !strcmp(name, "repo.owner")) 131 else if (ctx.repo && !strcmp(name, "repo.owner"))
117 ctx.repo->owner = xstrdup(value); 132 ctx.repo->owner = xstrdup(value);
118 else if (ctx.repo && !strcmp(name, "repo.defbranch")) 133 else if (ctx.repo && !strcmp(name, "repo.defbranch"))
119 ctx.repo->defbranch = xstrdup(value); 134 ctx.repo->defbranch = xstrdup(value);
120 else if (ctx.repo && !strcmp(name, "repo.snapshots")) 135 else if (ctx.repo && !strcmp(name, "repo.snapshots"))
121 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */ 136 ctx.repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value); /* XXX: &? */
122 else if (ctx.repo && !strcmp(name, "repo.enable-log-filecount")) 137 else if (ctx.repo && !strcmp(name, "repo.enable-log-filecount"))
123 ctx.repo->enable_log_filecount = ctx.cfg.enable_log_filecount * atoi(value); 138 ctx.repo->enable_log_filecount = ctx.cfg.enable_log_filecount * atoi(value);
124 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount")) 139 else if (ctx.repo && !strcmp(name, "repo.enable-log-linecount"))
125 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value); 140 ctx.repo->enable_log_linecount = ctx.cfg.enable_log_linecount * atoi(value);
126 else if (ctx.repo && !strcmp(name, "repo.max-stats")) 141 else if (ctx.repo && !strcmp(name, "repo.max-stats"))
127 ctx.repo->max_stats = cgit_find_stats_period(value, NULL); 142 ctx.repo->max_stats = cgit_find_stats_period(value, NULL);
128 else if (ctx.repo && !strcmp(name, "repo.module-link")) 143 else if (ctx.repo && !strcmp(name, "repo.module-link"))
129 ctx.repo->module_link= xstrdup(value); 144 ctx.repo->module_link= xstrdup(value);
130 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) { 145 else if (ctx.repo && !strcmp(name, "repo.readme") && value != NULL) {
131 if (*value == '/') 146 if (*value == '/')
132 ctx.repo->readme = xstrdup(value); 147 ctx.repo->readme = xstrdup(value);
133 else 148 else
134 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value)); 149 ctx.repo->readme = xstrdup(fmt("%s/%s", ctx.repo->path, value));
135 } else if (!strcmp(name, "include")) 150 } else if (!strcmp(name, "include"))
136 parse_configfile(value, config_cb); 151 parse_configfile(value, config_cb);
137} 152}
138 153
139static void querystring_cb(const char *name, const char *value) 154static void querystring_cb(const char *name, const char *value)
140{ 155{
141 if (!strcmp(name,"r")) { 156 if (!strcmp(name,"r")) {
142 ctx.qry.repo = xstrdup(value); 157 ctx.qry.repo = xstrdup(value);
143 ctx.repo = cgit_get_repoinfo(value); 158 ctx.repo = cgit_get_repoinfo(value);
144 } else if (!strcmp(name, "p")) { 159 } else if (!strcmp(name, "p")) {
145 ctx.qry.page = xstrdup(value); 160 ctx.qry.page = xstrdup(value);
146 } else if (!strcmp(name, "url")) { 161 } else if (!strcmp(name, "url")) {
147 ctx.qry.url = xstrdup(value); 162 ctx.qry.url = xstrdup(value);
diff --git a/cgit.h b/cgit.h
index 8c64efe..d0fff1f 100644
--- a/cgit.h
+++ b/cgit.h
@@ -4,249 +4,261 @@
4 4
5#include <git-compat-util.h> 5#include <git-compat-util.h>
6#include <cache.h> 6#include <cache.h>
7#include <grep.h> 7#include <grep.h>
8#include <object.h> 8#include <object.h>
9#include <tree.h> 9#include <tree.h>
10#include <commit.h> 10#include <commit.h>
11#include <tag.h> 11#include <tag.h>
12#include <diff.h> 12#include <diff.h>
13#include <diffcore.h> 13#include <diffcore.h>
14#include <refs.h> 14#include <refs.h>
15#include <revision.h> 15#include <revision.h>
16#include <log-tree.h> 16#include <log-tree.h>
17#include <archive.h> 17#include <archive.h>
18#include <xdiff-interface.h> 18#include <xdiff-interface.h>
19#include <xdiff/xdiff.h> 19#include <xdiff/xdiff.h>
20#include <utf8.h> 20#include <utf8.h>
21 21
22 22
23/* 23/*
24 * Dateformats used on misc. pages 24 * Dateformats used on misc. pages
25 */ 25 */
26#define FMT_LONGDATE "%Y-%m-%d %H:%M:%S (%Z)" 26#define FMT_LONGDATE "%Y-%m-%d %H:%M:%S (%Z)"
27#define FMT_SHORTDATE "%Y-%m-%d" 27#define FMT_SHORTDATE "%Y-%m-%d"
28#define FMT_ATOMDATE "%Y-%m-%dT%H:%M:%SZ" 28#define FMT_ATOMDATE "%Y-%m-%dT%H:%M:%SZ"
29 29
30 30
31/* 31/*
32 * Limits used for relative dates 32 * Limits used for relative dates
33 */ 33 */
34#define TM_MIN 60 34#define TM_MIN 60
35#define TM_HOUR (TM_MIN * 60) 35#define TM_HOUR (TM_MIN * 60)
36#define TM_DAY (TM_HOUR * 24) 36#define TM_DAY (TM_HOUR * 24)
37#define TM_WEEK (TM_DAY * 7) 37#define TM_WEEK (TM_DAY * 7)
38#define TM_YEAR (TM_DAY * 365) 38#define TM_YEAR (TM_DAY * 365)
39#define TM_MONTH (TM_YEAR / 12.0) 39#define TM_MONTH (TM_YEAR / 12.0)
40 40
41 41
42/* 42/*
43 * Default encoding 43 * Default encoding
44 */ 44 */
45#define PAGE_ENCODING "UTF-8" 45#define PAGE_ENCODING "UTF-8"
46 46
47typedef void (*configfn)(const char *name, const char *value); 47typedef void (*configfn)(const char *name, const char *value);
48typedef void (*filepair_fn)(struct diff_filepair *pair); 48typedef void (*filepair_fn)(struct diff_filepair *pair);
49typedef void (*linediff_fn)(char *line, int len); 49typedef void (*linediff_fn)(char *line, int len);
50 50
51struct cgit_repo { 51struct cgit_repo {
52 char *url; 52 char *url;
53 char *name; 53 char *name;
54 char *path; 54 char *path;
55 char *desc; 55 char *desc;
56 char *owner; 56 char *owner;
57 char *defbranch; 57 char *defbranch;
58 char *group; 58 char *group;
59 char *module_link; 59 char *module_link;
60 char *readme; 60 char *readme;
61 char *clone_url; 61 char *clone_url;
62 int snapshots; 62 int snapshots;
63 int enable_log_filecount; 63 int enable_log_filecount;
64 int enable_log_linecount; 64 int enable_log_linecount;
65 int max_stats; 65 int max_stats;
66 time_t mtime; 66 time_t mtime;
67}; 67};
68 68
69struct cgit_repolist { 69struct cgit_repolist {
70 int length; 70 int length;
71 int count; 71 int count;
72 struct cgit_repo *repos; 72 struct cgit_repo *repos;
73}; 73};
74 74
75struct commitinfo { 75struct commitinfo {
76 struct commit *commit; 76 struct commit *commit;
77 char *author; 77 char *author;
78 char *author_email; 78 char *author_email;
79 unsigned long author_date; 79 unsigned long author_date;
80 char *committer; 80 char *committer;
81 char *committer_email; 81 char *committer_email;
82 unsigned long committer_date; 82 unsigned long committer_date;
83 char *subject; 83 char *subject;
84 char *msg; 84 char *msg;
85 char *msg_encoding; 85 char *msg_encoding;
86}; 86};
87 87
88struct taginfo { 88struct taginfo {
89 char *tagger; 89 char *tagger;
90 char *tagger_email; 90 char *tagger_email;
91 unsigned long tagger_date; 91 unsigned long tagger_date;
92 char *msg; 92 char *msg;
93}; 93};
94 94
95struct refinfo { 95struct refinfo {
96 const char *refname; 96 const char *refname;
97 struct object *object; 97 struct object *object;
98 union { 98 union {
99 struct taginfo *tag; 99 struct taginfo *tag;
100 struct commitinfo *commit; 100 struct commitinfo *commit;
101 }; 101 };
102}; 102};
103 103
104struct reflist { 104struct reflist {
105 struct refinfo **refs; 105 struct refinfo **refs;
106 int alloc; 106 int alloc;
107 int count; 107 int count;
108}; 108};
109 109
110struct cgit_query { 110struct cgit_query {
111 int has_symref; 111 int has_symref;
112 int has_sha1; 112 int has_sha1;
113 char *raw; 113 char *raw;
114 char *repo; 114 char *repo;
115 char *page; 115 char *page;
116 char *search; 116 char *search;
117 char *grep; 117 char *grep;
118 char *head; 118 char *head;
119 char *sha1; 119 char *sha1;
120 char *sha2; 120 char *sha2;
121 char *path; 121 char *path;
122 char *name; 122 char *name;
123 char *mimetype; 123 char *mimetype;
124 char *url; 124 char *url;
125 char *period; 125 char *period;
126 int ofs; 126 int ofs;
127 int nohead; 127 int nohead;
128 char *sort; 128 char *sort;
129 int showmsg; 129 int showmsg;
130}; 130};
131 131
132struct cgit_filter {
133 char *cmd;
134 char **argv;
135 int old_stdout;
136 int pipe_fh[2];
137 int pid;
138 int exitstatus;
139};
140
132struct cgit_config { 141struct cgit_config {
133 char *agefile; 142 char *agefile;
134 char *cache_root; 143 char *cache_root;
135 char *clone_prefix; 144 char *clone_prefix;
136 char *css; 145 char *css;
137 char *favicon; 146 char *favicon;
138 char *footer; 147 char *footer;
139 char *head_include; 148 char *head_include;
140 char *header; 149 char *header;
141 char *index_header; 150 char *index_header;
142 char *index_info; 151 char *index_info;
143 char *logo; 152 char *logo;
144 char *logo_link; 153 char *logo_link;
145 char *module_link; 154 char *module_link;
146 char *repo_group; 155 char *repo_group;
147 char *robots; 156 char *robots;
148 char *root_title; 157 char *root_title;
149 char *root_desc; 158 char *root_desc;
150 char *root_readme; 159 char *root_readme;
151 char *script_name; 160 char *script_name;
152 char *virtual_root; 161 char *virtual_root;
153 int cache_size; 162 int cache_size;
154 int cache_dynamic_ttl; 163 int cache_dynamic_ttl;
155 int cache_max_create_time; 164 int cache_max_create_time;
156 int cache_repo_ttl; 165 int cache_repo_ttl;
157 int cache_root_ttl; 166 int cache_root_ttl;
158 int cache_static_ttl; 167 int cache_static_ttl;
159 int embedded; 168 int embedded;
160 int enable_index_links; 169 int enable_index_links;
161 int enable_log_filecount; 170 int enable_log_filecount;
162 int enable_log_linecount; 171 int enable_log_linecount;
163 int local_time; 172 int local_time;
164 int max_repo_count; 173 int max_repo_count;
165 int max_commit_count; 174 int max_commit_count;
166 int max_lock_attempts; 175 int max_lock_attempts;
167 int max_msg_len; 176 int max_msg_len;
168 int max_repodesc_len; 177 int max_repodesc_len;
169 int max_stats; 178 int max_stats;
170 int nocache; 179 int nocache;
171 int noheader; 180 int noheader;
172 int renamelimit; 181 int renamelimit;
173 int snapshots; 182 int snapshots;
174 int summary_branches; 183 int summary_branches;
175 int summary_log; 184 int summary_log;
176 int summary_tags; 185 int summary_tags;
177}; 186};
178 187
179struct cgit_page { 188struct cgit_page {
180 time_t modified; 189 time_t modified;
181 time_t expires; 190 time_t expires;
182 size_t size; 191 size_t size;
183 char *mimetype; 192 char *mimetype;
184 char *charset; 193 char *charset;
185 char *filename; 194 char *filename;
186 char *etag; 195 char *etag;
187 char *title; 196 char *title;
188 int status; 197 int status;
189 char *statusmsg; 198 char *statusmsg;
190}; 199};
191 200
192struct cgit_context { 201struct cgit_context {
193 struct cgit_query qry; 202 struct cgit_query qry;
194 struct cgit_config cfg; 203 struct cgit_config cfg;
195 struct cgit_repo *repo; 204 struct cgit_repo *repo;
196 struct cgit_page page; 205 struct cgit_page page;
197}; 206};
198 207
199struct cgit_snapshot_format { 208struct cgit_snapshot_format {
200 const char *suffix; 209 const char *suffix;
201 const char *mimetype; 210 const char *mimetype;
202 write_archive_fn_t write_func; 211 write_archive_fn_t write_func;
203 int bit; 212 int bit;
204}; 213};
205 214
206extern const char *cgit_version; 215extern const char *cgit_version;
207 216
208extern struct cgit_repolist cgit_repolist; 217extern struct cgit_repolist cgit_repolist;
209extern struct cgit_context ctx; 218extern struct cgit_context ctx;
210extern const struct cgit_snapshot_format cgit_snapshot_formats[]; 219extern const struct cgit_snapshot_format cgit_snapshot_formats[];
211 220
212extern struct cgit_repo *cgit_add_repo(const char *url); 221extern struct cgit_repo *cgit_add_repo(const char *url);
213extern struct cgit_repo *cgit_get_repoinfo(const char *url); 222extern struct cgit_repo *cgit_get_repoinfo(const char *url);
214extern void cgit_repo_config_cb(const char *name, const char *value); 223extern void cgit_repo_config_cb(const char *name, const char *value);
215 224
216extern int chk_zero(int result, char *msg); 225extern int chk_zero(int result, char *msg);
217extern int chk_positive(int result, char *msg); 226extern int chk_positive(int result, char *msg);
218extern int chk_non_negative(int result, char *msg); 227extern int chk_non_negative(int result, char *msg);
219 228
220extern char *trim_end(const char *str, char c); 229extern char *trim_end(const char *str, char c);
221extern char *strlpart(char *txt, int maxlen); 230extern char *strlpart(char *txt, int maxlen);
222extern char *strrpart(char *txt, int maxlen); 231extern char *strrpart(char *txt, int maxlen);
223 232
224extern void cgit_add_ref(struct reflist *list, struct refinfo *ref); 233extern void cgit_add_ref(struct reflist *list, struct refinfo *ref);
225extern int cgit_refs_cb(const char *refname, const unsigned char *sha1, 234extern int cgit_refs_cb(const char *refname, const unsigned char *sha1,
226 int flags, void *cb_data); 235 int flags, void *cb_data);
227 236
228extern void *cgit_free_commitinfo(struct commitinfo *info); 237extern void *cgit_free_commitinfo(struct commitinfo *info);
229 238
230extern int cgit_diff_files(const unsigned char *old_sha1, 239extern int cgit_diff_files(const unsigned char *old_sha1,
231 const unsigned char *new_sha1, 240 const unsigned char *new_sha1,
232 unsigned long *old_size, unsigned long *new_size, 241 unsigned long *old_size, unsigned long *new_size,
233 int *binary, linediff_fn fn); 242 int *binary, linediff_fn fn);
234 243
235extern void cgit_diff_tree(const unsigned char *old_sha1, 244extern void cgit_diff_tree(const unsigned char *old_sha1,
236 const unsigned char *new_sha1, 245 const unsigned char *new_sha1,
237 filepair_fn fn, const char *prefix); 246 filepair_fn fn, const char *prefix);
238 247
239extern void cgit_diff_commit(struct commit *commit, filepair_fn fn); 248extern void cgit_diff_commit(struct commit *commit, filepair_fn fn);
240 249
241extern char *fmt(const char *format,...); 250extern char *fmt(const char *format,...);
242 251
243extern struct commitinfo *cgit_parse_commit(struct commit *commit); 252extern struct commitinfo *cgit_parse_commit(struct commit *commit);
244extern struct taginfo *cgit_parse_tag(struct tag *tag); 253extern struct taginfo *cgit_parse_tag(struct tag *tag);
245extern void cgit_parse_url(const char *url); 254extern void cgit_parse_url(const char *url);
246 255
247extern const char *cgit_repobasename(const char *reponame); 256extern const char *cgit_repobasename(const char *reponame);
248 257
249extern int cgit_parse_snapshots_mask(const char *str); 258extern int cgit_parse_snapshots_mask(const char *str);
250 259
260extern int cgit_open_filter(struct cgit_filter *filter);
261extern int cgit_close_filter(struct cgit_filter *filter);
262
251 263
252#endif /* CGIT_H */ 264#endif /* CGIT_H */
diff --git a/shared.c b/shared.c
index cce0af4..288cfa2 100644
--- a/shared.c
+++ b/shared.c
@@ -230,128 +230,163 @@ int filediff_cb(void *priv, mmbuffer_t *mb, int nbuf)
230 if (mb[i].ptr[mb[i].size-1] != '\n') { 230 if (mb[i].ptr[mb[i].size-1] != '\n') {
231 /* Incomplete line */ 231 /* Incomplete line */
232 diffbuf = xrealloc(diffbuf, buflen + mb[i].size); 232 diffbuf = xrealloc(diffbuf, buflen + mb[i].size);
233 memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); 233 memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size);
234 buflen += mb[i].size; 234 buflen += mb[i].size;
235 continue; 235 continue;
236 } 236 }
237 237
238 /* we have a complete line */ 238 /* we have a complete line */
239 if (!diffbuf) { 239 if (!diffbuf) {
240 ((linediff_fn)priv)(mb[i].ptr, mb[i].size); 240 ((linediff_fn)priv)(mb[i].ptr, mb[i].size);
241 continue; 241 continue;
242 } 242 }
243 diffbuf = xrealloc(diffbuf, buflen + mb[i].size); 243 diffbuf = xrealloc(diffbuf, buflen + mb[i].size);
244 memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size); 244 memcpy(diffbuf + buflen, mb[i].ptr, mb[i].size);
245 ((linediff_fn)priv)(diffbuf, buflen + mb[i].size); 245 ((linediff_fn)priv)(diffbuf, buflen + mb[i].size);
246 free(diffbuf); 246 free(diffbuf);
247 diffbuf = NULL; 247 diffbuf = NULL;
248 buflen = 0; 248 buflen = 0;
249 } 249 }
250 if (diffbuf) { 250 if (diffbuf) {
251 ((linediff_fn)priv)(diffbuf, buflen); 251 ((linediff_fn)priv)(diffbuf, buflen);
252 free(diffbuf); 252 free(diffbuf);
253 diffbuf = NULL; 253 diffbuf = NULL;
254 buflen = 0; 254 buflen = 0;
255 } 255 }
256 return 0; 256 return 0;
257} 257}
258 258
259int cgit_diff_files(const unsigned char *old_sha1, 259int cgit_diff_files(const unsigned char *old_sha1,
260 const unsigned char *new_sha1, unsigned long *old_size, 260 const unsigned char *new_sha1, unsigned long *old_size,
261 unsigned long *new_size, int *binary, linediff_fn fn) 261 unsigned long *new_size, int *binary, linediff_fn fn)
262{ 262{
263 mmfile_t file1, file2; 263 mmfile_t file1, file2;
264 xpparam_t diff_params; 264 xpparam_t diff_params;
265 xdemitconf_t emit_params; 265 xdemitconf_t emit_params;
266 xdemitcb_t emit_cb; 266 xdemitcb_t emit_cb;
267 267
268 if (!load_mmfile(&file1, old_sha1) || !load_mmfile(&file2, new_sha1)) 268 if (!load_mmfile(&file1, old_sha1) || !load_mmfile(&file2, new_sha1))
269 return 1; 269 return 1;
270 270
271 *old_size = file1.size; 271 *old_size = file1.size;
272 *new_size = file2.size; 272 *new_size = file2.size;
273 273
274 if ((file1.ptr && buffer_is_binary(file1.ptr, file1.size)) || 274 if ((file1.ptr && buffer_is_binary(file1.ptr, file1.size)) ||
275 (file2.ptr && buffer_is_binary(file2.ptr, file2.size))) { 275 (file2.ptr && buffer_is_binary(file2.ptr, file2.size))) {
276 *binary = 1; 276 *binary = 1;
277 return 0; 277 return 0;
278 } 278 }
279 279
280 memset(&diff_params, 0, sizeof(diff_params)); 280 memset(&diff_params, 0, sizeof(diff_params));
281 memset(&emit_params, 0, sizeof(emit_params)); 281 memset(&emit_params, 0, sizeof(emit_params));
282 memset(&emit_cb, 0, sizeof(emit_cb)); 282 memset(&emit_cb, 0, sizeof(emit_cb));
283 diff_params.flags = XDF_NEED_MINIMAL; 283 diff_params.flags = XDF_NEED_MINIMAL;
284 emit_params.ctxlen = 3; 284 emit_params.ctxlen = 3;
285 emit_params.flags = XDL_EMIT_FUNCNAMES; 285 emit_params.flags = XDL_EMIT_FUNCNAMES;
286 emit_cb.outf = filediff_cb; 286 emit_cb.outf = filediff_cb;
287 emit_cb.priv = fn; 287 emit_cb.priv = fn;
288 xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb); 288 xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb);
289 return 0; 289 return 0;
290} 290}
291 291
292void cgit_diff_tree(const unsigned char *old_sha1, 292void cgit_diff_tree(const unsigned char *old_sha1,
293 const unsigned char *new_sha1, 293 const unsigned char *new_sha1,
294 filepair_fn fn, const char *prefix) 294 filepair_fn fn, const char *prefix)
295{ 295{
296 struct diff_options opt; 296 struct diff_options opt;
297 int ret; 297 int ret;
298 int prefixlen; 298 int prefixlen;
299 299
300 diff_setup(&opt); 300 diff_setup(&opt);
301 opt.output_format = DIFF_FORMAT_CALLBACK; 301 opt.output_format = DIFF_FORMAT_CALLBACK;
302 opt.detect_rename = 1; 302 opt.detect_rename = 1;
303 opt.rename_limit = ctx.cfg.renamelimit; 303 opt.rename_limit = ctx.cfg.renamelimit;
304 DIFF_OPT_SET(&opt, RECURSIVE); 304 DIFF_OPT_SET(&opt, RECURSIVE);
305 opt.format_callback = cgit_diff_tree_cb; 305 opt.format_callback = cgit_diff_tree_cb;
306 opt.format_callback_data = fn; 306 opt.format_callback_data = fn;
307 if (prefix) { 307 if (prefix) {
308 opt.nr_paths = 1; 308 opt.nr_paths = 1;
309 opt.paths = &prefix; 309 opt.paths = &prefix;
310 prefixlen = strlen(prefix); 310 prefixlen = strlen(prefix);
311 opt.pathlens = &prefixlen; 311 opt.pathlens = &prefixlen;
312 } 312 }
313 diff_setup_done(&opt); 313 diff_setup_done(&opt);
314 314
315 if (old_sha1 && !is_null_sha1(old_sha1)) 315 if (old_sha1 && !is_null_sha1(old_sha1))
316 ret = diff_tree_sha1(old_sha1, new_sha1, "", &opt); 316 ret = diff_tree_sha1(old_sha1, new_sha1, "", &opt);
317 else 317 else
318 ret = diff_root_tree_sha1(new_sha1, "", &opt); 318 ret = diff_root_tree_sha1(new_sha1, "", &opt);
319 diffcore_std(&opt); 319 diffcore_std(&opt);
320 diff_flush(&opt); 320 diff_flush(&opt);
321} 321}
322 322
323void cgit_diff_commit(struct commit *commit, filepair_fn fn) 323void cgit_diff_commit(struct commit *commit, filepair_fn fn)
324{ 324{
325 unsigned char *old_sha1 = NULL; 325 unsigned char *old_sha1 = NULL;
326 326
327 if (commit->parents) 327 if (commit->parents)
328 old_sha1 = commit->parents->item->object.sha1; 328 old_sha1 = commit->parents->item->object.sha1;
329 cgit_diff_tree(old_sha1, commit->object.sha1, fn, NULL); 329 cgit_diff_tree(old_sha1, commit->object.sha1, fn, NULL);
330} 330}
331 331
332int cgit_parse_snapshots_mask(const char *str) 332int cgit_parse_snapshots_mask(const char *str)
333{ 333{
334 const struct cgit_snapshot_format *f; 334 const struct cgit_snapshot_format *f;
335 static const char *delim = " \t,:/|;"; 335 static const char *delim = " \t,:/|;";
336 int tl, sl, rv = 0; 336 int tl, sl, rv = 0;
337 337
338 /* favor legacy setting */ 338 /* favor legacy setting */
339 if(atoi(str)) 339 if(atoi(str))
340 return 1; 340 return 1;
341 for(;;) { 341 for(;;) {
342 str += strspn(str,delim); 342 str += strspn(str,delim);
343 tl = strcspn(str,delim); 343 tl = strcspn(str,delim);
344 if (!tl) 344 if (!tl)
345 break; 345 break;
346 for (f = cgit_snapshot_formats; f->suffix; f++) { 346 for (f = cgit_snapshot_formats; f->suffix; f++) {
347 sl = strlen(f->suffix); 347 sl = strlen(f->suffix);
348 if((tl == sl && !strncmp(f->suffix, str, tl)) || 348 if((tl == sl && !strncmp(f->suffix, str, tl)) ||
349 (tl == sl-1 && !strncmp(f->suffix+1, str, tl-1))) { 349 (tl == sl-1 && !strncmp(f->suffix+1, str, tl-1))) {
350 rv |= f->bit; 350 rv |= f->bit;
351 break; 351 break;
352 } 352 }
353 } 353 }
354 str += tl; 354 str += tl;
355 } 355 }
356 return rv; 356 return rv;
357} 357}
358
359int cgit_open_filter(struct cgit_filter *filter)
360{
361
362 filter->old_stdout = chk_positive(dup(STDOUT_FILENO),
363 "Unable to duplicate STDOUT");
364 chk_zero(pipe(filter->pipe_fh), "Unable to create pipe to subprocess");
365 filter->pid = chk_non_negative(fork(), "Unable to create subprocess");
366 if (filter->pid == 0) {
367 close(filter->pipe_fh[1]);
368 chk_non_negative(dup2(filter->pipe_fh[0], STDIN_FILENO),
369 "Unable to use pipe as STDIN");
370 execvp(filter->cmd, filter->argv);
371 die("Unable to exec subprocess %s: %s (%d)", filter->cmd,
372 strerror(errno), errno);
373 }
374 close(filter->pipe_fh[0]);
375 chk_non_negative(dup2(filter->pipe_fh[1], STDOUT_FILENO),
376 "Unable to use pipe as STDOUT");
377 close(filter->pipe_fh[1]);
378 return 0;
379}
380
381int cgit_close_filter(struct cgit_filter *filter)
382{
383 chk_non_negative(dup2(filter->old_stdout, STDOUT_FILENO),
384 "Unable to restore STDOUT");
385 close(filter->old_stdout);
386 if (filter->pid < 0)
387 return 0;
388 waitpid(filter->pid, &filter->exitstatus, 0);
389 if (WIFEXITED(filter->exitstatus) && !WEXITSTATUS(filter->exitstatus))
390 return 0;
391 die("Subprocess %s exited abnormally", filter->cmd);
392}