Browse Source

gitweb: use highlight's shebang detection

The "highlight" binary can, in some cases, determine the language type
by the means of file contents, for example the shebang in the first line
for some scripting languages.  Make use of this autodetection for files
which syntax is not known by gitweb.  In that case, pass the blob
contents to "highlight --force"; the parameter is needed to make it
always generate HTML output (which includes HTML-escaping).

Although we now run highlight on files which do not end up highlighted,
performance is virtually unaffected because when we call highlight, it
is used for escaping HTML.  In the case that highlight is used, gitweb
calls sanitize() instead of esc_html(), and the latter is significantly
slower (it does more, being roughly a superset of sanitize()).  Simple
benchmark comparing performance of 'blob' view of files without syntax
highlighting in gitweb before and after this change indicates ±1%
difference in request time for all file types.  Benchmark was performed
on local instance on Debian, using Apache/2.4.23 web server and CGI.

Document the feature and improve syntax highlight documentation, add
test to ensure gitweb doesn't crash when language detection is used.

Signed-off-by: Ian Kelling <ian@iankelling.org>
Acked-by: Jakub Narębski <jnareb@gmail.com>
Signed-off-by: Junio C Hamano <gitster@pobox.com>
maint
Ian Kelling 8 years ago committed by Junio C Hamano
parent
commit
779a206632
  1. 21
      Documentation/gitweb.conf.txt
  2. 10
      gitweb/gitweb.perl
  3. 8
      t/t9500-gitweb-standalone-no-errors.sh

21
Documentation/gitweb.conf.txt

@ -246,13 +246,20 @@ $highlight_bin::
Note that 'highlight' feature must be set for gitweb to actually Note that 'highlight' feature must be set for gitweb to actually
use syntax highlighting. use syntax highlighting.
+ +
*NOTE*: if you want to add support for new file type (supported by *NOTE*: for a file to be highlighted, its syntax type must be detected
"highlight" but not used by gitweb), you need to modify `%highlight_ext` and that syntax must be supported by "highlight". The default syntax
or `%highlight_basename`, depending on whether you detect type of file detection is minimal, and there are many supported syntax types with no
based on extension (for example "sh") or on its basename (for example detection by default. There are three options for adding syntax
"Makefile"). The keys of these hashes are extension and basename, detection. The first and second priority are `%highlight_basename` and
respectively, and value for given key is name of syntax to be passed via `%highlight_ext`, which detect based on basename (the full filename, for
`--syntax <syntax>` to highlighter. example "Makefile") and extension (for example "sh"). The keys of these
hashes are the basename and extension, respectively, and the value for a
given key is the name of the syntax to be passed via `--syntax <syntax>`
to "highlight". The last priority is the "highlight" configuration of
`Shebang` regular expressions to detect the language based on the first
line in the file, (for example, matching the line "#!/bin/bash"). See
the highlight documentation and the default config at
/etc/highlight/filetypes.conf for more details.
+ +
For example if repositories you are hosting use "phtml" extension for For example if repositories you are hosting use "phtml" extension for
PHP files, and you want to have correct syntax-highlighting for those PHP files, and you want to have correct syntax-highlighting for those

10
gitweb/gitweb.perl

@ -3931,15 +3931,16 @@ sub guess_file_syntax {
# or return original FD if no highlighting # or return original FD if no highlighting
sub run_highlighter { sub run_highlighter {
my ($fd, $highlight, $syntax) = @_; my ($fd, $highlight, $syntax) = @_;
return $fd unless ($highlight && defined $syntax); return $fd unless ($highlight);


close $fd; close $fd;
my $syntax_arg = (defined $syntax) ? "--syntax $syntax" : "--force";
open $fd, quote_command(git_cmd(), "cat-file", "blob", $hash)." | ". open $fd, quote_command(git_cmd(), "cat-file", "blob", $hash)." | ".
quote_command($^X, '-CO', '-MEncode=decode,FB_DEFAULT', '-pse', quote_command($^X, '-CO', '-MEncode=decode,FB_DEFAULT', '-pse',
'$_ = decode($fe, $_, FB_DEFAULT) if !utf8::decode($_);', '$_ = decode($fe, $_, FB_DEFAULT) if !utf8::decode($_);',
'--', "-fe=$fallback_encoding")." | ". '--', "-fe=$fallback_encoding")." | ".
quote_command($highlight_bin). quote_command($highlight_bin).
" --replace-tabs=8 --fragment --syntax $syntax |" " --replace-tabs=8 --fragment $syntax_arg |"
or die_error(500, "Couldn't open file or run syntax highlighter"); or die_error(500, "Couldn't open file or run syntax highlighter");
return $fd; return $fd;
} }
@ -7063,8 +7064,7 @@ sub git_blob {


my $highlight = gitweb_check_feature('highlight'); my $highlight = gitweb_check_feature('highlight');
my $syntax = guess_file_syntax($highlight, $file_name); my $syntax = guess_file_syntax($highlight, $file_name);
$fd = run_highlighter($fd, $highlight, $syntax) $fd = run_highlighter($fd, $highlight, $syntax);
if $syntax;


git_header_html(undef, $expires); git_header_html(undef, $expires);
my $formats_nav = ''; my $formats_nav = '';
@ -7117,7 +7117,7 @@ sub git_blob {
$line = untabify($line); $line = untabify($line);
printf qq!<div class="pre"><a id="l%i" href="%s#l%i" class="linenr">%4i</a> %s</div>\n!, printf qq!<div class="pre"><a id="l%i" href="%s#l%i" class="linenr">%4i</a> %s</div>\n!,
$nr, esc_attr(href(-replay => 1)), $nr, $nr, $nr, esc_attr(href(-replay => 1)), $nr, $nr,
$syntax ? sanitize($line) : esc_html($line, -nbsp=>1); $highlight ? sanitize($line) : esc_html($line, -nbsp=>1);
} }
} }
close $fd close $fd

8
t/t9500-gitweb-standalone-no-errors.sh

@ -709,6 +709,14 @@ test_expect_success HIGHLIGHT \
git commit -m "Add test.sh" && git commit -m "Add test.sh" &&
gitweb_run "p=.git;a=blob;f=test.sh"' gitweb_run "p=.git;a=blob;f=test.sh"'


test_expect_success HIGHLIGHT \
'syntax highlighting (highlighter language autodetection)' \
'git config gitweb.highlight yes &&
echo "#!/usr/bin/perl" > test &&
git add test &&
git commit -m "Add test" &&
gitweb_run "p=.git;a=blob;f=test"'

# ---------------------------------------------------------------------- # ----------------------------------------------------------------------
# forks of projects # forks of projects



Loading…
Cancel
Save