require "htmlentities" module Metanorma module Utils class Log attr_writer :xml def initialize @log = {} @c = HTMLEntities.new @mapid = {} end # severity: 0: abort; 1: serious; 2: not serious def add(category, loc, msg, severity: 2) @novalid and return @log[category] ||= [] item = create_entry(loc, msg, severity) @log[category] << item loc = loc.nil? ? "" : "(#{current_location(loc)}): " suppress_display?(category, loc, msg) or warn "#{category}: #{loc}#{msg}" end def abort_messages @log.values.each_with_object([]) do |v, m| v.each do |e| e[:severity].zero? and m << e[:message] end end end def suppress_display?(category, _loc, _msg) ["Metanorma XML Syntax"].include?(category) end def create_entry(loc, msg, severity) msg = msg.encode("UTF-8", invalid: :replace, undef: :replace) item = { location: current_location(loc), severity: severity, message: msg, context: context(loc), line: line(loc, msg) } if item[:message].include?(" :: ") a = item[:message].split(" :: ", 2) item[:context] = a[1] item[:message] = a[0] end item end def current_location(node) if node.nil? then "" elsif node.respond_to?(:id) && !node.id.nil? then "ID #{node.id}" elsif node.respond_to?(:id) && node.id.nil? && node.respond_to?(:parent) while !node.nil? && node.id.nil? node = node.parent end node.nil? ? "" : "ID #{node.id}" elsif node.respond_to?(:to_xml) && node.respond_to?(:parent) while !node.nil? && node["id"].nil? && node.respond_to?(:parent) node = node.parent end node.respond_to?(:parent) ? "ID #{node['id']}" : "" elsif node.is_a? String then node elsif node.respond_to?(:lineno) && !node.lineno.nil? && !node.lineno.empty? "Asciidoctor Line #{'%06d' % node.lineno}" elsif node.respond_to?(:line) && !node.line.nil? "XML Line #{'%06d' % node.line}" elsif node.respond_to?(:parent) while !node.nil? && (!node.respond_to?(:level) || node.level.positive?) && (!node.respond_to?(:context) || node.context != :section) node = node.parent return "Section: #{node.title}" if node.respond_to?(:context) && node&.context == :section end "??" else "??" end end def line(node, msg) if node.respond_to?(:line) && !node.line.nil? "#{'%06d' % node.line}" elsif /^XML Line /.match?(msg) msg.sub(/^XML Line /, "").sub(/:.*$/, "") else "000000" end end def context(node) node.is_a? String and return nil node.respond_to?(:to_xml) and return human_readable_xml(node) node.respond_to?(:to_s) and return node.to_s nil end # try to approximate input, at least for maths def human_readable_xml(node) ret = node.dup ret.xpath(".//*[local-name() = 'stem']").each do |s| sub = s.at("./*[local-name() = 'asciimath'] | " \ "./*[local-name() = 'latexmath']") sub and s.replace(sub) end ret.to_xml end def log_hdr(file) <<~HTML #{file} errors

#{file} errors

HTML end def write(file) @filename = file.sub(".err.html", ".html") File.open(file, "w:UTF-8") do |f| f.puts log_hdr(file) @log.each_key { |key| write_key(f, key) } f.puts "\n" end end def write_key(file, key) file.puts <<~HTML

#{key}

\n HTML @log[key].sort_by { |a| [a[:line], a[:location], a[:message]] } .each do |n| write_entry(file, render_preproc_entry(n)) end file.puts "
LineID MessageContextSeverity
\n" end def render_preproc_entry(entry) ret = entry.dup ret[:line] = nil if ret[:line] == "000000" ret[:location] = loc_link(entry) ret[:message] = break_up_long_str(entry[:message], 10, 2) .gsub(/`([^`]+)`/, "\\1") ret[:context] = context_render(entry) ret.compact end def context_render(entry) entry[:context] or return nil entry[:context].split("\n").first(5) .join("\n").gsub("><", "> <") end def mapid(old, new) @mapid[old] = new end def loc_link(entry) loc = entry[:location] loc.nil? || loc.empty? and loc = "--" if /^ID /.match?(loc) loc.sub!(/^ID /, "") loc = @mapid[loc] while @mapid[loc] url = "#{@filename}##{loc}" end loc &&= break_up_long_str(loc, 10, 2) url and loc = "#{loc}" loc end def break_up_long_str(str, threshold, punct) Metanorma::Utils.break_up_long_str(str, threshold, punct) end def write_entry(file, entry) entry[:context] &&= @c.encode(break_up_long_str(entry[:context], 40, 2)) file.print <<~HTML #{entry[:line]}#{entry[:location]} #{entry[:message]}
#{entry[:context]}
#{entry[:severity]} HTML end end end end