mirror of
https://git.zx2c4.com/cgit
synced 2025-08-17 16:01:06 +02:00
Compare commits
109 commits
Author | SHA1 | Date | |
---|---|---|---|
|
09d24d7cd0 | ||
|
fb87de795b | ||
|
34c30d12fc | ||
|
b2c939af4b | ||
|
92a8f1676a | ||
|
dbadd856ba | ||
|
8905003cba | ||
|
63d35e556f | ||
|
793c420897 | ||
|
a95762af1a | ||
|
2f50b47c72 | ||
|
a6da40bf84 | ||
|
0e6744b308 | ||
|
2593cd8138 | ||
|
4dee601bb6 | ||
|
6feb1b669b | ||
|
00ecfaadea | ||
|
907134b7a2 | ||
|
aee39b4e9a | ||
|
093ac96970 | ||
|
91f25909b9 | ||
|
852cb3b0e2 | ||
|
4c520cefc9 | ||
|
d071f28cfa | ||
|
c1a1d23111 | ||
|
a0f6669bdb | ||
|
ce2062d9e2 | ||
|
4e4b30effb | ||
|
3295155a0c | ||
|
e32f8416e8 | ||
|
afffc3e772 | ||
|
fd20a5475e | ||
|
bcffc52366 | ||
|
cc6d9cc7fc | ||
|
e10159691e | ||
|
979cf4a753 | ||
|
6ac984b51d | ||
|
33efb5fec5 | ||
|
e5c868f109 | ||
|
43df01c10f | ||
|
89ee517122 | ||
|
2486d70752 | ||
|
bcdfb2197f | ||
|
b9ff119549 | ||
|
bb02e24ec2 | ||
|
cc9b717c87 | ||
|
9761994243 | ||
|
bbbaa29a96 | ||
|
73e98c16e8 | ||
|
11be5b8182 | ||
|
b8f2b675df | ||
|
45eff40655 | ||
|
5258c297ba | ||
|
6dbbffe015 | ||
|
62eb8db452 | ||
|
d889cae811 | ||
|
4ffadc1e0c | ||
|
bd6f5683f6 | ||
|
f69626c68e | ||
|
cef27b670a | ||
|
b1739247b1 | ||
|
fe99c76ee4 | ||
|
adcc4f822f | ||
|
a1039ab175 | ||
|
a4de0e810b | ||
|
779631c6dc | ||
|
629659d2cf | ||
|
205837d468 | ||
|
f780396c0a | ||
|
0462f08d85 | ||
|
55fa25adb0 | ||
|
6a8d6d4b50 | ||
|
892ba8c3cc | ||
|
cc230bf044 | ||
|
06671f4b21 | ||
|
fde897b817 | ||
|
5e49023b01 | ||
|
fa146ccabd | ||
|
bd68c98879 | ||
|
ca98c9e7bf | ||
|
d8e5dd25a0 | ||
|
583aa5d80e | ||
|
bfabd4519c | ||
|
8fc0c81bbb | ||
|
034e3c7d56 | ||
|
e1ad15d368 | ||
|
27a6d69ab3 | ||
|
985fba80d0 | ||
|
68de710c1c | ||
|
ccba7eb9d0 | ||
|
54c407a74a | ||
|
bd0293f570 | ||
|
5bd7e9bc1b | ||
|
7d87cd3a21 | ||
|
e23f63461f | ||
|
55ebd5e97c | ||
|
441dac1d74 | ||
|
898b9e19e0 | ||
|
a22855747e | ||
|
2c9f56f3e1 | ||
|
a96f2890f4 | ||
|
0899eb644f | ||
|
2fc008d6de | ||
|
edb3403f00 | ||
|
7f75647b55 | ||
|
71ba7187e5 | ||
|
60a930044d | ||
|
7cde5885d8 | ||
|
b0fc647fe6 |
49 changed files with 968 additions and 475 deletions
9
Makefile
9
Makefile
|
@ -1,6 +1,6 @@
|
||||||
all::
|
all::
|
||||||
|
|
||||||
CGIT_VERSION = v1.2.1
|
CGIT_VERSION = v1.2.3
|
||||||
CGIT_SCRIPT_NAME = cgit.cgi
|
CGIT_SCRIPT_NAME = cgit.cgi
|
||||||
CGIT_SCRIPT_PATH = /var/www/htdocs/cgit
|
CGIT_SCRIPT_PATH = /var/www/htdocs/cgit
|
||||||
CGIT_DATA_PATH = $(CGIT_SCRIPT_PATH)
|
CGIT_DATA_PATH = $(CGIT_SCRIPT_PATH)
|
||||||
|
@ -14,8 +14,8 @@ htmldir = $(docdir)
|
||||||
pdfdir = $(docdir)
|
pdfdir = $(docdir)
|
||||||
mandir = $(prefix)/share/man
|
mandir = $(prefix)/share/man
|
||||||
SHA1_HEADER = <openssl/sha.h>
|
SHA1_HEADER = <openssl/sha.h>
|
||||||
GIT_VER = 2.18.0
|
GIT_VER = 2.46.0
|
||||||
GIT_URL = https://www.kernel.org/pub/software/scm/git/git-$(GIT_VER).tar.gz
|
GIT_URL = https://www.kernel.org/pub/software/scm/git/git-$(GIT_VER).tar.xz
|
||||||
INSTALL = install
|
INSTALL = install
|
||||||
COPYTREE = cp -r
|
COPYTREE = cp -r
|
||||||
MAN5_TXT = $(wildcard *.5.txt)
|
MAN5_TXT = $(wildcard *.5.txt)
|
||||||
|
@ -87,6 +87,7 @@ install: all
|
||||||
$(INSTALL) -m 0755 cgit $(DESTDIR)$(CGIT_SCRIPT_PATH)/$(CGIT_SCRIPT_NAME)
|
$(INSTALL) -m 0755 cgit $(DESTDIR)$(CGIT_SCRIPT_PATH)/$(CGIT_SCRIPT_NAME)
|
||||||
$(INSTALL) -m 0755 -d $(DESTDIR)$(CGIT_DATA_PATH)
|
$(INSTALL) -m 0755 -d $(DESTDIR)$(CGIT_DATA_PATH)
|
||||||
$(INSTALL) -m 0644 cgit.css $(DESTDIR)$(CGIT_DATA_PATH)/cgit.css
|
$(INSTALL) -m 0644 cgit.css $(DESTDIR)$(CGIT_DATA_PATH)/cgit.css
|
||||||
|
$(INSTALL) -m 0644 cgit.js $(DESTDIR)$(CGIT_DATA_PATH)/cgit.js
|
||||||
$(INSTALL) -m 0644 cgit.png $(DESTDIR)$(CGIT_DATA_PATH)/cgit.png
|
$(INSTALL) -m 0644 cgit.png $(DESTDIR)$(CGIT_DATA_PATH)/cgit.png
|
||||||
$(INSTALL) -m 0644 favicon.ico $(DESTDIR)$(CGIT_DATA_PATH)/favicon.ico
|
$(INSTALL) -m 0644 favicon.ico $(DESTDIR)$(CGIT_DATA_PATH)/favicon.ico
|
||||||
$(INSTALL) -m 0644 robots.txt $(DESTDIR)$(CGIT_DATA_PATH)/robots.txt
|
$(INSTALL) -m 0644 robots.txt $(DESTDIR)$(CGIT_DATA_PATH)/robots.txt
|
||||||
|
@ -157,7 +158,7 @@ clean-doc:
|
||||||
$(RM) cgitrc.5 cgitrc.5.html cgitrc.5.pdf cgitrc.5.xml cgitrc.5.fo
|
$(RM) cgitrc.5 cgitrc.5.html cgitrc.5.pdf cgitrc.5.xml cgitrc.5.fo
|
||||||
|
|
||||||
get-git:
|
get-git:
|
||||||
curl -L $(GIT_URL) | tar -xzf - && rm -rf git && mv git-$(GIT_VER) git
|
curl -L $(GIT_URL) | tar -xJf - && rm -rf git && mv git-$(GIT_VER) git
|
||||||
|
|
||||||
tags:
|
tags:
|
||||||
$(QUIET_TAGS)find . -name '*.[ch]' | xargs ctags
|
$(QUIET_TAGS)find . -name '*.[ch]' | xargs ctags
|
||||||
|
|
45
cache.c
45
cache.c
|
@ -85,40 +85,45 @@ static int close_slot(struct cache_slot *slot)
|
||||||
/* Print the content of the active cache slot (but skip the key). */
|
/* Print the content of the active cache slot (but skip the key). */
|
||||||
static int print_slot(struct cache_slot *slot)
|
static int print_slot(struct cache_slot *slot)
|
||||||
{
|
{
|
||||||
|
off_t off;
|
||||||
#ifdef HAVE_LINUX_SENDFILE
|
#ifdef HAVE_LINUX_SENDFILE
|
||||||
off_t start_off;
|
off_t size;
|
||||||
int ret;
|
#endif
|
||||||
|
|
||||||
start_off = slot->keylen + 1;
|
off = slot->keylen + 1;
|
||||||
|
|
||||||
|
#ifdef HAVE_LINUX_SENDFILE
|
||||||
|
size = slot->cache_st.st_size;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
ret = sendfile(STDOUT_FILENO, slot->cache_fd, &start_off,
|
ssize_t ret;
|
||||||
slot->cache_st.st_size - start_off);
|
ret = sendfile(STDOUT_FILENO, slot->cache_fd, &off, size - off);
|
||||||
if (ret < 0) {
|
if (ret < 0) {
|
||||||
if (errno == EAGAIN || errno == EINTR)
|
if (errno == EAGAIN || errno == EINTR)
|
||||||
continue;
|
continue;
|
||||||
|
/* Fall back to read/write on EINVAL or ENOSYS */
|
||||||
|
if (errno == EINVAL || errno == ENOSYS)
|
||||||
|
break;
|
||||||
return errno;
|
return errno;
|
||||||
}
|
}
|
||||||
|
if (off == size)
|
||||||
return 0;
|
return 0;
|
||||||
} while (1);
|
} while (1);
|
||||||
#else
|
#endif
|
||||||
ssize_t i, j;
|
|
||||||
|
|
||||||
i = lseek(slot->cache_fd, slot->keylen + 1, SEEK_SET);
|
if (lseek(slot->cache_fd, off, SEEK_SET) != off)
|
||||||
if (i != slot->keylen + 1)
|
|
||||||
return errno;
|
return errno;
|
||||||
|
|
||||||
do {
|
do {
|
||||||
i = j = xread(slot->cache_fd, slot->buf, sizeof(slot->buf));
|
ssize_t ret;
|
||||||
if (i > 0)
|
ret = xread(slot->cache_fd, slot->buf, sizeof(slot->buf));
|
||||||
j = xwrite(STDOUT_FILENO, slot->buf, i);
|
if (ret < 0)
|
||||||
} while (i > 0 && j == i);
|
|
||||||
|
|
||||||
if (i < 0 || j != i)
|
|
||||||
return errno;
|
return errno;
|
||||||
else
|
if (ret == 0)
|
||||||
return 0;
|
return 0;
|
||||||
#endif
|
if (write_in_full(STDOUT_FILENO, slot->buf, ret) < 0)
|
||||||
|
return errno;
|
||||||
|
} while (1);
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Check if the slot has expired */
|
/* Check if the slot has expired */
|
||||||
|
@ -401,12 +406,12 @@ int cache_process(int size, const char *path, const char *key, int ttl,
|
||||||
static char *sprintftime(const char *format, time_t time)
|
static char *sprintftime(const char *format, time_t time)
|
||||||
{
|
{
|
||||||
static char buf[64];
|
static char buf[64];
|
||||||
struct tm *tm;
|
struct tm tm;
|
||||||
|
|
||||||
if (!time)
|
if (!time)
|
||||||
return NULL;
|
return NULL;
|
||||||
tm = gmtime(&time);
|
gmtime_r(&time, &tm);
|
||||||
strftime(buf, sizeof(buf)-1, format, tm);
|
strftime(buf, sizeof(buf)-1, format, &tm);
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
72
cgit.c
72
cgit.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "cache.h"
|
#include "cache.h"
|
||||||
#include "cmd.h"
|
#include "cmd.h"
|
||||||
|
@ -19,6 +21,16 @@
|
||||||
|
|
||||||
const char *cgit_version = CGIT_VERSION;
|
const char *cgit_version = CGIT_VERSION;
|
||||||
|
|
||||||
|
__attribute__((constructor))
|
||||||
|
static void constructor_environment()
|
||||||
|
{
|
||||||
|
/* Do not look in /etc/ for gitconfig and gitattributes. */
|
||||||
|
setenv("GIT_CONFIG_NOSYSTEM", "1", 1);
|
||||||
|
setenv("GIT_ATTR_NOSYSTEM", "1", 1);
|
||||||
|
unsetenv("HOME");
|
||||||
|
unsetenv("XDG_CONFIG_HOME");
|
||||||
|
}
|
||||||
|
|
||||||
static void add_mimetype(const char *name, const char *value)
|
static void add_mimetype(const char *name, const char *value)
|
||||||
{
|
{
|
||||||
struct string_list_item *item;
|
struct string_list_item *item;
|
||||||
|
@ -50,6 +62,8 @@ static void repo_config(struct cgit_repo *repo, const char *name, const char *va
|
||||||
repo->extra_head_content = xstrdup(value);
|
repo->extra_head_content = xstrdup(value);
|
||||||
else if (!strcmp(name, "snapshots"))
|
else if (!strcmp(name, "snapshots"))
|
||||||
repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value);
|
repo->snapshots = ctx.cfg.snapshots & cgit_parse_snapshots_mask(value);
|
||||||
|
else if (!strcmp(name, "enable-blame"))
|
||||||
|
repo->enable_blame = atoi(value);
|
||||||
else if (!strcmp(name, "enable-commit-graph"))
|
else if (!strcmp(name, "enable-commit-graph"))
|
||||||
repo->enable_commit_graph = atoi(value);
|
repo->enable_commit_graph = atoi(value);
|
||||||
else if (!strcmp(name, "enable-log-filecount"))
|
else if (!strcmp(name, "enable-log-filecount"))
|
||||||
|
@ -130,7 +144,9 @@ static void config_cb(const char *name, const char *value)
|
||||||
else if (!strcmp(name, "root-readme"))
|
else if (!strcmp(name, "root-readme"))
|
||||||
ctx.cfg.root_readme = xstrdup(value);
|
ctx.cfg.root_readme = xstrdup(value);
|
||||||
else if (!strcmp(name, "css"))
|
else if (!strcmp(name, "css"))
|
||||||
ctx.cfg.css = xstrdup(value);
|
string_list_append(&ctx.cfg.css, xstrdup(value));
|
||||||
|
else if (!strcmp(name, "js"))
|
||||||
|
string_list_append(&ctx.cfg.js, xstrdup(value));
|
||||||
else if (!strcmp(name, "favicon"))
|
else if (!strcmp(name, "favicon"))
|
||||||
ctx.cfg.favicon = xstrdup(value);
|
ctx.cfg.favicon = xstrdup(value);
|
||||||
else if (!strcmp(name, "footer"))
|
else if (!strcmp(name, "footer"))
|
||||||
|
@ -225,9 +241,11 @@ static void config_cb(const char *name, const char *value)
|
||||||
ctx.cfg.max_repodesc_len = atoi(value);
|
ctx.cfg.max_repodesc_len = atoi(value);
|
||||||
else if (!strcmp(name, "max-blob-size"))
|
else if (!strcmp(name, "max-blob-size"))
|
||||||
ctx.cfg.max_blob_size = atoi(value);
|
ctx.cfg.max_blob_size = atoi(value);
|
||||||
else if (!strcmp(name, "max-repo-count"))
|
else if (!strcmp(name, "max-repo-count")) {
|
||||||
ctx.cfg.max_repo_count = atoi(value);
|
ctx.cfg.max_repo_count = atoi(value);
|
||||||
else if (!strcmp(name, "max-commit-count"))
|
if (ctx.cfg.max_repo_count <= 0)
|
||||||
|
ctx.cfg.max_repo_count = INT_MAX;
|
||||||
|
} else if (!strcmp(name, "max-commit-count"))
|
||||||
ctx.cfg.max_commit_count = atoi(value);
|
ctx.cfg.max_commit_count = atoi(value);
|
||||||
else if (!strcmp(name, "project-list"))
|
else if (!strcmp(name, "project-list"))
|
||||||
ctx.cfg.project_list = xstrdup(expand_macros(value));
|
ctx.cfg.project_list = xstrdup(expand_macros(value));
|
||||||
|
@ -312,11 +330,11 @@ static void querystring_cb(const char *name, const char *value)
|
||||||
ctx.qry.head = xstrdup(value);
|
ctx.qry.head = xstrdup(value);
|
||||||
ctx.qry.has_symref = 1;
|
ctx.qry.has_symref = 1;
|
||||||
} else if (!strcmp(name, "id")) {
|
} else if (!strcmp(name, "id")) {
|
||||||
ctx.qry.sha1 = xstrdup(value);
|
ctx.qry.oid = xstrdup(value);
|
||||||
ctx.qry.has_sha1 = 1;
|
ctx.qry.has_oid = 1;
|
||||||
} else if (!strcmp(name, "id2")) {
|
} else if (!strcmp(name, "id2")) {
|
||||||
ctx.qry.sha2 = xstrdup(value);
|
ctx.qry.oid2 = xstrdup(value);
|
||||||
ctx.qry.has_sha1 = 1;
|
ctx.qry.has_oid = 1;
|
||||||
} else if (!strcmp(name, "ofs")) {
|
} else if (!strcmp(name, "ofs")) {
|
||||||
ctx.qry.ofs = atoi(value);
|
ctx.qry.ofs = atoi(value);
|
||||||
} else if (!strcmp(name, "path")) {
|
} else if (!strcmp(name, "path")) {
|
||||||
|
@ -364,7 +382,6 @@ static void prepare_context(void)
|
||||||
ctx.cfg.case_sensitive_sort = 1;
|
ctx.cfg.case_sensitive_sort = 1;
|
||||||
ctx.cfg.branch_sort = 0;
|
ctx.cfg.branch_sort = 0;
|
||||||
ctx.cfg.commit_sort = 0;
|
ctx.cfg.commit_sort = 0;
|
||||||
ctx.cfg.css = "/cgit.css";
|
|
||||||
ctx.cfg.logo = "/cgit.png";
|
ctx.cfg.logo = "/cgit.png";
|
||||||
ctx.cfg.favicon = "/favicon.ico";
|
ctx.cfg.favicon = "/favicon.ico";
|
||||||
ctx.cfg.local_time = 0;
|
ctx.cfg.local_time = 0;
|
||||||
|
@ -416,7 +433,7 @@ static void prepare_context(void)
|
||||||
ctx.page.modified = time(NULL);
|
ctx.page.modified = time(NULL);
|
||||||
ctx.page.expires = ctx.page.modified;
|
ctx.page.expires = ctx.page.modified;
|
||||||
ctx.page.etag = NULL;
|
ctx.page.etag = NULL;
|
||||||
string_list_init(&ctx.cfg.mimetypes, 1);
|
string_list_init_dup(&ctx.cfg.mimetypes);
|
||||||
if (ctx.env.script_name)
|
if (ctx.env.script_name)
|
||||||
ctx.cfg.script_name = xstrdup(ctx.env.script_name);
|
ctx.cfg.script_name = xstrdup(ctx.env.script_name);
|
||||||
if (ctx.env.query_string)
|
if (ctx.env.query_string)
|
||||||
|
@ -458,7 +475,8 @@ static char *find_default_branch(struct cgit_repo *repo)
|
||||||
info.req_ref = repo->defbranch;
|
info.req_ref = repo->defbranch;
|
||||||
info.first_ref = NULL;
|
info.first_ref = NULL;
|
||||||
info.match = 0;
|
info.match = 0;
|
||||||
for_each_branch_ref(find_current_ref, &info);
|
refs_for_each_branch_ref(get_main_ref_store(the_repository),
|
||||||
|
find_current_ref, &info);
|
||||||
if (info.match)
|
if (info.match)
|
||||||
ref = info.req_ref;
|
ref = info.req_ref;
|
||||||
else
|
else
|
||||||
|
@ -475,7 +493,8 @@ static char *guess_defbranch(void)
|
||||||
const char *ref, *refname;
|
const char *ref, *refname;
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
|
|
||||||
ref = resolve_ref_unsafe("HEAD", 0, &oid, NULL);
|
ref = refs_resolve_ref_unsafe(get_main_ref_store(the_repository),
|
||||||
|
"HEAD", 0, &oid, NULL);
|
||||||
if (!ref || !skip_prefix(ref, "refs/heads/", &refname))
|
if (!ref || !skip_prefix(ref, "refs/heads/", &refname))
|
||||||
return "master";
|
return "master";
|
||||||
return xstrdup(refname);
|
return xstrdup(refname);
|
||||||
|
@ -495,9 +514,11 @@ static inline void parse_readme(const char *readme, char **filename, char **ref,
|
||||||
/* Check if the readme is tracked in the git repo. */
|
/* Check if the readme is tracked in the git repo. */
|
||||||
colon = strchr(readme, ':');
|
colon = strchr(readme, ':');
|
||||||
if (colon && strlen(colon) > 1) {
|
if (colon && strlen(colon) > 1) {
|
||||||
/* If it starts with a colon, we want to use
|
/* If it starts with a colon, we want to use head given
|
||||||
* the default branch */
|
* from query or the default branch */
|
||||||
if (colon == readme && repo->defbranch)
|
if (colon == readme && ctx.qry.head)
|
||||||
|
*ref = xstrdup(ctx.qry.head);
|
||||||
|
else if (colon == readme && repo->defbranch)
|
||||||
*ref = xstrdup(repo->defbranch);
|
*ref = xstrdup(repo->defbranch);
|
||||||
else
|
else
|
||||||
*ref = xstrndup(readme, colon - readme);
|
*ref = xstrndup(readme, colon - readme);
|
||||||
|
@ -563,18 +584,13 @@ static void prepare_repo_env(int *nongit)
|
||||||
/* The path to the git repository. */
|
/* The path to the git repository. */
|
||||||
setenv("GIT_DIR", ctx.repo->path, 1);
|
setenv("GIT_DIR", ctx.repo->path, 1);
|
||||||
|
|
||||||
/* Do not look in /etc/ for gitconfig and gitattributes. */
|
|
||||||
setenv("GIT_CONFIG_NOSYSTEM", "1", 1);
|
|
||||||
setenv("GIT_ATTR_NOSYSTEM", "1", 1);
|
|
||||||
unsetenv("HOME");
|
|
||||||
unsetenv("XDG_CONFIG_HOME");
|
|
||||||
|
|
||||||
/* Setup the git directory and initialize the notes system. Both of these
|
/* Setup the git directory and initialize the notes system. Both of these
|
||||||
* load local configuration from the git repository, so we do them both while
|
* load local configuration from the git repository, so we do them both while
|
||||||
* the HOME variables are unset. */
|
* the HOME variables are unset. */
|
||||||
setup_git_directory_gently(nongit);
|
setup_git_directory_gently(nongit);
|
||||||
init_display_notes(NULL);
|
load_display_notes(NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int prepare_repo_cmd(int nongit)
|
static int prepare_repo_cmd(int nongit)
|
||||||
{
|
{
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
|
@ -619,7 +635,7 @@ static int prepare_repo_cmd(int nongit)
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (get_oid(ctx.qry.head, &oid)) {
|
if (repo_get_oid(the_repository, ctx.qry.head, &oid)) {
|
||||||
char *old_head = ctx.qry.head;
|
char *old_head = ctx.qry.head;
|
||||||
ctx.qry.head = xstrdup(ctx.repo->defbranch);
|
ctx.qry.head = xstrdup(ctx.repo->defbranch);
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
|
@ -645,7 +661,7 @@ static inline void open_auth_filter(const char *function)
|
||||||
ctx.env.https ? ctx.env.https : "",
|
ctx.env.https ? ctx.env.https : "",
|
||||||
ctx.qry.repo ? ctx.qry.repo : "",
|
ctx.qry.repo ? ctx.qry.repo : "",
|
||||||
ctx.qry.page ? ctx.qry.page : "",
|
ctx.qry.page ? ctx.qry.page : "",
|
||||||
ctx.qry.url ? ctx.qry.url : "",
|
cgit_currentfullurl(),
|
||||||
cgit_loginurl());
|
cgit_loginurl());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -809,6 +825,8 @@ static void print_repo(FILE *f, struct cgit_repo *repo)
|
||||||
fprintf(f, "repo.homepage=%s\n", repo->homepage);
|
fprintf(f, "repo.homepage=%s\n", repo->homepage);
|
||||||
if (repo->clone_url)
|
if (repo->clone_url)
|
||||||
fprintf(f, "repo.clone-url=%s\n", repo->clone_url);
|
fprintf(f, "repo.clone-url=%s\n", repo->clone_url);
|
||||||
|
fprintf(f, "repo.enable-blame=%d\n",
|
||||||
|
repo->enable_blame);
|
||||||
fprintf(f, "repo.enable-commit-graph=%d\n",
|
fprintf(f, "repo.enable-commit-graph=%d\n",
|
||||||
repo->enable_commit_graph);
|
repo->enable_commit_graph);
|
||||||
fprintf(f, "repo.enable-log-filecount=%d\n",
|
fprintf(f, "repo.enable-log-filecount=%d\n",
|
||||||
|
@ -983,9 +1001,9 @@ static void cgit_parse_args(int argc, const char **argv)
|
||||||
} else if (skip_prefix(argv[i], "--head=", &arg)) {
|
} else if (skip_prefix(argv[i], "--head=", &arg)) {
|
||||||
ctx.qry.head = xstrdup(arg);
|
ctx.qry.head = xstrdup(arg);
|
||||||
ctx.qry.has_symref = 1;
|
ctx.qry.has_symref = 1;
|
||||||
} else if (skip_prefix(argv[i], "--sha1=", &arg)) {
|
} else if (skip_prefix(argv[i], "--oid=", &arg)) {
|
||||||
ctx.qry.sha1 = xstrdup(arg);
|
ctx.qry.oid = xstrdup(arg);
|
||||||
ctx.qry.has_sha1 = 1;
|
ctx.qry.has_oid = 1;
|
||||||
} else if (skip_prefix(argv[i], "--ofs=", &arg)) {
|
} else if (skip_prefix(argv[i], "--ofs=", &arg)) {
|
||||||
ctx.qry.ofs = atoi(arg);
|
ctx.qry.ofs = atoi(arg);
|
||||||
} else if (skip_prefix(argv[i], "--scan-tree=", &arg) ||
|
} else if (skip_prefix(argv[i], "--scan-tree=", &arg) ||
|
||||||
|
@ -1028,7 +1046,7 @@ static int calc_ttl(void)
|
||||||
if (!strcmp(ctx.qry.page, "snapshot"))
|
if (!strcmp(ctx.qry.page, "snapshot"))
|
||||||
return ctx.cfg.cache_snapshot_ttl;
|
return ctx.cfg.cache_snapshot_ttl;
|
||||||
|
|
||||||
if (ctx.qry.has_sha1)
|
if (ctx.qry.has_oid)
|
||||||
return ctx.cfg.cache_static_ttl;
|
return ctx.cfg.cache_static_ttl;
|
||||||
|
|
||||||
if (ctx.qry.has_symref)
|
if (ctx.qry.has_symref)
|
||||||
|
|
6
cgit.css
6
cgit.css
|
@ -363,6 +363,10 @@ div#cgit table.blame td.lines > div > pre {
|
||||||
top: 0;
|
top: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
div#cgit table.blame .oid {
|
||||||
|
font-size: 100%;
|
||||||
|
}
|
||||||
|
|
||||||
div#cgit table.bin-blob {
|
div#cgit table.bin-blob {
|
||||||
margin-top: 0.5em;
|
margin-top: 0.5em;
|
||||||
border: solid 1px black;
|
border: solid 1px black;
|
||||||
|
@ -561,7 +565,7 @@ div#cgit table.diff td div.del {
|
||||||
color: red;
|
color: red;
|
||||||
}
|
}
|
||||||
|
|
||||||
div#cgit .sha1 {
|
div#cgit .oid {
|
||||||
font-family: monospace;
|
font-family: monospace;
|
||||||
font-size: 90%;
|
font-size: 90%;
|
||||||
}
|
}
|
||||||
|
|
46
cgit.h
46
cgit.h
|
@ -1,29 +1,35 @@
|
||||||
#ifndef CGIT_H
|
#ifndef CGIT_H
|
||||||
#define CGIT_H
|
#define CGIT_H
|
||||||
|
|
||||||
|
|
||||||
#include <git-compat-util.h>
|
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
#include <cache.h>
|
#include <git-compat-util.h>
|
||||||
#include <grep.h>
|
|
||||||
#include <object.h>
|
#include <archive.h>
|
||||||
#include <tree.h>
|
|
||||||
#include <commit.h>
|
#include <commit.h>
|
||||||
#include <tag.h>
|
|
||||||
#include <diff.h>
|
|
||||||
#include <diffcore.h>
|
#include <diffcore.h>
|
||||||
#include <argv-array.h>
|
#include <diff.h>
|
||||||
|
#include <environment.h>
|
||||||
|
#include <graph.h>
|
||||||
|
#include <grep.h>
|
||||||
|
#include <hex.h>
|
||||||
|
#include <log-tree.h>
|
||||||
|
#include <notes.h>
|
||||||
|
#include <object.h>
|
||||||
|
#include <object-name.h>
|
||||||
|
#include <object-store.h>
|
||||||
|
#include <path.h>
|
||||||
#include <refs.h>
|
#include <refs.h>
|
||||||
#include <revision.h>
|
#include <revision.h>
|
||||||
#include <log-tree.h>
|
#include <setup.h>
|
||||||
#include <archive.h>
|
|
||||||
#include <string-list.h>
|
#include <string-list.h>
|
||||||
|
#include <strvec.h>
|
||||||
|
#include <tag.h>
|
||||||
|
#include <tree.h>
|
||||||
|
#include <utf8.h>
|
||||||
|
#include <wrapper.h>
|
||||||
#include <xdiff-interface.h>
|
#include <xdiff-interface.h>
|
||||||
#include <xdiff/xdiff.h>
|
#include <xdiff/xdiff.h>
|
||||||
#include <utf8.h>
|
|
||||||
#include <notes.h>
|
|
||||||
#include <graph.h>
|
|
||||||
|
|
||||||
/* Add isgraph(x) to Git's sane ctype support (see git-compat-util.h) */
|
/* Add isgraph(x) to Git's sane ctype support (see git-compat-util.h) */
|
||||||
#undef isgraph
|
#undef isgraph
|
||||||
|
@ -63,7 +69,7 @@ typedef enum {
|
||||||
struct cgit_filter {
|
struct cgit_filter {
|
||||||
int (*open)(struct cgit_filter *, va_list ap);
|
int (*open)(struct cgit_filter *, va_list ap);
|
||||||
int (*close)(struct cgit_filter *);
|
int (*close)(struct cgit_filter *);
|
||||||
void (*fprintf)(struct cgit_filter *, FILE *, const char *prefix);
|
void (*fprintfp)(struct cgit_filter *, FILE *, const char *prefix);
|
||||||
void (*cleanup)(struct cgit_filter *);
|
void (*cleanup)(struct cgit_filter *);
|
||||||
int argument_count;
|
int argument_count;
|
||||||
};
|
};
|
||||||
|
@ -93,6 +99,7 @@ struct cgit_repo {
|
||||||
char *logo_link;
|
char *logo_link;
|
||||||
char *snapshot_prefix;
|
char *snapshot_prefix;
|
||||||
int snapshots;
|
int snapshots;
|
||||||
|
int enable_blame;
|
||||||
int enable_commit_graph;
|
int enable_commit_graph;
|
||||||
int enable_log_filecount;
|
int enable_log_filecount;
|
||||||
int enable_log_linecount;
|
int enable_log_linecount;
|
||||||
|
@ -162,7 +169,7 @@ struct reflist {
|
||||||
|
|
||||||
struct cgit_query {
|
struct cgit_query {
|
||||||
int has_symref;
|
int has_symref;
|
||||||
int has_sha1;
|
int has_oid;
|
||||||
int has_difftype;
|
int has_difftype;
|
||||||
char *raw;
|
char *raw;
|
||||||
char *repo;
|
char *repo;
|
||||||
|
@ -170,8 +177,8 @@ struct cgit_query {
|
||||||
char *search;
|
char *search;
|
||||||
char *grep;
|
char *grep;
|
||||||
char *head;
|
char *head;
|
||||||
char *sha1;
|
char *oid;
|
||||||
char *sha2;
|
char *oid2;
|
||||||
char *path;
|
char *path;
|
||||||
char *name;
|
char *name;
|
||||||
char *url;
|
char *url;
|
||||||
|
@ -193,7 +200,6 @@ struct cgit_config {
|
||||||
char *cache_root;
|
char *cache_root;
|
||||||
char *clone_prefix;
|
char *clone_prefix;
|
||||||
char *clone_url;
|
char *clone_url;
|
||||||
char *css;
|
|
||||||
char *favicon;
|
char *favicon;
|
||||||
char *footer;
|
char *footer;
|
||||||
char *head_include;
|
char *head_include;
|
||||||
|
@ -204,6 +210,7 @@ struct cgit_config {
|
||||||
char *module_link;
|
char *module_link;
|
||||||
char *project_list;
|
char *project_list;
|
||||||
struct string_list readme;
|
struct string_list readme;
|
||||||
|
struct string_list css;
|
||||||
char *robots;
|
char *robots;
|
||||||
char *root_title;
|
char *root_title;
|
||||||
char *root_desc;
|
char *root_desc;
|
||||||
|
@ -262,6 +269,7 @@ struct cgit_config {
|
||||||
int branch_sort;
|
int branch_sort;
|
||||||
int commit_sort;
|
int commit_sort;
|
||||||
struct string_list mimetypes;
|
struct string_list mimetypes;
|
||||||
|
struct string_list js;
|
||||||
struct cgit_filter *about_filter;
|
struct cgit_filter *about_filter;
|
||||||
struct cgit_filter *commit_filter;
|
struct cgit_filter *commit_filter;
|
||||||
struct cgit_filter *source_filter;
|
struct cgit_filter *source_filter;
|
||||||
|
|
68
cgit.js
Normal file
68
cgit.js
Normal file
|
@ -0,0 +1,68 @@
|
||||||
|
/* cgit.js: javacript functions for cgit
|
||||||
|
*
|
||||||
|
* Copyright (C) 2006-2018 cgit Development Team <cgit@lists.zx2c4.com>
|
||||||
|
*
|
||||||
|
* Licensed under GNU General Public License v2
|
||||||
|
* (see COPYING for full license text)
|
||||||
|
*/
|
||||||
|
|
||||||
|
(function () {
|
||||||
|
|
||||||
|
/* This follows the logic and suffixes used in ui-shared.c */
|
||||||
|
|
||||||
|
var age_classes = [ "age-mins", "age-hours", "age-days", "age-weeks", "age-months", "age-years" ];
|
||||||
|
var age_suffix = [ "min.", "hours", "days", "weeks", "months", "years", "years" ];
|
||||||
|
var age_next = [ 60, 3600, 24 * 3600, 7 * 24 * 3600, 30 * 24 * 3600, 365 * 24 * 3600, 365 * 24 * 3600 ];
|
||||||
|
var age_limit = [ 7200, 24 * 7200, 7 * 24 * 7200, 30 * 24 * 7200, 365 * 25 * 7200, 365 * 25 * 7200 ];
|
||||||
|
var update_next = [ 10, 5 * 60, 1800, 24 * 3600, 24 * 3600, 24 * 3600, 24 * 3600 ];
|
||||||
|
|
||||||
|
function render_age(e, age) {
|
||||||
|
var t, n;
|
||||||
|
|
||||||
|
for (n = 0; n < age_classes.length; n++)
|
||||||
|
if (age < age_limit[n])
|
||||||
|
break;
|
||||||
|
|
||||||
|
t = Math.round(age / age_next[n]) + " " + age_suffix[n];
|
||||||
|
|
||||||
|
if (e.textContent != t) {
|
||||||
|
e.textContent = t;
|
||||||
|
if (n == age_classes.length)
|
||||||
|
n--;
|
||||||
|
if (e.className != age_classes[n])
|
||||||
|
e.className = age_classes[n];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function aging() {
|
||||||
|
var n, next = 24 * 3600,
|
||||||
|
now_ut = Math.round((new Date().getTime() / 1000));
|
||||||
|
|
||||||
|
for (n = 0; n < age_classes.length; n++) {
|
||||||
|
var m, elems = document.getElementsByClassName(age_classes[n]);
|
||||||
|
|
||||||
|
if (elems.length && update_next[n] < next)
|
||||||
|
next = update_next[n];
|
||||||
|
|
||||||
|
for (m = 0; m < elems.length; m++) {
|
||||||
|
var age = now_ut - elems[m].getAttribute("data-ut");
|
||||||
|
|
||||||
|
render_age(elems[m], age);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* We only need to come back when the age might have changed.
|
||||||
|
* Eg, if everything is counted in hours already, once per
|
||||||
|
* 5 minutes is accurate enough.
|
||||||
|
*/
|
||||||
|
|
||||||
|
window.setTimeout(aging, next * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", function() {
|
||||||
|
/* we can do the aging on DOM content load since no layout dependency */
|
||||||
|
aging();
|
||||||
|
}, false);
|
||||||
|
|
||||||
|
})();
|
34
cgitrc.5.txt
34
cgitrc.5.txt
|
@ -126,7 +126,8 @@ commit-sort::
|
||||||
|
|
||||||
css::
|
css::
|
||||||
Url which specifies the css document to include in all cgit pages.
|
Url which specifies the css document to include in all cgit pages.
|
||||||
Default value: "/cgit.css".
|
Default value: "/cgit.css". May be given multiple times, each
|
||||||
|
css URL path is added in the head section of the document in turn.
|
||||||
|
|
||||||
email-filter::
|
email-filter::
|
||||||
Specifies a command which will be invoked to format names and email
|
Specifies a command which will be invoked to format names and email
|
||||||
|
@ -238,6 +239,11 @@ include::
|
||||||
Name of a configfile to include before the rest of the current config-
|
Name of a configfile to include before the rest of the current config-
|
||||||
file is parsed. Default value: none. See also: "MACRO EXPANSION".
|
file is parsed. Default value: none. See also: "MACRO EXPANSION".
|
||||||
|
|
||||||
|
js::
|
||||||
|
Url which specifies the javascript script document to include in all cgit
|
||||||
|
pages. Default value: "/cgit.js". Setting this to an empty string will
|
||||||
|
disable generation of the link to this file in the head section.
|
||||||
|
|
||||||
local-time::
|
local-time::
|
||||||
Flag which, if set to "1", makes cgit print commit and tag times in the
|
Flag which, if set to "1", makes cgit print commit and tag times in the
|
||||||
servers timezone. Default value: "0".
|
servers timezone. Default value: "0".
|
||||||
|
@ -269,7 +275,8 @@ max-message-length::
|
||||||
|
|
||||||
max-repo-count::
|
max-repo-count::
|
||||||
Specifies the number of entries to list per page on the repository
|
Specifies the number of entries to list per page on the repository
|
||||||
index page. Default value: "50".
|
index page. The value "0" shows all repositories without limitation.
|
||||||
|
Default value: "50".
|
||||||
|
|
||||||
max-repodesc-length::
|
max-repodesc-length::
|
||||||
Specifies the maximum number of repo description characters to display
|
Specifies the maximum number of repo description characters to display
|
||||||
|
@ -407,9 +414,12 @@ side-by-side-diffs::
|
||||||
snapshots::
|
snapshots::
|
||||||
Text which specifies the default set of snapshot formats that cgit
|
Text which specifies the default set of snapshot formats that cgit
|
||||||
generates links for. The value is a space-separated list of zero or
|
generates links for. The value is a space-separated list of zero or
|
||||||
more of the values "tar", "tar.gz", "tar.bz2", "tar.xz" and "zip".
|
more of the values "tar", "tar.gz", "tar.bz2", "tar.lz", "tar.xz",
|
||||||
The special value "all" enables all snapshot formats.
|
"tar.zst" and "zip". The special value "all" enables all snapshot
|
||||||
Default value: none.
|
formats. Default value: none.
|
||||||
|
All compressors use default settings. Some settings can be influenced
|
||||||
|
with environment variables, for example set ZSTD_CLEVEL=10 in web
|
||||||
|
server environment for higher (but slower) zstd compression.
|
||||||
|
|
||||||
source-filter::
|
source-filter::
|
||||||
Specifies a command which will be invoked to format plaintext blobs
|
Specifies a command which will be invoked to format plaintext blobs
|
||||||
|
@ -485,6 +495,10 @@ repo.email-filter::
|
||||||
Override the default email-filter. Default value: none. See also:
|
Override the default email-filter. Default value: none. See also:
|
||||||
"enable-filter-overrides". See also: "FILTER API".
|
"enable-filter-overrides". See also: "FILTER API".
|
||||||
|
|
||||||
|
repo.enable-blame::
|
||||||
|
A flag which can be used to disable the global setting
|
||||||
|
`enable-blame'. Default value: none.
|
||||||
|
|
||||||
repo.enable-commit-graph::
|
repo.enable-commit-graph::
|
||||||
A flag which can be used to disable the global setting
|
A flag which can be used to disable the global setting
|
||||||
`enable-commit-graph'. Default value: none.
|
`enable-commit-graph'. Default value: none.
|
||||||
|
@ -572,11 +586,11 @@ repo.readme::
|
||||||
verbatim as the "About" page for this repo. You may also specify a
|
verbatim as the "About" page for this repo. You may also specify a
|
||||||
git refspec by head or by hash by prepending the refspec followed by
|
git refspec by head or by hash by prepending the refspec followed by
|
||||||
a colon. For example, "master:docs/readme.mkd". If the value begins
|
a colon. For example, "master:docs/readme.mkd". If the value begins
|
||||||
with a colon, i.e. ":docs/readme.rst", the default branch of the
|
with a colon, i.e. ":docs/readme.rst", the head giving in query or
|
||||||
repository will be used. Sharing any file will expose that entire
|
the default branch of the repository will be used. Sharing any file
|
||||||
directory tree to the "/about/PATH" endpoints, so be sure that there
|
will expose that entire directory tree to the "/about/PATH" endpoints,
|
||||||
are no non-public files located in the same directory as the readme
|
so be sure that there are no non-public files located in the same
|
||||||
file. Default value: <readme>.
|
directory as the readme file. Default value: <readme>.
|
||||||
|
|
||||||
repo.section::
|
repo.section::
|
||||||
Override the current section name for this repository. Default value:
|
Override the current section name for this repository. Default value:
|
||||||
|
|
20
cmd.c
20
cmd.c
|
@ -66,7 +66,7 @@ static void about_fn(void)
|
||||||
|
|
||||||
static void blame_fn(void)
|
static void blame_fn(void)
|
||||||
{
|
{
|
||||||
if (ctx.cfg.enable_blame)
|
if (ctx.repo->enable_blame)
|
||||||
cgit_print_blame();
|
cgit_print_blame();
|
||||||
else
|
else
|
||||||
cgit_print_error_page(403, "Forbidden", "Blame is disabled");
|
cgit_print_error_page(403, "Forbidden", "Blame is disabled");
|
||||||
|
@ -74,22 +74,22 @@ static void blame_fn(void)
|
||||||
|
|
||||||
static void blob_fn(void)
|
static void blob_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_blob(ctx.qry.sha1, ctx.qry.path, ctx.qry.head, 0);
|
cgit_print_blob(ctx.qry.oid, ctx.qry.path, ctx.qry.head, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void commit_fn(void)
|
static void commit_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_commit(ctx.qry.sha1, ctx.qry.path);
|
cgit_print_commit(ctx.qry.oid, ctx.qry.path);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void diff_fn(void)
|
static void diff_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_diff(ctx.qry.sha1, ctx.qry.sha2, ctx.qry.path, 1, 0);
|
cgit_print_diff(ctx.qry.oid, ctx.qry.oid2, ctx.qry.path, 1, 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void rawdiff_fn(void)
|
static void rawdiff_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_diff(ctx.qry.sha1, ctx.qry.sha2, ctx.qry.path, 1, 1);
|
cgit_print_diff(ctx.qry.oid, ctx.qry.oid2, ctx.qry.path, 1, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void info_fn(void)
|
static void info_fn(void)
|
||||||
|
@ -99,7 +99,7 @@ static void info_fn(void)
|
||||||
|
|
||||||
static void log_fn(void)
|
static void log_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_log(ctx.qry.sha1, ctx.qry.ofs, ctx.cfg.max_commit_count,
|
cgit_print_log(ctx.qry.oid, ctx.qry.ofs, ctx.cfg.max_commit_count,
|
||||||
ctx.qry.grep, ctx.qry.search, ctx.qry.path, 1,
|
ctx.qry.grep, ctx.qry.search, ctx.qry.path, 1,
|
||||||
ctx.repo->enable_commit_graph,
|
ctx.repo->enable_commit_graph,
|
||||||
ctx.repo->commit_sort);
|
ctx.repo->commit_sort);
|
||||||
|
@ -125,7 +125,7 @@ static void repolist_fn(void)
|
||||||
|
|
||||||
static void patch_fn(void)
|
static void patch_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_patch(ctx.qry.sha1, ctx.qry.sha2, ctx.qry.path);
|
cgit_print_patch(ctx.qry.oid, ctx.qry.oid2, ctx.qry.path);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void plain_fn(void)
|
static void plain_fn(void)
|
||||||
|
@ -140,7 +140,7 @@ static void refs_fn(void)
|
||||||
|
|
||||||
static void snapshot_fn(void)
|
static void snapshot_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_snapshot(ctx.qry.head, ctx.qry.sha1, ctx.qry.path,
|
cgit_print_snapshot(ctx.qry.head, ctx.qry.oid, ctx.qry.path,
|
||||||
ctx.qry.nohead);
|
ctx.qry.nohead);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,12 +156,12 @@ static void summary_fn(void)
|
||||||
|
|
||||||
static void tag_fn(void)
|
static void tag_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_tag(ctx.qry.sha1);
|
cgit_print_tag(ctx.qry.oid);
|
||||||
}
|
}
|
||||||
|
|
||||||
static void tree_fn(void)
|
static void tree_fn(void)
|
||||||
{
|
{
|
||||||
cgit_print_tree(ctx.qry.sha1, ctx.qry.path);
|
cgit_print_tree(ctx.qry.oid, ctx.qry.path);
|
||||||
}
|
}
|
||||||
|
|
||||||
#define def_cmd(name, want_repo, want_vpath, is_clone) \
|
#define def_cmd(name, want_repo, want_vpath, is_clone) \
|
||||||
|
|
6
filter.c
6
filter.c
|
@ -128,7 +128,7 @@ void cgit_exec_filter_init(struct cgit_exec_filter *filter, char *cmd, char **ar
|
||||||
memset(filter, 0, sizeof(*filter));
|
memset(filter, 0, sizeof(*filter));
|
||||||
filter->base.open = open_exec_filter;
|
filter->base.open = open_exec_filter;
|
||||||
filter->base.close = close_exec_filter;
|
filter->base.close = close_exec_filter;
|
||||||
filter->base.fprintf = fprintf_exec_filter;
|
filter->base.fprintfp = fprintf_exec_filter;
|
||||||
filter->base.cleanup = cleanup_exec_filter;
|
filter->base.cleanup = cleanup_exec_filter;
|
||||||
filter->cmd = cmd;
|
filter->cmd = cmd;
|
||||||
filter->argv = argv;
|
filter->argv = argv;
|
||||||
|
@ -353,7 +353,7 @@ static struct cgit_filter *new_lua_filter(const char *cmd, int argument_count)
|
||||||
memset(filter, 0, sizeof(*filter));
|
memset(filter, 0, sizeof(*filter));
|
||||||
filter->base.open = open_lua_filter;
|
filter->base.open = open_lua_filter;
|
||||||
filter->base.close = close_lua_filter;
|
filter->base.close = close_lua_filter;
|
||||||
filter->base.fprintf = fprintf_lua_filter;
|
filter->base.fprintfp = fprintf_lua_filter;
|
||||||
filter->base.cleanup = cleanup_lua_filter;
|
filter->base.cleanup = cleanup_lua_filter;
|
||||||
filter->base.argument_count = argument_count;
|
filter->base.argument_count = argument_count;
|
||||||
filter->script_file = xstrdup(cmd);
|
filter->script_file = xstrdup(cmd);
|
||||||
|
@ -385,7 +385,7 @@ int cgit_close_filter(struct cgit_filter *filter)
|
||||||
|
|
||||||
void cgit_fprintf_filter(struct cgit_filter *filter, FILE *f, const char *prefix)
|
void cgit_fprintf_filter(struct cgit_filter *filter, FILE *f, const char *prefix)
|
||||||
{
|
{
|
||||||
filter->fprintf(filter, f, prefix);
|
filter->fprintfp(filter, f, prefix);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -19,7 +19,7 @@ regex=''
|
||||||
|
|
||||||
# This expression generates links to commits referenced by their SHA1.
|
# This expression generates links to commits referenced by their SHA1.
|
||||||
regex=$regex'
|
regex=$regex'
|
||||||
s|\b([0-9a-fA-F]{7,40})\b|<a href="./?id=\1">\1</a>|g'
|
s|\b([0-9a-fA-F]{7,64})\b|<a href="./?id=\1">\1</a>|g'
|
||||||
|
|
||||||
# This expression generates links to a fictional bugtracker.
|
# This expression generates links to a fictional bugtracker.
|
||||||
regex=$regex'
|
regex=$regex'
|
||||||
|
|
|
@ -3,15 +3,24 @@
|
||||||
-- prefix in filters. It is much faster than the corresponding python script.
|
-- prefix in filters. It is much faster than the corresponding python script.
|
||||||
--
|
--
|
||||||
-- Requirements:
|
-- Requirements:
|
||||||
-- luacrypto >= 0.3
|
-- luaossl
|
||||||
-- <http://mkottman.github.io/luacrypto/>
|
-- <http://25thandclement.com/~william/projects/luaossl.html>
|
||||||
--
|
--
|
||||||
|
|
||||||
local crypto = require("crypto")
|
local digest = require("openssl.digest")
|
||||||
|
|
||||||
|
function md5_hex(input)
|
||||||
|
local b = digest.new("md5"):final(input)
|
||||||
|
local x = ""
|
||||||
|
for i = 1, #b do
|
||||||
|
x = x .. string.format("%.2x", string.byte(b, i))
|
||||||
|
end
|
||||||
|
return x
|
||||||
|
end
|
||||||
|
|
||||||
function filter_open(email, page)
|
function filter_open(email, page)
|
||||||
buffer = ""
|
buffer = ""
|
||||||
md5 = crypto.digest("md5", email:sub(2, -2):lower())
|
md5 = md5_hex(email:sub(2, -2):lower())
|
||||||
end
|
end
|
||||||
|
|
||||||
function filter_close()
|
function filter_close()
|
||||||
|
|
|
@ -3,15 +3,24 @@
|
||||||
-- prefix in filters.
|
-- prefix in filters.
|
||||||
--
|
--
|
||||||
-- Requirements:
|
-- Requirements:
|
||||||
-- luacrypto >= 0.3
|
-- luaossl
|
||||||
-- <http://mkottman.github.io/luacrypto/>
|
-- <http://25thandclement.com/~william/projects/luaossl.html>
|
||||||
--
|
--
|
||||||
|
|
||||||
local crypto = require("crypto")
|
local digest = require("openssl.digest")
|
||||||
|
|
||||||
|
function md5_hex(input)
|
||||||
|
local b = digest.new("md5"):final(input)
|
||||||
|
local x = ""
|
||||||
|
for i = 1, #b do
|
||||||
|
x = x .. string.format("%.2x", string.byte(b, i))
|
||||||
|
end
|
||||||
|
return x
|
||||||
|
end
|
||||||
|
|
||||||
function filter_open(email, page)
|
function filter_open(email, page)
|
||||||
buffer = ""
|
buffer = ""
|
||||||
md5 = crypto.digest("md5", email:sub(2, -2):lower())
|
md5 = md5_hex(email:sub(2, -2):lower())
|
||||||
end
|
end
|
||||||
|
|
||||||
function filter_close()
|
function filter_close()
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
-- This script may be used with the auth-filter.
|
-- This script may be used with the auth-filter.
|
||||||
--
|
--
|
||||||
-- Requirements:
|
-- Requirements:
|
||||||
-- luacrypto >= 0.3
|
-- luaossl
|
||||||
-- <http://mkottman.github.io/luacrypto/>
|
-- <http://25thandclement.com/~william/projects/luaossl.html>
|
||||||
-- luaposix
|
-- luaposix
|
||||||
-- <https://github.com/luaposix/luaposix>
|
-- <https://github.com/luaposix/luaposix>
|
||||||
--
|
--
|
||||||
local sysstat = require("posix.sys.stat")
|
local sysstat = require("posix.sys.stat")
|
||||||
local unistd = require("posix.unistd")
|
local unistd = require("posix.unistd")
|
||||||
local crypto = require("crypto")
|
local rand = require("openssl.rand")
|
||||||
|
local hmac = require("openssl.hmac")
|
||||||
|
|
||||||
-- This file should contain a series of lines in the form of:
|
-- This file should contain a series of lines in the form of:
|
||||||
-- username1:hash1
|
-- username1:hash1
|
||||||
|
@ -225,6 +225,13 @@ function get_cookie(cookies, name)
|
||||||
return url_decode(string.match(cookies, ";" .. name .. "=(.-);"))
|
return url_decode(string.match(cookies, ";" .. name .. "=(.-);"))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function tohex(b)
|
||||||
|
local x = ""
|
||||||
|
for i = 1, #b do
|
||||||
|
x = x .. string.format("%.2x", string.byte(b, i))
|
||||||
|
end
|
||||||
|
return x
|
||||||
|
end
|
||||||
|
|
||||||
--
|
--
|
||||||
--
|
--
|
||||||
|
@ -242,12 +249,12 @@ function get_secret()
|
||||||
local secret_file = io.open(secret_filename, "r")
|
local secret_file = io.open(secret_filename, "r")
|
||||||
if secret_file == nil then
|
if secret_file == nil then
|
||||||
local old_umask = sysstat.umask(63)
|
local old_umask = sysstat.umask(63)
|
||||||
local temporary_filename = secret_filename .. ".tmp." .. crypto.hex(crypto.rand.bytes(16))
|
local temporary_filename = secret_filename .. ".tmp." .. tohex(rand.bytes(16))
|
||||||
local temporary_file = io.open(temporary_filename, "w")
|
local temporary_file = io.open(temporary_filename, "w")
|
||||||
if temporary_file == nil then
|
if temporary_file == nil then
|
||||||
os.exit(177)
|
os.exit(177)
|
||||||
end
|
end
|
||||||
temporary_file:write(crypto.hex(crypto.rand.bytes(32)))
|
temporary_file:write(tohex(rand.bytes(32)))
|
||||||
temporary_file:close()
|
temporary_file:close()
|
||||||
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
||||||
unistd.unlink(temporary_filename)
|
unistd.unlink(temporary_filename)
|
||||||
|
@ -272,7 +279,7 @@ function validate_value(expected_field, cookie)
|
||||||
local field = ""
|
local field = ""
|
||||||
local expiration = 0
|
local expiration = 0
|
||||||
local salt = ""
|
local salt = ""
|
||||||
local hmac = ""
|
local chmac = ""
|
||||||
|
|
||||||
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
||||||
return nil
|
return nil
|
||||||
|
@ -291,19 +298,19 @@ function validate_value(expected_field, cookie)
|
||||||
elseif i == 3 then
|
elseif i == 3 then
|
||||||
salt = component
|
salt = component
|
||||||
elseif i == 4 then
|
elseif i == 4 then
|
||||||
hmac = component
|
chmac = component
|
||||||
else
|
else
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
i = i + 1
|
i = i + 1
|
||||||
end
|
end
|
||||||
|
|
||||||
if hmac == nil or hmac:len() == 0 then
|
if chmac == nil or chmac:len() == 0 then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Lua hashes strings, so these comparisons are time invariant.
|
-- Lua hashes strings, so these comparisons are time invariant.
|
||||||
if hmac ~= crypto.hmac.digest("sha256", field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt, get_secret()) then
|
if chmac ~= tohex(hmac.new(get_secret(), "sha256"):final(field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt)) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -324,11 +331,11 @@ function secure_value(field, value, expiration)
|
||||||
end
|
end
|
||||||
|
|
||||||
local authstr = ""
|
local authstr = ""
|
||||||
local salt = crypto.hex(crypto.rand.bytes(16))
|
local salt = tohex(rand.bytes(16))
|
||||||
value = url_encode(value)
|
value = url_encode(value)
|
||||||
field = url_encode(field)
|
field = url_encode(field)
|
||||||
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
||||||
authstr = authstr .. "|" .. crypto.hmac.digest("sha256", authstr, get_secret())
|
authstr = authstr .. "|" .. tohex(hmac.new(get_secret(), "sha256"):final(authstr))
|
||||||
return authstr
|
return authstr
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -1,8 +1,8 @@
|
||||||
-- This script may be used with the auth-filter. Be sure to configure it as you wish.
|
-- This script may be used with the auth-filter. Be sure to configure it as you wish.
|
||||||
--
|
--
|
||||||
-- Requirements:
|
-- Requirements:
|
||||||
-- luacrypto >= 0.3
|
-- luaossl
|
||||||
-- <http://mkottman.github.io/luacrypto/>
|
-- <http://25thandclement.com/~william/projects/luaossl.html>
|
||||||
-- lualdap >= 1.2
|
-- lualdap >= 1.2
|
||||||
-- <https://git.zx2c4.com/lualdap/about/>
|
-- <https://git.zx2c4.com/lualdap/about/>
|
||||||
-- luaposix
|
-- luaposix
|
||||||
|
@ -10,9 +10,9 @@
|
||||||
--
|
--
|
||||||
local sysstat = require("posix.sys.stat")
|
local sysstat = require("posix.sys.stat")
|
||||||
local unistd = require("posix.unistd")
|
local unistd = require("posix.unistd")
|
||||||
local crypto = require("crypto")
|
|
||||||
local lualdap = require("lualdap")
|
local lualdap = require("lualdap")
|
||||||
|
local rand = require("openssl.rand")
|
||||||
|
local hmac = require("openssl.hmac")
|
||||||
|
|
||||||
--
|
--
|
||||||
--
|
--
|
||||||
|
@ -226,6 +226,13 @@ function get_cookie(cookies, name)
|
||||||
return string.match(cookies, ";" .. name .. "=(.-);")
|
return string.match(cookies, ";" .. name .. "=(.-);")
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function tohex(b)
|
||||||
|
local x = ""
|
||||||
|
for i = 1, #b do
|
||||||
|
x = x .. string.format("%.2x", string.byte(b, i))
|
||||||
|
end
|
||||||
|
return x
|
||||||
|
end
|
||||||
|
|
||||||
--
|
--
|
||||||
--
|
--
|
||||||
|
@ -243,12 +250,12 @@ function get_secret()
|
||||||
local secret_file = io.open(secret_filename, "r")
|
local secret_file = io.open(secret_filename, "r")
|
||||||
if secret_file == nil then
|
if secret_file == nil then
|
||||||
local old_umask = sysstat.umask(63)
|
local old_umask = sysstat.umask(63)
|
||||||
local temporary_filename = secret_filename .. ".tmp." .. crypto.hex(crypto.rand.bytes(16))
|
local temporary_filename = secret_filename .. ".tmp." .. tohex(rand.bytes(16))
|
||||||
local temporary_file = io.open(temporary_filename, "w")
|
local temporary_file = io.open(temporary_filename, "w")
|
||||||
if temporary_file == nil then
|
if temporary_file == nil then
|
||||||
os.exit(177)
|
os.exit(177)
|
||||||
end
|
end
|
||||||
temporary_file:write(crypto.hex(crypto.rand.bytes(32)))
|
temporary_file:write(tohex(rand.bytes(32)))
|
||||||
temporary_file:close()
|
temporary_file:close()
|
||||||
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
||||||
unistd.unlink(temporary_filename)
|
unistd.unlink(temporary_filename)
|
||||||
|
@ -273,7 +280,7 @@ function validate_value(expected_field, cookie)
|
||||||
local field = ""
|
local field = ""
|
||||||
local expiration = 0
|
local expiration = 0
|
||||||
local salt = ""
|
local salt = ""
|
||||||
local hmac = ""
|
local chmac = ""
|
||||||
|
|
||||||
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
||||||
return nil
|
return nil
|
||||||
|
@ -292,19 +299,19 @@ function validate_value(expected_field, cookie)
|
||||||
elseif i == 3 then
|
elseif i == 3 then
|
||||||
salt = component
|
salt = component
|
||||||
elseif i == 4 then
|
elseif i == 4 then
|
||||||
hmac = component
|
chmac = component
|
||||||
else
|
else
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
i = i + 1
|
i = i + 1
|
||||||
end
|
end
|
||||||
|
|
||||||
if hmac == nil or hmac:len() == 0 then
|
if chmac == nil or chmac:len() == 0 then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Lua hashes strings, so these comparisons are time invariant.
|
-- Lua hashes strings, so these comparisons are time invariant.
|
||||||
if hmac ~= crypto.hmac.digest("sha256", field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt, get_secret()) then
|
if chmac ~= tohex(hmac.new(get_secret(), "sha256"):final(field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt)) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -325,11 +332,11 @@ function secure_value(field, value, expiration)
|
||||||
end
|
end
|
||||||
|
|
||||||
local authstr = ""
|
local authstr = ""
|
||||||
local salt = crypto.hex(crypto.rand.bytes(16))
|
local salt = tohex(rand.bytes(16))
|
||||||
value = url_encode(value)
|
value = url_encode(value)
|
||||||
field = url_encode(field)
|
field = url_encode(field)
|
||||||
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
||||||
authstr = authstr .. "|" .. crypto.hmac.digest("sha256", authstr, get_secret())
|
authstr = authstr .. "|" .. tohex(hmac.new(get_secret(), "sha256"):final(authstr))
|
||||||
return authstr
|
return authstr
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
|
@ -3,6 +3,7 @@ import markdown
|
||||||
import sys
|
import sys
|
||||||
import io
|
import io
|
||||||
from pygments.formatters import HtmlFormatter
|
from pygments.formatters import HtmlFormatter
|
||||||
|
from markdown.extensions.toc import TocExtension
|
||||||
sys.stdin = io.TextIOWrapper(sys.stdin.buffer, encoding='utf-8')
|
sys.stdin = io.TextIOWrapper(sys.stdin.buffer, encoding='utf-8')
|
||||||
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
|
sys.stdout = io.TextIOWrapper(sys.stdout.buffer, encoding='utf-8')
|
||||||
sys.stdout.write('''
|
sys.stdout.write('''
|
||||||
|
@ -48,10 +49,14 @@ sys.stdout.write('''
|
||||||
line-height: 1;
|
line-height: 1;
|
||||||
padding-left: 0;
|
padding-left: 0;
|
||||||
margin-left: -22px;
|
margin-left: -22px;
|
||||||
top: 15%}
|
top: 15%;
|
||||||
|
}
|
||||||
.markdown-body h1:hover a.anchor .mini-icon-link, .markdown-body h2:hover a.anchor .mini-icon-link, .markdown-body h3:hover a.anchor .mini-icon-link, .markdown-body h4:hover a.anchor .mini-icon-link, .markdown-body h5:hover a.anchor .mini-icon-link, .markdown-body h6:hover a.anchor .mini-icon-link {
|
.markdown-body h1:hover a.anchor .mini-icon-link, .markdown-body h2:hover a.anchor .mini-icon-link, .markdown-body h3:hover a.anchor .mini-icon-link, .markdown-body h4:hover a.anchor .mini-icon-link, .markdown-body h5:hover a.anchor .mini-icon-link, .markdown-body h6:hover a.anchor .mini-icon-link {
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
div#cgit .markdown-body h1 a.toclink, div#cgit .markdown-body h2 a.toclink, div#cgit .markdown-body h3 a.toclink, div#cgit .markdown-body h4 a.toclink, div#cgit .markdown-body h5 a.toclink, div#cgit .markdown-body h6 a.toclink {
|
||||||
|
color: black;
|
||||||
|
}
|
||||||
.markdown-body h1 tt, .markdown-body h1 code, .markdown-body h2 tt, .markdown-body h2 code, .markdown-body h3 tt, .markdown-body h3 code, .markdown-body h4 tt, .markdown-body h4 code, .markdown-body h5 tt, .markdown-body h5 code, .markdown-body h6 tt, .markdown-body h6 code {
|
.markdown-body h1 tt, .markdown-body h1 code, .markdown-body h2 tt, .markdown-body h2 code, .markdown-body h3 tt, .markdown-body h3 code, .markdown-body h4 tt, .markdown-body h4 code, .markdown-body h5 tt, .markdown-body h5 code, .markdown-body h6 tt, .markdown-body h6 code {
|
||||||
font-size: inherit;
|
font-size: inherit;
|
||||||
}
|
}
|
||||||
|
@ -81,11 +86,7 @@ sys.stdout.write('''
|
||||||
margin: 15px 0;
|
margin: 15px 0;
|
||||||
}
|
}
|
||||||
.markdown-body hr {
|
.markdown-body hr {
|
||||||
background: transparent url("/dirty-shade.png") repeat-x 0 0;
|
border: 2px solid #ccc;
|
||||||
border: 0 none;
|
|
||||||
color: #ccc;
|
|
||||||
height: 4px;
|
|
||||||
padding: 0;
|
|
||||||
}
|
}
|
||||||
.markdown-body>h2:first-child, .markdown-body>h1:first-child, .markdown-body>h1:first-child+h2, .markdown-body>h3:first-child, .markdown-body>h4:first-child, .markdown-body>h5:first-child, .markdown-body>h6:first-child {
|
.markdown-body>h2:first-child, .markdown-body>h1:first-child, .markdown-body>h1:first-child+h2, .markdown-body>h3:first-child, .markdown-body>h4:first-child, .markdown-body>h5:first-child, .markdown-body>h6:first-child {
|
||||||
margin-top: 0;
|
margin-top: 0;
|
||||||
|
@ -290,5 +291,14 @@ sys.stdout.write('''
|
||||||
sys.stdout.write("<div class='markdown-body'>")
|
sys.stdout.write("<div class='markdown-body'>")
|
||||||
sys.stdout.flush()
|
sys.stdout.flush()
|
||||||
# Note: you may want to run this through bleach for sanitization
|
# Note: you may want to run this through bleach for sanitization
|
||||||
markdown.markdownFromFile(output_format="html5", extensions=["markdown.extensions.fenced_code", "markdown.extensions.codehilite", "markdown.extensions.tables"], extension_configs={"markdown.extensions.codehilite":{"css_class":"highlight"}})
|
markdown.markdownFromFile(
|
||||||
|
output_format="html5",
|
||||||
|
extensions=[
|
||||||
|
"markdown.extensions.fenced_code",
|
||||||
|
"markdown.extensions.codehilite",
|
||||||
|
"markdown.extensions.tables",
|
||||||
|
"markdown.extensions.sane_lists",
|
||||||
|
TocExtension(anchorlink=True)],
|
||||||
|
extension_configs={
|
||||||
|
"markdown.extensions.codehilite":{"css_class":"highlight"}})
|
||||||
sys.stdout.write("</div>")
|
sys.stdout.write("</div>")
|
||||||
|
|
|
@ -1,15 +1,15 @@
|
||||||
-- This script may be used with the auth-filter. Be sure to configure it as you wish.
|
-- This script may be used with the auth-filter. Be sure to configure it as you wish.
|
||||||
--
|
--
|
||||||
-- Requirements:
|
-- Requirements:
|
||||||
-- luacrypto >= 0.3
|
-- luaossl
|
||||||
-- <http://mkottman.github.io/luacrypto/>
|
-- <http://25thandclement.com/~william/projects/luaossl.html>
|
||||||
-- luaposix
|
-- luaposix
|
||||||
-- <https://github.com/luaposix/luaposix>
|
-- <https://github.com/luaposix/luaposix>
|
||||||
--
|
--
|
||||||
local sysstat = require("posix.sys.stat")
|
local sysstat = require("posix.sys.stat")
|
||||||
local unistd = require("posix.unistd")
|
local unistd = require("posix.unistd")
|
||||||
local crypto = require("crypto")
|
local rand = require("openssl.rand")
|
||||||
|
local hmac = require("openssl.hmac")
|
||||||
|
|
||||||
--
|
--
|
||||||
--
|
--
|
||||||
|
@ -180,6 +180,13 @@ function get_cookie(cookies, name)
|
||||||
return url_decode(string.match(cookies, ";" .. name .. "=(.-);"))
|
return url_decode(string.match(cookies, ";" .. name .. "=(.-);"))
|
||||||
end
|
end
|
||||||
|
|
||||||
|
function tohex(b)
|
||||||
|
local x = ""
|
||||||
|
for i = 1, #b do
|
||||||
|
x = x .. string.format("%.2x", string.byte(b, i))
|
||||||
|
end
|
||||||
|
return x
|
||||||
|
end
|
||||||
|
|
||||||
--
|
--
|
||||||
--
|
--
|
||||||
|
@ -197,12 +204,12 @@ function get_secret()
|
||||||
local secret_file = io.open(secret_filename, "r")
|
local secret_file = io.open(secret_filename, "r")
|
||||||
if secret_file == nil then
|
if secret_file == nil then
|
||||||
local old_umask = sysstat.umask(63)
|
local old_umask = sysstat.umask(63)
|
||||||
local temporary_filename = secret_filename .. ".tmp." .. crypto.hex(crypto.rand.bytes(16))
|
local temporary_filename = secret_filename .. ".tmp." .. tohex(rand.bytes(16))
|
||||||
local temporary_file = io.open(temporary_filename, "w")
|
local temporary_file = io.open(temporary_filename, "w")
|
||||||
if temporary_file == nil then
|
if temporary_file == nil then
|
||||||
os.exit(177)
|
os.exit(177)
|
||||||
end
|
end
|
||||||
temporary_file:write(crypto.hex(crypto.rand.bytes(32)))
|
temporary_file:write(tohex(rand.bytes(32)))
|
||||||
temporary_file:close()
|
temporary_file:close()
|
||||||
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
unistd.link(temporary_filename, secret_filename) -- Intentionally fails in the case that another process is doing the same.
|
||||||
unistd.unlink(temporary_filename)
|
unistd.unlink(temporary_filename)
|
||||||
|
@ -227,7 +234,7 @@ function validate_value(expected_field, cookie)
|
||||||
local field = ""
|
local field = ""
|
||||||
local expiration = 0
|
local expiration = 0
|
||||||
local salt = ""
|
local salt = ""
|
||||||
local hmac = ""
|
local chmac = ""
|
||||||
|
|
||||||
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
if cookie == nil or cookie:len() < 3 or cookie:sub(1, 1) == "|" then
|
||||||
return nil
|
return nil
|
||||||
|
@ -246,19 +253,19 @@ function validate_value(expected_field, cookie)
|
||||||
elseif i == 3 then
|
elseif i == 3 then
|
||||||
salt = component
|
salt = component
|
||||||
elseif i == 4 then
|
elseif i == 4 then
|
||||||
hmac = component
|
chmac = component
|
||||||
else
|
else
|
||||||
break
|
break
|
||||||
end
|
end
|
||||||
i = i + 1
|
i = i + 1
|
||||||
end
|
end
|
||||||
|
|
||||||
if hmac == nil or hmac:len() == 0 then
|
if chmac == nil or chmac:len() == 0 then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
-- Lua hashes strings, so these comparisons are time invariant.
|
-- Lua hashes strings, so these comparisons are time invariant.
|
||||||
if hmac ~= crypto.hmac.digest("sha256", field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt, get_secret()) then
|
if chmac ~= tohex(hmac.new(get_secret(), "sha256"):final(field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt)) then
|
||||||
return nil
|
return nil
|
||||||
end
|
end
|
||||||
|
|
||||||
|
@ -279,11 +286,11 @@ function secure_value(field, value, expiration)
|
||||||
end
|
end
|
||||||
|
|
||||||
local authstr = ""
|
local authstr = ""
|
||||||
local salt = crypto.hex(crypto.rand.bytes(16))
|
local salt = tohex(rand.bytes(16))
|
||||||
value = url_encode(value)
|
value = url_encode(value)
|
||||||
field = url_encode(field)
|
field = url_encode(field)
|
||||||
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
authstr = field .. "|" .. value .. "|" .. tostring(expiration) .. "|" .. salt
|
||||||
authstr = authstr .. "|" .. crypto.hmac.digest("sha256", authstr, get_secret())
|
authstr = authstr .. "|" .. tohex(hmac.new(get_secret(), "sha256"):final(authstr))
|
||||||
return authstr
|
return authstr
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
2
git
2
git
|
@ -1 +1 @@
|
||||||
Subproject commit 53f9a3e157dbbc901a02ac2c73346d375e24978c
|
Subproject commit 39bf06adf96da25b87c9aa7d35a32ef3683eb4a4
|
2
html.c
2
html.c
|
@ -59,7 +59,7 @@ char *fmt(const char *format, ...)
|
||||||
va_start(args, format);
|
va_start(args, format);
|
||||||
len = vsnprintf(buf[bufidx], sizeof(buf[bufidx]), format, args);
|
len = vsnprintf(buf[bufidx], sizeof(buf[bufidx]), format, args);
|
||||||
va_end(args);
|
va_end(args);
|
||||||
if (len > sizeof(buf[bufidx])) {
|
if (len >= sizeof(buf[bufidx])) {
|
||||||
fprintf(stderr, "[html.c] string truncated: %s\n", format);
|
fprintf(stderr, "[html.c] string truncated: %s\n", format);
|
||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
16
parsing.c
16
parsing.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
|
|
||||||
/*
|
/*
|
||||||
|
@ -63,8 +65,7 @@ static char *substr(const char *head, const char *tail)
|
||||||
if (tail < head)
|
if (tail < head)
|
||||||
return xstrdup("");
|
return xstrdup("");
|
||||||
buf = xmalloc(tail - head + 1);
|
buf = xmalloc(tail - head + 1);
|
||||||
strncpy(buf, head, tail - head);
|
strlcpy(buf, head, tail - head + 1);
|
||||||
buf[tail - head] = '\0';
|
|
||||||
return buf;
|
return buf;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -78,7 +79,7 @@ static void parse_user(const char *t, char **name, char **email, unsigned long *
|
||||||
|
|
||||||
email_len = ident.mail_end - ident.mail_begin;
|
email_len = ident.mail_end - ident.mail_begin;
|
||||||
*email = xmalloc(strlen("<") + email_len + strlen(">") + 1);
|
*email = xmalloc(strlen("<") + email_len + strlen(">") + 1);
|
||||||
sprintf(*email, "<%.*s>", email_len, ident.mail_begin);
|
xsnprintf(*email, email_len + 3, "<%.*s>", email_len, ident.mail_begin);
|
||||||
|
|
||||||
if (ident.date_begin)
|
if (ident.date_begin)
|
||||||
*date = strtoul(ident.date_begin, NULL, 10);
|
*date = strtoul(ident.date_begin, NULL, 10);
|
||||||
|
@ -128,9 +129,8 @@ static int end_of_header(const char *p)
|
||||||
|
|
||||||
struct commitinfo *cgit_parse_commit(struct commit *commit)
|
struct commitinfo *cgit_parse_commit(struct commit *commit)
|
||||||
{
|
{
|
||||||
const int sha1hex_len = 40;
|
|
||||||
struct commitinfo *ret;
|
struct commitinfo *ret;
|
||||||
const char *p = get_cached_commit_buffer(commit, NULL);
|
const char *p = repo_get_commit_buffer(the_repository, commit, NULL);
|
||||||
const char *t;
|
const char *t;
|
||||||
|
|
||||||
ret = xcalloc(1, sizeof(struct commitinfo));
|
ret = xcalloc(1, sizeof(struct commitinfo));
|
||||||
|
@ -141,10 +141,10 @@ struct commitinfo *cgit_parse_commit(struct commit *commit)
|
||||||
|
|
||||||
if (!skip_prefix(p, "tree ", &p))
|
if (!skip_prefix(p, "tree ", &p))
|
||||||
die("Bad commit: %s", oid_to_hex(&commit->object.oid));
|
die("Bad commit: %s", oid_to_hex(&commit->object.oid));
|
||||||
p += sha1hex_len + 1;
|
p += the_hash_algo->hexsz + 1;
|
||||||
|
|
||||||
while (skip_prefix(p, "parent ", &p))
|
while (skip_prefix(p, "parent ", &p))
|
||||||
p += sha1hex_len + 1;
|
p += the_hash_algo->hexsz + 1;
|
||||||
|
|
||||||
if (p && skip_prefix(p, "author ", &p)) {
|
if (p && skip_prefix(p, "author ", &p)) {
|
||||||
parse_user(p, &ret->author, &ret->author_email,
|
parse_user(p, &ret->author, &ret->author_email,
|
||||||
|
@ -200,7 +200,7 @@ struct taginfo *cgit_parse_tag(struct tag *tag)
|
||||||
const char *p;
|
const char *p;
|
||||||
struct taginfo *ret = NULL;
|
struct taginfo *ret = NULL;
|
||||||
|
|
||||||
data = read_object_file(&tag->object.oid, &type, &size);
|
data = repo_read_object_file(the_repository, &tag->object.oid, &type, &size);
|
||||||
if (!data || type != OBJ_TAG)
|
if (!data || type != OBJ_TAG)
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
User-agent: *
|
User-agent: *
|
||||||
Disallow: /*/snapshot/*
|
Disallow: /*/snapshot/*
|
||||||
|
Disallow: /*/blame/*
|
||||||
Allow: /
|
Allow: /
|
||||||
|
|
|
@ -54,7 +54,7 @@ static void scan_tree_repo_config(const char *name, const char *value)
|
||||||
config_fn(repo, name, value);
|
config_fn(repo, name, value);
|
||||||
}
|
}
|
||||||
|
|
||||||
static int gitconfig_config(const char *key, const char *value, void *cb)
|
static int gitconfig_config(const char *key, const char *value, const struct config_context *, void *cb)
|
||||||
{
|
{
|
||||||
const char *name;
|
const char *name;
|
||||||
|
|
||||||
|
|
36
shared.c
36
shared.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
|
|
||||||
struct cgit_repolist cgit_repolist;
|
struct cgit_repolist cgit_repolist;
|
||||||
|
@ -58,6 +60,7 @@ struct cgit_repo *cgit_add_repo(const char *url)
|
||||||
ret->homepage = NULL;
|
ret->homepage = NULL;
|
||||||
ret->section = ctx.cfg.section;
|
ret->section = ctx.cfg.section;
|
||||||
ret->snapshots = ctx.cfg.snapshots;
|
ret->snapshots = ctx.cfg.snapshots;
|
||||||
|
ret->enable_blame = ctx.cfg.enable_blame;
|
||||||
ret->enable_commit_graph = ctx.cfg.enable_commit_graph;
|
ret->enable_commit_graph = ctx.cfg.enable_commit_graph;
|
||||||
ret->enable_log_filecount = ctx.cfg.enable_log_filecount;
|
ret->enable_log_filecount = ctx.cfg.enable_log_filecount;
|
||||||
ret->enable_log_linecount = ctx.cfg.enable_log_linecount;
|
ret->enable_log_linecount = ctx.cfg.enable_log_linecount;
|
||||||
|
@ -161,7 +164,7 @@ static struct refinfo *cgit_mk_refinfo(const char *refname, const struct object_
|
||||||
|
|
||||||
ref = xmalloc(sizeof (struct refinfo));
|
ref = xmalloc(sizeof (struct refinfo));
|
||||||
ref->refname = xstrdup(refname);
|
ref->refname = xstrdup(refname);
|
||||||
ref->object = parse_object(oid);
|
ref->object = parse_object(the_repository, oid);
|
||||||
switch (ref->object->type) {
|
switch (ref->object->type) {
|
||||||
case OBJ_TAG:
|
case OBJ_TAG:
|
||||||
ref->tag = cgit_parse_tag((struct tag *)ref->object);
|
ref->tag = cgit_parse_tag((struct tag *)ref->object);
|
||||||
|
@ -240,7 +243,7 @@ static int load_mmfile(mmfile_t *file, const struct object_id *oid)
|
||||||
file->ptr = (char *)"";
|
file->ptr = (char *)"";
|
||||||
file->size = 0;
|
file->size = 0;
|
||||||
} else {
|
} else {
|
||||||
file->ptr = read_object_file(oid, &type,
|
file->ptr = repo_read_object_file(the_repository, oid, &type,
|
||||||
(unsigned long *)&file->size);
|
(unsigned long *)&file->size);
|
||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -325,7 +328,7 @@ int cgit_diff_files(const struct object_id *old_oid,
|
||||||
diff_params.flags |= XDF_IGNORE_WHITESPACE;
|
diff_params.flags |= XDF_IGNORE_WHITESPACE;
|
||||||
emit_params.ctxlen = context > 0 ? context : 3;
|
emit_params.ctxlen = context > 0 ? context : 3;
|
||||||
emit_params.flags = XDL_EMIT_FUNCNAMES;
|
emit_params.flags = XDL_EMIT_FUNCNAMES;
|
||||||
emit_cb.outf = filediff_cb;
|
emit_cb.out_line = filediff_cb;
|
||||||
emit_cb.priv = fn;
|
emit_cb.priv = fn;
|
||||||
xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb);
|
xdl_diff(&file1, &file2, &diff_params, &emit_params, &emit_cb);
|
||||||
if (file1.size)
|
if (file1.size)
|
||||||
|
@ -340,10 +343,9 @@ void cgit_diff_tree(const struct object_id *old_oid,
|
||||||
filepair_fn fn, const char *prefix, int ignorews)
|
filepair_fn fn, const char *prefix, int ignorews)
|
||||||
{
|
{
|
||||||
struct diff_options opt;
|
struct diff_options opt;
|
||||||
struct pathspec_item item;
|
struct pathspec_item *item;
|
||||||
|
|
||||||
memset(&item, 0, sizeof(item));
|
repo_diff_setup(the_repository, &opt);
|
||||||
diff_setup(&opt);
|
|
||||||
opt.output_format = DIFF_FORMAT_CALLBACK;
|
opt.output_format = DIFF_FORMAT_CALLBACK;
|
||||||
opt.detect_rename = 1;
|
opt.detect_rename = 1;
|
||||||
opt.rename_limit = ctx.cfg.renamelimit;
|
opt.rename_limit = ctx.cfg.renamelimit;
|
||||||
|
@ -353,10 +355,11 @@ void cgit_diff_tree(const struct object_id *old_oid,
|
||||||
opt.format_callback = cgit_diff_tree_cb;
|
opt.format_callback = cgit_diff_tree_cb;
|
||||||
opt.format_callback_data = fn;
|
opt.format_callback_data = fn;
|
||||||
if (prefix) {
|
if (prefix) {
|
||||||
item.match = xstrdup(prefix);
|
item = xcalloc(1, sizeof(*item));
|
||||||
item.len = strlen(prefix);
|
item->match = xstrdup(prefix);
|
||||||
|
item->len = strlen(prefix);
|
||||||
opt.pathspec.nr = 1;
|
opt.pathspec.nr = 1;
|
||||||
opt.pathspec.items = &item;
|
opt.pathspec.items = item;
|
||||||
}
|
}
|
||||||
diff_setup_done(&opt);
|
diff_setup_done(&opt);
|
||||||
|
|
||||||
|
@ -366,8 +369,6 @@ void cgit_diff_tree(const struct object_id *old_oid,
|
||||||
diff_root_tree_oid(new_oid, "", &opt);
|
diff_root_tree_oid(new_oid, "", &opt);
|
||||||
diffcore_std(&opt);
|
diffcore_std(&opt);
|
||||||
diff_flush(&opt);
|
diff_flush(&opt);
|
||||||
|
|
||||||
free(item.match);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_diff_commit(struct commit *commit, filepair_fn fn, const char *prefix)
|
void cgit_diff_commit(struct commit *commit, filepair_fn fn, const char *prefix)
|
||||||
|
@ -540,7 +541,9 @@ char *expand_macros(const char *txt)
|
||||||
|
|
||||||
char *get_mimetype_for_filename(const char *filename)
|
char *get_mimetype_for_filename(const char *filename)
|
||||||
{
|
{
|
||||||
char *ext, *mimetype, *token, line[1024], *saveptr;
|
char *ext, *mimetype, line[1024];
|
||||||
|
struct string_list list = STRING_LIST_INIT_NODUP;
|
||||||
|
int i;
|
||||||
FILE *file;
|
FILE *file;
|
||||||
struct string_list_item *mime;
|
struct string_list_item *mime;
|
||||||
|
|
||||||
|
@ -565,13 +568,16 @@ char *get_mimetype_for_filename(const char *filename)
|
||||||
while (fgets(line, sizeof(line), file)) {
|
while (fgets(line, sizeof(line), file)) {
|
||||||
if (!line[0] || line[0] == '#')
|
if (!line[0] || line[0] == '#')
|
||||||
continue;
|
continue;
|
||||||
mimetype = strtok_r(line, " \t\r\n", &saveptr);
|
string_list_split_in_place(&list, line, " \t\r\n", -1);
|
||||||
while ((token = strtok_r(NULL, " \t\r\n", &saveptr))) {
|
string_list_remove_empty_items(&list, 0);
|
||||||
if (!strcasecmp(ext, token)) {
|
mimetype = list.items[0].string;
|
||||||
|
for (i = 1; i < list.nr; i++) {
|
||||||
|
if (!strcasecmp(ext, list.items[i].string)) {
|
||||||
fclose(file);
|
fclose(file);
|
||||||
return xstrdup(mimetype);
|
return xstrdup(mimetype);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
string_list_clear(&list, 0);
|
||||||
}
|
}
|
||||||
fclose(file);
|
fclose(file);
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
|
@ -80,13 +80,17 @@ mkrepo() {
|
||||||
git commit -m "commit $n"
|
git commit -m "commit $n"
|
||||||
n=$(expr $n + 1)
|
n=$(expr $n + 1)
|
||||||
done
|
done
|
||||||
if test "$3" = "testplus"
|
case "$3" in
|
||||||
then
|
testplus)
|
||||||
echo "hello" >a+b
|
echo "hello" >a+b
|
||||||
git add a+b
|
git add a+b
|
||||||
git commit -m "add a+b"
|
git commit -m "add a+b"
|
||||||
git branch "1+2"
|
git branch "1+2"
|
||||||
fi
|
;;
|
||||||
|
commit-graph)
|
||||||
|
git commit-graph write
|
||||||
|
;;
|
||||||
|
esac
|
||||||
)
|
)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -95,7 +99,7 @@ setup_repos()
|
||||||
rm -rf cache
|
rm -rf cache
|
||||||
mkdir -p cache
|
mkdir -p cache
|
||||||
mkrepo repos/foo 5 >/dev/null
|
mkrepo repos/foo 5 >/dev/null
|
||||||
mkrepo repos/bar 50 >/dev/null
|
mkrepo repos/bar 50 commit-graph >/dev/null
|
||||||
mkrepo repos/foo+bar 10 testplus >/dev/null
|
mkrepo repos/foo+bar 10 testplus >/dev/null
|
||||||
mkrepo "repos/with space" 2 >/dev/null
|
mkrepo "repos/with space" 2 >/dev/null
|
||||||
mkrepo repos/filter 5 testplus >/dev/null
|
mkrepo repos/filter 5 testplus >/dev/null
|
||||||
|
@ -104,7 +108,7 @@ virtual-root=/
|
||||||
cache-root=$PWD/cache
|
cache-root=$PWD/cache
|
||||||
|
|
||||||
cache-size=1021
|
cache-size=1021
|
||||||
snapshots=tar.gz tar.bz zip
|
snapshots=tar.gz tar.bz tar.lz tar.xz tar.zst zip
|
||||||
enable-log-filecount=1
|
enable-log-filecount=1
|
||||||
enable-log-linecount=1
|
enable-log-linecount=1
|
||||||
summary-log=5
|
summary-log=5
|
||||||
|
|
|
@ -1,5 +1,9 @@
|
||||||
#!/bin/sh
|
#!/bin/sh
|
||||||
|
|
||||||
|
if [ "${CGIT_TEST_NO_GIT_VERSION}" = "YesPlease" ]; then
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
test_description='Check Git version is correct'
|
test_description='Check Git version is correct'
|
||||||
CGIT_TEST_NO_CREATE_REPOS=YesPlease
|
CGIT_TEST_NO_CREATE_REPOS=YesPlease
|
||||||
. ./setup.sh
|
. ./setup.sh
|
||||||
|
@ -29,11 +33,11 @@ test_expect_success 'test submodule version matches Makefile' '
|
||||||
else
|
else
|
||||||
(
|
(
|
||||||
cd ../.. &&
|
cd ../.. &&
|
||||||
sm_sha1=$(git ls-files --stage -- git |
|
sm_oid=$(git ls-files --stage -- git |
|
||||||
sed -e "s/^[0-9]* \\([0-9a-f]*\\) [0-9] .*$/\\1/") &&
|
sed -e "s/^[0-9]* \\([0-9a-f]*\\) [0-9] .*$/\\1/") &&
|
||||||
cd git &&
|
cd git &&
|
||||||
git describe --match "v[0-9]*" $sm_sha1
|
git describe --match "v[0-9]*" $sm_oid
|
||||||
) | sed -e "s/^v//" >sm_version &&
|
) | sed -e "s/^v//" -e "s/-/./" >sm_version &&
|
||||||
test_cmp sm_version makefile_version
|
test_cmp sm_version makefile_version
|
||||||
fi
|
fi
|
||||||
'
|
'
|
||||||
|
|
|
@ -25,7 +25,7 @@ test_expect_success 'get root commit' '
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'root commit contains diffstat' '
|
test_expect_success 'root commit contains diffstat' '
|
||||||
grep "<a href=./foo/diff/file-1.id=[0-9a-f]\{40\}.>file-1</a>" tmp
|
grep "<a href=./foo/diff/file-1.id=[0-9a-f]\{40,64\}.>file-1</a>" tmp
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'root commit contains diff' '
|
test_expect_success 'root commit contains diff' '
|
||||||
|
|
|
@ -25,7 +25,7 @@ test_expect_success 'verify gzip format' '
|
||||||
|
|
||||||
test_expect_success 'untar' '
|
test_expect_success 'untar' '
|
||||||
rm -rf master &&
|
rm -rf master &&
|
||||||
tar -xzf master.tar.gz
|
gzip -dc master.tar.gz | tar -xf -
|
||||||
'
|
'
|
||||||
|
|
||||||
test_expect_success 'count files' '
|
test_expect_success 'count files' '
|
||||||
|
@ -38,6 +38,129 @@ test_expect_success 'verify untarred file-5' '
|
||||||
test_line_count = 1 master/file-5
|
test_line_count = 1 master/file-5
|
||||||
'
|
'
|
||||||
|
|
||||||
|
if test -n "$(which lzip 2>/dev/null)"; then
|
||||||
|
test_set_prereq LZIP
|
||||||
|
else
|
||||||
|
say 'Skipping LZIP validation tests: lzip not found'
|
||||||
|
fi
|
||||||
|
|
||||||
|
test_expect_success LZIP 'get foo/snapshot/master.tar.lz' '
|
||||||
|
cgit_url "foo/snapshot/master.tar.lz" >tmp
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'check html headers' '
|
||||||
|
head -n 1 tmp |
|
||||||
|
grep "Content-Type: application/x-lzip" &&
|
||||||
|
|
||||||
|
head -n 2 tmp |
|
||||||
|
grep "Content-Disposition: inline; filename=.master.tar.lz."
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'strip off the header lines' '
|
||||||
|
strip_headers <tmp >master.tar.lz
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'verify lzip format' '
|
||||||
|
lzip --test master.tar.lz
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'untar' '
|
||||||
|
rm -rf master &&
|
||||||
|
lzip -dc master.tar.lz | tar -xf -
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'count files' '
|
||||||
|
ls master/ >output &&
|
||||||
|
test_line_count = 5 output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success LZIP 'verify untarred file-5' '
|
||||||
|
grep "^5$" master/file-5 &&
|
||||||
|
test_line_count = 1 master/file-5
|
||||||
|
'
|
||||||
|
|
||||||
|
if test -n "$(which xz 2>/dev/null)"; then
|
||||||
|
test_set_prereq XZ
|
||||||
|
else
|
||||||
|
say 'Skipping XZ validation tests: xz not found'
|
||||||
|
fi
|
||||||
|
|
||||||
|
test_expect_success XZ 'get foo/snapshot/master.tar.xz' '
|
||||||
|
cgit_url "foo/snapshot/master.tar.xz" >tmp
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'check html headers' '
|
||||||
|
head -n 1 tmp |
|
||||||
|
grep "Content-Type: application/x-xz" &&
|
||||||
|
|
||||||
|
head -n 2 tmp |
|
||||||
|
grep "Content-Disposition: inline; filename=.master.tar.xz."
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'strip off the header lines' '
|
||||||
|
strip_headers <tmp >master.tar.xz
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'verify xz format' '
|
||||||
|
xz --test master.tar.xz
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'untar' '
|
||||||
|
rm -rf master &&
|
||||||
|
xz -dc master.tar.xz | tar -xf -
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'count files' '
|
||||||
|
ls master/ >output &&
|
||||||
|
test_line_count = 5 output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success XZ 'verify untarred file-5' '
|
||||||
|
grep "^5$" master/file-5 &&
|
||||||
|
test_line_count = 1 master/file-5
|
||||||
|
'
|
||||||
|
|
||||||
|
if test -n "$(which zstd 2>/dev/null)"; then
|
||||||
|
test_set_prereq ZSTD
|
||||||
|
else
|
||||||
|
say 'Skipping ZSTD validation tests: zstd not found'
|
||||||
|
fi
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'get foo/snapshot/master.tar.zst' '
|
||||||
|
cgit_url "foo/snapshot/master.tar.zst" >tmp
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'check html headers' '
|
||||||
|
head -n 1 tmp |
|
||||||
|
grep "Content-Type: application/x-zstd" &&
|
||||||
|
|
||||||
|
head -n 2 tmp |
|
||||||
|
grep "Content-Disposition: inline; filename=.master.tar.zst."
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'strip off the header lines' '
|
||||||
|
strip_headers <tmp >master.tar.zst
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'verify zstd format' '
|
||||||
|
zstd --test master.tar.zst
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'untar' '
|
||||||
|
rm -rf master &&
|
||||||
|
zstd -dc master.tar.zst | tar -xf -
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'count files' '
|
||||||
|
ls master/ >output &&
|
||||||
|
test_line_count = 5 output
|
||||||
|
'
|
||||||
|
|
||||||
|
test_expect_success ZSTD 'verify untarred file-5' '
|
||||||
|
grep "^5$" master/file-5 &&
|
||||||
|
test_line_count = 1 master/file-5
|
||||||
|
'
|
||||||
|
|
||||||
test_expect_success 'get foo/snapshot/master.zip' '
|
test_expect_success 'get foo/snapshot/master.zip' '
|
||||||
cgit_url "foo/snapshot/master.zip" >tmp
|
cgit_url "foo/snapshot/master.zip" >tmp
|
||||||
'
|
'
|
||||||
|
|
|
@ -9,6 +9,12 @@ test -n "$(which strace 2>/dev/null)" || {
|
||||||
exit
|
exit
|
||||||
}
|
}
|
||||||
|
|
||||||
|
strace true 2>/dev/null || {
|
||||||
|
skip_all='Skipping access validation tests: strace not functional'
|
||||||
|
test_done
|
||||||
|
exit
|
||||||
|
}
|
||||||
|
|
||||||
test_no_home_access () {
|
test_no_home_access () {
|
||||||
non_existent_path="/path/to/some/place/that/does/not/possibly/exist"
|
non_existent_path="/path/to/some/place/that/does/not/possibly/exist"
|
||||||
while test -d "$non_existent_path"; do
|
while test -d "$non_existent_path"; do
|
||||||
|
@ -19,7 +25,7 @@ test_no_home_access () {
|
||||||
-E CGIT_CONFIG="$PWD/cgitrc" \
|
-E CGIT_CONFIG="$PWD/cgitrc" \
|
||||||
-E QUERY_STRING="url=$1" \
|
-E QUERY_STRING="url=$1" \
|
||||||
-e access -f -o strace.out cgit &&
|
-e access -f -o strace.out cgit &&
|
||||||
test_must_fail grep "$non_existent_path" strace.out
|
! grep "$non_existent_path" strace.out
|
||||||
}
|
}
|
||||||
|
|
||||||
test_no_home_access_success() {
|
test_no_home_access_success() {
|
||||||
|
|
42
ui-atom.c
42
ui-atom.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-atom.h"
|
#include "ui-atom.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -67,29 +69,25 @@ static void add_entry(struct commit *commit, const char *host)
|
||||||
html("'/>\n");
|
html("'/>\n");
|
||||||
free(pageurl);
|
free(pageurl);
|
||||||
}
|
}
|
||||||
htmlf("<id>%s</id>\n", hex);
|
html("<id>");
|
||||||
|
html_txtf("urn:%s:%s", the_hash_algo->name, hex);
|
||||||
|
html("</id>\n");
|
||||||
html("<content type='text'>\n");
|
html("<content type='text'>\n");
|
||||||
html_txt(info->msg);
|
html_txt(info->msg);
|
||||||
html("</content>\n");
|
html("</content>\n");
|
||||||
html("<content type='xhtml'>\n");
|
|
||||||
html("<div xmlns='http://www.w3.org/1999/xhtml'>\n");
|
|
||||||
html("<pre>\n");
|
|
||||||
html_txt(info->msg);
|
|
||||||
html("</pre>\n");
|
|
||||||
html("</div>\n");
|
|
||||||
html("</content>\n");
|
|
||||||
html("</entry>\n");
|
html("</entry>\n");
|
||||||
cgit_free_commitinfo(info);
|
cgit_free_commitinfo(info);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
void cgit_print_atom(char *tip, char *path, int max_count)
|
void cgit_print_atom(char *tip, const char *path, int max_count)
|
||||||
{
|
{
|
||||||
char *host;
|
char *host;
|
||||||
const char *argv[] = {NULL, tip, NULL, NULL, NULL};
|
const char *argv[] = {NULL, tip, NULL, NULL, NULL};
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct rev_info rev;
|
struct rev_info rev;
|
||||||
int argc = 2;
|
int argc = 2;
|
||||||
|
bool first = true;
|
||||||
|
|
||||||
if (ctx.qry.show_all)
|
if (ctx.qry.show_all)
|
||||||
argv[1] = "--all";
|
argv[1] = "--all";
|
||||||
|
@ -101,7 +99,7 @@ void cgit_print_atom(char *tip, char *path, int max_count)
|
||||||
argv[argc++] = path;
|
argv[argc++] = path;
|
||||||
}
|
}
|
||||||
|
|
||||||
init_revisions(&rev, NULL);
|
repo_init_revisions(the_repository, &rev, NULL);
|
||||||
rev.abbrev = DEFAULT_ABBREV;
|
rev.abbrev = DEFAULT_ABBREV;
|
||||||
rev.commit_format = CMIT_FMT_DEFAULT;
|
rev.commit_format = CMIT_FMT_DEFAULT;
|
||||||
rev.verbose_header = 1;
|
rev.verbose_header = 1;
|
||||||
|
@ -130,18 +128,30 @@ void cgit_print_atom(char *tip, char *path, int max_count)
|
||||||
html_txt(ctx.repo->desc);
|
html_txt(ctx.repo->desc);
|
||||||
html("</subtitle>\n");
|
html("</subtitle>\n");
|
||||||
if (host) {
|
if (host) {
|
||||||
|
char *fullurl = cgit_currentfullurl();
|
||||||
char *repourl = cgit_repourl(ctx.repo->url);
|
char *repourl = cgit_repourl(ctx.repo->url);
|
||||||
html("<link rel='alternate' type='text/html' href='");
|
html("<id>");
|
||||||
html(cgit_httpscheme());
|
html_txtf("%s%s%s", cgit_httpscheme(), host, fullurl);
|
||||||
html_attr(host);
|
html("</id>\n");
|
||||||
html_attr(repourl);
|
html("<link rel='self' href='");
|
||||||
|
html_attrf("%s%s%s", cgit_httpscheme(), host, fullurl);
|
||||||
html("'/>\n");
|
html("'/>\n");
|
||||||
|
html("<link rel='alternate' type='text/html' href='");
|
||||||
|
html_attrf("%s%s%s", cgit_httpscheme(), host, repourl);
|
||||||
|
html("'/>\n");
|
||||||
|
free(fullurl);
|
||||||
free(repourl);
|
free(repourl);
|
||||||
}
|
}
|
||||||
while ((commit = get_revision(&rev)) != NULL) {
|
while ((commit = get_revision(&rev)) != NULL) {
|
||||||
|
if (first) {
|
||||||
|
html("<updated>");
|
||||||
|
html_txt(show_date(commit->date, 0,
|
||||||
|
date_mode_from_type(DATE_ISO8601_STRICT)));
|
||||||
|
html("</updated>\n");
|
||||||
|
first = false;
|
||||||
|
}
|
||||||
add_entry(commit, host);
|
add_entry(commit, host);
|
||||||
free_commit_buffer(commit);
|
release_commit_memory(the_repository->parsed_objects, commit);
|
||||||
free_commit_list(commit->parents);
|
|
||||||
commit->parents = NULL;
|
commit->parents = NULL;
|
||||||
}
|
}
|
||||||
html("</feed>\n");
|
html("</feed>\n");
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
#ifndef UI_ATOM_H
|
#ifndef UI_ATOM_H
|
||||||
#define UI_ATOM_H
|
#define UI_ATOM_H
|
||||||
|
|
||||||
extern void cgit_print_atom(char *tip, char *path, int max_count);
|
extern void cgit_print_atom(char *tip, const char *path, int max_count);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
|
52
ui-blame.c
52
ui-blame.c
|
@ -6,11 +6,13 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-blame.h"
|
#include "ui-blame.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
#include "ui-shared.h"
|
#include "ui-shared.h"
|
||||||
#include "argv-array.h"
|
#include "strvec.h"
|
||||||
#include "blame.h"
|
#include "blame.h"
|
||||||
|
|
||||||
|
|
||||||
|
@ -48,12 +50,21 @@ static void emit_blame_entry_hash(struct blame_entry *ent)
|
||||||
unsigned long line = 0;
|
unsigned long line = 0;
|
||||||
|
|
||||||
char *detail = emit_suspect_detail(suspect);
|
char *detail = emit_suspect_detail(suspect);
|
||||||
html("<span class='sha1'>");
|
html("<span class='oid'>");
|
||||||
cgit_commit_link(find_unique_abbrev(oid, DEFAULT_ABBREV), detail,
|
cgit_commit_link(repo_find_unique_abbrev(the_repository, oid, DEFAULT_ABBREV), detail,
|
||||||
NULL, ctx.qry.head, oid_to_hex(oid), suspect->path);
|
NULL, ctx.qry.head, oid_to_hex(oid), suspect->path);
|
||||||
html("</span>");
|
html("</span>");
|
||||||
free(detail);
|
free(detail);
|
||||||
|
|
||||||
|
if (!repo_parse_commit(the_repository, suspect->commit) && suspect->commit->parents) {
|
||||||
|
struct commit *parent = suspect->commit->parents->item;
|
||||||
|
|
||||||
|
html(" ");
|
||||||
|
cgit_blame_link("^", "Blame the previous revision", NULL,
|
||||||
|
ctx.qry.head, oid_to_hex(&parent->object.oid),
|
||||||
|
suspect->path);
|
||||||
|
}
|
||||||
|
|
||||||
while (line++ < ent->num_lines)
|
while (line++ < ent->num_lines)
|
||||||
html("\n");
|
html("\n");
|
||||||
}
|
}
|
||||||
|
@ -104,7 +115,7 @@ static void print_object(const struct object_id *oid, const char *path,
|
||||||
enum object_type type;
|
enum object_type type;
|
||||||
char *buf;
|
char *buf;
|
||||||
unsigned long size;
|
unsigned long size;
|
||||||
struct argv_array rev_argv = ARGV_ARRAY_INIT;
|
struct strvec rev_argv = STRVEC_INIT;
|
||||||
struct rev_info revs;
|
struct rev_info revs;
|
||||||
struct blame_scoreboard sb;
|
struct blame_scoreboard sb;
|
||||||
struct blame_origin *o;
|
struct blame_origin *o;
|
||||||
|
@ -117,21 +128,23 @@ static void print_object(const struct object_id *oid, const char *path,
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
buf = read_object_file(oid, &type, &size);
|
buf = repo_read_object_file(the_repository, oid, &type, &size);
|
||||||
if (!buf) {
|
if (!buf) {
|
||||||
cgit_print_error_page(500, "Internal server error",
|
cgit_print_error_page(500, "Internal server error",
|
||||||
"Error reading object %s", oid_to_hex(oid));
|
"Error reading object %s", oid_to_hex(oid));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
argv_array_push(&rev_argv, "blame");
|
strvec_push(&rev_argv, "blame");
|
||||||
argv_array_push(&rev_argv, rev);
|
strvec_push(&rev_argv, rev);
|
||||||
init_revisions(&revs, NULL);
|
repo_init_revisions(the_repository, &revs, NULL);
|
||||||
revs.diffopt.flags.allow_textconv = 1;
|
revs.diffopt.flags.allow_textconv = 1;
|
||||||
setup_revisions(rev_argv.argc, rev_argv.argv, &revs, NULL);
|
setup_revisions(rev_argv.nr, rev_argv.v, &revs, NULL);
|
||||||
init_scoreboard(&sb);
|
init_scoreboard(&sb);
|
||||||
sb.revs = &revs;
|
sb.revs = &revs;
|
||||||
setup_scoreboard(&sb, path, &o);
|
sb.repo = the_repository;
|
||||||
|
sb.path = path;
|
||||||
|
setup_scoreboard(&sb, &o);
|
||||||
o->suspects = blame_entry_prepend(NULL, 0, sb.num_lines, o);
|
o->suspects = blame_entry_prepend(NULL, 0, sb.num_lines, o);
|
||||||
prio_queue_put(&sb.commits, o->commit);
|
prio_queue_put(&sb.commits, o->commit);
|
||||||
blame_origin_decref(o);
|
blame_origin_decref(o);
|
||||||
|
@ -150,6 +163,10 @@ static void print_object(const struct object_id *oid, const char *path,
|
||||||
cgit_tree_link("tree", NULL, NULL, ctx.qry.head, rev, path);
|
cgit_tree_link("tree", NULL, NULL, ctx.qry.head, rev, path);
|
||||||
html(")\n");
|
html(")\n");
|
||||||
|
|
||||||
|
if (buffer_is_binary(buf, size)) {
|
||||||
|
html("<div class='error'>blob is binary.</div>");
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
if (ctx.cfg.max_blob_size && size / 1024 > ctx.cfg.max_blob_size) {
|
if (ctx.cfg.max_blob_size && size / 1024 > ctx.cfg.max_blob_size) {
|
||||||
htmlf("<div class='error'>blob size (%ldKB)"
|
htmlf("<div class='error'>blob size (%ldKB)"
|
||||||
" exceeds display size limit (%dKB).</div>",
|
" exceeds display size limit (%dKB).</div>",
|
||||||
|
@ -219,8 +236,7 @@ cleanup:
|
||||||
}
|
}
|
||||||
|
|
||||||
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage,
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
void *cbdata)
|
|
||||||
{
|
{
|
||||||
struct walk_tree_context *walk_tree_ctx = cbdata;
|
struct walk_tree_context *walk_tree_ctx = cbdata;
|
||||||
|
|
||||||
|
@ -255,7 +271,7 @@ static int basedir_len(const char *path)
|
||||||
|
|
||||||
void cgit_print_blame(void)
|
void cgit_print_blame(void)
|
||||||
{
|
{
|
||||||
const char *rev = ctx.qry.sha1;
|
const char *rev = ctx.qry.oid;
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct pathspec_item path_items = {
|
struct pathspec_item path_items = {
|
||||||
|
@ -273,13 +289,13 @@ void cgit_print_blame(void)
|
||||||
if (!rev)
|
if (!rev)
|
||||||
rev = ctx.qry.head;
|
rev = ctx.qry.head;
|
||||||
|
|
||||||
if (get_oid(rev, &oid)) {
|
if (repo_get_oid(the_repository, rev, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Invalid revision name: %s", rev);
|
"Invalid revision name: %s", rev);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
if (!commit || parse_commit(commit)) {
|
if (!commit || repo_parse_commit(the_repository, commit)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Invalid commit reference: %s", rev);
|
"Invalid commit reference: %s", rev);
|
||||||
return;
|
return;
|
||||||
|
@ -289,8 +305,8 @@ void cgit_print_blame(void)
|
||||||
walk_tree_ctx.match_baselen = (path_items.match) ?
|
walk_tree_ctx.match_baselen = (path_items.match) ?
|
||||||
basedir_len(path_items.match) : -1;
|
basedir_len(path_items.match) : -1;
|
||||||
|
|
||||||
read_tree_recursive(commit->maybe_tree, "", 0, 0, &paths, walk_tree,
|
read_tree(the_repository, repo_get_commit_tree(the_repository, commit),
|
||||||
&walk_tree_ctx);
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
if (!walk_tree_ctx.state)
|
if (!walk_tree_ctx.state)
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
else if (walk_tree_ctx.state == 2)
|
else if (walk_tree_ctx.state == 2)
|
||||||
|
|
28
ui-blob.c
28
ui-blob.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-blob.h"
|
#include "ui-blob.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -19,7 +21,7 @@ struct walk_tree_context {
|
||||||
};
|
};
|
||||||
|
|
||||||
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage, void *cbdata)
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
{
|
{
|
||||||
struct walk_tree_context *walk_tree_ctx = cbdata;
|
struct walk_tree_context *walk_tree_ctx = cbdata;
|
||||||
|
|
||||||
|
@ -52,11 +54,13 @@ int cgit_ref_path_exists(const char *path, const char *ref, int file_only)
|
||||||
.file_only = file_only
|
.file_only = file_only
|
||||||
};
|
};
|
||||||
|
|
||||||
if (get_oid(ref, &oid))
|
if (repo_get_oid(the_repository, ref, &oid))
|
||||||
goto done;
|
goto done;
|
||||||
if (oid_object_info(the_repository, &oid, &size) != OBJ_COMMIT)
|
if (oid_object_info(the_repository, &oid, &size) != OBJ_COMMIT)
|
||||||
goto done;
|
goto done;
|
||||||
read_tree_recursive(lookup_commit_reference(&oid)->maybe_tree, "", 0, 0, &paths, walk_tree, &walk_tree_ctx);
|
read_tree(the_repository,
|
||||||
|
repo_get_commit_tree(the_repository, lookup_commit_reference(the_repository, &oid)),
|
||||||
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
|
|
||||||
done:
|
done:
|
||||||
free(path_items.match);
|
free(path_items.match);
|
||||||
|
@ -85,19 +89,20 @@ int cgit_print_file(char *path, const char *head, int file_only)
|
||||||
.file_only = file_only
|
.file_only = file_only
|
||||||
};
|
};
|
||||||
|
|
||||||
if (get_oid(head, &oid))
|
if (repo_get_oid(the_repository, head, &oid))
|
||||||
return -1;
|
return -1;
|
||||||
type = oid_object_info(the_repository, &oid, &size);
|
type = oid_object_info(the_repository, &oid, &size);
|
||||||
if (type == OBJ_COMMIT) {
|
if (type == OBJ_COMMIT) {
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
read_tree_recursive(commit->maybe_tree, "", 0, 0, &paths, walk_tree, &walk_tree_ctx);
|
read_tree(the_repository, repo_get_commit_tree(the_repository, commit),
|
||||||
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
if (!walk_tree_ctx.found_path)
|
if (!walk_tree_ctx.found_path)
|
||||||
return -1;
|
return -1;
|
||||||
type = oid_object_info(the_repository, &oid, &size);
|
type = oid_object_info(the_repository, &oid, &size);
|
||||||
}
|
}
|
||||||
if (type == OBJ_BAD)
|
if (type == OBJ_BAD)
|
||||||
return -1;
|
return -1;
|
||||||
buf = read_object_file(&oid, &type, &size);
|
buf = repo_read_object_file(the_repository, &oid, &type, &size);
|
||||||
if (!buf)
|
if (!buf)
|
||||||
return -1;
|
return -1;
|
||||||
buf[size] = '\0';
|
buf[size] = '\0';
|
||||||
|
@ -135,7 +140,7 @@ void cgit_print_blob(const char *hex, char *path, const char *head, int file_onl
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
if (get_oid(head, &oid)) {
|
if (repo_get_oid(the_repository, head, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad ref: %s", head);
|
"Bad ref: %s", head);
|
||||||
return;
|
return;
|
||||||
|
@ -145,8 +150,9 @@ void cgit_print_blob(const char *hex, char *path, const char *head, int file_onl
|
||||||
type = oid_object_info(the_repository, &oid, &size);
|
type = oid_object_info(the_repository, &oid, &size);
|
||||||
|
|
||||||
if ((!hex) && type == OBJ_COMMIT && path) {
|
if ((!hex) && type == OBJ_COMMIT && path) {
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
read_tree_recursive(commit->maybe_tree, "", 0, 0, &paths, walk_tree, &walk_tree_ctx);
|
read_tree(the_repository, repo_get_commit_tree(the_repository, commit),
|
||||||
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
type = oid_object_info(the_repository, &oid, &size);
|
type = oid_object_info(the_repository, &oid, &size);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -156,7 +162,7 @@ void cgit_print_blob(const char *hex, char *path, const char *head, int file_onl
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
buf = read_object_file(&oid, &type, &size);
|
buf = repo_read_object_file(the_repository, &oid, &type, &size);
|
||||||
if (!buf) {
|
if (!buf) {
|
||||||
cgit_print_error_page(500, "Internal server error",
|
cgit_print_error_page(500, "Internal server error",
|
||||||
"Error reading object %s", hex);
|
"Error reading object %s", hex);
|
||||||
|
|
|
@ -7,6 +7,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-clone.h"
|
#include "ui-clone.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -19,12 +21,12 @@ static int print_ref_info(const char *refname, const struct object_id *oid,
|
||||||
{
|
{
|
||||||
struct object *obj;
|
struct object *obj;
|
||||||
|
|
||||||
if (!(obj = parse_object(oid)))
|
if (!(obj = parse_object(the_repository, oid)))
|
||||||
return 0;
|
return 0;
|
||||||
|
|
||||||
htmlf("%s\t%s\n", oid_to_hex(oid), refname);
|
htmlf("%s\t%s\n", oid_to_hex(oid), refname);
|
||||||
if (obj->type == OBJ_TAG) {
|
if (obj->type == OBJ_TAG) {
|
||||||
if (!(obj = deref_tag(obj, refname, 0)))
|
if (!(obj = deref_tag(the_repository, obj, refname, 0)))
|
||||||
return 0;
|
return 0;
|
||||||
htmlf("%s\t%s^{}\n", oid_to_hex(&obj->oid), refname);
|
htmlf("%s\t%s^{}\n", oid_to_hex(&obj->oid), refname);
|
||||||
}
|
}
|
||||||
|
@ -87,7 +89,8 @@ void cgit_clone_info(void)
|
||||||
ctx.page.mimetype = "text/plain";
|
ctx.page.mimetype = "text/plain";
|
||||||
ctx.page.filename = "info/refs";
|
ctx.page.filename = "info/refs";
|
||||||
cgit_print_http_headers();
|
cgit_print_http_headers();
|
||||||
for_each_ref(print_ref_info, NULL);
|
refs_for_each_ref(get_main_ref_store(the_repository),
|
||||||
|
print_ref_info, NULL);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_clone_objects(void)
|
void cgit_clone_objects(void)
|
||||||
|
|
23
ui-commit.c
23
ui-commit.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-commit.h"
|
#include "ui-commit.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -26,12 +28,12 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
if (!hex)
|
if (!hex)
|
||||||
hex = ctx.qry.head;
|
hex = ctx.qry.head;
|
||||||
|
|
||||||
if (get_oid(hex, &oid)) {
|
if (repo_get_oid(the_repository, hex, &oid)) {
|
||||||
cgit_print_error_page(400, "Bad request",
|
cgit_print_error_page(400, "Bad request",
|
||||||
"Bad object id: %s", hex);
|
"Bad object id: %s", hex);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
if (!commit) {
|
if (!commit) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad commit reference: %s", hex);
|
"Bad commit reference: %s", hex);
|
||||||
|
@ -39,10 +41,11 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
}
|
}
|
||||||
info = cgit_parse_commit(commit);
|
info = cgit_parse_commit(commit);
|
||||||
|
|
||||||
format_display_notes(&oid, ¬es, PAGE_ENCODING, 0);
|
format_display_notes(&oid, ¬es, PAGE_ENCODING, 1);
|
||||||
|
|
||||||
load_ref_decorations(NULL, DECORATE_FULL_REFS);
|
load_ref_decorations(NULL, DECORATE_FULL_REFS);
|
||||||
|
|
||||||
|
ctx.page.title = fmtalloc("%s - %s", info->subject, ctx.page.title);
|
||||||
cgit_print_layout_start();
|
cgit_print_layout_start();
|
||||||
cgit_print_diff_ctrls();
|
cgit_print_diff_ctrls();
|
||||||
html("<table summary='commit info' class='commit-info'>\n");
|
html("<table summary='commit info' class='commit-info'>\n");
|
||||||
|
@ -70,15 +73,15 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
html_txt(show_date(info->committer_date, info->committer_tz,
|
html_txt(show_date(info->committer_date, info->committer_tz,
|
||||||
cgit_date_mode(DATE_ISO8601)));
|
cgit_date_mode(DATE_ISO8601)));
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
html("<tr><th>commit</th><td colspan='2' class='sha1'>");
|
html("<tr><th>commit</th><td colspan='2' class='oid'>");
|
||||||
tmp = oid_to_hex(&commit->object.oid);
|
tmp = oid_to_hex(&commit->object.oid);
|
||||||
cgit_commit_link(tmp, NULL, NULL, ctx.qry.head, tmp, prefix);
|
cgit_commit_link(tmp, NULL, NULL, ctx.qry.head, tmp, prefix);
|
||||||
html(" (");
|
html(" (");
|
||||||
cgit_patch_link("patch", NULL, NULL, NULL, tmp, prefix);
|
cgit_patch_link("patch", NULL, NULL, NULL, tmp, prefix);
|
||||||
html(")</td></tr>\n");
|
html(")</td></tr>\n");
|
||||||
html("<tr><th>tree</th><td colspan='2' class='sha1'>");
|
html("<tr><th>tree</th><td colspan='2' class='oid'>");
|
||||||
tmp = xstrdup(hex);
|
tmp = xstrdup(hex);
|
||||||
cgit_tree_link(oid_to_hex(&commit->maybe_tree->object.oid), NULL, NULL,
|
cgit_tree_link(oid_to_hex(get_commit_tree_oid(commit)), NULL, NULL,
|
||||||
ctx.qry.head, tmp, NULL);
|
ctx.qry.head, tmp, NULL);
|
||||||
if (prefix) {
|
if (prefix) {
|
||||||
html(" /");
|
html(" /");
|
||||||
|
@ -87,7 +90,7 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
free(tmp);
|
free(tmp);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
for (p = commit->parents; p; p = p->next) {
|
for (p = commit->parents; p; p = p->next) {
|
||||||
parent = lookup_commit_reference(&p->item->object.oid);
|
parent = lookup_commit_reference(the_repository, &p->item->object.oid);
|
||||||
if (!parent) {
|
if (!parent) {
|
||||||
html("<tr><td colspan='3'>");
|
html("<tr><td colspan='3'>");
|
||||||
cgit_print_error("Error reading parent commit");
|
cgit_print_error("Error reading parent commit");
|
||||||
|
@ -95,7 +98,7 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
html("<tr><th>parent</th>"
|
html("<tr><th>parent</th>"
|
||||||
"<td colspan='2' class='sha1'>");
|
"<td colspan='2' class='oid'>");
|
||||||
tmp = tmp2 = oid_to_hex(&p->item->object.oid);
|
tmp = tmp2 = oid_to_hex(&p->item->object.oid);
|
||||||
if (ctx.repo->enable_subject_links) {
|
if (ctx.repo->enable_subject_links) {
|
||||||
parent_info = cgit_parse_commit(parent);
|
parent_info = cgit_parse_commit(parent);
|
||||||
|
@ -109,7 +112,7 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
parents++;
|
parents++;
|
||||||
}
|
}
|
||||||
if (ctx.repo->snapshots) {
|
if (ctx.repo->snapshots) {
|
||||||
html("<tr><th>download</th><td colspan='2' class='sha1'>");
|
html("<tr><th>download</th><td colspan='2' class='oid'>");
|
||||||
cgit_print_snapshot_links(ctx.repo, hex, "<br/>");
|
cgit_print_snapshot_links(ctx.repo, hex, "<br/>");
|
||||||
html("</td></tr>");
|
html("</td></tr>");
|
||||||
}
|
}
|
||||||
|
@ -139,7 +142,7 @@ void cgit_print_commit(char *hex, const char *prefix)
|
||||||
tmp = oid_to_hex(&commit->parents->item->object.oid);
|
tmp = oid_to_hex(&commit->parents->item->object.oid);
|
||||||
else
|
else
|
||||||
tmp = NULL;
|
tmp = NULL;
|
||||||
cgit_print_diff(ctx.qry.sha1, tmp, prefix, 0, 0);
|
cgit_print_diff(ctx.qry.oid, tmp, prefix, 0, 0);
|
||||||
}
|
}
|
||||||
strbuf_release(¬es);
|
strbuf_release(¬es);
|
||||||
cgit_free_commitinfo(info);
|
cgit_free_commitinfo(info);
|
||||||
|
|
36
ui-diff.c
36
ui-diff.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-diff.h"
|
#include "ui-diff.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -82,7 +84,7 @@ static void print_fileinfo(struct fileinfo *info)
|
||||||
}
|
}
|
||||||
|
|
||||||
html("<tr>");
|
html("<tr>");
|
||||||
htmlf("<td class='mode'>");
|
html("<td class='mode'>");
|
||||||
if (is_null_oid(info->new_oid)) {
|
if (is_null_oid(info->new_oid)) {
|
||||||
cgit_print_filemode(info->old_mode);
|
cgit_print_filemode(info->old_mode);
|
||||||
} else {
|
} else {
|
||||||
|
@ -97,8 +99,8 @@ static void print_fileinfo(struct fileinfo *info)
|
||||||
html("]</span>");
|
html("]</span>");
|
||||||
}
|
}
|
||||||
htmlf("</td><td class='%s'>", class);
|
htmlf("</td><td class='%s'>", class);
|
||||||
cgit_diff_link(info->new_path, NULL, NULL, ctx.qry.head, ctx.qry.sha1,
|
cgit_diff_link(info->new_path, NULL, NULL, ctx.qry.head, ctx.qry.oid,
|
||||||
ctx.qry.sha2, info->new_path);
|
ctx.qry.oid2, info->new_path);
|
||||||
if (info->status == DIFF_STATUS_COPIED || info->status == DIFF_STATUS_RENAMED) {
|
if (info->status == DIFF_STATUS_COPIED || info->status == DIFF_STATUS_RENAMED) {
|
||||||
htmlf(" (%s from ",
|
htmlf(" (%s from ",
|
||||||
info->status == DIFF_STATUS_COPIED ? "copied" : "renamed");
|
info->status == DIFF_STATUS_COPIED ? "copied" : "renamed");
|
||||||
|
@ -194,8 +196,8 @@ static void cgit_print_diffstat(const struct object_id *old_oid,
|
||||||
int i;
|
int i;
|
||||||
|
|
||||||
html("<div class='diffstat-header'>");
|
html("<div class='diffstat-header'>");
|
||||||
cgit_diff_link("Diffstat", NULL, NULL, ctx.qry.head, ctx.qry.sha1,
|
cgit_diff_link("Diffstat", NULL, NULL, ctx.qry.head, ctx.qry.oid,
|
||||||
ctx.qry.sha2, NULL);
|
ctx.qry.oid2, NULL);
|
||||||
if (prefix) {
|
if (prefix) {
|
||||||
html(" (limited to '");
|
html(" (limited to '");
|
||||||
html_txt(prefix);
|
html_txt(prefix);
|
||||||
|
@ -258,8 +260,8 @@ static void header(const struct object_id *oid1, char *path1, int mode1,
|
||||||
htmlf("<br/>deleted file mode %.6o", mode1);
|
htmlf("<br/>deleted file mode %.6o", mode1);
|
||||||
|
|
||||||
if (!subproject) {
|
if (!subproject) {
|
||||||
abbrev1 = xstrdup(find_unique_abbrev(oid1, DEFAULT_ABBREV));
|
abbrev1 = xstrdup(repo_find_unique_abbrev(the_repository, oid1, DEFAULT_ABBREV));
|
||||||
abbrev2 = xstrdup(find_unique_abbrev(oid2, DEFAULT_ABBREV));
|
abbrev2 = xstrdup(repo_find_unique_abbrev(the_repository, oid2, DEFAULT_ABBREV));
|
||||||
htmlf("<br/>index %s..%s", abbrev1, abbrev2);
|
htmlf("<br/>index %s..%s", abbrev1, abbrev2);
|
||||||
free(abbrev1);
|
free(abbrev1);
|
||||||
free(abbrev2);
|
free(abbrev2);
|
||||||
|
@ -402,21 +404,21 @@ void cgit_print_diff(const char *new_rev, const char *old_rev,
|
||||||
|
|
||||||
if (!new_rev)
|
if (!new_rev)
|
||||||
new_rev = ctx.qry.head;
|
new_rev = ctx.qry.head;
|
||||||
if (get_oid(new_rev, new_rev_oid)) {
|
if (repo_get_oid(the_repository, new_rev, new_rev_oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad object name: %s", new_rev);
|
"Bad object name: %s", new_rev);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(new_rev_oid);
|
commit = lookup_commit_reference(the_repository, new_rev_oid);
|
||||||
if (!commit || parse_commit(commit)) {
|
if (!commit || repo_parse_commit(the_repository, commit)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad commit: %s", oid_to_hex(new_rev_oid));
|
"Bad commit: %s", oid_to_hex(new_rev_oid));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
new_tree_oid = &commit->maybe_tree->object.oid;
|
new_tree_oid = get_commit_tree_oid(commit);
|
||||||
|
|
||||||
if (old_rev) {
|
if (old_rev) {
|
||||||
if (get_oid(old_rev, old_rev_oid)) {
|
if (repo_get_oid(the_repository, old_rev, old_rev_oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad object name: %s", old_rev);
|
"Bad object name: %s", old_rev);
|
||||||
return;
|
return;
|
||||||
|
@ -424,17 +426,17 @@ void cgit_print_diff(const char *new_rev, const char *old_rev,
|
||||||
} else if (commit->parents && commit->parents->item) {
|
} else if (commit->parents && commit->parents->item) {
|
||||||
oidcpy(old_rev_oid, &commit->parents->item->object.oid);
|
oidcpy(old_rev_oid, &commit->parents->item->object.oid);
|
||||||
} else {
|
} else {
|
||||||
oidclr(old_rev_oid);
|
oidclr(old_rev_oid, the_repository->hash_algo);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!is_null_oid(old_rev_oid)) {
|
if (!is_null_oid(old_rev_oid)) {
|
||||||
commit2 = lookup_commit_reference(old_rev_oid);
|
commit2 = lookup_commit_reference(the_repository, old_rev_oid);
|
||||||
if (!commit2 || parse_commit(commit2)) {
|
if (!commit2 || repo_parse_commit(the_repository, commit2)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad commit: %s", oid_to_hex(old_rev_oid));
|
"Bad commit: %s", oid_to_hex(old_rev_oid));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
old_tree_oid = &commit2->maybe_tree->object.oid;
|
old_tree_oid = get_commit_tree_oid(commit2);
|
||||||
} else {
|
} else {
|
||||||
old_tree_oid = NULL;
|
old_tree_oid = NULL;
|
||||||
}
|
}
|
||||||
|
@ -442,7 +444,7 @@ void cgit_print_diff(const char *new_rev, const char *old_rev,
|
||||||
if (raw) {
|
if (raw) {
|
||||||
struct diff_options diffopt;
|
struct diff_options diffopt;
|
||||||
|
|
||||||
diff_setup(&diffopt);
|
repo_diff_setup(the_repository, &diffopt);
|
||||||
diffopt.output_format = DIFF_FORMAT_PATCH;
|
diffopt.output_format = DIFF_FORMAT_PATCH;
|
||||||
diffopt.flags.recursive = 1;
|
diffopt.flags.recursive = 1;
|
||||||
diff_setup_done(&diffopt);
|
diff_setup_done(&diffopt);
|
||||||
|
|
70
ui-log.c
70
ui-log.c
|
@ -6,11 +6,13 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-log.h"
|
#include "ui-log.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
#include "ui-shared.h"
|
#include "ui-shared.h"
|
||||||
#include "argv-array.h"
|
#include "strvec.h"
|
||||||
|
|
||||||
static int files, add_lines, rem_lines, lines_counted;
|
static int files, add_lines, rem_lines, lines_counted;
|
||||||
|
|
||||||
|
@ -65,9 +67,10 @@ void show_commit_decorations(struct commit *commit)
|
||||||
return;
|
return;
|
||||||
html("<span class='decoration'>");
|
html("<span class='decoration'>");
|
||||||
while (deco) {
|
while (deco) {
|
||||||
struct object_id peeled;
|
struct object_id oid_tag, peeled;
|
||||||
int is_annotated = 0;
|
int is_annotated = 0;
|
||||||
strncpy(buf, prettify_refname(deco->name), sizeof(buf) - 1);
|
|
||||||
|
strlcpy(buf, prettify_refname(deco->name), sizeof(buf));
|
||||||
switch(deco->type) {
|
switch(deco->type) {
|
||||||
case DECORATION_NONE:
|
case DECORATION_NONE:
|
||||||
/* If the git-core doesn't recognize it,
|
/* If the git-core doesn't recognize it,
|
||||||
|
@ -79,8 +82,9 @@ void show_commit_decorations(struct commit *commit)
|
||||||
ctx.qry.showmsg, 0);
|
ctx.qry.showmsg, 0);
|
||||||
break;
|
break;
|
||||||
case DECORATION_REF_TAG:
|
case DECORATION_REF_TAG:
|
||||||
if (!peel_ref(deco->name, &peeled))
|
if (!refs_read_ref(get_main_ref_store(the_repository), deco->name, &oid_tag) &&
|
||||||
is_annotated = !oidcmp(&commit->object.oid, &peeled);
|
!peel_iterated_oid(the_repository, &oid_tag, &peeled))
|
||||||
|
is_annotated = !oideq(&oid_tag, &peeled);
|
||||||
cgit_tag_link(buf, NULL, is_annotated ? "tag-annotated-deco" : "tag-deco", buf);
|
cgit_tag_link(buf, NULL, is_annotated ? "tag-annotated-deco" : "tag-deco", buf);
|
||||||
break;
|
break;
|
||||||
case DECORATION_REF_REMOTE:
|
case DECORATION_REF_REMOTE:
|
||||||
|
@ -145,7 +149,7 @@ static int show_commit(struct commit *commit, struct rev_info *revs)
|
||||||
/* When we get here we have precisely one parent. */
|
/* When we get here we have precisely one parent. */
|
||||||
parent = parents->item;
|
parent = parents->item;
|
||||||
/* If we can't parse the commit, let print_commit() report an error. */
|
/* If we can't parse the commit, let print_commit() report an error. */
|
||||||
if (parse_commit(parent))
|
if (repo_parse_commit(the_repository, parent))
|
||||||
return 1;
|
return 1;
|
||||||
|
|
||||||
files = 0;
|
files = 0;
|
||||||
|
@ -153,12 +157,12 @@ static int show_commit(struct commit *commit, struct rev_info *revs)
|
||||||
rem_lines = 0;
|
rem_lines = 0;
|
||||||
|
|
||||||
revs->diffopt.flags.recursive = 1;
|
revs->diffopt.flags.recursive = 1;
|
||||||
diff_tree_oid(&parent->maybe_tree->object.oid,
|
diff_tree_oid(get_commit_tree_oid(parent),
|
||||||
&commit->maybe_tree->object.oid,
|
get_commit_tree_oid(commit),
|
||||||
"", &revs->diffopt);
|
"", &revs->diffopt);
|
||||||
diffcore_std(&revs->diffopt);
|
diffcore_std(&revs->diffopt);
|
||||||
|
|
||||||
found = !diff_queue_is_empty();
|
found = !diff_queue_is_empty(&revs->diffopt);
|
||||||
saved_fmt = revs->diffopt.output_format;
|
saved_fmt = revs->diffopt.output_format;
|
||||||
revs->diffopt.output_format = DIFF_FORMAT_CALLBACK;
|
revs->diffopt.output_format = DIFF_FORMAT_CALLBACK;
|
||||||
revs->diffopt.format_callback = cgit_diff_tree_cb;
|
revs->diffopt.format_callback = cgit_diff_tree_cb;
|
||||||
|
@ -234,7 +238,7 @@ static void print_commit(struct commit *commit, struct rev_info *revs)
|
||||||
strbuf_add(&msgbuf, "\n\n", 2);
|
strbuf_add(&msgbuf, "\n\n", 2);
|
||||||
|
|
||||||
/* Place wrap_symbol at position i in info->subject */
|
/* Place wrap_symbol at position i in info->subject */
|
||||||
strcpy(info->subject + i, wrap_symbol);
|
strlcpy(info->subject + i, wrap_symbol, subject_len - i + 1);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head,
|
cgit_commit_link(info->subject, NULL, NULL, ctx.qry.head,
|
||||||
|
@ -329,7 +333,7 @@ static const char *disambiguate_ref(const char *ref, int *must_free_result)
|
||||||
struct strbuf longref = STRBUF_INIT;
|
struct strbuf longref = STRBUF_INIT;
|
||||||
|
|
||||||
strbuf_addf(&longref, "refs/heads/%s", ref);
|
strbuf_addf(&longref, "refs/heads/%s", ref);
|
||||||
if (get_oid(longref.buf, &oid) == 0) {
|
if (repo_get_oid(the_repository, longref.buf, &oid) == 0) {
|
||||||
*must_free_result = 1;
|
*must_free_result = 1;
|
||||||
return strbuf_detach(&longref, NULL);
|
return strbuf_detach(&longref, NULL);
|
||||||
}
|
}
|
||||||
|
@ -362,27 +366,27 @@ static char *next_token(char **src)
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern,
|
void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern,
|
||||||
char *path, int pager, int commit_graph, int commit_sort)
|
const char *path, int pager, int commit_graph, int commit_sort)
|
||||||
{
|
{
|
||||||
struct rev_info rev;
|
struct rev_info rev;
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct argv_array rev_argv = ARGV_ARRAY_INIT;
|
struct strvec rev_argv = STRVEC_INIT;
|
||||||
int i, columns = commit_graph ? 4 : 3;
|
int i, columns = commit_graph ? 4 : 3;
|
||||||
int must_free_tip = 0;
|
int must_free_tip = 0;
|
||||||
|
|
||||||
/* rev_argv.argv[0] will be ignored by setup_revisions */
|
/* rev_argv.argv[0] will be ignored by setup_revisions */
|
||||||
argv_array_push(&rev_argv, "log_rev_setup");
|
strvec_push(&rev_argv, "log_rev_setup");
|
||||||
|
|
||||||
if (!tip)
|
if (!tip)
|
||||||
tip = ctx.qry.head;
|
tip = ctx.qry.head;
|
||||||
tip = disambiguate_ref(tip, &must_free_tip);
|
tip = disambiguate_ref(tip, &must_free_tip);
|
||||||
argv_array_push(&rev_argv, tip);
|
strvec_push(&rev_argv, tip);
|
||||||
|
|
||||||
if (grep && pattern && *pattern) {
|
if (grep && pattern && *pattern) {
|
||||||
pattern = xstrdup(pattern);
|
pattern = xstrdup(pattern);
|
||||||
if (!strcmp(grep, "grep") || !strcmp(grep, "author") ||
|
if (!strcmp(grep, "grep") || !strcmp(grep, "author") ||
|
||||||
!strcmp(grep, "committer")) {
|
!strcmp(grep, "committer")) {
|
||||||
argv_array_pushf(&rev_argv, "--%s=%s", grep, pattern);
|
strvec_pushf(&rev_argv, "--%s=%s", grep, pattern);
|
||||||
} else if (!strcmp(grep, "range")) {
|
} else if (!strcmp(grep, "range")) {
|
||||||
char *arg;
|
char *arg;
|
||||||
/* Split the pattern at whitespace and add each token
|
/* Split the pattern at whitespace and add each token
|
||||||
|
@ -390,14 +394,14 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
* rev-list options. Also, replace the previously
|
* rev-list options. Also, replace the previously
|
||||||
* pushed tip (it's no longer relevant).
|
* pushed tip (it's no longer relevant).
|
||||||
*/
|
*/
|
||||||
argv_array_pop(&rev_argv);
|
strvec_pop(&rev_argv);
|
||||||
while ((arg = next_token(&pattern))) {
|
while ((arg = next_token(&pattern))) {
|
||||||
if (*arg == '-') {
|
if (*arg == '-') {
|
||||||
fprintf(stderr, "Bad range expr: %s\n",
|
fprintf(stderr, "Bad range expr: %s\n",
|
||||||
arg);
|
arg);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
argv_array_push(&rev_argv, arg);
|
strvec_push(&rev_argv, arg);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -412,31 +416,31 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
}
|
}
|
||||||
|
|
||||||
if (commit_graph && !ctx.qry.follow) {
|
if (commit_graph && !ctx.qry.follow) {
|
||||||
argv_array_push(&rev_argv, "--graph");
|
strvec_push(&rev_argv, "--graph");
|
||||||
argv_array_push(&rev_argv, "--color");
|
strvec_push(&rev_argv, "--color");
|
||||||
graph_set_column_colors(column_colors_html,
|
graph_set_column_colors(column_colors_html,
|
||||||
COLUMN_COLORS_HTML_MAX);
|
COLUMN_COLORS_HTML_MAX);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (commit_sort == 1)
|
if (commit_sort == 1)
|
||||||
argv_array_push(&rev_argv, "--date-order");
|
strvec_push(&rev_argv, "--date-order");
|
||||||
else if (commit_sort == 2)
|
else if (commit_sort == 2)
|
||||||
argv_array_push(&rev_argv, "--topo-order");
|
strvec_push(&rev_argv, "--topo-order");
|
||||||
|
|
||||||
if (path && ctx.qry.follow)
|
if (path && ctx.qry.follow)
|
||||||
argv_array_push(&rev_argv, "--follow");
|
strvec_push(&rev_argv, "--follow");
|
||||||
argv_array_push(&rev_argv, "--");
|
strvec_push(&rev_argv, "--");
|
||||||
if (path)
|
if (path)
|
||||||
argv_array_push(&rev_argv, path);
|
strvec_push(&rev_argv, path);
|
||||||
|
|
||||||
init_revisions(&rev, NULL);
|
repo_init_revisions(the_repository, &rev, NULL);
|
||||||
rev.abbrev = DEFAULT_ABBREV;
|
rev.abbrev = DEFAULT_ABBREV;
|
||||||
rev.commit_format = CMIT_FMT_DEFAULT;
|
rev.commit_format = CMIT_FMT_DEFAULT;
|
||||||
rev.verbose_header = 1;
|
rev.verbose_header = 1;
|
||||||
rev.show_root_diff = 0;
|
rev.show_root_diff = 0;
|
||||||
rev.ignore_missing = 1;
|
rev.ignore_missing = 1;
|
||||||
rev.simplify_history = 1;
|
rev.simplify_history = 1;
|
||||||
setup_revisions(rev_argv.argc, rev_argv.argv, &rev, NULL);
|
setup_revisions(rev_argv.nr, rev_argv.v, &rev, NULL);
|
||||||
load_ref_decorations(NULL, DECORATE_FULL_REFS);
|
load_ref_decorations(NULL, DECORATE_FULL_REFS);
|
||||||
rev.show_decorations = 1;
|
rev.show_decorations = 1;
|
||||||
rev.grep_filter.ignore_case = 1;
|
rev.grep_filter.ignore_case = 1;
|
||||||
|
@ -463,7 +467,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
if (pager) {
|
if (pager) {
|
||||||
html(" (");
|
html(" (");
|
||||||
cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL,
|
cgit_log_link(ctx.qry.showmsg ? "Collapse" : "Expand", NULL,
|
||||||
NULL, ctx.qry.head, ctx.qry.sha1,
|
NULL, ctx.qry.head, ctx.qry.oid,
|
||||||
ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep,
|
ctx.qry.vpath, ctx.qry.ofs, ctx.qry.grep,
|
||||||
ctx.qry.search, ctx.qry.showmsg ? 0 : 1,
|
ctx.qry.search, ctx.qry.showmsg ? 0 : 1,
|
||||||
ctx.qry.follow);
|
ctx.qry.follow);
|
||||||
|
@ -488,8 +492,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; /* nop */) {
|
for (i = 0; i < ofs && (commit = get_revision(&rev)) != NULL; /* nop */) {
|
||||||
if (show_commit(commit, &rev))
|
if (show_commit(commit, &rev))
|
||||||
i++;
|
i++;
|
||||||
free_commit_buffer(commit);
|
release_commit_memory(the_repository->parsed_objects, commit);
|
||||||
free_commit_list(commit->parents);
|
|
||||||
commit->parents = NULL;
|
commit->parents = NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -510,8 +513,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
i++;
|
i++;
|
||||||
print_commit(commit, &rev);
|
print_commit(commit, &rev);
|
||||||
}
|
}
|
||||||
free_commit_buffer(commit);
|
release_commit_memory(the_repository->parsed_objects, commit);
|
||||||
free_commit_list(commit->parents);
|
|
||||||
commit->parents = NULL;
|
commit->parents = NULL;
|
||||||
}
|
}
|
||||||
if (pager) {
|
if (pager) {
|
||||||
|
@ -519,7 +521,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
if (ofs > 0) {
|
if (ofs > 0) {
|
||||||
html("<li>");
|
html("<li>");
|
||||||
cgit_log_link("[prev]", NULL, NULL, ctx.qry.head,
|
cgit_log_link("[prev]", NULL, NULL, ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.vpath,
|
ctx.qry.oid, ctx.qry.vpath,
|
||||||
ofs - cnt, ctx.qry.grep,
|
ofs - cnt, ctx.qry.grep,
|
||||||
ctx.qry.search, ctx.qry.showmsg,
|
ctx.qry.search, ctx.qry.showmsg,
|
||||||
ctx.qry.follow);
|
ctx.qry.follow);
|
||||||
|
@ -528,7 +530,7 @@ void cgit_print_log(const char *tip, int ofs, int cnt, char *grep, char *pattern
|
||||||
if ((commit = get_revision(&rev)) != NULL) {
|
if ((commit = get_revision(&rev)) != NULL) {
|
||||||
html("<li>");
|
html("<li>");
|
||||||
cgit_log_link("[next]", NULL, NULL, ctx.qry.head,
|
cgit_log_link("[next]", NULL, NULL, ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.vpath,
|
ctx.qry.oid, ctx.qry.vpath,
|
||||||
ofs + cnt, ctx.qry.grep,
|
ofs + cnt, ctx.qry.grep,
|
||||||
ctx.qry.search, ctx.qry.showmsg,
|
ctx.qry.search, ctx.qry.showmsg,
|
||||||
ctx.qry.follow);
|
ctx.qry.follow);
|
||||||
|
|
2
ui-log.h
2
ui-log.h
|
@ -2,7 +2,7 @@
|
||||||
#define UI_LOG_H
|
#define UI_LOG_H
|
||||||
|
|
||||||
extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep,
|
extern void cgit_print_log(const char *tip, int ofs, int cnt, char *grep,
|
||||||
char *pattern, char *path, int pager,
|
char *pattern, const char *path, int pager,
|
||||||
int commit_graph, int commit_sort);
|
int commit_graph, int commit_sort);
|
||||||
extern void show_commit_decorations(struct commit *commit);
|
extern void show_commit_decorations(struct commit *commit);
|
||||||
|
|
||||||
|
|
23
ui-patch.c
23
ui-patch.c
|
@ -6,18 +6,23 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-patch.h"
|
#include "ui-patch.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
#include "ui-shared.h"
|
#include "ui-shared.h"
|
||||||
|
|
||||||
|
/* two commit hashes with two dots in between and termination */
|
||||||
|
#define REV_RANGE_LEN 2 * GIT_MAX_HEXSZ + 3
|
||||||
|
|
||||||
void cgit_print_patch(const char *new_rev, const char *old_rev,
|
void cgit_print_patch(const char *new_rev, const char *old_rev,
|
||||||
const char *prefix)
|
const char *prefix)
|
||||||
{
|
{
|
||||||
struct rev_info rev;
|
struct rev_info rev;
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct object_id new_rev_oid, old_rev_oid;
|
struct object_id new_rev_oid, old_rev_oid;
|
||||||
char rev_range[2 * 40 + 3];
|
char rev_range[REV_RANGE_LEN];
|
||||||
const char *rev_argv[] = { NULL, "--reverse", "--format=email", rev_range, "--", prefix, NULL };
|
const char *rev_argv[] = { NULL, "--reverse", "--format=email", rev_range, "--", prefix, NULL };
|
||||||
int rev_argc = ARRAY_SIZE(rev_argv) - 1;
|
int rev_argc = ARRAY_SIZE(rev_argv) - 1;
|
||||||
char *patchname;
|
char *patchname;
|
||||||
|
@ -28,12 +33,12 @@ void cgit_print_patch(const char *new_rev, const char *old_rev,
|
||||||
if (!new_rev)
|
if (!new_rev)
|
||||||
new_rev = ctx.qry.head;
|
new_rev = ctx.qry.head;
|
||||||
|
|
||||||
if (get_oid(new_rev, &new_rev_oid)) {
|
if (repo_get_oid(the_repository, new_rev, &new_rev_oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad object id: %s", new_rev);
|
"Bad object id: %s", new_rev);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(&new_rev_oid);
|
commit = lookup_commit_reference(the_repository, &new_rev_oid);
|
||||||
if (!commit) {
|
if (!commit) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad commit reference: %s", new_rev);
|
"Bad commit reference: %s", new_rev);
|
||||||
|
@ -41,12 +46,12 @@ void cgit_print_patch(const char *new_rev, const char *old_rev,
|
||||||
}
|
}
|
||||||
|
|
||||||
if (old_rev) {
|
if (old_rev) {
|
||||||
if (get_oid(old_rev, &old_rev_oid)) {
|
if (repo_get_oid(the_repository, old_rev, &old_rev_oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad object id: %s", old_rev);
|
"Bad object id: %s", old_rev);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!lookup_commit_reference(&old_rev_oid)) {
|
if (!lookup_commit_reference(the_repository, &old_rev_oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad commit reference: %s", old_rev);
|
"Bad commit reference: %s", old_rev);
|
||||||
return;
|
return;
|
||||||
|
@ -54,13 +59,13 @@ void cgit_print_patch(const char *new_rev, const char *old_rev,
|
||||||
} else if (commit->parents && commit->parents->item) {
|
} else if (commit->parents && commit->parents->item) {
|
||||||
oidcpy(&old_rev_oid, &commit->parents->item->object.oid);
|
oidcpy(&old_rev_oid, &commit->parents->item->object.oid);
|
||||||
} else {
|
} else {
|
||||||
oidclr(&old_rev_oid);
|
oidclr(&old_rev_oid, the_repository->hash_algo);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (is_null_oid(&old_rev_oid)) {
|
if (is_null_oid(&old_rev_oid)) {
|
||||||
memcpy(rev_range, oid_to_hex(&new_rev_oid), GIT_SHA1_HEXSZ + 1);
|
memcpy(rev_range, oid_to_hex(&new_rev_oid), the_hash_algo->hexsz + 1);
|
||||||
} else {
|
} else {
|
||||||
sprintf(rev_range, "%s..%s", oid_to_hex(&old_rev_oid),
|
xsnprintf(rev_range, REV_RANGE_LEN, "%s..%s", oid_to_hex(&old_rev_oid),
|
||||||
oid_to_hex(&new_rev_oid));
|
oid_to_hex(&new_rev_oid));
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -75,7 +80,7 @@ void cgit_print_patch(const char *new_rev, const char *old_rev,
|
||||||
"%s%n%n%w(0)%b";
|
"%s%n%n%w(0)%b";
|
||||||
}
|
}
|
||||||
|
|
||||||
init_revisions(&rev, NULL);
|
repo_init_revisions(the_repository, &rev, NULL);
|
||||||
rev.abbrev = DEFAULT_ABBREV;
|
rev.abbrev = DEFAULT_ABBREV;
|
||||||
rev.verbose_header = 1;
|
rev.verbose_header = 1;
|
||||||
rev.diff = 1;
|
rev.diff = 1;
|
||||||
|
|
23
ui-plain.c
23
ui-plain.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-plain.h"
|
#include "ui-plain.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -28,7 +30,7 @@ static int print_object(const struct object_id *oid, const char *path)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
buf = read_object_file(oid, &type, &size);
|
buf = repo_read_object_file(the_repository, oid, &type, &size);
|
||||||
if (!buf) {
|
if (!buf) {
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -99,7 +101,7 @@ static void print_dir(const struct object_id *oid, const char *base,
|
||||||
fullpath = NULL;
|
fullpath = NULL;
|
||||||
}
|
}
|
||||||
html("<li>");
|
html("<li>");
|
||||||
cgit_plain_link("../", NULL, NULL, ctx.qry.head, ctx.qry.sha1,
|
cgit_plain_link("../", NULL, NULL, ctx.qry.head, ctx.qry.oid,
|
||||||
fullpath);
|
fullpath);
|
||||||
html("</li>\n");
|
html("</li>\n");
|
||||||
}
|
}
|
||||||
|
@ -118,7 +120,7 @@ static void print_dir_entry(const struct object_id *oid, const char *base,
|
||||||
if (S_ISGITLINK(mode)) {
|
if (S_ISGITLINK(mode)) {
|
||||||
cgit_submodule_link(NULL, fullpath, oid_to_hex(oid));
|
cgit_submodule_link(NULL, fullpath, oid_to_hex(oid));
|
||||||
} else
|
} else
|
||||||
cgit_plain_link(path, NULL, NULL, ctx.qry.head, ctx.qry.sha1,
|
cgit_plain_link(path, NULL, NULL, ctx.qry.head, ctx.qry.oid,
|
||||||
fullpath);
|
fullpath);
|
||||||
html("</li>\n");
|
html("</li>\n");
|
||||||
free(fullpath);
|
free(fullpath);
|
||||||
|
@ -130,7 +132,7 @@ static void print_dir_tail(void)
|
||||||
}
|
}
|
||||||
|
|
||||||
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage, void *cbdata)
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
{
|
{
|
||||||
struct walk_tree_context *walk_tree_ctx = cbdata;
|
struct walk_tree_context *walk_tree_ctx = cbdata;
|
||||||
|
|
||||||
|
@ -163,7 +165,7 @@ static int basedir_len(const char *path)
|
||||||
|
|
||||||
void cgit_print_plain(void)
|
void cgit_print_plain(void)
|
||||||
{
|
{
|
||||||
const char *rev = ctx.qry.sha1;
|
const char *rev = ctx.qry.oid;
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
struct commit *commit;
|
struct commit *commit;
|
||||||
struct pathspec_item path_items = {
|
struct pathspec_item path_items = {
|
||||||
|
@ -181,24 +183,25 @@ void cgit_print_plain(void)
|
||||||
if (!rev)
|
if (!rev)
|
||||||
rev = ctx.qry.head;
|
rev = ctx.qry.head;
|
||||||
|
|
||||||
if (get_oid(rev, &oid)) {
|
if (repo_get_oid(the_repository, rev, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
if (!commit || parse_commit(commit)) {
|
if (!commit || repo_parse_commit(the_repository, commit)) {
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (!path_items.match) {
|
if (!path_items.match) {
|
||||||
path_items.match = "";
|
path_items.match = "";
|
||||||
walk_tree_ctx.match_baselen = -1;
|
walk_tree_ctx.match_baselen = -1;
|
||||||
print_dir(&commit->maybe_tree->object.oid, "", 0, "");
|
print_dir(get_commit_tree_oid(commit), "", 0, "");
|
||||||
walk_tree_ctx.match = 2;
|
walk_tree_ctx.match = 2;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
walk_tree_ctx.match_baselen = basedir_len(path_items.match);
|
walk_tree_ctx.match_baselen = basedir_len(path_items.match);
|
||||||
read_tree_recursive(commit->maybe_tree, "", 0, 0, &paths, walk_tree, &walk_tree_ctx);
|
read_tree(the_repository, repo_get_commit_tree(the_repository, commit),
|
||||||
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
if (!walk_tree_ctx.match)
|
if (!walk_tree_ctx.match)
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
else if (walk_tree_ctx.match == 2)
|
else if (walk_tree_ctx.match == 2)
|
||||||
|
|
13
ui-refs.c
13
ui-refs.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-refs.h"
|
#include "ui-refs.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -136,7 +138,7 @@ static int print_tag(struct refinfo *ref)
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_refs_link(char *path)
|
static void print_refs_link(const char *path)
|
||||||
{
|
{
|
||||||
html("<tr class='nohover'><td colspan='5'>");
|
html("<tr class='nohover'><td colspan='5'>");
|
||||||
cgit_refs_link("[...]", NULL, NULL, ctx.qry.head, NULL, path);
|
cgit_refs_link("[...]", NULL, NULL, ctx.qry.head, NULL, path);
|
||||||
|
@ -155,9 +157,11 @@ void cgit_print_branches(int maxcount)
|
||||||
|
|
||||||
list.refs = NULL;
|
list.refs = NULL;
|
||||||
list.alloc = list.count = 0;
|
list.alloc = list.count = 0;
|
||||||
for_each_branch_ref(cgit_refs_cb, &list);
|
refs_for_each_branch_ref(get_main_ref_store(the_repository),
|
||||||
|
cgit_refs_cb, &list);
|
||||||
if (ctx.repo->enable_remote_branches)
|
if (ctx.repo->enable_remote_branches)
|
||||||
for_each_remote_ref(cgit_refs_cb, &list);
|
refs_for_each_remote_ref(get_main_ref_store(the_repository),
|
||||||
|
cgit_refs_cb, &list);
|
||||||
|
|
||||||
if (maxcount == 0 || maxcount > list.count)
|
if (maxcount == 0 || maxcount > list.count)
|
||||||
maxcount = list.count;
|
maxcount = list.count;
|
||||||
|
@ -182,7 +186,8 @@ void cgit_print_tags(int maxcount)
|
||||||
|
|
||||||
list.refs = NULL;
|
list.refs = NULL;
|
||||||
list.alloc = list.count = 0;
|
list.alloc = list.count = 0;
|
||||||
for_each_tag_ref(cgit_refs_cb, &list);
|
refs_for_each_tag_ref(get_main_ref_store(the_repository),
|
||||||
|
cgit_refs_cb, &list);
|
||||||
if (list.count == 0)
|
if (list.count == 0)
|
||||||
return;
|
return;
|
||||||
qsort(list.refs, list.count, sizeof(*list.refs), cmp_tag_age);
|
qsort(list.refs, list.count, sizeof(*list.refs), cmp_tag_age);
|
||||||
|
|
|
@ -11,7 +11,7 @@
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
#include "ui-shared.h"
|
#include "ui-shared.h"
|
||||||
|
|
||||||
static time_t read_agefile(char *path)
|
static time_t read_agefile(const char *path)
|
||||||
{
|
{
|
||||||
time_t result;
|
time_t result;
|
||||||
size_t size;
|
size_t size;
|
||||||
|
@ -20,7 +20,7 @@ static time_t read_agefile(char *path)
|
||||||
|
|
||||||
if (readfile(path, &buf, &size)) {
|
if (readfile(path, &buf, &size)) {
|
||||||
free(buf);
|
free(buf);
|
||||||
return -1;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (parse_date(buf, &date_buf) == 0)
|
if (parse_date(buf, &date_buf) == 0)
|
||||||
|
@ -321,7 +321,7 @@ void cgit_print_repolist(void)
|
||||||
}
|
}
|
||||||
htmlf("<tr><td class='%s'>",
|
htmlf("<tr><td class='%s'>",
|
||||||
!sorted && section ? "sublevel-repo" : "toplevel-repo");
|
!sorted && section ? "sublevel-repo" : "toplevel-repo");
|
||||||
cgit_summary_link(ctx.repo->name, ctx.repo->name, NULL, NULL);
|
cgit_summary_link(ctx.repo->name, NULL, NULL, NULL);
|
||||||
html("</td><td>");
|
html("</td><td>");
|
||||||
repourl = cgit_repourl(ctx.repo->url);
|
repourl = cgit_repourl(ctx.repo->url);
|
||||||
html_link_open(repourl, NULL, NULL);
|
html_link_open(repourl, NULL, NULL);
|
||||||
|
|
196
ui-shared.c
196
ui-shared.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-shared.h"
|
#include "ui-shared.h"
|
||||||
#include "cmd.h"
|
#include "cmd.h"
|
||||||
|
@ -22,10 +24,11 @@ static char *http_date(time_t t)
|
||||||
static char month[][4] =
|
static char month[][4] =
|
||||||
{"Jan", "Feb", "Mar", "Apr", "May", "Jun",
|
{"Jan", "Feb", "Mar", "Apr", "May", "Jun",
|
||||||
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
|
"Jul", "Aug", "Sep", "Oct", "Nov", "Dec"};
|
||||||
struct tm *tm = gmtime(&t);
|
struct tm tm;
|
||||||
return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm->tm_wday],
|
gmtime_r(&t, &tm);
|
||||||
tm->tm_mday, month[tm->tm_mon], 1900 + tm->tm_year,
|
return fmt("%s, %02d %s %04d %02d:%02d:%02d GMT", day[tm.tm_wday],
|
||||||
tm->tm_hour, tm->tm_min, tm->tm_sec);
|
tm.tm_mday, month[tm.tm_mon], 1900 + tm.tm_year,
|
||||||
|
tm.tm_hour, tm.tm_min, tm.tm_sec);
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_print_error(const char *fmt, ...)
|
void cgit_print_error(const char *fmt, ...)
|
||||||
|
@ -68,15 +71,48 @@ char *cgit_hosturl(void)
|
||||||
char *cgit_currenturl(void)
|
char *cgit_currenturl(void)
|
||||||
{
|
{
|
||||||
const char *root = cgit_rooturl();
|
const char *root = cgit_rooturl();
|
||||||
size_t len = strlen(root);
|
|
||||||
|
|
||||||
if (!ctx.qry.url)
|
if (!ctx.qry.url)
|
||||||
return xstrdup(root);
|
return xstrdup(root);
|
||||||
if (len && root[len - 1] == '/')
|
if (root[0] && root[strlen(root) - 1] == '/')
|
||||||
return fmtalloc("%s%s", root, ctx.qry.url);
|
return fmtalloc("%s%s", root, ctx.qry.url);
|
||||||
return fmtalloc("%s/%s", root, ctx.qry.url);
|
return fmtalloc("%s/%s", root, ctx.qry.url);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
char *cgit_currentfullurl(void)
|
||||||
|
{
|
||||||
|
const char *root = cgit_rooturl();
|
||||||
|
const char *orig_query = ctx.env.query_string ? ctx.env.query_string : "";
|
||||||
|
size_t len = strlen(orig_query);
|
||||||
|
char *query = xmalloc(len + 2), *start_url, *ret;
|
||||||
|
|
||||||
|
/* Remove all url=... parts from query string */
|
||||||
|
memcpy(query + 1, orig_query, len + 1);
|
||||||
|
query[0] = '?';
|
||||||
|
start_url = query;
|
||||||
|
while ((start_url = strstr(start_url, "url=")) != NULL) {
|
||||||
|
if (start_url[-1] == '?' || start_url[-1] == '&') {
|
||||||
|
const char *end_url = strchr(start_url, '&');
|
||||||
|
if (end_url)
|
||||||
|
memmove(start_url, end_url + 1, strlen(end_url));
|
||||||
|
else
|
||||||
|
start_url[0] = '\0';
|
||||||
|
} else
|
||||||
|
++start_url;
|
||||||
|
}
|
||||||
|
if (!query[1])
|
||||||
|
query[0] = '\0';
|
||||||
|
|
||||||
|
if (!ctx.qry.url)
|
||||||
|
ret = fmtalloc("%s%s", root, query);
|
||||||
|
else if (root[0] && root[strlen(root) - 1] == '/')
|
||||||
|
ret = fmtalloc("%s%s%s", root, ctx.qry.url, query);
|
||||||
|
else
|
||||||
|
ret = fmtalloc("%s/%s%s", root, ctx.qry.url, query);
|
||||||
|
free(query);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
const char *cgit_rooturl(void)
|
const char *cgit_rooturl(void)
|
||||||
{
|
{
|
||||||
if (ctx.cfg.virtual_root)
|
if (ctx.cfg.virtual_root)
|
||||||
|
@ -488,45 +524,45 @@ static void cgit_self_link(char *name, const char *title, const char *class)
|
||||||
else if (!strcmp(ctx.qry.page, "summary"))
|
else if (!strcmp(ctx.qry.page, "summary"))
|
||||||
cgit_summary_link(name, title, class, ctx.qry.head);
|
cgit_summary_link(name, title, class, ctx.qry.head);
|
||||||
else if (!strcmp(ctx.qry.page, "tag"))
|
else if (!strcmp(ctx.qry.page, "tag"))
|
||||||
cgit_tag_link(name, title, class, ctx.qry.has_sha1 ?
|
cgit_tag_link(name, title, class, ctx.qry.has_oid ?
|
||||||
ctx.qry.sha1 : ctx.qry.head);
|
ctx.qry.oid : ctx.qry.head);
|
||||||
else if (!strcmp(ctx.qry.page, "tree"))
|
else if (!strcmp(ctx.qry.page, "tree"))
|
||||||
cgit_tree_link(name, title, class, ctx.qry.head,
|
cgit_tree_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "plain"))
|
else if (!strcmp(ctx.qry.page, "plain"))
|
||||||
cgit_plain_link(name, title, class, ctx.qry.head,
|
cgit_plain_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "blame"))
|
else if (!strcmp(ctx.qry.page, "blame"))
|
||||||
cgit_blame_link(name, title, class, ctx.qry.head,
|
cgit_blame_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "log"))
|
else if (!strcmp(ctx.qry.page, "log"))
|
||||||
cgit_log_link(name, title, class, ctx.qry.head,
|
cgit_log_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path, ctx.qry.ofs,
|
ctx.qry.path, ctx.qry.ofs,
|
||||||
ctx.qry.grep, ctx.qry.search,
|
ctx.qry.grep, ctx.qry.search,
|
||||||
ctx.qry.showmsg, ctx.qry.follow);
|
ctx.qry.showmsg, ctx.qry.follow);
|
||||||
else if (!strcmp(ctx.qry.page, "commit"))
|
else if (!strcmp(ctx.qry.page, "commit"))
|
||||||
cgit_commit_link(name, title, class, ctx.qry.head,
|
cgit_commit_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "patch"))
|
else if (!strcmp(ctx.qry.page, "patch"))
|
||||||
cgit_patch_link(name, title, class, ctx.qry.head,
|
cgit_patch_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "refs"))
|
else if (!strcmp(ctx.qry.page, "refs"))
|
||||||
cgit_refs_link(name, title, class, ctx.qry.head,
|
cgit_refs_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "snapshot"))
|
else if (!strcmp(ctx.qry.page, "snapshot"))
|
||||||
cgit_snapshot_link(name, title, class, ctx.qry.head,
|
cgit_snapshot_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.has_sha1 ? ctx.qry.sha1 : NULL,
|
ctx.qry.has_oid ? ctx.qry.oid : NULL,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "diff"))
|
else if (!strcmp(ctx.qry.page, "diff"))
|
||||||
cgit_diff_link(name, title, class, ctx.qry.head,
|
cgit_diff_link(name, title, class, ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.sha2,
|
ctx.qry.oid, ctx.qry.oid2,
|
||||||
ctx.qry.path);
|
ctx.qry.path);
|
||||||
else if (!strcmp(ctx.qry.page, "stats"))
|
else if (!strcmp(ctx.qry.page, "stats"))
|
||||||
cgit_stats_link(name, title, class, ctx.qry.head,
|
cgit_stats_link(name, title, class, ctx.qry.head,
|
||||||
|
@ -628,18 +664,18 @@ void cgit_submodule_link(const char *class, char *path, const char *rev)
|
||||||
path[len - 1] = tail;
|
path[len - 1] = tail;
|
||||||
}
|
}
|
||||||
|
|
||||||
const struct date_mode *cgit_date_mode(enum date_mode_type type)
|
const struct date_mode cgit_date_mode(enum date_mode_type type)
|
||||||
{
|
{
|
||||||
static struct date_mode mode;
|
static struct date_mode mode;
|
||||||
mode.type = type;
|
mode.type = type;
|
||||||
mode.local = ctx.cfg.local_time;
|
mode.local = ctx.cfg.local_time;
|
||||||
return &mode;
|
return mode;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_rel_date(time_t t, int tz, double value,
|
static void print_rel_date(time_t t, int tz, double value,
|
||||||
const char *class, const char *suffix)
|
const char *class, const char *suffix)
|
||||||
{
|
{
|
||||||
htmlf("<span class='%s' title='", class);
|
htmlf("<span class='%s' data-ut='%" PRIu64 "' title='", class, (uint64_t)t);
|
||||||
html_attr(show_date(t, tz, cgit_date_mode(DATE_ISO8601)));
|
html_attr(show_date(t, tz, cgit_date_mode(DATE_ISO8601)));
|
||||||
htmlf("'>%.0f %s</span>", value, suffix);
|
htmlf("'>%.0f %s</span>", value, suffix);
|
||||||
}
|
}
|
||||||
|
@ -734,6 +770,38 @@ static void print_rel_vcs_link(const char *url)
|
||||||
html(" Git repository'/>\n");
|
html(" Git repository'/>\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int emit_css_link(struct string_list_item *s, void *arg)
|
||||||
|
{
|
||||||
|
/* Do not emit anything if css= is specified. */
|
||||||
|
if (s && *s->string == '\0')
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
html("<link rel='stylesheet' type='text/css' href='");
|
||||||
|
if (s)
|
||||||
|
html_attr(s->string);
|
||||||
|
else
|
||||||
|
html_attr((const char *)arg);
|
||||||
|
html("'/>\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
static int emit_js_link(struct string_list_item *s, void *arg)
|
||||||
|
{
|
||||||
|
/* Do not emit anything if js= is specified. */
|
||||||
|
if (s && *s->string == '\0')
|
||||||
|
return 0;
|
||||||
|
|
||||||
|
html("<script type='text/javascript' src='");
|
||||||
|
if (s)
|
||||||
|
html_attr(s->string);
|
||||||
|
else
|
||||||
|
html_attr((const char *)arg);
|
||||||
|
html("'></script>\n");
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
void cgit_print_docstart(void)
|
void cgit_print_docstart(void)
|
||||||
{
|
{
|
||||||
char *host = cgit_hosturl();
|
char *host = cgit_hosturl();
|
||||||
|
@ -753,9 +821,17 @@ void cgit_print_docstart(void)
|
||||||
htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
|
htmlf("<meta name='generator' content='cgit %s'/>\n", cgit_version);
|
||||||
if (ctx.cfg.robots && *ctx.cfg.robots)
|
if (ctx.cfg.robots && *ctx.cfg.robots)
|
||||||
htmlf("<meta name='robots' content='%s'/>\n", ctx.cfg.robots);
|
htmlf("<meta name='robots' content='%s'/>\n", ctx.cfg.robots);
|
||||||
html("<link rel='stylesheet' type='text/css' href='");
|
|
||||||
html_attr(ctx.cfg.css);
|
if (ctx.cfg.css.items)
|
||||||
html("'/>\n");
|
for_each_string_list(&ctx.cfg.css, emit_css_link, NULL);
|
||||||
|
else
|
||||||
|
emit_css_link(NULL, "/cgit.css");
|
||||||
|
|
||||||
|
if (ctx.cfg.js.items)
|
||||||
|
for_each_string_list(&ctx.cfg.js, emit_js_link, NULL);
|
||||||
|
else
|
||||||
|
emit_js_link(NULL, "/cgit.js");
|
||||||
|
|
||||||
if (ctx.cfg.favicon) {
|
if (ctx.cfg.favicon) {
|
||||||
html("<link rel='shortcut icon' href='");
|
html("<link rel='shortcut icon' href='");
|
||||||
html_attr(ctx.cfg.favicon);
|
html_attr(ctx.cfg.favicon);
|
||||||
|
@ -885,10 +961,10 @@ void cgit_add_hidden_formfields(int incl_head, int incl_search,
|
||||||
strcmp(ctx.qry.head, ctx.repo->defbranch))
|
strcmp(ctx.qry.head, ctx.repo->defbranch))
|
||||||
html_hidden("h", ctx.qry.head);
|
html_hidden("h", ctx.qry.head);
|
||||||
|
|
||||||
if (ctx.qry.sha1)
|
if (ctx.qry.oid)
|
||||||
html_hidden("id", ctx.qry.sha1);
|
html_hidden("id", ctx.qry.oid);
|
||||||
if (ctx.qry.sha2)
|
if (ctx.qry.oid2)
|
||||||
html_hidden("id2", ctx.qry.sha2);
|
html_hidden("id2", ctx.qry.oid2);
|
||||||
if (ctx.qry.showmsg)
|
if (ctx.qry.showmsg)
|
||||||
html_hidden("showmsg", "1");
|
html_hidden("showmsg", "1");
|
||||||
|
|
||||||
|
@ -912,12 +988,13 @@ static void cgit_print_path_crumbs(char *path)
|
||||||
{
|
{
|
||||||
char *old_path = ctx.qry.path;
|
char *old_path = ctx.qry.path;
|
||||||
char *p = path, *q, *end = path + strlen(path);
|
char *p = path, *q, *end = path + strlen(path);
|
||||||
|
int levels = 0;
|
||||||
|
|
||||||
ctx.qry.path = NULL;
|
ctx.qry.path = NULL;
|
||||||
cgit_self_link("root", NULL, NULL);
|
cgit_self_link("root", NULL, NULL);
|
||||||
ctx.qry.path = p = path;
|
ctx.qry.path = p = path;
|
||||||
while (p < end) {
|
while (p < end) {
|
||||||
if (!(q = strchr(p, '/')))
|
if (!(q = strchr(p, '/')) || levels > 15)
|
||||||
q = end;
|
q = end;
|
||||||
*q = '\0';
|
*q = '\0';
|
||||||
html_txt("/");
|
html_txt("/");
|
||||||
|
@ -925,6 +1002,7 @@ static void cgit_print_path_crumbs(char *path)
|
||||||
if (q < end)
|
if (q < end)
|
||||||
*q = '/';
|
*q = '/';
|
||||||
p = q + 1;
|
p = q + 1;
|
||||||
|
++levels;
|
||||||
}
|
}
|
||||||
ctx.qry.path = old_path;
|
ctx.qry.path = old_path;
|
||||||
}
|
}
|
||||||
|
@ -959,15 +1037,17 @@ static void print_header(void)
|
||||||
if (ctx.repo) {
|
if (ctx.repo) {
|
||||||
cgit_index_link("index", NULL, NULL, NULL, NULL, 0, 1);
|
cgit_index_link("index", NULL, NULL, NULL, NULL, 0, 1);
|
||||||
html(" : ");
|
html(" : ");
|
||||||
cgit_summary_link(ctx.repo->name, ctx.repo->name, NULL, NULL);
|
cgit_summary_link(ctx.repo->name, NULL, NULL, NULL);
|
||||||
if (ctx.env.authenticated) {
|
if (ctx.env.authenticated) {
|
||||||
html("</td><td class='form'>");
|
html("</td><td class='form'>");
|
||||||
html("<form method='get'>\n");
|
html("<form method='get'>\n");
|
||||||
cgit_add_hidden_formfields(0, 1, ctx.qry.page);
|
cgit_add_hidden_formfields(0, 1, ctx.qry.page);
|
||||||
html("<select name='h' onchange='this.form.submit();'>\n");
|
html("<select name='h' onchange='this.form.submit();'>\n");
|
||||||
for_each_branch_ref(print_branch_option, ctx.qry.head);
|
refs_for_each_branch_ref(get_main_ref_store(the_repository),
|
||||||
|
print_branch_option, ctx.qry.head);
|
||||||
if (ctx.repo->enable_remote_branches)
|
if (ctx.repo->enable_remote_branches)
|
||||||
for_each_remote_ref(print_branch_option, ctx.qry.head);
|
refs_for_each_remote_ref(get_main_ref_store(the_repository),
|
||||||
|
print_branch_option, ctx.qry.head);
|
||||||
html("</select> ");
|
html("</select> ");
|
||||||
html("<input type='submit' value='switch'/>");
|
html("<input type='submit' value='switch'/>");
|
||||||
html("</form>");
|
html("</form>");
|
||||||
|
@ -980,7 +1060,13 @@ static void print_header(void)
|
||||||
if (ctx.repo) {
|
if (ctx.repo) {
|
||||||
html_txt(ctx.repo->desc);
|
html_txt(ctx.repo->desc);
|
||||||
html("</td><td class='sub right'>");
|
html("</td><td class='sub right'>");
|
||||||
|
if (ctx.repo->owner_filter) {
|
||||||
|
cgit_open_filter(ctx.repo->owner_filter);
|
||||||
html_txt(ctx.repo->owner);
|
html_txt(ctx.repo->owner);
|
||||||
|
cgit_close_filter(ctx.repo->owner_filter);
|
||||||
|
} else {
|
||||||
|
html_txt(ctx.repo->owner);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
if (ctx.cfg.root_desc)
|
if (ctx.cfg.root_desc)
|
||||||
html_txt(ctx.cfg.root_desc);
|
html_txt(ctx.cfg.root_desc);
|
||||||
|
@ -1003,20 +1089,20 @@ void cgit_print_pageheader(void)
|
||||||
cgit_summary_link("summary", NULL, hc("summary"),
|
cgit_summary_link("summary", NULL, hc("summary"),
|
||||||
ctx.qry.head);
|
ctx.qry.head);
|
||||||
cgit_refs_link("refs", NULL, hc("refs"), ctx.qry.head,
|
cgit_refs_link("refs", NULL, hc("refs"), ctx.qry.head,
|
||||||
ctx.qry.sha1, NULL);
|
ctx.qry.oid, NULL);
|
||||||
cgit_log_link("log", NULL, hc("log"), ctx.qry.head,
|
cgit_log_link("log", NULL, hc("log"), ctx.qry.head,
|
||||||
NULL, ctx.qry.vpath, 0, NULL, NULL,
|
NULL, ctx.qry.vpath, 0, NULL, NULL,
|
||||||
ctx.qry.showmsg, ctx.qry.follow);
|
ctx.qry.showmsg, ctx.qry.follow);
|
||||||
if (ctx.qry.page && !strcmp(ctx.qry.page, "blame"))
|
if (ctx.qry.page && !strcmp(ctx.qry.page, "blame"))
|
||||||
cgit_blame_link("blame", NULL, hc("blame"), ctx.qry.head,
|
cgit_blame_link("blame", NULL, hc("blame"), ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.vpath);
|
ctx.qry.oid, ctx.qry.vpath);
|
||||||
else
|
else
|
||||||
cgit_tree_link("tree", NULL, hc("tree"), ctx.qry.head,
|
cgit_tree_link("tree", NULL, hc("tree"), ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.vpath);
|
ctx.qry.oid, ctx.qry.vpath);
|
||||||
cgit_commit_link("commit", NULL, hc("commit"),
|
cgit_commit_link("commit", NULL, hc("commit"),
|
||||||
ctx.qry.head, ctx.qry.sha1, ctx.qry.vpath);
|
ctx.qry.head, ctx.qry.oid, ctx.qry.vpath);
|
||||||
cgit_diff_link("diff", NULL, hc("diff"), ctx.qry.head,
|
cgit_diff_link("diff", NULL, hc("diff"), ctx.qry.head,
|
||||||
ctx.qry.sha1, ctx.qry.sha2, ctx.qry.vpath);
|
ctx.qry.oid, ctx.qry.oid2, ctx.qry.vpath);
|
||||||
if (ctx.repo->max_stats)
|
if (ctx.repo->max_stats)
|
||||||
cgit_stats_link("stats", NULL, hc("stats"),
|
cgit_stats_link("stats", NULL, hc("stats"),
|
||||||
ctx.qry.head, ctx.qry.vpath);
|
ctx.qry.head, ctx.qry.vpath);
|
||||||
|
@ -1106,11 +1192,11 @@ void cgit_compose_snapshot_prefix(struct strbuf *filename, const char *base,
|
||||||
* name starts with {v,V}[0-9] and the prettify mapping is injective,
|
* name starts with {v,V}[0-9] and the prettify mapping is injective,
|
||||||
* i.e. each stripped tag can be inverted without ambiguities.
|
* i.e. each stripped tag can be inverted without ambiguities.
|
||||||
*/
|
*/
|
||||||
if (get_oid(fmt("refs/tags/%s", ref), &oid) == 0 &&
|
if (repo_get_oid(the_repository, fmt("refs/tags/%s", ref), &oid) == 0 &&
|
||||||
(ref[0] == 'v' || ref[0] == 'V') && isdigit(ref[1]) &&
|
(ref[0] == 'v' || ref[0] == 'V') && isdigit(ref[1]) &&
|
||||||
((get_oid(fmt("refs/tags/%s", ref + 1), &oid) == 0) +
|
((repo_get_oid(the_repository, fmt("refs/tags/%s", ref + 1), &oid) == 0) +
|
||||||
(get_oid(fmt("refs/tags/v%s", ref + 1), &oid) == 0) +
|
(repo_get_oid(the_repository, fmt("refs/tags/v%s", ref + 1), &oid) == 0) +
|
||||||
(get_oid(fmt("refs/tags/V%s", ref + 1), &oid) == 0) == 1))
|
(repo_get_oid(the_repository, fmt("refs/tags/V%s", ref + 1), &oid) == 0) == 1))
|
||||||
ref++;
|
ref++;
|
||||||
|
|
||||||
strbuf_addf(filename, "%s-%s", base, ref);
|
strbuf_addf(filename, "%s-%s", base, ref);
|
||||||
|
@ -1159,31 +1245,17 @@ void cgit_print_snapshot_links(const struct cgit_repo *repo, const char *ref,
|
||||||
|
|
||||||
void cgit_set_title_from_path(const char *path)
|
void cgit_set_title_from_path(const char *path)
|
||||||
{
|
{
|
||||||
size_t path_len, path_index, path_last_end;
|
struct strbuf sb = STRBUF_INIT;
|
||||||
char *new_title;
|
const char *slash, *last_slash;
|
||||||
|
|
||||||
if (!path)
|
if (!path)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
path_len = strlen(path);
|
for (last_slash = path + strlen(path); (slash = memrchr(path, '/', last_slash - path)) != NULL; last_slash = slash) {
|
||||||
new_title = xmalloc(path_len + 3 + strlen(ctx.page.title) + 1);
|
strbuf_add(&sb, slash + 1, last_slash - slash - 1);
|
||||||
new_title[0] = '\0';
|
strbuf_addstr(&sb, " \xc2\xab ");
|
||||||
|
|
||||||
for (path_index = path_len, path_last_end = path_len; path_index-- > 0;) {
|
|
||||||
if (path[path_index] == '/') {
|
|
||||||
if (path_index == path_len - 1) {
|
|
||||||
path_last_end = path_index - 1;
|
|
||||||
continue;
|
|
||||||
}
|
}
|
||||||
strncat(new_title, &path[path_index + 1], path_last_end - path_index - 1);
|
strbuf_add(&sb, path, last_slash - path);
|
||||||
strcat(new_title, "\\");
|
strbuf_addf(&sb, " - %s", ctx.page.title);
|
||||||
path_last_end = path_index;
|
ctx.page.title = strbuf_detach(&sb, NULL);
|
||||||
}
|
|
||||||
}
|
|
||||||
if (path_last_end)
|
|
||||||
strncat(new_title, path, path_last_end);
|
|
||||||
|
|
||||||
strcat(new_title, " - ");
|
|
||||||
strcat(new_title, ctx.page.title);
|
|
||||||
ctx.page.title = new_title;
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -5,6 +5,7 @@ extern const char *cgit_httpscheme(void);
|
||||||
extern char *cgit_hosturl(void);
|
extern char *cgit_hosturl(void);
|
||||||
extern const char *cgit_rooturl(void);
|
extern const char *cgit_rooturl(void);
|
||||||
extern char *cgit_currenturl(void);
|
extern char *cgit_currenturl(void);
|
||||||
|
extern char *cgit_currentfullurl(void);
|
||||||
extern const char *cgit_loginurl(void);
|
extern const char *cgit_loginurl(void);
|
||||||
extern char *cgit_repourl(const char *reponame);
|
extern char *cgit_repourl(const char *reponame);
|
||||||
extern char *cgit_fileurl(const char *reponame, const char *pagename,
|
extern char *cgit_fileurl(const char *reponame, const char *pagename,
|
||||||
|
@ -64,7 +65,7 @@ __attribute__((format (printf,1,2)))
|
||||||
extern void cgit_print_error(const char *fmt, ...);
|
extern void cgit_print_error(const char *fmt, ...);
|
||||||
__attribute__((format (printf,1,0)))
|
__attribute__((format (printf,1,0)))
|
||||||
extern void cgit_vprint_error(const char *fmt, va_list ap);
|
extern void cgit_vprint_error(const char *fmt, va_list ap);
|
||||||
extern const struct date_mode *cgit_date_mode(enum date_mode_type type);
|
extern const struct date_mode cgit_date_mode(enum date_mode_type type);
|
||||||
extern void cgit_print_age(time_t t, int tz, time_t max_relative);
|
extern void cgit_print_age(time_t t, int tz, time_t max_relative);
|
||||||
extern void cgit_print_http_headers(void);
|
extern void cgit_print_http_headers(void);
|
||||||
extern void cgit_redirect(const char *url, bool permanent);
|
extern void cgit_redirect(const char *url, bool permanent);
|
||||||
|
|
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-snapshot.h"
|
#include "ui-snapshot.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -13,32 +15,32 @@
|
||||||
|
|
||||||
static int write_archive_type(const char *format, const char *hex, const char *prefix)
|
static int write_archive_type(const char *format, const char *hex, const char *prefix)
|
||||||
{
|
{
|
||||||
struct argv_array argv = ARGV_ARRAY_INIT;
|
struct strvec argv = STRVEC_INIT;
|
||||||
const char **nargv;
|
const char **nargv;
|
||||||
int result;
|
int result;
|
||||||
argv_array_push(&argv, "snapshot");
|
strvec_push(&argv, "snapshot");
|
||||||
argv_array_push(&argv, format);
|
strvec_push(&argv, format);
|
||||||
if (prefix) {
|
if (prefix) {
|
||||||
struct strbuf buf = STRBUF_INIT;
|
struct strbuf buf = STRBUF_INIT;
|
||||||
strbuf_addstr(&buf, prefix);
|
strbuf_addstr(&buf, prefix);
|
||||||
strbuf_addch(&buf, '/');
|
strbuf_addch(&buf, '/');
|
||||||
argv_array_push(&argv, "--prefix");
|
strvec_push(&argv, "--prefix");
|
||||||
argv_array_push(&argv, buf.buf);
|
strvec_push(&argv, buf.buf);
|
||||||
strbuf_release(&buf);
|
strbuf_release(&buf);
|
||||||
}
|
}
|
||||||
argv_array_push(&argv, hex);
|
strvec_push(&argv, hex);
|
||||||
/*
|
/*
|
||||||
* Now we need to copy the pointers to arguments into a new
|
* Now we need to copy the pointers to arguments into a new
|
||||||
* structure because write_archive will rearrange its arguments
|
* structure because write_archive will rearrange its arguments
|
||||||
* which may result in duplicated/missing entries causing leaks
|
* which may result in duplicated/missing entries causing leaks
|
||||||
* or double-frees in argv_array_clear.
|
* or double-frees in strvec_clear.
|
||||||
*/
|
*/
|
||||||
nargv = xmalloc(sizeof(char *) * (argv.argc + 1));
|
nargv = xmalloc(sizeof(char *) * (argv.nr + 1));
|
||||||
/* argv_array guarantees a trailing NULL entry. */
|
/* strvec guarantees a trailing NULL entry. */
|
||||||
memcpy(nargv, argv.argv, sizeof(char *) * (argv.argc + 1));
|
memcpy(nargv, argv.v, sizeof(char *) * (argv.nr + 1));
|
||||||
|
|
||||||
result = write_archive(argv.argc, nargv, NULL, NULL, 0);
|
result = write_archive(argv.nr, nargv, NULL, the_repository, NULL, 0);
|
||||||
argv_array_clear(&argv);
|
strvec_clear(&argv);
|
||||||
free(nargv);
|
free(nargv);
|
||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
@ -79,18 +81,32 @@ static int write_tar_bzip2_archive(const char *hex, const char *prefix)
|
||||||
return write_compressed_tar_archive(hex, prefix, argv);
|
return write_compressed_tar_archive(hex, prefix, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int write_tar_lzip_archive(const char *hex, const char *prefix)
|
||||||
|
{
|
||||||
|
char *argv[] = { "lzip", NULL };
|
||||||
|
return write_compressed_tar_archive(hex, prefix, argv);
|
||||||
|
}
|
||||||
|
|
||||||
static int write_tar_xz_archive(const char *hex, const char *prefix)
|
static int write_tar_xz_archive(const char *hex, const char *prefix)
|
||||||
{
|
{
|
||||||
char *argv[] = { "xz", NULL };
|
char *argv[] = { "xz", NULL };
|
||||||
return write_compressed_tar_archive(hex, prefix, argv);
|
return write_compressed_tar_archive(hex, prefix, argv);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static int write_tar_zstd_archive(const char *hex, const char *prefix)
|
||||||
|
{
|
||||||
|
char *argv[] = { "zstd", "-T0", NULL };
|
||||||
|
return write_compressed_tar_archive(hex, prefix, argv);
|
||||||
|
}
|
||||||
|
|
||||||
const struct cgit_snapshot_format cgit_snapshot_formats[] = {
|
const struct cgit_snapshot_format cgit_snapshot_formats[] = {
|
||||||
/* .tar must remain the 0 index */
|
/* .tar must remain the 0 index */
|
||||||
{ ".tar", "application/x-tar", write_tar_archive },
|
{ ".tar", "application/x-tar", write_tar_archive },
|
||||||
{ ".tar.gz", "application/x-gzip", write_tar_gzip_archive },
|
{ ".tar.gz", "application/x-gzip", write_tar_gzip_archive },
|
||||||
{ ".tar.bz2", "application/x-bzip2", write_tar_bzip2_archive },
|
{ ".tar.bz2", "application/x-bzip2", write_tar_bzip2_archive },
|
||||||
|
{ ".tar.lz", "application/x-lzip", write_tar_lzip_archive },
|
||||||
{ ".tar.xz", "application/x-xz", write_tar_xz_archive },
|
{ ".tar.xz", "application/x-xz", write_tar_xz_archive },
|
||||||
|
{ ".tar.zst", "application/x-zstd", write_tar_zstd_archive },
|
||||||
{ ".zip", "application/x-zip", write_zip_archive },
|
{ ".zip", "application/x-zip", write_zip_archive },
|
||||||
{ NULL }
|
{ NULL }
|
||||||
};
|
};
|
||||||
|
@ -103,7 +119,7 @@ const struct object_id *cgit_snapshot_get_sig(const char *ref,
|
||||||
struct notes_tree *tree;
|
struct notes_tree *tree;
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
|
|
||||||
if (get_oid(ref, &oid))
|
if (repo_get_oid(the_repository, ref, &oid))
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
tree = &snapshot_sig_notes[f - &cgit_snapshot_formats[0]];
|
tree = &snapshot_sig_notes[f - &cgit_snapshot_formats[0]];
|
||||||
|
@ -142,12 +158,12 @@ static int make_snapshot(const struct cgit_snapshot_format *format,
|
||||||
{
|
{
|
||||||
struct object_id oid;
|
struct object_id oid;
|
||||||
|
|
||||||
if (get_oid(hex, &oid)) {
|
if (repo_get_oid(the_repository, hex, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad object id: %s", hex);
|
"Bad object id: %s", hex);
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
if (!lookup_commit_reference(&oid)) {
|
if (!lookup_commit_reference(the_repository, &oid)) {
|
||||||
cgit_print_error_page(400, "Bad request",
|
cgit_print_error_page(400, "Bad request",
|
||||||
"Not a commit reference: %s", hex);
|
"Not a commit reference: %s", hex);
|
||||||
return 1;
|
return 1;
|
||||||
|
@ -156,6 +172,7 @@ static int make_snapshot(const struct cgit_snapshot_format *format,
|
||||||
ctx.page.mimetype = xstrdup(format->mimetype);
|
ctx.page.mimetype = xstrdup(format->mimetype);
|
||||||
ctx.page.filename = xstrdup(filename);
|
ctx.page.filename = xstrdup(filename);
|
||||||
cgit_print_http_headers();
|
cgit_print_http_headers();
|
||||||
|
init_archivers();
|
||||||
format->write_func(hex, prefix);
|
format->write_func(hex, prefix);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@ -175,7 +192,7 @@ static int write_sig(const struct cgit_snapshot_format *format,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
buf = read_object_file(note, &type, &size);
|
buf = repo_read_object_file(the_repository, note, &type, &size);
|
||||||
if (!buf) {
|
if (!buf) {
|
||||||
cgit_print_error_page(404, "Not found", "Not found");
|
cgit_print_error_page(404, "Not found", "Not found");
|
||||||
return 0;
|
return 0;
|
||||||
|
@ -215,7 +232,7 @@ static const char *get_ref_from_filename(const struct cgit_repo *repo,
|
||||||
strbuf_addstr(&snapshot, filename);
|
strbuf_addstr(&snapshot, filename);
|
||||||
strbuf_setlen(&snapshot, snapshot.len - strlen(format->suffix));
|
strbuf_setlen(&snapshot, snapshot.len - strlen(format->suffix));
|
||||||
|
|
||||||
if (get_oid(snapshot.buf, &oid) == 0)
|
if (repo_get_oid(the_repository, snapshot.buf, &oid) == 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
reponame = cgit_snapshot_prefix(repo);
|
reponame = cgit_snapshot_prefix(repo);
|
||||||
|
@ -227,15 +244,15 @@ static const char *get_ref_from_filename(const struct cgit_repo *repo,
|
||||||
strbuf_splice(&snapshot, 0, new_start - snapshot.buf, "", 0);
|
strbuf_splice(&snapshot, 0, new_start - snapshot.buf, "", 0);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (get_oid(snapshot.buf, &oid) == 0)
|
if (repo_get_oid(the_repository, snapshot.buf, &oid) == 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
strbuf_insert(&snapshot, 0, "v", 1);
|
strbuf_insert(&snapshot, 0, "v", 1);
|
||||||
if (get_oid(snapshot.buf, &oid) == 0)
|
if (repo_get_oid(the_repository, snapshot.buf, &oid) == 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
strbuf_splice(&snapshot, 0, 1, "V", 1);
|
strbuf_splice(&snapshot, 0, 1, "V", 1);
|
||||||
if (get_oid(snapshot.buf, &oid) == 0)
|
if (repo_get_oid(the_repository, snapshot.buf, &oid) == 0)
|
||||||
goto out;
|
goto out;
|
||||||
|
|
||||||
result = 0;
|
result = 0;
|
||||||
|
|
23
ui-ssdiff.c
23
ui-ssdiff.c
|
@ -103,8 +103,7 @@ static int line_from_hunk(char *line, char type)
|
||||||
return 0;
|
return 0;
|
||||||
len = buf2 - buf1;
|
len = buf2 - buf1;
|
||||||
buf2 = xmalloc(len + 1);
|
buf2 = xmalloc(len + 1);
|
||||||
strncpy(buf2, buf1, len);
|
strlcpy(buf2, buf1, len + 1);
|
||||||
buf2[len] = '\0';
|
|
||||||
res = atoi(buf2);
|
res = atoi(buf2);
|
||||||
free(buf2);
|
free(buf2);
|
||||||
return res;
|
return res;
|
||||||
|
@ -118,6 +117,7 @@ static char *replace_tabs(char *line)
|
||||||
int n_tabs = 0;
|
int n_tabs = 0;
|
||||||
int i;
|
int i;
|
||||||
char *result;
|
char *result;
|
||||||
|
size_t result_len;
|
||||||
|
|
||||||
if (linelen == 0) {
|
if (linelen == 0) {
|
||||||
result = xmalloc(1);
|
result = xmalloc(1);
|
||||||
|
@ -129,16 +129,19 @@ static char *replace_tabs(char *line)
|
||||||
if (line[i] == '\t')
|
if (line[i] == '\t')
|
||||||
n_tabs += 1;
|
n_tabs += 1;
|
||||||
}
|
}
|
||||||
result = xmalloc(linelen + n_tabs * 8 + 1);
|
result_len = linelen + n_tabs * 8;
|
||||||
|
result = xmalloc(result_len + 1);
|
||||||
result[0] = '\0';
|
result[0] = '\0';
|
||||||
|
|
||||||
for (;;) {
|
for (;;) {
|
||||||
cur_buf = strchr(prev_buf, '\t');
|
cur_buf = strchr(prev_buf, '\t');
|
||||||
if (!cur_buf) {
|
if (!cur_buf) {
|
||||||
strcat(result, prev_buf);
|
linelen = strlen(result);
|
||||||
|
strlcpy(&result[linelen], prev_buf, result_len - linelen + 1);
|
||||||
break;
|
break;
|
||||||
} else {
|
} else {
|
||||||
strncat(result, prev_buf, cur_buf - prev_buf);
|
linelen = strlen(result);
|
||||||
|
strlcpy(&result[linelen], prev_buf, cur_buf - prev_buf + 1);
|
||||||
linelen = strlen(result);
|
linelen = strlen(result);
|
||||||
memset(&result[linelen], ' ', 8 - (linelen % 8));
|
memset(&result[linelen], ' ', 8 - (linelen % 8));
|
||||||
result[linelen + 8 - (linelen % 8)] = '\0';
|
result[linelen + 8 - (linelen % 8)] = '\0';
|
||||||
|
@ -206,11 +209,13 @@ static void print_part_with_lcs(char *class, char *line, char *lcs)
|
||||||
}
|
}
|
||||||
} else if (line[i] == lcs[j]) {
|
} else if (line[i] == lcs[j]) {
|
||||||
same = 1;
|
same = 1;
|
||||||
htmlf("</span>");
|
html("</span>");
|
||||||
j += 1;
|
j += 1;
|
||||||
}
|
}
|
||||||
html_txt(c);
|
html_txt(c);
|
||||||
}
|
}
|
||||||
|
if (!same)
|
||||||
|
html("</span>");
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_ssdiff_line(char *class,
|
static void print_ssdiff_line(char *class,
|
||||||
|
@ -235,7 +240,7 @@ static void print_ssdiff_line(char *class,
|
||||||
char *fileurl = cgit_fileurl(ctx.repo->url, "tree", old_file->path, id_str);
|
char *fileurl = cgit_fileurl(ctx.repo->url, "tree", old_file->path, id_str);
|
||||||
html("<td class='lineno'><a href='");
|
html("<td class='lineno'><a href='");
|
||||||
html(fileurl);
|
html(fileurl);
|
||||||
htmlf("' id='%s'>%s</a>", lineno_str, lineno_str + 1);
|
htmlf("'>%s</a>", lineno_str + 1);
|
||||||
html("</td>");
|
html("</td>");
|
||||||
htmlf("<td class='%s'>", class);
|
htmlf("<td class='%s'>", class);
|
||||||
free(fileurl);
|
free(fileurl);
|
||||||
|
@ -258,7 +263,7 @@ static void print_ssdiff_line(char *class,
|
||||||
char *fileurl = cgit_fileurl(ctx.repo->url, "tree", new_file->path, id_str);
|
char *fileurl = cgit_fileurl(ctx.repo->url, "tree", new_file->path, id_str);
|
||||||
html("<td class='lineno'><a href='");
|
html("<td class='lineno'><a href='");
|
||||||
html(fileurl);
|
html(fileurl);
|
||||||
htmlf("' id='%s'>%s</a>", lineno_str, lineno_str + 1);
|
htmlf("'>%s</a>", lineno_str + 1);
|
||||||
html("</td>");
|
html("</td>");
|
||||||
htmlf("<td class='%s'>", class);
|
htmlf("<td class='%s'>", class);
|
||||||
free(fileurl);
|
free(fileurl);
|
||||||
|
@ -404,7 +409,7 @@ void cgit_ssdiff_header_begin(void)
|
||||||
|
|
||||||
void cgit_ssdiff_header_end(void)
|
void cgit_ssdiff_header_end(void)
|
||||||
{
|
{
|
||||||
html("</td><tr>");
|
html("</td></tr>");
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_ssdiff_footer(void)
|
void cgit_ssdiff_footer(void)
|
||||||
|
|
63
ui-stats.c
63
ui-stats.c
|
@ -1,3 +1,13 @@
|
||||||
|
/* ui-stats.c: generate stats view
|
||||||
|
*
|
||||||
|
* Copyright (C) 2006-2014 cgit Development Team <cgit@lists.zx2c4.com>
|
||||||
|
*
|
||||||
|
* Licensed under GNU General Public License v2
|
||||||
|
* (see COPYING for full license text)
|
||||||
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-stats.h"
|
#include "ui-stats.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -166,7 +176,7 @@ static void add_commit(struct string_list *authors, struct commit *commit,
|
||||||
struct authorstat *authorstat;
|
struct authorstat *authorstat;
|
||||||
struct string_list *items;
|
struct string_list *items;
|
||||||
char *tmp;
|
char *tmp;
|
||||||
struct tm *date;
|
struct tm date;
|
||||||
time_t t;
|
time_t t;
|
||||||
uintptr_t *counter;
|
uintptr_t *counter;
|
||||||
|
|
||||||
|
@ -180,9 +190,9 @@ static void add_commit(struct string_list *authors, struct commit *commit,
|
||||||
authorstat = author->util;
|
authorstat = author->util;
|
||||||
items = &authorstat->list;
|
items = &authorstat->list;
|
||||||
t = info->committer_date;
|
t = info->committer_date;
|
||||||
date = gmtime(&t);
|
gmtime_r(&t, &date);
|
||||||
period->trunc(date);
|
period->trunc(&date);
|
||||||
tmp = xstrdup(period->pretty(date));
|
tmp = xstrdup(period->pretty(&date));
|
||||||
item = string_list_insert(items, tmp);
|
item = string_list_insert(items, tmp);
|
||||||
counter = (uintptr_t *)&item->util;
|
counter = (uintptr_t *)&item->util;
|
||||||
if (*counter)
|
if (*counter)
|
||||||
|
@ -215,22 +225,22 @@ static struct string_list collect_stats(const struct cgit_period *period)
|
||||||
int argc = 3;
|
int argc = 3;
|
||||||
time_t now;
|
time_t now;
|
||||||
long i;
|
long i;
|
||||||
struct tm *tm;
|
struct tm tm;
|
||||||
char tmp[11];
|
char tmp[11];
|
||||||
|
|
||||||
time(&now);
|
time(&now);
|
||||||
tm = gmtime(&now);
|
gmtime_r(&now, &tm);
|
||||||
period->trunc(tm);
|
period->trunc(&tm);
|
||||||
for (i = 1; i < period->count; i++)
|
for (i = 1; i < period->count; i++)
|
||||||
period->dec(tm);
|
period->dec(&tm);
|
||||||
strftime(tmp, sizeof(tmp), "%Y-%m-%d", tm);
|
strftime(tmp, sizeof(tmp), "%Y-%m-%d", &tm);
|
||||||
argv[2] = xstrdup(fmt("--since=%s", tmp));
|
argv[2] = xstrdup(fmt("--since=%s", tmp));
|
||||||
if (ctx.qry.path) {
|
if (ctx.qry.path) {
|
||||||
argv[3] = "--";
|
argv[3] = "--";
|
||||||
argv[4] = ctx.qry.path;
|
argv[4] = ctx.qry.path;
|
||||||
argc += 2;
|
argc += 2;
|
||||||
}
|
}
|
||||||
init_revisions(&rev, NULL);
|
repo_init_revisions(the_repository, &rev, NULL);
|
||||||
rev.abbrev = DEFAULT_ABBREV;
|
rev.abbrev = DEFAULT_ABBREV;
|
||||||
rev.commit_format = CMIT_FMT_DEFAULT;
|
rev.commit_format = CMIT_FMT_DEFAULT;
|
||||||
rev.max_parents = 1;
|
rev.max_parents = 1;
|
||||||
|
@ -241,8 +251,7 @@ static struct string_list collect_stats(const struct cgit_period *period)
|
||||||
memset(&authors, 0, sizeof(authors));
|
memset(&authors, 0, sizeof(authors));
|
||||||
while ((commit = get_revision(&rev)) != NULL) {
|
while ((commit = get_revision(&rev)) != NULL) {
|
||||||
add_commit(&authors, commit, period);
|
add_commit(&authors, commit, period);
|
||||||
free_commit_buffer(commit);
|
release_commit_memory(the_repository->parsed_objects, commit);
|
||||||
free_commit_list(commit->parents);
|
|
||||||
commit->parents = NULL;
|
commit->parents = NULL;
|
||||||
}
|
}
|
||||||
return authors;
|
return authors;
|
||||||
|
@ -261,21 +270,21 @@ static void print_combined_authorrow(struct string_list *authors, int from,
|
||||||
struct string_list_item *date;
|
struct string_list_item *date;
|
||||||
time_t now;
|
time_t now;
|
||||||
long i, j, total, subtotal;
|
long i, j, total, subtotal;
|
||||||
struct tm *tm;
|
struct tm tm;
|
||||||
char *tmp;
|
char *tmp;
|
||||||
|
|
||||||
time(&now);
|
time(&now);
|
||||||
tm = gmtime(&now);
|
gmtime_r(&now, &tm);
|
||||||
period->trunc(tm);
|
period->trunc(&tm);
|
||||||
for (i = 1; i < period->count; i++)
|
for (i = 1; i < period->count; i++)
|
||||||
period->dec(tm);
|
period->dec(&tm);
|
||||||
|
|
||||||
total = 0;
|
total = 0;
|
||||||
htmlf("<tr><td class='%s'>%s</td>", leftclass,
|
htmlf("<tr><td class='%s'>%s</td>", leftclass,
|
||||||
fmt(name, to - from + 1));
|
fmt(name, to - from + 1));
|
||||||
for (j = 0; j < period->count; j++) {
|
for (j = 0; j < period->count; j++) {
|
||||||
tmp = period->pretty(tm);
|
tmp = period->pretty(&tm);
|
||||||
period->inc(tm);
|
period->inc(&tm);
|
||||||
subtotal = 0;
|
subtotal = 0;
|
||||||
for (i = from; i <= to; i++) {
|
for (i = from; i <= to; i++) {
|
||||||
author = &authors->items[i];
|
author = &authors->items[i];
|
||||||
|
@ -300,20 +309,20 @@ static void print_authors(struct string_list *authors, int top,
|
||||||
struct string_list_item *date;
|
struct string_list_item *date;
|
||||||
time_t now;
|
time_t now;
|
||||||
long i, j, total;
|
long i, j, total;
|
||||||
struct tm *tm;
|
struct tm tm;
|
||||||
char *tmp;
|
char *tmp;
|
||||||
|
|
||||||
time(&now);
|
time(&now);
|
||||||
tm = gmtime(&now);
|
gmtime_r(&now, &tm);
|
||||||
period->trunc(tm);
|
period->trunc(&tm);
|
||||||
for (i = 1; i < period->count; i++)
|
for (i = 1; i < period->count; i++)
|
||||||
period->dec(tm);
|
period->dec(&tm);
|
||||||
|
|
||||||
html("<table class='stats'><tr><th>Author</th>");
|
html("<table class='stats'><tr><th>Author</th>");
|
||||||
for (j = 0; j < period->count; j++) {
|
for (j = 0; j < period->count; j++) {
|
||||||
tmp = period->pretty(tm);
|
tmp = period->pretty(&tm);
|
||||||
htmlf("<th>%s</th>", tmp);
|
htmlf("<th>%s</th>", tmp);
|
||||||
period->inc(tm);
|
period->inc(&tm);
|
||||||
}
|
}
|
||||||
html("<th>Total</th></tr>\n");
|
html("<th>Total</th></tr>\n");
|
||||||
|
|
||||||
|
@ -329,10 +338,10 @@ static void print_authors(struct string_list *authors, int top,
|
||||||
items = &authorstat->list;
|
items = &authorstat->list;
|
||||||
total = 0;
|
total = 0;
|
||||||
for (j = 0; j < period->count; j++)
|
for (j = 0; j < period->count; j++)
|
||||||
period->dec(tm);
|
period->dec(&tm);
|
||||||
for (j = 0; j < period->count; j++) {
|
for (j = 0; j < period->count; j++) {
|
||||||
tmp = period->pretty(tm);
|
tmp = period->pretty(&tm);
|
||||||
period->inc(tm);
|
period->inc(&tm);
|
||||||
date = string_list_lookup(items, tmp);
|
date = string_list_lookup(items, tmp);
|
||||||
if (!date)
|
if (!date)
|
||||||
html("<td>0</td>");
|
html("<td>0</td>");
|
||||||
|
|
|
@ -99,7 +99,7 @@ static char* append_readme_path(const char *filename, const char *ref, const cha
|
||||||
return full_path;
|
return full_path;
|
||||||
}
|
}
|
||||||
|
|
||||||
void cgit_print_repo_readme(char *path)
|
void cgit_print_repo_readme(const char *path)
|
||||||
{
|
{
|
||||||
char *filename, *ref, *mimetype;
|
char *filename, *ref, *mimetype;
|
||||||
int free_filename = 0;
|
int free_filename = 0;
|
||||||
|
|
|
@ -2,6 +2,6 @@
|
||||||
#define UI_SUMMARY_H
|
#define UI_SUMMARY_H
|
||||||
|
|
||||||
extern void cgit_print_summary(void);
|
extern void cgit_print_summary(void);
|
||||||
extern void cgit_print_repo_readme(char *path);
|
extern void cgit_print_repo_readme(const char *path);
|
||||||
|
|
||||||
#endif /* UI_SUMMARY_H */
|
#endif /* UI_SUMMARY_H */
|
||||||
|
|
18
ui-tag.c
18
ui-tag.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-tag.h"
|
#include "ui-tag.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -33,7 +35,7 @@ static void print_tag_content(char *buf)
|
||||||
|
|
||||||
static void print_download_links(char *revname)
|
static void print_download_links(char *revname)
|
||||||
{
|
{
|
||||||
html("<tr><th>download</th><td class='sha1'>");
|
html("<tr><th>download</th><td class='oid'>");
|
||||||
cgit_print_snapshot_links(ctx.repo, revname, "<br/>");
|
cgit_print_snapshot_links(ctx.repo, revname, "<br/>");
|
||||||
html("</td></tr>");
|
html("</td></tr>");
|
||||||
}
|
}
|
||||||
|
@ -48,12 +50,12 @@ void cgit_print_tag(char *revname)
|
||||||
revname = ctx.qry.head;
|
revname = ctx.qry.head;
|
||||||
|
|
||||||
strbuf_addf(&fullref, "refs/tags/%s", revname);
|
strbuf_addf(&fullref, "refs/tags/%s", revname);
|
||||||
if (get_oid(fullref.buf, &oid)) {
|
if (repo_get_oid(the_repository, fullref.buf, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Bad tag reference: %s", revname);
|
"Bad tag reference: %s", revname);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
obj = parse_object(&oid);
|
obj = parse_object(the_repository, &oid);
|
||||||
if (!obj) {
|
if (!obj) {
|
||||||
cgit_print_error_page(500, "Internal server error",
|
cgit_print_error_page(500, "Internal server error",
|
||||||
"Bad object id: %s", oid_to_hex(&oid));
|
"Bad object id: %s", oid_to_hex(&oid));
|
||||||
|
@ -63,7 +65,7 @@ void cgit_print_tag(char *revname)
|
||||||
struct tag *tag;
|
struct tag *tag;
|
||||||
struct taginfo *info;
|
struct taginfo *info;
|
||||||
|
|
||||||
tag = lookup_tag(&oid);
|
tag = lookup_tag(the_repository, &oid);
|
||||||
if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag))) {
|
if (!tag || parse_tag(tag) || !(info = cgit_parse_tag(tag))) {
|
||||||
cgit_print_error_page(500, "Internal server error",
|
cgit_print_error_page(500, "Internal server error",
|
||||||
"Bad tag object: %s", revname);
|
"Bad tag object: %s", revname);
|
||||||
|
@ -71,7 +73,7 @@ void cgit_print_tag(char *revname)
|
||||||
}
|
}
|
||||||
cgit_print_layout_start();
|
cgit_print_layout_start();
|
||||||
html("<table class='commit-info'>\n");
|
html("<table class='commit-info'>\n");
|
||||||
htmlf("<tr><td>tag name</td><td>");
|
html("<tr><td>tag name</td><td>");
|
||||||
html_txt(revname);
|
html_txt(revname);
|
||||||
htmlf(" (%s)</td></tr>\n", oid_to_hex(&oid));
|
htmlf(" (%s)</td></tr>\n", oid_to_hex(&oid));
|
||||||
if (info->tagger_date > 0) {
|
if (info->tagger_date > 0) {
|
||||||
|
@ -91,7 +93,7 @@ void cgit_print_tag(char *revname)
|
||||||
cgit_close_filter(ctx.repo->email_filter);
|
cgit_close_filter(ctx.repo->email_filter);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
}
|
}
|
||||||
html("<tr><td>tagged object</td><td class='sha1'>");
|
html("<tr><td>tagged object</td><td class='oid'>");
|
||||||
cgit_object_link(tag->tagged);
|
cgit_object_link(tag->tagged);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
if (ctx.repo->snapshots)
|
if (ctx.repo->snapshots)
|
||||||
|
@ -103,10 +105,10 @@ void cgit_print_tag(char *revname)
|
||||||
} else {
|
} else {
|
||||||
cgit_print_layout_start();
|
cgit_print_layout_start();
|
||||||
html("<table class='commit-info'>\n");
|
html("<table class='commit-info'>\n");
|
||||||
htmlf("<tr><td>tag name</td><td>");
|
html("<tr><td>tag name</td><td>");
|
||||||
html_txt(revname);
|
html_txt(revname);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
html("<tr><td>tagged object</td><td class='sha1'>");
|
html("<tr><td>tagged object</td><td class='oid'>");
|
||||||
cgit_object_link(obj);
|
cgit_object_link(obj);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
if (ctx.repo->snapshots)
|
if (ctx.repo->snapshots)
|
||||||
|
|
65
ui-tree.c
65
ui-tree.c
|
@ -6,6 +6,8 @@
|
||||||
* (see COPYING for full license text)
|
* (see COPYING for full license text)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
#define USE_THE_REPOSITORY_VARIABLE
|
||||||
|
|
||||||
#include "cgit.h"
|
#include "cgit.h"
|
||||||
#include "ui-tree.h"
|
#include "ui-tree.h"
|
||||||
#include "html.h"
|
#include "html.h"
|
||||||
|
@ -84,11 +86,12 @@ static void print_binary_buffer(char *buf, unsigned long size)
|
||||||
html("</table>\n");
|
html("</table>\n");
|
||||||
}
|
}
|
||||||
|
|
||||||
static void print_object(const struct object_id *oid, char *path, const char *basename, const char *rev)
|
static void print_object(const struct object_id *oid, const char *path, const char *basename, const char *rev)
|
||||||
{
|
{
|
||||||
enum object_type type;
|
enum object_type type;
|
||||||
char *buf;
|
char *buf;
|
||||||
unsigned long size;
|
unsigned long size;
|
||||||
|
bool is_binary;
|
||||||
|
|
||||||
type = oid_object_info(the_repository, oid, &size);
|
type = oid_object_info(the_repository, oid, &size);
|
||||||
if (type == OBJ_BAD) {
|
if (type == OBJ_BAD) {
|
||||||
|
@ -97,12 +100,13 @@ static void print_object(const struct object_id *oid, char *path, const char *ba
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
buf = read_object_file(oid, &type, &size);
|
buf = repo_read_object_file(the_repository, oid, &type, &size);
|
||||||
if (!buf) {
|
if (!buf) {
|
||||||
cgit_print_error_page(500, "Internal server error",
|
cgit_print_error_page(500, "Internal server error",
|
||||||
"Error reading object %s", oid_to_hex(oid));
|
"Error reading object %s", oid_to_hex(oid));
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
is_binary = buffer_is_binary(buf, size);
|
||||||
|
|
||||||
cgit_set_title_from_path(path);
|
cgit_set_title_from_path(path);
|
||||||
|
|
||||||
|
@ -110,7 +114,7 @@ static void print_object(const struct object_id *oid, char *path, const char *ba
|
||||||
htmlf("blob: %s (", oid_to_hex(oid));
|
htmlf("blob: %s (", oid_to_hex(oid));
|
||||||
cgit_plain_link("plain", NULL, NULL, ctx.qry.head,
|
cgit_plain_link("plain", NULL, NULL, ctx.qry.head,
|
||||||
rev, path);
|
rev, path);
|
||||||
if (ctx.cfg.enable_blame) {
|
if (ctx.repo->enable_blame && !is_binary) {
|
||||||
html(") (");
|
html(") (");
|
||||||
cgit_blame_link("blame", NULL, NULL, ctx.qry.head,
|
cgit_blame_link("blame", NULL, NULL, ctx.qry.head,
|
||||||
rev, path);
|
rev, path);
|
||||||
|
@ -123,7 +127,7 @@ static void print_object(const struct object_id *oid, char *path, const char *ba
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (buffer_is_binary(buf, size))
|
if (is_binary)
|
||||||
print_binary_buffer(buf, size);
|
print_binary_buffer(buf, size);
|
||||||
else
|
else
|
||||||
print_text_buffer(basename, buf, size);
|
print_text_buffer(basename, buf, size);
|
||||||
|
@ -139,8 +143,7 @@ struct single_tree_ctx {
|
||||||
};
|
};
|
||||||
|
|
||||||
static int single_tree_cb(const struct object_id *oid, struct strbuf *base,
|
static int single_tree_cb(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage,
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
void *cbdata)
|
|
||||||
{
|
{
|
||||||
struct single_tree_ctx *ctx = cbdata;
|
struct single_tree_ctx *ctx = cbdata;
|
||||||
|
|
||||||
|
@ -177,7 +180,7 @@ static void write_tree_link(const struct object_id *oid, char *name,
|
||||||
cgit_tree_link(name, NULL, "ls-dir", ctx.qry.head, rev,
|
cgit_tree_link(name, NULL, "ls-dir", ctx.qry.head, rev,
|
||||||
fullpath->buf);
|
fullpath->buf);
|
||||||
|
|
||||||
tree = lookup_tree(&tree_ctx.oid);
|
tree = lookup_tree(the_repository, &tree_ctx.oid);
|
||||||
if (!tree)
|
if (!tree)
|
||||||
return;
|
return;
|
||||||
|
|
||||||
|
@ -185,8 +188,7 @@ static void write_tree_link(const struct object_id *oid, char *name,
|
||||||
tree_ctx.name = NULL;
|
tree_ctx.name = NULL;
|
||||||
tree_ctx.count = 0;
|
tree_ctx.count = 0;
|
||||||
|
|
||||||
read_tree_recursive(tree, "", 0, 1, &paths, single_tree_cb,
|
read_tree(the_repository, tree, &paths, single_tree_cb, &tree_ctx);
|
||||||
&tree_ctx);
|
|
||||||
|
|
||||||
if (tree_ctx.count != 1)
|
if (tree_ctx.count != 1)
|
||||||
break;
|
break;
|
||||||
|
@ -199,14 +201,16 @@ static void write_tree_link(const struct object_id *oid, char *name,
|
||||||
}
|
}
|
||||||
|
|
||||||
static int ls_item(const struct object_id *oid, struct strbuf *base,
|
static int ls_item(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage, void *cbdata)
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
{
|
{
|
||||||
struct walk_tree_context *walk_tree_ctx = cbdata;
|
struct walk_tree_context *walk_tree_ctx = cbdata;
|
||||||
char *name;
|
char *name;
|
||||||
struct strbuf fullpath = STRBUF_INIT;
|
struct strbuf fullpath = STRBUF_INIT;
|
||||||
|
struct strbuf linkpath = STRBUF_INIT;
|
||||||
struct strbuf class = STRBUF_INIT;
|
struct strbuf class = STRBUF_INIT;
|
||||||
enum object_type type;
|
enum object_type type;
|
||||||
unsigned long size = 0;
|
unsigned long size = 0;
|
||||||
|
char *buf;
|
||||||
|
|
||||||
name = xstrdup(pathname);
|
name = xstrdup(pathname);
|
||||||
strbuf_addf(&fullpath, "%s%s%s", ctx.qry.path ? ctx.qry.path : "",
|
strbuf_addf(&fullpath, "%s%s%s", ctx.qry.path ? ctx.qry.path : "",
|
||||||
|
@ -218,8 +222,7 @@ static int ls_item(const struct object_id *oid, struct strbuf *base,
|
||||||
htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>",
|
htmlf("<tr><td colspan='3'>Bad object: %s %s</td></tr>",
|
||||||
name,
|
name,
|
||||||
oid_to_hex(oid));
|
oid_to_hex(oid));
|
||||||
free(name);
|
goto cleanup;
|
||||||
return 0;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -239,6 +242,21 @@ static int ls_item(const struct object_id *oid, struct strbuf *base,
|
||||||
cgit_tree_link(name, NULL, class.buf, ctx.qry.head,
|
cgit_tree_link(name, NULL, class.buf, ctx.qry.head,
|
||||||
walk_tree_ctx->curr_rev, fullpath.buf);
|
walk_tree_ctx->curr_rev, fullpath.buf);
|
||||||
}
|
}
|
||||||
|
if (S_ISLNK(mode)) {
|
||||||
|
html(" -> ");
|
||||||
|
buf = repo_read_object_file(the_repository, oid, &type, &size);
|
||||||
|
if (!buf) {
|
||||||
|
htmlf("Error reading object: %s", oid_to_hex(oid));
|
||||||
|
goto cleanup;
|
||||||
|
}
|
||||||
|
strbuf_addbuf(&linkpath, &fullpath);
|
||||||
|
strbuf_addf(&linkpath, "/../%s", buf);
|
||||||
|
strbuf_normalize_path(&linkpath);
|
||||||
|
cgit_tree_link(buf, NULL, class.buf, ctx.qry.head,
|
||||||
|
walk_tree_ctx->curr_rev, linkpath.buf);
|
||||||
|
free(buf);
|
||||||
|
strbuf_release(&linkpath);
|
||||||
|
}
|
||||||
htmlf("</td><td class='ls-size'>%li</td>", size);
|
htmlf("</td><td class='ls-size'>%li</td>", size);
|
||||||
|
|
||||||
html("<td>");
|
html("<td>");
|
||||||
|
@ -251,10 +269,12 @@ static int ls_item(const struct object_id *oid, struct strbuf *base,
|
||||||
if (!S_ISGITLINK(mode))
|
if (!S_ISGITLINK(mode))
|
||||||
cgit_plain_link("plain", NULL, "button", ctx.qry.head,
|
cgit_plain_link("plain", NULL, "button", ctx.qry.head,
|
||||||
walk_tree_ctx->curr_rev, fullpath.buf);
|
walk_tree_ctx->curr_rev, fullpath.buf);
|
||||||
if (!S_ISDIR(mode) && ctx.cfg.enable_blame)
|
if (!S_ISDIR(mode) && ctx.repo->enable_blame)
|
||||||
cgit_blame_link("blame", NULL, "button", ctx.qry.head,
|
cgit_blame_link("blame", NULL, "button", ctx.qry.head,
|
||||||
walk_tree_ctx->curr_rev, fullpath.buf);
|
walk_tree_ctx->curr_rev, fullpath.buf);
|
||||||
html("</td></tr>\n");
|
html("</td></tr>\n");
|
||||||
|
|
||||||
|
cleanup:
|
||||||
free(name);
|
free(name);
|
||||||
strbuf_release(&fullpath);
|
strbuf_release(&fullpath);
|
||||||
strbuf_release(&class);
|
strbuf_release(&class);
|
||||||
|
@ -279,7 +299,7 @@ static void ls_tail(void)
|
||||||
cgit_print_layout_end();
|
cgit_print_layout_end();
|
||||||
}
|
}
|
||||||
|
|
||||||
static void ls_tree(const struct object_id *oid, char *path, struct walk_tree_context *walk_tree_ctx)
|
static void ls_tree(const struct object_id *oid, const char *path, struct walk_tree_context *walk_tree_ctx)
|
||||||
{
|
{
|
||||||
struct tree *tree;
|
struct tree *tree;
|
||||||
struct pathspec paths = {
|
struct pathspec paths = {
|
||||||
|
@ -294,13 +314,13 @@ static void ls_tree(const struct object_id *oid, char *path, struct walk_tree_co
|
||||||
}
|
}
|
||||||
|
|
||||||
ls_head();
|
ls_head();
|
||||||
read_tree_recursive(tree, "", 0, 1, &paths, ls_item, walk_tree_ctx);
|
read_tree(the_repository, tree, &paths, ls_item, walk_tree_ctx);
|
||||||
ls_tail();
|
ls_tail();
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
||||||
const char *pathname, unsigned mode, int stage, void *cbdata)
|
const char *pathname, unsigned mode, void *cbdata)
|
||||||
{
|
{
|
||||||
struct walk_tree_context *walk_tree_ctx = cbdata;
|
struct walk_tree_context *walk_tree_ctx = cbdata;
|
||||||
|
|
||||||
|
@ -325,7 +345,7 @@ static int walk_tree(const struct object_id *oid, struct strbuf *base,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
ls_item(oid, base, pathname, mode, stage, walk_tree_ctx);
|
ls_item(oid, base, pathname, mode, walk_tree_ctx);
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -354,13 +374,13 @@ void cgit_print_tree(const char *rev, char *path)
|
||||||
if (!rev)
|
if (!rev)
|
||||||
rev = ctx.qry.head;
|
rev = ctx.qry.head;
|
||||||
|
|
||||||
if (get_oid(rev, &oid)) {
|
if (repo_get_oid(the_repository, rev, &oid)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Invalid revision name: %s", rev);
|
"Invalid revision name: %s", rev);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
commit = lookup_commit_reference(&oid);
|
commit = lookup_commit_reference(the_repository, &oid);
|
||||||
if (!commit || parse_commit(commit)) {
|
if (!commit || repo_parse_commit(the_repository, commit)) {
|
||||||
cgit_print_error_page(404, "Not found",
|
cgit_print_error_page(404, "Not found",
|
||||||
"Invalid commit reference: %s", rev);
|
"Invalid commit reference: %s", rev);
|
||||||
return;
|
return;
|
||||||
|
@ -369,11 +389,12 @@ void cgit_print_tree(const char *rev, char *path)
|
||||||
walk_tree_ctx.curr_rev = xstrdup(rev);
|
walk_tree_ctx.curr_rev = xstrdup(rev);
|
||||||
|
|
||||||
if (path == NULL) {
|
if (path == NULL) {
|
||||||
ls_tree(&commit->maybe_tree->object.oid, NULL, &walk_tree_ctx);
|
ls_tree(get_commit_tree_oid(commit), NULL, &walk_tree_ctx);
|
||||||
goto cleanup;
|
goto cleanup;
|
||||||
}
|
}
|
||||||
|
|
||||||
read_tree_recursive(commit->maybe_tree, "", 0, 0, &paths, walk_tree, &walk_tree_ctx);
|
read_tree(the_repository, repo_get_commit_tree(the_repository, commit),
|
||||||
|
&paths, walk_tree, &walk_tree_ctx);
|
||||||
if (walk_tree_ctx.state == 1)
|
if (walk_tree_ctx.state == 1)
|
||||||
ls_tail();
|
ls_tail();
|
||||||
else if (walk_tree_ctx.state == 2)
|
else if (walk_tree_ctx.state == 2)
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue