summaryrefslogtreecommitdiff
path: root/haml_lint/linter
diff options
context:
space:
mode:
authorGitLab Bot <gitlab-bot@gitlab.com>2020-07-15 18:09:09 +0000
committerGitLab Bot <gitlab-bot@gitlab.com>2020-07-15 18:09:09 +0000
commitda1962d9ac710f95d350d2645c87f5a663123cf2 (patch)
tree1725ade126a9b4ae0148cd100cee94c44f9ce9f3 /haml_lint/linter
parente69e3f1eb695b4e852c56e7ddf8c52915ae2631b (diff)
downloadgitlab-ce-da1962d9ac710f95d350d2645c87f5a663123cf2.tar.gz
Add latest changes from gitlab-org/gitlab@master
Diffstat (limited to 'haml_lint/linter')
-rw-r--r--haml_lint/linter/documentation_links.rb100
1 files changed, 100 insertions, 0 deletions
diff --git a/haml_lint/linter/documentation_links.rb b/haml_lint/linter/documentation_links.rb
new file mode 100644
index 00000000000..f8e0eec5cdc
--- /dev/null
+++ b/haml_lint/linter/documentation_links.rb
@@ -0,0 +1,100 @@
+# frozen_string_literal: true
+
+require_relative '../../lib/gitlab/utils/markdown'
+
+module HamlLint
+ class Linter
+ # This class is responsible for detection of help_page_path helpers
+ # with incorrect links or anchors
+ class DocumentationLinks < Linter
+ include ::HamlLint::LinterRegistry
+ include ::Gitlab::Utils::Markdown
+
+ DOCS_DIRECTORY = File.join(File.expand_path('../..', __dir__), 'doc')
+
+ HELP_PATH_LINK_PATTERN = <<~PATTERN
+ `(send nil? :help_page_path $...)
+ PATTERN
+
+ MARKDOWN_HEADER = %r{\A\#{1,6}\s+(?<header>.+)\Z}.freeze
+
+ def visit_script(node)
+ check(node)
+ end
+
+ def visit_silent_script(node)
+ check(node)
+ end
+
+ def visit_tag(node)
+ check(node)
+ end
+
+ private
+
+ def check(node)
+ match = extract_link_and_anchor(node)
+
+ return if match.empty?
+
+ path_to_file = detect_path_to_file(match[:link])
+
+ unless File.file?(path_to_file)
+ record_lint(node, "help_page_path points to the unknown location: #{path_to_file}")
+ return
+ end
+
+ unless correct_anchor?(path_to_file, match[:anchor])
+ record_lint(node, "anchor (#{match[:anchor]}) is missing in: #{path_to_file}")
+ end
+ end
+
+ def extract_link_and_anchor(node)
+ ast_tree = fetch_ast_tree(node)
+
+ return {} unless ast_tree
+
+ link_match, attributes_match = ::RuboCop::NodePattern.new(HELP_PATH_LINK_PATTERN).match(ast_tree)
+
+ { link: fetch_link(link_match), anchor: fetch_anchor(attributes_match) }.compact
+ end
+
+ def fetch_ast_tree(node)
+ # Sometimes links are provided via data attributes in html tag
+ return node.parsed_attributes.syntax_tree if node.type == :tag
+
+ node.parsed_script.syntax_tree
+ end
+
+ def detect_path_to_file(link)
+ path = File.join(DOCS_DIRECTORY, link)
+ path += '.md' unless path.end_with?('.md')
+ path
+ end
+
+ def fetch_link(link_match)
+ return unless link_match && link_match.str_type?
+
+ link_match.value
+ end
+
+ def fetch_anchor(attributes_match)
+ return unless attributes_match
+
+ attributes_match.each_pair do |pkey, pvalue|
+ break pvalue.value if pkey.value == :anchor
+ end
+ end
+
+ def correct_anchor?(path_to_file, anchor)
+ return true unless anchor
+
+ File.open(path_to_file).any? do |line|
+ result = line.match(MARKDOWN_HEADER)
+
+ string_to_anchor(result[:header]) == anchor if result
+ end
+ end
+ end
+ end
+end