gitweb: Refactor syntax highlighting support
This refactoring (adding guess_file_syntax and run_highlighter subroutines) is meant to make it easier in the future to add support for other syntax highlighing solutions, or make it smarter by not re-running `git cat-file` second time. Instead of looping over list of regexps (keys of %highlight_type hash), make use of the fact that choosing syntax is based either on full basename (%highlight_basename), or on file extension (%highlight_ext). Add some basic test of syntax highlighting (with 'highlight' as prerequisite) to t/t9500-gitweb-standalone-no-errors.sh test. While at it make git_blob Perl style prettier. Signed-off-by: Jakub Narebski <jnareb@gmail.com> Acked-by: Petr Baudis <pasky@suse.cz> Signed-off-by: Junio C Hamano <gitster@pobox.com>
This commit is contained in:
parent
b331fe5476
commit
592ea4173a
@ -227,36 +227,6 @@ our %avatar_size = (
|
|||||||
# Leave it undefined (or set to 'undef') to turn off load checking.
|
# Leave it undefined (or set to 'undef') to turn off load checking.
|
||||||
our $maxload = 300;
|
our $maxload = 300;
|
||||||
|
|
||||||
# syntax highlighting
|
|
||||||
our %highlight_type = (
|
|
||||||
# match by basename
|
|
||||||
'SConstruct' => 'py',
|
|
||||||
'Program' => 'py',
|
|
||||||
'Library' => 'py',
|
|
||||||
'Makefile' => 'make',
|
|
||||||
# match by extension
|
|
||||||
'\.py$' => 'py', # Python
|
|
||||||
'\.c$' => 'c',
|
|
||||||
'\.h$' => 'c',
|
|
||||||
'\.cpp$' => 'cpp',
|
|
||||||
'\.cxx$' => 'cpp',
|
|
||||||
'\.rb$' => 'ruby',
|
|
||||||
'\.java$' => 'java',
|
|
||||||
'\.css$' => 'css',
|
|
||||||
'\.php3?$' => 'php',
|
|
||||||
'\.sh$' => 'sh', # Bash / shell script
|
|
||||||
'\.pl$' => 'pl', # Perl
|
|
||||||
'\.js$' => 'js', # JavaScript
|
|
||||||
'\.tex$' => 'tex', # TeX and LaTeX
|
|
||||||
'\.bib$' => 'bib', # BibTeX
|
|
||||||
'\.x?html$' => 'xml',
|
|
||||||
'\.xml$' => 'xml',
|
|
||||||
'\.awk$' => 'awk',
|
|
||||||
'\.bat$' => 'bat', # DOS Batch script
|
|
||||||
'\.ini$' => 'ini',
|
|
||||||
'\.spec$' => 'spec', # RPM Spec
|
|
||||||
);
|
|
||||||
|
|
||||||
# You define site-wide feature defaults here; override them with
|
# You define site-wide feature defaults here; override them with
|
||||||
# $GITWEB_CONFIG as necessary.
|
# $GITWEB_CONFIG as necessary.
|
||||||
our %feature = (
|
our %feature = (
|
||||||
@ -478,8 +448,8 @@ our %feature = (
|
|||||||
|
|
||||||
# Syntax highlighting support. This is based on Daniel Svensson's
|
# Syntax highlighting support. This is based on Daniel Svensson's
|
||||||
# and Sham Chukoury's work in gitweb-xmms2.git.
|
# and Sham Chukoury's work in gitweb-xmms2.git.
|
||||||
# It requires the 'highlight' program, and therefore is disabled
|
# It requires the 'highlight' program present in $PATH,
|
||||||
# by default.
|
# and therefore is disabled by default.
|
||||||
|
|
||||||
# To enable system wide have in $GITWEB_CONFIG
|
# To enable system wide have in $GITWEB_CONFIG
|
||||||
# $feature{'highlight'}{'default'} = [1];
|
# $feature{'highlight'}{'default'} = [1];
|
||||||
@ -3198,6 +3168,61 @@ sub blob_contenttype {
|
|||||||
return $type;
|
return $type;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
# guess file syntax for syntax highlighting; return undef if no highlighting
|
||||||
|
# the name of syntax can (in the future) depend on syntax highlighter used
|
||||||
|
sub guess_file_syntax {
|
||||||
|
my ($highlight, $mimetype, $file_name) = @_;
|
||||||
|
return undef unless ($highlight && defined $file_name);
|
||||||
|
|
||||||
|
# configuration for 'highlight' (http://www.andre-simon.de/)
|
||||||
|
# match by basename
|
||||||
|
my %highlight_basename = (
|
||||||
|
#'Program' => 'py',
|
||||||
|
#'Library' => 'py',
|
||||||
|
'SConstruct' => 'py', # SCons equivalent of Makefile
|
||||||
|
'Makefile' => 'make',
|
||||||
|
);
|
||||||
|
# match by extension
|
||||||
|
my %highlight_ext = (
|
||||||
|
# main extensions, defining name of syntax;
|
||||||
|
# see files in /usr/share/highlight/langDefs/ directory
|
||||||
|
map { $_ => $_ }
|
||||||
|
qw(py c cpp rb java css php sh pl js tex bib xml awk bat ini spec tcl),
|
||||||
|
# alternate extensions, see /etc/highlight/filetypes.conf
|
||||||
|
'h' => 'c',
|
||||||
|
map { $_ => 'cpp' } qw(cxx c++ cc),
|
||||||
|
map { $_ => 'php' } qw(php3 php4),
|
||||||
|
map { $_ => 'pl' } qw(perl pm), # perhaps also 'cgi'
|
||||||
|
'mak' => 'make',
|
||||||
|
map { $_ => 'xml' } qw(xhtml html htm),
|
||||||
|
);
|
||||||
|
|
||||||
|
my $basename = basename($file_name, '.in');
|
||||||
|
return $highlight_basename{$basename}
|
||||||
|
if exists $highlight_basename{$basename};
|
||||||
|
|
||||||
|
$basename =~ /\.([^.]*)$/;
|
||||||
|
my $ext = $1 or return undef;
|
||||||
|
return $highlight_ext{$ext}
|
||||||
|
if exists $highlight_ext{$ext};
|
||||||
|
|
||||||
|
return undef;
|
||||||
|
}
|
||||||
|
|
||||||
|
# run highlighter and return FD of its output,
|
||||||
|
# or return original FD if no highlighting
|
||||||
|
sub run_highlighter {
|
||||||
|
my ($fd, $highlight, $syntax) = @_;
|
||||||
|
return $fd unless ($highlight && defined $syntax);
|
||||||
|
|
||||||
|
close $fd
|
||||||
|
or die_error(404, "Reading blob failed");
|
||||||
|
open $fd, quote_command(git_cmd(), "cat-file", "blob", $hash)." | ".
|
||||||
|
"highlight --xhtml --fragment --syntax $syntax |"
|
||||||
|
or die_error(500, "Couldn't open file or run syntax highlighter");
|
||||||
|
return $fd;
|
||||||
|
}
|
||||||
|
|
||||||
## ======================================================================
|
## ======================================================================
|
||||||
## functions printing HTML: header, footer, error page
|
## functions printing HTML: header, footer, error page
|
||||||
|
|
||||||
@ -5397,24 +5422,10 @@ sub git_blob {
|
|||||||
# we can have blame only for text/* mimetype
|
# we can have blame only for text/* mimetype
|
||||||
$have_blame &&= ($mimetype =~ m!^text/!);
|
$have_blame &&= ($mimetype =~ m!^text/!);
|
||||||
|
|
||||||
my $have_highlight = gitweb_check_feature('highlight');
|
my $highlight = gitweb_check_feature('highlight');
|
||||||
my $syntax;
|
my $syntax = guess_file_syntax($highlight, $mimetype, $file_name);
|
||||||
if ($have_highlight && defined($file_name)) {
|
$fd = run_highlighter($fd, $highlight, $syntax)
|
||||||
my $basename = basename($file_name, '.in');
|
if $syntax;
|
||||||
foreach my $regexp (keys %highlight_type) {
|
|
||||||
if ($basename =~ /$regexp/) {
|
|
||||||
$syntax = $highlight_type{$regexp};
|
|
||||||
last;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($syntax) {
|
|
||||||
close $fd;
|
|
||||||
open $fd, quote_command(git_cmd(), "cat-file", "blob", $hash)." | ".
|
|
||||||
"highlight --xhtml --fragment -t 8 --syntax $syntax |"
|
|
||||||
or die_error(500, "Couldn't open file or run syntax highlighter");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
git_header_html(undef, $expires);
|
git_header_html(undef, $expires);
|
||||||
my $formats_nav = '';
|
my $formats_nav = '';
|
||||||
@ -5465,9 +5476,8 @@ sub git_blob {
|
|||||||
chomp $line;
|
chomp $line;
|
||||||
$nr++;
|
$nr++;
|
||||||
$line = untabify($line);
|
$line = untabify($line);
|
||||||
printf "<div class=\"pre\"><a id=\"l%i\" href=\"" . href(-replay => 1)
|
printf qq!<div class="pre"><a id="l%i" href="%s#l%i" class="linenr">%4i</a> %s</div>\n!,
|
||||||
. "#l%i\" class=\"linenr\">%4i</a> %s</div>\n",
|
$nr, href(-replay => 1), $nr, $nr, $syntax ? $line : esc_html($line, -nbsp=>1);
|
||||||
$nr, $nr, $nr, $syntax ? $line : esc_html($line, -nbsp=>1);
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
close $fd
|
close $fd
|
||||||
|
@ -647,4 +647,33 @@ test_expect_success \
|
|||||||
gitweb_run "p=.git;a=summary"'
|
gitweb_run "p=.git;a=summary"'
|
||||||
test_debug 'cat gitweb.log'
|
test_debug 'cat gitweb.log'
|
||||||
|
|
||||||
|
# ----------------------------------------------------------------------
|
||||||
|
# syntax highlighting
|
||||||
|
|
||||||
|
cat >>gitweb_config.perl <<\EOF
|
||||||
|
$feature{'highlight'}{'override'} = 1;
|
||||||
|
EOF
|
||||||
|
|
||||||
|
highlight --version >/dev/null 2>&1
|
||||||
|
if [ $? -eq 127 ]; then
|
||||||
|
say "Skipping syntax highlighting test, because 'highlight' was not found"
|
||||||
|
else
|
||||||
|
test_set_prereq HIGHLIGHT
|
||||||
|
fi
|
||||||
|
|
||||||
|
test_expect_success HIGHLIGHT \
|
||||||
|
'syntax highlighting (no highlight)' \
|
||||||
|
'git config gitweb.highlight yes &&
|
||||||
|
gitweb_run "p=.git;a=blob;f=file"'
|
||||||
|
test_debug 'cat gitweb.log'
|
||||||
|
|
||||||
|
test_expect_success HIGHLIGHT \
|
||||||
|
'syntax highlighting (highlighted)' \
|
||||||
|
'git config gitweb.highlight yes &&
|
||||||
|
echo "#!/usr/bin/sh" > test.sh &&
|
||||||
|
git add test.sh &&
|
||||||
|
git commit -m "Add test.sh" &&
|
||||||
|
gitweb_run "p=.git;a=blob;f=test.sh"'
|
||||||
|
test_debug 'cat gitweb.log'
|
||||||
|
|
||||||
test_done
|
test_done
|
||||||
|
Loading…
Reference in New Issue
Block a user