platformos-check 0.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (240) hide show
  1. checksums.yaml +7 -0
  2. data/.dockerignore +2 -0
  3. data/.gitignore +22 -0
  4. data/.rubocop.yml +555 -0
  5. data/CHANGELOG.md +5 -0
  6. data/CODE_OF_CONDUCT.md +74 -0
  7. data/CONTRIBUTING.md +209 -0
  8. data/Gemfile +33 -0
  9. data/Guardfile +7 -0
  10. data/LICENSE.md +10 -0
  11. data/Makefile +18 -0
  12. data/README.md +189 -0
  13. data/RELEASING.md +35 -0
  14. data/Rakefile +83 -0
  15. data/TROUBLESHOOTING.md +35 -0
  16. data/bin/platformos-check +29 -0
  17. data/bin/platformos-check-language-server +29 -0
  18. data/config/default.yml +98 -0
  19. data/config/nothing.yml +11 -0
  20. data/data/platformos_liquid/built_in_liquid_objects.json +66 -0
  21. data/data/platformos_liquid/deprecated_filters.json +22 -0
  22. data/data/platformos_liquid/documentation/filters.json +6 -0
  23. data/data/platformos_liquid/documentation/latest.json +2 -0
  24. data/data/platformos_liquid/documentation/objects.json +6 -0
  25. data/data/platformos_liquid/documentation/tags.json +6 -0
  26. data/docker/check.Dockerfile +3 -0
  27. data/docker/lsp.Dockerfile +21 -0
  28. data/docs/api/check.md +15 -0
  29. data/docs/api/html_check.md +46 -0
  30. data/docs/api/liquid_check.md +115 -0
  31. data/docs/api/yaml_check.md +19 -0
  32. data/docs/checks/TEMPLATE.md.erb +52 -0
  33. data/docs/checks/convert_include_to_render.md +48 -0
  34. data/docs/checks/deprecated_filter.md +30 -0
  35. data/docs/checks/html_parsing_error.md +50 -0
  36. data/docs/checks/img_lazy_loading.md +63 -0
  37. data/docs/checks/img_width_and_height.md +79 -0
  38. data/docs/checks/invalid_args.md +56 -0
  39. data/docs/checks/liquid_tag.md +65 -0
  40. data/docs/checks/missing_enable_comment.md +50 -0
  41. data/docs/checks/missing_template.md +65 -0
  42. data/docs/checks/parse_json_format.md +76 -0
  43. data/docs/checks/parser_blocking_javascript.md +97 -0
  44. data/docs/checks/required_layout_object.md +28 -0
  45. data/docs/checks/space_inside_braces.md +89 -0
  46. data/docs/checks/syntax_error.md +49 -0
  47. data/docs/checks/template_length.md +45 -0
  48. data/docs/checks/undefined_object.md +71 -0
  49. data/docs/checks/unknown_filter.md +46 -0
  50. data/docs/checks/unused_assign.md +63 -0
  51. data/docs/checks/unused_partial.md +32 -0
  52. data/docs/checks/valid_yaml.md +48 -0
  53. data/docs/flamegraph.svg +18488 -0
  54. data/docs/language_server/code-action-command-palette.png +0 -0
  55. data/docs/language_server/code-action-flow.png +0 -0
  56. data/docs/language_server/code-action-keyboard.png +0 -0
  57. data/docs/language_server/code-action-light-bulb.png +0 -0
  58. data/docs/language_server/code-action-problem.png +0 -0
  59. data/docs/language_server/code-action-quickfix.png +0 -0
  60. data/docs/language_server/how_to_correct_code_with_code_actions_and_execute_command.md +197 -0
  61. data/docs/preview.png +0 -0
  62. data/exe/platformos-check +6 -0
  63. data/exe/platformos-check-language-server +7 -0
  64. data/lib/platformos_check/analyzer.rb +178 -0
  65. data/lib/platformos_check/api_call_file.rb +9 -0
  66. data/lib/platformos_check/app.rb +138 -0
  67. data/lib/platformos_check/app_file.rb +89 -0
  68. data/lib/platformos_check/app_file_rewriter.rb +79 -0
  69. data/lib/platformos_check/asset_file.rb +34 -0
  70. data/lib/platformos_check/bug.rb +23 -0
  71. data/lib/platformos_check/check.rb +163 -0
  72. data/lib/platformos_check/checks/TEMPLATE.rb.erb +11 -0
  73. data/lib/platformos_check/checks/convert_include_to_render.rb +17 -0
  74. data/lib/platformos_check/checks/deprecated_filter.rb +123 -0
  75. data/lib/platformos_check/checks/html_parsing_error.rb +13 -0
  76. data/lib/platformos_check/checks/img_lazy_loading.rb +18 -0
  77. data/lib/platformos_check/checks/img_width_and_height.rb +46 -0
  78. data/lib/platformos_check/checks/invalid_args.rb +81 -0
  79. data/lib/platformos_check/checks/liquid_tag.rb +47 -0
  80. data/lib/platformos_check/checks/missing_enable_comment.rb +37 -0
  81. data/lib/platformos_check/checks/missing_template.rb +107 -0
  82. data/lib/platformos_check/checks/parse_json_format.rb +31 -0
  83. data/lib/platformos_check/checks/parser_blocking_javascript.rb +17 -0
  84. data/lib/platformos_check/checks/required_layout_object.rb +41 -0
  85. data/lib/platformos_check/checks/space_inside_braces.rb +150 -0
  86. data/lib/platformos_check/checks/syntax_error.rb +31 -0
  87. data/lib/platformos_check/checks/template_length.rb +20 -0
  88. data/lib/platformos_check/checks/undefined_object.rb +206 -0
  89. data/lib/platformos_check/checks/unknown_filter.rb +27 -0
  90. data/lib/platformos_check/checks/unused_assign.rb +101 -0
  91. data/lib/platformos_check/checks/unused_partial.rb +93 -0
  92. data/lib/platformos_check/checks/valid_yaml.rb +16 -0
  93. data/lib/platformos_check/checks.rb +73 -0
  94. data/lib/platformos_check/checks_tracking.rb +9 -0
  95. data/lib/platformos_check/cli.rb +239 -0
  96. data/lib/platformos_check/config.rb +219 -0
  97. data/lib/platformos_check/config_file.rb +6 -0
  98. data/lib/platformos_check/corrector.rb +68 -0
  99. data/lib/platformos_check/disabled_check.rb +44 -0
  100. data/lib/platformos_check/disabled_checks.rb +96 -0
  101. data/lib/platformos_check/email_file.rb +9 -0
  102. data/lib/platformos_check/exceptions.rb +36 -0
  103. data/lib/platformos_check/file_system_storage.rb +93 -0
  104. data/lib/platformos_check/graphql_file.rb +68 -0
  105. data/lib/platformos_check/html_check.rb +8 -0
  106. data/lib/platformos_check/html_node.rb +210 -0
  107. data/lib/platformos_check/html_visitor.rb +36 -0
  108. data/lib/platformos_check/in_memory_storage.rb +68 -0
  109. data/lib/platformos_check/json_file.rb +57 -0
  110. data/lib/platformos_check/json_helper.rb +73 -0
  111. data/lib/platformos_check/json_helpers.rb +24 -0
  112. data/lib/platformos_check/json_printer.rb +32 -0
  113. data/lib/platformos_check/language_server/bridge.rb +167 -0
  114. data/lib/platformos_check/language_server/channel.rb +69 -0
  115. data/lib/platformos_check/language_server/client_capabilities.rb +27 -0
  116. data/lib/platformos_check/language_server/code_action_engine.rb +32 -0
  117. data/lib/platformos_check/language_server/code_action_provider.rb +41 -0
  118. data/lib/platformos_check/language_server/code_action_providers/quickfix_code_action_provider.rb +85 -0
  119. data/lib/platformos_check/language_server/code_action_providers/source_fix_all_code_action_provider.rb +41 -0
  120. data/lib/platformos_check/language_server/completion_context.rb +52 -0
  121. data/lib/platformos_check/language_server/completion_engine.rb +32 -0
  122. data/lib/platformos_check/language_server/completion_helper.rb +26 -0
  123. data/lib/platformos_check/language_server/completion_provider.rb +53 -0
  124. data/lib/platformos_check/language_server/completion_providers/assignments_completion_provider.rb +40 -0
  125. data/lib/platformos_check/language_server/completion_providers/filter_completion_provider.rb +102 -0
  126. data/lib/platformos_check/language_server/completion_providers/object_attribute_completion_provider.rb +48 -0
  127. data/lib/platformos_check/language_server/completion_providers/object_completion_provider.rb +38 -0
  128. data/lib/platformos_check/language_server/completion_providers/render_snippet_completion_provider.rb +50 -0
  129. data/lib/platformos_check/language_server/completion_providers/tag_completion_provider.rb +41 -0
  130. data/lib/platformos_check/language_server/configuration.rb +89 -0
  131. data/lib/platformos_check/language_server/constants.rb +29 -0
  132. data/lib/platformos_check/language_server/diagnostic.rb +129 -0
  133. data/lib/platformos_check/language_server/diagnostics_engine.rb +131 -0
  134. data/lib/platformos_check/language_server/diagnostics_manager.rb +184 -0
  135. data/lib/platformos_check/language_server/document_change_corrector.rb +271 -0
  136. data/lib/platformos_check/language_server/document_link_engine.rb +21 -0
  137. data/lib/platformos_check/language_server/document_link_provider.rb +71 -0
  138. data/lib/platformos_check/language_server/document_link_providers/asset_document_link_provider.rb +11 -0
  139. data/lib/platformos_check/language_server/document_link_providers/include_document_link_provider.rb +11 -0
  140. data/lib/platformos_check/language_server/document_link_providers/render_document_link_provider.rb +11 -0
  141. data/lib/platformos_check/language_server/document_link_providers/section_document_link_provider.rb +11 -0
  142. data/lib/platformos_check/language_server/execute_command_engine.rb +19 -0
  143. data/lib/platformos_check/language_server/execute_command_provider.rb +30 -0
  144. data/lib/platformos_check/language_server/execute_command_providers/correction_execute_command_provider.rb +48 -0
  145. data/lib/platformos_check/language_server/execute_command_providers/run_checks_execute_command_provider.rb +28 -0
  146. data/lib/platformos_check/language_server/handler.rb +310 -0
  147. data/lib/platformos_check/language_server/hover_engine.rb +32 -0
  148. data/lib/platformos_check/language_server/hover_provider.rb +53 -0
  149. data/lib/platformos_check/language_server/hover_providers/filter_hover_provider.rb +113 -0
  150. data/lib/platformos_check/language_server/io_messenger.rb +109 -0
  151. data/lib/platformos_check/language_server/messenger.rb +27 -0
  152. data/lib/platformos_check/language_server/protocol.rb +55 -0
  153. data/lib/platformos_check/language_server/server.rb +188 -0
  154. data/lib/platformos_check/language_server/tokens.rb +55 -0
  155. data/lib/platformos_check/language_server/type_helper.rb +22 -0
  156. data/lib/platformos_check/language_server/uri_helper.rb +39 -0
  157. data/lib/platformos_check/language_server/variable_lookup_finder/assignments_finder/node_handler.rb +87 -0
  158. data/lib/platformos_check/language_server/variable_lookup_finder/assignments_finder/scope.rb +60 -0
  159. data/lib/platformos_check/language_server/variable_lookup_finder/assignments_finder/scope_visitor.rb +44 -0
  160. data/lib/platformos_check/language_server/variable_lookup_finder/assignments_finder.rb +76 -0
  161. data/lib/platformos_check/language_server/variable_lookup_finder/constants.rb +44 -0
  162. data/lib/platformos_check/language_server/variable_lookup_finder/liquid_fixer.rb +103 -0
  163. data/lib/platformos_check/language_server/variable_lookup_finder/potential_lookup.rb +10 -0
  164. data/lib/platformos_check/language_server/variable_lookup_finder/tolerant_parser.rb +94 -0
  165. data/lib/platformos_check/language_server/variable_lookup_finder.rb +262 -0
  166. data/lib/platformos_check/language_server/variable_lookup_traverser.rb +70 -0
  167. data/lib/platformos_check/language_server/versioned_in_memory_storage.rb +84 -0
  168. data/lib/platformos_check/language_server.rb +71 -0
  169. data/lib/platformos_check/layout_file.rb +15 -0
  170. data/lib/platformos_check/liquid_check.rb +10 -0
  171. data/lib/platformos_check/liquid_file.rb +102 -0
  172. data/lib/platformos_check/liquid_node.rb +570 -0
  173. data/lib/platformos_check/liquid_visitor.rb +39 -0
  174. data/lib/platformos_check/migration_file.rb +9 -0
  175. data/lib/platformos_check/node.rb +53 -0
  176. data/lib/platformos_check/offense.rb +228 -0
  177. data/lib/platformos_check/page_file.rb +9 -0
  178. data/lib/platformos_check/parsing_helpers.rb +21 -0
  179. data/lib/platformos_check/partial_file.rb +15 -0
  180. data/lib/platformos_check/platformos_liquid/deprecated_filter.rb +31 -0
  181. data/lib/platformos_check/platformos_liquid/documentation/markdown_template.rb +51 -0
  182. data/lib/platformos_check/platformos_liquid/documentation.rb +45 -0
  183. data/lib/platformos_check/platformos_liquid/filter.rb +19 -0
  184. data/lib/platformos_check/platformos_liquid/object.rb +15 -0
  185. data/lib/platformos_check/platformos_liquid/source_index/base_entry.rb +66 -0
  186. data/lib/platformos_check/platformos_liquid/source_index/base_state.rb +23 -0
  187. data/lib/platformos_check/platformos_liquid/source_index/filter_entry.rb +26 -0
  188. data/lib/platformos_check/platformos_liquid/source_index/filter_state.rb +11 -0
  189. data/lib/platformos_check/platformos_liquid/source_index/object_entry.rb +20 -0
  190. data/lib/platformos_check/platformos_liquid/source_index/object_state.rb +11 -0
  191. data/lib/platformos_check/platformos_liquid/source_index/parameter_entry.rb +25 -0
  192. data/lib/platformos_check/platformos_liquid/source_index/property_entry.rb +21 -0
  193. data/lib/platformos_check/platformos_liquid/source_index/return_type_entry.rb +41 -0
  194. data/lib/platformos_check/platformos_liquid/source_index/tag_entry.rb +24 -0
  195. data/lib/platformos_check/platformos_liquid/source_index/tag_state.rb +11 -0
  196. data/lib/platformos_check/platformos_liquid/source_index.rb +79 -0
  197. data/lib/platformos_check/platformos_liquid/source_manager.rb +116 -0
  198. data/lib/platformos_check/platformos_liquid/tag.rb +59 -0
  199. data/lib/platformos_check/platformos_liquid.rb +21 -0
  200. data/lib/platformos_check/position.rb +180 -0
  201. data/lib/platformos_check/position_helper.rb +57 -0
  202. data/lib/platformos_check/printer.rb +87 -0
  203. data/lib/platformos_check/regex_helpers.rb +21 -0
  204. data/lib/platformos_check/releaser.rb +43 -0
  205. data/lib/platformos_check/schema_file.rb +6 -0
  206. data/lib/platformos_check/sms_file.rb +9 -0
  207. data/lib/platformos_check/storage.rb +29 -0
  208. data/lib/platformos_check/string_helpers.rb +48 -0
  209. data/lib/platformos_check/tags/background.rb +67 -0
  210. data/lib/platformos_check/tags/base.rb +14 -0
  211. data/lib/platformos_check/tags/base_block.rb +14 -0
  212. data/lib/platformos_check/tags/base_tag_methods.rb +59 -0
  213. data/lib/platformos_check/tags/cache.rb +13 -0
  214. data/lib/platformos_check/tags/export.rb +30 -0
  215. data/lib/platformos_check/tags/form.rb +19 -0
  216. data/lib/platformos_check/tags/function.rb +58 -0
  217. data/lib/platformos_check/tags/graphql.rb +70 -0
  218. data/lib/platformos_check/tags/hash_assign.rb +75 -0
  219. data/lib/platformos_check/tags/log.rb +15 -0
  220. data/lib/platformos_check/tags/parse_json.rb +24 -0
  221. data/lib/platformos_check/tags/print.rb +20 -0
  222. data/lib/platformos_check/tags/redirect_to.rb +15 -0
  223. data/lib/platformos_check/tags/render.rb +60 -0
  224. data/lib/platformos_check/tags/response_headers.rb +20 -0
  225. data/lib/platformos_check/tags/response_status.rb +20 -0
  226. data/lib/platformos_check/tags/return.rb +20 -0
  227. data/lib/platformos_check/tags/session.rb +27 -0
  228. data/lib/platformos_check/tags/sign_in.rb +27 -0
  229. data/lib/platformos_check/tags/spam_protection.rb +15 -0
  230. data/lib/platformos_check/tags/theme_render.rb +58 -0
  231. data/lib/platformos_check/tags/try.rb +59 -0
  232. data/lib/platformos_check/tags.rb +65 -0
  233. data/lib/platformos_check/translation_file.rb +6 -0
  234. data/lib/platformos_check/user_schema_file.rb +6 -0
  235. data/lib/platformos_check/version.rb +5 -0
  236. data/lib/platformos_check/yaml_check.rb +11 -0
  237. data/lib/platformos_check/yaml_file.rb +57 -0
  238. data/lib/platformos_check.rb +106 -0
  239. data/platformos-check.gemspec +34 -0
  240. metadata +329 -0
