summaryrefslogtreecommitdiff
path: root/cgitcgi/ui-blame.c
diff options
context:
space:
mode:
authorkx <kx@radix.pro>2023-03-24 03:51:10 +0300
committerkx <kx@radix.pro>2023-03-24 03:51:10 +0300
commit05d292b208dfe01324826b4c87bbc4da3389a0d5 (patch)
treeb10a2269e9320785f3b61189e75f6778fa167986 /cgitcgi/ui-blame.c
parent40ab18a661ff6ada40e73969be293918d346a2f5 (diff)
downloadcgit-ui-05d292b208dfe01324826b4c87bbc4da3389a0d5.tar.xz
Version 0.1.7
Diffstat (limited to 'cgitcgi/ui-blame.c')
-rw-r--r--cgitcgi/ui-blame.c160
1 files changed, 160 insertions, 0 deletions
diff --git a/cgitcgi/ui-blame.c b/cgitcgi/ui-blame.c
new file mode 100644
index 0000000..faf4a33
--- /dev/null
+++ b/cgitcgi/ui-blame.c
@@ -0,0 +1,160 @@
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#include <stdlib.h>
+#include <stdio.h>
+#include <sys/sysinfo.h>
+#include <sys/types.h>
+#ifdef HAVE_INTTYPES_H
+#include <inttypes.h>
+#else
+#include <stdint.h>
+#endif
+#include <stddef.h> /* offsetof(3) */
+#include <dirent.h>
+#include <sys/stat.h> /* chmod(2) */
+#include <sys/file.h>
+#include <sys/mman.h>
+#include <fcntl.h>
+#include <limits.h>
+#include <string.h> /* strdup(3) */
+#include <libgen.h> /* basename(3) */
+#include <ctype.h> /* tolower(3) */
+#include <errno.h>
+#include <time.h>
+#include <sys/time.h>
+#include <pwd.h>
+#include <grp.h>
+#include <stdarg.h>
+#include <locale.h>
+#include <unistd.h>
+
+#include <git2.h>
+
+#include <nls.h>
+
+#include <defs.h>
+
+#include <fatal.h>
+#include <http.h>
+#include <html.h>
+
+#include <dlist.h>
+#include <strbuf.h>
+#include <repolist.h>
+#include <wrapper.h>
+#include <system.h>
+#include <date.h>
+
+#include <ctx.h>
+#include <git-shared.h>
+#include <ui-shared.h>
+
+
+static void cgit_print_blame( struct strbuf *sb, const char *relative_path, const char *revision )
+{
+ const char *name = NULL, *git_root = NULL, *repo_root = NULL;
+
+ if( !sb || !relative_path ) return;
+
+ name = ctx.repo.name;
+ git_root = ctx.repo.git_root;
+ repo_root = ctx.repo.repo_root;
+
+ strbuf_addf( sb, "<div class=\"blame\">" );
+ strbuf_addf( sb, " <pre><code class='language-PlainText'>" );
+
+ if( name && git_root )
+ {
+ char ref[PATH_MAX] = { 0 };
+ char rpath[PATH_MAX] = { 0 };
+
+ char path[PATH_MAX] = { 0 };
+ char cmd[PATH_MAX];
+ struct strbuf buf = STRBUF_INIT;
+ char *raw = NULL;
+ pid_t p = (pid_t) -1;
+ int rc;
+
+ sprintf( (char *)&path[0], "%s/", git_root );
+ if( repo_root && *repo_root )
+ {
+ strcat( (char *)&path[0], repo_root );
+ strcat( (char *)&path[0], "/" );
+ }
+ strcat( (char *)&path[0], name );
+
+ if( !is_bare( (char *)&path[0] ) )
+ {
+ strcat( (char *)&path[0], "/.git" );
+ }
+
+ parse_relative_path( (char *)&ref[0], (char *)&rpath[0], relative_path );
+
+ if( revision && *revision )
+ snprintf( (char *)&cmd[0], 1024, "git --git-dir=%s blame %s -- %s 2>/dev/null", (char *)&path[0], revision, (char *)&rpath[0] );
+ else
+ snprintf( (char *)&cmd[0], 1024, "git --git-dir=%s blame -- %s 2>/dev/null", (char *)&path[0], (char *)&rpath[0] );
+ p = sys_exec_command( &buf, cmd );
+ rc = sys_wait_command( p, NULL );
+ if( rc != 0 )
+ {
+ strbuf_release( &buf );
+ return;
+ }
+
+ if( buf.buf[0] )
+ {
+ raw = strbuf_detach( &buf, NULL );
+ strbuf_addstr_xml_quoted( &buf, (const char *)raw );
+ free( raw );
+
+ strbuf_addbuf( sb, (const struct strbuf *)&buf );
+ }
+ strbuf_release( &buf );
+ }
+
+ strbuf_addstr( sb, "\n </code></pre>\n" );
+ strbuf_addstr( sb, "</div>\n" );
+
+ return;
+}
+
+void cgit_print_blame_page( void )
+{
+ FILE *fp;
+ struct strbuf buf = STRBUF_INIT;
+
+ fp = xfopen( ctx.page.header, "r" );
+ (void)strbuf_env_fread( &buf, fp );
+ fclose( fp );
+
+ strbuf_addf( &buf, " <div class=\"content segment\">\n" );
+ strbuf_addf( &buf, " <div class=\"container\">\n" );
+ strbuf_addf( &buf, " <div class=\"cgit-main-content\">\n" );
+
+ if( ctx.repo.name )
+ {
+ cgit_print_blame( &buf, ctx.repo.relative_path, (!strcmp( ctx.query.rev, "0" )) ? (const char *)&ctx.repo.relative_info.revision[0] : ctx.query.rev );
+ }
+ else
+ {
+ strbuf_addf( &buf, " <h1>Requested resource cannot be shown</h1>\n" );
+ strbuf_addf( &buf, " <p class='leading'>Repository '%s' not found.</p>\n", ctx.repo.name );
+ }
+
+ strbuf_addf( &buf, " </div> <!-- End of cgit-main-content -->\n" );
+ strbuf_addf( &buf, " </div> <!-- End of container -->\n" );
+ strbuf_addf( &buf, " </div> <!-- End of content segment -->\n" );
+
+ fp = xfopen( ctx.page.footer, "r" );
+ (void)strbuf_env_fread( &buf, fp );
+ fclose( fp );
+
+ ctx.page.size = buf.len;
+ cgit_print_http_headers();
+ strbuf_write( &buf, STDOUT_FILENO );
+ strbuf_release( &buf );
+}