ref: b38b770691bdd012bd66c9f51da7af2b41c46e7a
dir: /query.c/
#include <u.h> #include <libc.h> #include "git.h" #pragma varargck type "P" void int fullpath; int changes; char *path[128]; int npath; int Pfmt(Fmt *f) { int i, n; n = 0; for(i = 0; i < npath; i++) n += fmtprint(f, "%s/", path[i]); return n; } void showdir(Hash dh, char *dname, char m) { Dirent *p, *e; Object *d; path[npath++] = dname; if((d = readobject(dh)) == nil) sysfatal("bad hash %H", dh); assert(d->type == GTree); p = d->tree->ent; e = p + d->tree->nent; for(; p != e; p++){ if(p->ismod) continue; if(p->mode & DMDIR) showdir(p->h, p->name, m); else print("%c %P%s\n", m, p->name); } unref(d); npath--; } void show(Dirent *e, char m) { if(e->mode & DMDIR) showdir(e->h, e->name, m); else print("%c %P%s\n", m, e->name); } void difftrees(Hash ah, Hash bh) { Dirent *ap, *bp, *ae, *be; Object *a, *b; int c; if((a = readobject(ah)) == nil) sysfatal("bad hash %H", ah); if((b = readobject(bh)) == nil) sysfatal("bad hash %H", bh); if(a->type != b->type) return; switch(a->type){ case GCommit: difftrees(a->commit->tree, b->commit->tree); break; case GTree: ap = a->tree->ent; ae = ap + a->tree->nent; bp = b->tree->ent; be = bp + b->tree->nent; while(ap != ae && bp != be){ c = strcmp(ap->name, bp->name); if(c == 0){ if(ap->mode == bp->mode && hasheq(&ap->h, &bp->h)) goto next; if(ap->mode != bp->mode) print("! %P%s\n", ap->name); else if(!(ap->mode & DMDIR) || !(bp->mode & DMDIR)) print("@ %P%s\n", ap->name); if((ap->mode & DMDIR) && (bp->mode & DMDIR)){ if(npath >= nelem(path)) sysfatal("path too deep"); path[npath++] = ap->name; difftrees(ap->h, bp->h); npath--; } next: ap++; bp++; }else if(c < 0) { show(ap, '-'); ap++; }else if(c > 0){ show(bp, '+'); bp++; } } for(; ap != ae; ap++) show(ap, '-'); for(; bp != be; bp++) show(bp, '+'); break; } unref(a); unref(b); } void usage(void) { fprint(2, "usage: %s [-pc] query\n", argv0); exits("usage"); } void main(int argc, char **argv) { int i, j, n; Hash *h; char *p, *e, *s; char query[2048], repo[512]; ARGBEGIN{ case 'p': fullpath++; break; case 'c': changes++; break; default: usage(); break; }ARGEND; gitinit(); fmtinstall('P', Pfmt); if(argc == 0) usage(); if(findrepo(repo, sizeof(repo)) == -1) sysfatal("find root: %r"); if(chdir(repo) == -1) sysfatal("chdir: %r"); s = ""; p = query; e = query + nelem(query); for(i = 0; i < argc; i++){ p = seprint(p, e, "%s%s", s, argv[i]); s = " "; } if((n = resolverefs(&h, query)) == -1) sysfatal("resolve: %r"); if(changes){ if(n != 2) sysfatal("diff: need 2 commits, got %d", n); difftrees(h[0], h[1]); }else{ p = (fullpath ? "/mnt/git/object/" : ""); for(j = 0; j < n; j++) print("%s%H\n", p, h[j]); } exits(nil); }