@@ -0,0 +1,101 @@
1
+ # frozen_string_literal: true
2
+
3
+ module PlatformosCheck
4
+ # Checks unused {% assign x = ... %}
5
+ class UnusedAssign < LiquidCheck
6
+ severity :suggestion
7
+ category :liquid
8
+ doc docs_url(__FILE__)
9
+
10
+ class TemplateInfo < Struct.new(:used_assigns, :assign_nodes, :includes)
11
+ def collect_used_assigns(templates, visited = Set.new)
12
+ collected = used_assigns
13
+ # Check recursively inside included snippets for use
14
+ includes.each do |name|
15
+ if templates[name] && !visited.include?(name)
16
+ visited << name
17
+ collected += templates[name].collect_used_assigns(templates, visited)
18
+ end
19
+ end
20
+ collected
21
+ end
22
+ end
23
+
24
+ def initialize
25
+ @templates = {}
26
+ end
27
+
28
+ def on_document(node)
29
+ @templates[node.app_file.name] = TemplateInfo.new(Set.new, {}, Set.new)
30
+ end
31
+
32
+ def on_assign(node)
33
+ @templates[node.app_file.name].assign_nodes[node.value.to] = node
34
+ end
35
+
36
+ def on_parse_json(node)
37
+ @templates[node.app_file.name].assign_nodes[node.value.to] = node
38
+ end
39
+
40
+ def on_function(node)
41
+ return if node.value.to.start_with?('_')
42
+
43
+ @templates[node.app_file.name].assign_nodes[node.value.to] = node
44
+ end
45
+
46
+ def on_graphql(node)
47
+ return if node.value.to.start_with?('_')
48
+
49
+ @templates[node.app_file.name].assign_nodes[node.value.to] = node
50
+ end
51
+
52
+ def on_include(node)
53
+ return unless node.value.template_name_expr.is_a?(String)
54
+
55
+ @templates[node.app_file.name].includes << node.value.template_name_expr
56
+ end
57
+
58
+ def on_variable_lookup(node)
59
+ @templates[node.app_file.name].used_assigns << case node.value.name
60
+ when Liquid::VariableLookup
61
+ node.value.name.name
62
+ else
63
+ node.value.name
64
+ end
65
+ end
66
+
67
+ def on_end
68
+ @templates.each_pair do |_, info|
69
+ used = info.collect_used_assigns(@templates)
70
+ info.assign_nodes.each_pair do |name, node|
71
+ next if used.include?(name)
72
+
73
+ add_offense("`#{name}` is never used", node:) do |corrector|
74
+ case node.type_name
75
+ when :graphql
76
+ offset = node.markup.match(/^graphql\s+/)[0].size
77
+
78
+ corrector.insert_before(
79
+ node,
80
+ '_',
81
+ (node.start_index + offset)...(node.start_index + offset)
82
+ )
83
+ when :function
84
+ offset = node.markup.match(/^function\s+/)[0].size
85
+
86
+ corrector.insert_before(
87
+ node,
88
+ '_',
89
+ (node.start_index + offset)...(node.start_index + offset)
90
+ )
91
+ when :parse_json
92
+ # noop
93
+ else
94
+ corrector.remove(node)
95
+ end
96
+ end
97
+ end
98
+ end
99
+ end
100
+ end
101
+ end
@@ -0,0 +1,93 @@
1
+ # frozen_string_literal: true
2
+
3
+ module PlatformosCheck
4
+ class UnusedPartial < LiquidCheck
5
+ severity :suggestion
6
+ category :liquid
7
+ doc docs_url(__FILE__)
8
+
9
+ def initialize
10
+ @used_partials = Set.new
11
+ end
12
+
13
+ def on_render(node)
14
+ if node.value.template_name_expr.is_a?(String)
15
+ @used_partials << node.value.template_name_expr
16
+
17
+ elsif might_have_a_block_as_variable_lookup?(node)
18
+ # We ignore this case, because that's a "proper" use case for
19
+ # the render tag with OS 2.0
20
+ # {% render block %} shouldn't turn off the UnusedPartial check
21
+
22
+ else
23
+ # Can't reliably track unused partials if an expression is used, ignore this check
24
+ @used_partials.clear
25
+ ignore!
26
+ end
27
+ end
28
+ alias on_include on_render
29
+
30
+ def on_function(node)
31
+ if node.value.from.is_a?(String)
32
+ @used_partials << node.value.from
33
+ else
34
+ # Can't reliably track unused partials if an expression is used, ignore this check
35
+ @used_partials.clear
36
+ ignore!
37
+ end
38
+ end
39
+
40
+ def on_end
41
+ missing_partials.each do |app_file|
42
+ add_offense("This partial is not used", app_file:) do |corrector|
43
+ corrector.remove_file(@platformos_app.storage, app_file.relative_path.to_s)
44
+ end
45
+ end
46
+ end
47
+
48
+ def missing_partials
49
+ platformos_app.partials.reject { |t| @used_partials.include?(t.name) }
50
+ end
51
+
52
+ private
53
+
54
+ # This function returns true when the render node passed might have a
55
+ # variable lookup that refers to a block as template_name_expr.
56
+ #
57
+ # e.g.
58
+ #
59
+ # {% for block in col %}
60
+ # {% render block %}
61
+ # {% endfor %}
62
+ #
63
+ # In this case, the `block` variable_lookup in the render tag might be
64
+ # a Block because col might be an array of blocks.
65
+ #
66
+ # @param node [Node]
67
+ def might_have_a_block_as_variable_lookup?(node)
68
+ return false unless node.type_name == :render
69
+
70
+ return false unless node.value.template_name_expr.is_a?(Liquid::VariableLookup)
71
+
72
+ name = node.value.template_name_expr.name
73
+ return false unless name.is_a?(String)
74
+
75
+ # We're going through all the parents of the nodes until we find
76
+ # a For node with variable_name === to the template_name_expr's name
77
+ find_parent(node.parent) do |parent_node|
78
+ next false unless parent_node.type_name == :for
79
+
80
+ parent_node.value.variable_name == name
81
+ end
82
+ end
83
+
84
+ # @param node [Node]
85
+ def find_parent(node, &)
86
+ return nil unless node
87
+
88
+ return node if yield node
89
+
90
+ find_parent(node.parent, &)
91
+ end
92
+ end
93
+ end
@@ -0,0 +1,16 @@
1
+ # frozen_string_literal: true
2
+
3
+ module PlatformosCheck
4
+ class ValidYaml < YamlCheck
5
+ severity :error
6
+ category :yaml
7
+ doc docs_url(__FILE__)
8
+
9
+ def on_file(file)
10
+ return unless file.parse_error
11
+
12
+ message = file.parse_error
13
+ add_offense(message, app_file: file)
14
+ end
15
+ end
16
+ end
@@ -0,0 +1,73 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "pp"
4
+ require "timeout"
5
+
6
+ module PlatformosCheck
7
+ class Checks < Array
8
+ CHECK_METHOD_TIMEOUT = 5 # sec
9
+
10
+ def call(method, *)
11
+ each do |check|
12
+ call_check_method(check, method, *)
13
+ end
14
+ end
15
+
16
+ def disableable
17
+ @disableable ||= self.class.new(select(&:can_disable?))
18
+ end
19
+
20
+ def whole_platformos_app
21
+ @whole_platformos_app ||= self.class.new(select(&:whole_platformos_app?))
22
+ end
23
+
24
+ def single_file
25
+ @single_file ||= self.class.new(select(&:single_file?))
26
+ end
27
+
28
+ private
29
+
30
+ def call_check_method(check, method, *args)
31
+ return unless check.respond_to?(method) && !check.ignored?
32
+
33
+ # If you want to use binding.pry in unit tests, define the
34
+ # PLATFORMOS_CHECK_DEBUG environment variable. e.g.
35
+ #
36
+ # $ export PLATFORMOS_CHECK_DEBUG=true
37
+ # $ bundle exec rake tests:in_memory
38
+ #
39
+ if ENV['PLATFORMOS_CHECK_DEBUG']
40
+ check.send(method, *args)
41
+ else
42
+ Timeout.timeout(CHECK_METHOD_TIMEOUT) do
43
+ check.send(method, *args)
44
+ end
45
+ end
46
+ rescue Liquid::Error, PlatformosCheckError
47
+ raise
48
+ rescue StandardError => e
49
+ node = args.first
50
+ app_file = node.respond_to?(:app_file) ? node.app_file.relative_path : "?"
51
+ markup = node.respond_to?(:markup) ? node.markup : ""
52
+ node_class = node.respond_to?(:value) ? node.value.class : "?"
53
+ line_number = node.respond_to?(:line_number) ? node.line_number : "?"
54
+
55
+ PlatformosCheck.bug(<<~EOS)
56
+ Exception while running `#{check.code_name}##{method}`:
57
+ ```
58
+ #{e.class}: #{e.message}
59
+ #{e.backtrace.join("\n ")}
60
+ ```
61
+
62
+ Platformos App File: `#{app_file}`
63
+ Node: `#{node_class}`
64
+ Markup:
65
+ ```
66
+ #{markup}
67
+ ```
68
+ Line number: #{line_number}
69
+ Check options: `#{check.options.pretty_inspect}`
70
+ EOS
71
+ end
72
+ end
73
+ end
@@ -0,0 +1,9 @@
1
+ # frozen_string_literal: true
2
+
3
+ module PlatformosCheck
4
+ module ChecksTracking
5
+ def inherited(klass)
6
+ Check.all << klass
7
+ end
8
+ end
9
+ end
@@ -0,0 +1,239 @@
1
+ # frozen_string_literal: true
2
+
3
+ require "optparse"
4
+
5
+ module PlatformosCheck
6
+ class Cli
7
+ class Abort < StandardError; end
8
+
9
+ FORMATS = %i[text json]
10
+
11
+ attr_accessor :path
12
+
13
+ def initialize
14
+ @path = "."
15
+ @command = :check
16
+ @include_categories = []
17
+ @exclude_categories = []
18
+ @auto_correct = false
19
+ @update_docs = false
20
+ @config_path = nil
21
+ @fail_level = :error
22
+ @format = :text
23
+ end
24
+
25
+ def option_parser(parser = OptionParser.new, help: true)
26
+ return @option_parser if defined?(@option_parser)
27
+
28
+ @option_parser = parser
29
+ @option_parser.banner = "Usage: platformos-check [options] [/path/to/your/platformos_app]"
30
+
31
+ @option_parser.separator("")
32
+ @option_parser.separator("Basic Options:")
33
+ @option_parser.on(
34
+ "-C", "--config PATH",
35
+ "Use the config provided, overriding .platformos-check.yml if present",
36
+ "Use :platformos_app_app_extension to use default checks for app extensions"
37
+ ) { |path| @config_path = path }
38
+ @option_parser.on(
39
+ "-o", "--output FORMAT", FORMATS,
40
+ "The output format to use. (text|json, default: text)"
41
+ ) { |format| @format = format.to_sym }
42
+ @option_parser.on(
43
+ "-c", "--category CATEGORY", Check::CATEGORIES, "Only run this category of checks",
44
+ "Runs checks matching all categories when specified more than once"
45
+ ) { |category| @include_categories << category.to_sym }
46
+ @option_parser.on(
47
+ "-x", "--exclude-category CATEGORY", Check::CATEGORIES, "Exclude this category of checks",
48
+ "Excludes checks matching any category when specified more than once"
49
+ ) { |category| @exclude_categories << category.to_sym }
50
+ @option_parser.on(
51
+ "-a", "--auto-correct",
52
+ "Automatically fix offenses"
53
+ ) { @auto_correct = true }
54
+ @option_parser.on(
55
+ "--fail-level SEVERITY", [:crash] + Check::SEVERITIES,
56
+ "Minimum severity (error|suggestion|style) for exit with error code"
57
+ ) do |severity|
58
+ @fail_level = severity.to_sym
59
+ end
60
+
61
+ @option_parser.separator("")
62
+ @option_parser.separator("Miscellaneous:")
63
+ @option_parser.on(
64
+ "--init",
65
+ "Generate a .platformos-check.yml file"
66
+ ) { @command = :init }
67
+ @option_parser.on(
68
+ "--print",
69
+ "Output active config to STDOUT"
70
+ ) { @command = :print }
71
+ @option_parser.on(
72
+ "--update-docs",
73
+ "Update PlatformOS Check docs (objects, filters, and tags)"
74
+ ) { @update_docs = true }
75
+ @option_parser.on(
76
+ "-h", "--help",
77
+ "Show this. Hi!"
78
+ ) { @command = :help } if help
79
+ @option_parser.on(
80
+ "-l", "--list",
81
+ "List enabled checks"
82
+ ) { @command = :list }
83
+ @option_parser.on(
84
+ "-v", "--version",
85
+ "Print PlatformOS Check version"
86
+ ) { @command = :version }
87
+
88
+ if ENV["PLATFORMOS_CHECK_DEBUG"]
89
+ @option_parser.separator("")
90
+ @option_parser.separator("Debugging:")
91
+ @option_parser.on(
92
+ "--profile",
93
+ "Output a profile to STDOUT compatible with FlameGraph."
94
+ ) { @command = :profile }
95
+ end
96
+
97
+ @option_parser.separator("")
98
+ @option_parser.separator(<<~EOS)
99
+ Description:
100
+ PlatformOS Check helps you follow platformOS best practices by analyzing the
101
+ Liquid & JSON inside your app.
102
+
103
+ You can configure checks in the .platformos-check.yml file of your platformos_app root directory.
104
+ EOS
105
+
106
+ @option_parser
107
+ end
108
+
109
+ def parse(argv)
110
+ @path = option_parser.parse(argv).first || "."
111
+ rescue OptionParser::InvalidArgument => e
112
+ abort(e.message)
113
+ end
114
+
115
+ def run!
116
+ unless %i[version init help].include?(@command)
117
+ @config = if @config_path
118
+ PlatformosCheck::Config.new(
119
+ root: @path,
120
+ configuration: PlatformosCheck::Config.load_config(@config_path)
121
+ )
122
+ else
123
+ PlatformosCheck::Config.from_path(@path)
124
+ end
125
+ @config.include_categories = @include_categories unless @include_categories.empty?
126
+ @config.exclude_categories = @exclude_categories unless @exclude_categories.empty?
127
+ @config.auto_correct = @auto_correct
128
+ end
129
+
130
+ send(@command)
131
+ end
132
+
133
+ def run
134
+ run!
135
+ exit(0)
136
+ rescue Abort => e
137
+ if e.message.empty?
138
+ exit(1)
139
+ else
140
+ abort(e.message)
141
+ end
142
+ rescue PlatformosCheckError => e
143
+ warn(e.message)
144
+ exit(2)
145
+ end
146
+
147
+ def self.parse_and_run!(argv)
148
+ cli = new
149
+ cli.parse(argv)
150
+ cli.run!
151
+ end
152
+
153
+ def self.parse_and_run(argv)
154
+ cli = new
155
+ cli.parse(argv)
156
+ cli.run
157
+ end
158
+
159
+ def list
160
+ puts @config.enabled_checks
161
+ end
162
+
163
+ def version
164
+ puts PlatformosCheck::VERSION
165
+ end
166
+
167
+ def init
168
+ dotfile_path = PlatformosCheck::Config.find(@path)
169
+ raise Abort, "#{PlatformosCheck::Config::DOTFILE} already exists at #{@path}" unless dotfile_path.nil?
170
+
171
+ config_name = @config_path || "default"
172
+ File.write(
173
+ File.join(@path, PlatformosCheck::Config::DOTFILE),
174
+ File.read(PlatformosCheck::Config.bundled_config_path(config_name))
175
+ )
176
+
177
+ puts "Writing new #{PlatformosCheck::Config::DOTFILE} to #{@path}"
178
+ end
179
+
180
+ def print
181
+ puts YAML.dump(@config.to_h)
182
+ end
183
+
184
+ def help
185
+ puts option_parser
186
+ end
187
+
188
+ def check(out_stream = STDOUT)
189
+ update_docs
190
+
191
+ warn "Checking #{@config.root} ..."
192
+ storage = PlatformosCheck::FileSystemStorage.new(@config.root, ignored_patterns: @config.ignored_patterns)
193
+ platformos_app = PlatformosCheck::App.new(storage)
194
+ raise Abort, "No platformos_app files found." if platformos_app.all.empty?
195
+
196
+ analyzer = PlatformosCheck::Analyzer.new(platformos_app, @config.enabled_checks, @config.auto_correct)
197
+ analyzer.analyze_platformos_app
198
+ analyzer.correct_offenses
199
+ print_with_format(platformos_app, analyzer, out_stream)
200
+ # corrections are committed after printing so that the
201
+ # source_excerpts are still pointing to the uncorrected source.
202
+ analyzer.write_corrections
203
+ raise Abort, "" if analyzer.uncorrectable_offenses.any? do |offense|
204
+ offense.check.severity_value <= Check.severity_value(@fail_level)
205
+ end
206
+ end
207
+
208
+ def update_docs
209
+ return unless @update_docs
210
+
211
+ warn 'Updating documentation...'
212
+
213
+ PlatformosCheck::PlatformosLiquid::SourceManager.download
214
+ end
215
+
216
+ def profile
217
+ require 'ruby-prof-flamegraph'
218
+
219
+ result = RubyProf.profile do
220
+ check(STDERR)
221
+ end
222
+
223
+ # Print a graph profile to text
224
+ printer = RubyProf::FlameGraphPrinter.new(result)
225
+ printer.print(STDOUT, {})
226
+ rescue LoadError
227
+ warn "Profiling is only available in development"
228
+ end
229
+
230
+ def print_with_format(platformos_app, analyzer, out_stream)
231
+ case @format
232
+ when :text
233
+ PlatformosCheck::Printer.new(out_stream).print(platformos_app, analyzer.offenses, @config.auto_correct)
234
+ when :json
235
+ PlatformosCheck::JsonPrinter.new(out_stream).print(analyzer.offenses)
236
+ end
237
+ end
238
+ end
239
+ end