debian-mirror-gitlab/lib/banzai/filter/syntax_highlight_filter.rb

101 lines
3 KiB
Ruby
Raw Normal View History

2018-11-18 11:00:15 +05:30
# frozen_string_literal: true
2018-03-27 19:54:05 +05:30
require 'rouge/plugins/common_mark'
2021-02-22 17:27:13 +05:30
require "asciidoctor/extensions/asciidoctor_kroki/extension"
2015-09-25 12:07:36 +05:30
2019-03-02 22:35:43 +05:30
# Generated HTML is transformed back to GFM by app/assets/javascripts/behaviors/markdown/nodes/code_block.js
2015-12-23 02:04:40 +05:30
module Banzai
module Filter
2015-09-25 12:07:36 +05:30
# HTML Filter to highlight fenced code blocks
#
class SyntaxHighlightFilter < HTML::Pipeline::Filter
2019-07-07 11:18:12 +05:30
include OutputSafety
2019-12-04 20:38:33 +05:30
PARAMS_DELIMITER = ':'
LANG_PARAMS_ATTR = 'data-lang-params'
2019-07-07 11:18:12 +05:30
2021-06-02 17:11:27 +05:30
CSS = 'pre:not([data-math-style]):not([data-mermaid-style]):not([data-kroki-style]) > code'
XPATH = Gitlab::Utils::Nokogiri.css_to_xpath(CSS).freeze
2015-09-25 12:07:36 +05:30
def call
2021-06-02 17:11:27 +05:30
doc.xpath(XPATH).each do |node|
2015-09-25 12:07:36 +05:30
highlight_node(node)
end
doc
end
def highlight_node(node)
2018-11-18 11:00:15 +05:30
css_classes = +'code highlight js-syntax-highlight'
2019-07-07 11:18:12 +05:30
lang, lang_params = parse_lang_params(node.attr('lang'))
2021-11-18 22:05:49 +05:30
sourcepos = node.parent.attr('data-sourcepos')
2018-03-17 18:26:18 +05:30
retried = false
2016-08-24 12:49:21 +05:30
2018-03-17 18:26:18 +05:30
if use_rouge?(lang)
lexer = lexer_for(lang)
language = lexer.tag
else
lexer = Rouge::Lexers::PlainText.new
language = lang
end
2016-08-24 12:49:21 +05:30
2018-03-17 18:26:18 +05:30
begin
code = Rouge::Formatters::HTMLGitlab.format(lex(lexer, node.text), tag: language)
2021-04-29 21:17:54 +05:30
css_classes << " language-#{language}" if language
2021-06-08 01:23:25 +05:30
rescue StandardError
2018-03-17 18:26:18 +05:30
# Gracefully handle syntax highlighter bugs/errors to ensure users can
# still access an issue/comment/etc. First, retry with the plain text
# filter. If that fails, then just skip this entirely, but that would
# be a pretty bad upstream bug.
return if retried
language = nil
lexer = Rouge::Lexers::PlainText.new
retried = true
retry
2015-09-25 12:07:36 +05:30
end
2021-11-18 22:05:49 +05:30
sourcepos_attr = sourcepos ? "data-sourcepos=\"#{sourcepos}\"" : ""
highlighted = %(<pre #{sourcepos_attr} class="#{css_classes}"
2019-07-07 11:18:12 +05:30
lang="#{language}"
#{lang_params}
v-pre="true"><code>#{code}</code></pre>)
2016-08-24 12:49:21 +05:30
# Extracted to a method to measure it
replace_parent_pre_element(node, highlighted)
2015-09-25 12:07:36 +05:30
end
private
2019-07-07 11:18:12 +05:30
def parse_lang_params(language)
return unless language
lang, params = language.split(PARAMS_DELIMITER, 2)
formatted_params = %(#{LANG_PARAMS_ATTR}="#{escape_once(params)}") if params
[lang, formatted_params]
end
2016-09-13 17:45:13 +05:30
# Separate method so it can be instrumented.
def lex(lexer, code)
lexer.lex(code)
end
def lexer_for(language)
(Rouge::Lexer.find(language) || Rouge::Lexers::PlainText).new
end
2016-08-24 12:49:21 +05:30
def replace_parent_pre_element(node, highlighted)
# Replace the parent `pre` element with the entire highlighted block
node.parent.replace(highlighted)
end
2018-03-17 18:26:18 +05:30
def use_rouge?(language)
2021-02-22 17:27:13 +05:30
(%w(math suggestion) + ::AsciidoctorExtensions::Kroki::SUPPORTED_DIAGRAM_NAMES).exclude?(language)
2018-03-17 18:26:18 +05:30
end
2015-09-25 12:07:36 +05:30
end
end
end