packwerk 1.1.0 → 1.3.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (133) hide show
  1. checksums.yaml +4 -4
  2. data/.github/workflows/ci.yml +17 -8
  3. data/.ruby-version +1 -1
  4. data/Gemfile +1 -1
  5. data/Gemfile.lock +129 -111
  6. data/README.md +10 -3
  7. data/TROUBLESHOOT.md +2 -2
  8. data/USAGE.md +30 -30
  9. data/bin/m +29 -0
  10. data/bin/rake +29 -0
  11. data/bin/rubocop +29 -0
  12. data/bin/srb +29 -0
  13. data/bin/tapioca +29 -0
  14. data/dev.yml +7 -7
  15. data/exe/packwerk +1 -1
  16. data/gemfiles/Gemfile-rails-6-0 +22 -0
  17. data/lib/packwerk.rb +72 -34
  18. data/lib/packwerk/application_load_paths.rb +21 -10
  19. data/lib/packwerk/application_validator.rb +104 -84
  20. data/lib/packwerk/association_inspector.rb +23 -11
  21. data/lib/packwerk/checker.rb +4 -7
  22. data/lib/packwerk/cli.rb +36 -129
  23. data/lib/packwerk/configuration.rb +10 -2
  24. data/lib/packwerk/const_node_inspector.rb +13 -14
  25. data/lib/packwerk/constant_discovery.rb +2 -0
  26. data/lib/packwerk/constant_name_inspector.rb +0 -1
  27. data/lib/packwerk/dependency_checker.rb +12 -17
  28. data/lib/packwerk/deprecated_references.rb +8 -10
  29. data/lib/packwerk/file_processor.rb +0 -4
  30. data/lib/packwerk/formatters/offenses_formatter.rb +52 -0
  31. data/lib/packwerk/formatters/progress_formatter.rb +9 -4
  32. data/lib/packwerk/generators/configuration_file.rb +0 -1
  33. data/lib/packwerk/inflector.rb +0 -2
  34. data/lib/packwerk/node.rb +9 -2
  35. data/lib/packwerk/node_processor.rb +15 -32
  36. data/lib/packwerk/node_processor_factory.rb +0 -5
  37. data/lib/packwerk/node_visitor.rb +1 -4
  38. data/lib/packwerk/offense.rb +2 -8
  39. data/lib/packwerk/offense_collection.rb +84 -0
  40. data/lib/packwerk/offenses_formatter.rb +19 -0
  41. data/lib/packwerk/output_style.rb +20 -0
  42. data/lib/packwerk/output_styles/coloured.rb +29 -0
  43. data/lib/packwerk/output_styles/plain.rb +26 -0
  44. data/lib/packwerk/package.rb +17 -1
  45. data/lib/packwerk/package_set.rb +2 -3
  46. data/lib/packwerk/parse_run.rb +106 -0
  47. data/lib/packwerk/parsed_constant_definitions.rb +2 -4
  48. data/lib/packwerk/parsers.rb +0 -2
  49. data/lib/packwerk/parsers/erb.rb +0 -2
  50. data/lib/packwerk/parsers/factory.rb +1 -3
  51. data/lib/packwerk/privacy_checker.rb +22 -17
  52. data/lib/packwerk/reference_extractor.rb +0 -8
  53. data/lib/packwerk/reference_offense.rb +49 -0
  54. data/lib/packwerk/result.rb +9 -0
  55. data/lib/packwerk/run_context.rb +4 -21
  56. data/lib/packwerk/sanity_checker.rb +1 -3
  57. data/lib/packwerk/version.rb +1 -1
  58. data/lib/packwerk/violation_type.rb +0 -2
  59. data/library.yml +1 -1
  60. data/packwerk.gemspec +1 -0
  61. data/service.yml +1 -4
  62. data/shipit.rubygems.yml +5 -1
  63. data/sorbet/rbi/gems/{actioncable@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actioncable@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +56 -36
  64. data/sorbet/rbi/gems/{actionmailbox@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actionmailbox@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +25 -28
  65. data/sorbet/rbi/gems/{actionmailer@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actionmailer@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +43 -24
  66. data/sorbet/rbi/gems/{actionpack@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actionpack@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +382 -284
  67. data/sorbet/rbi/gems/{actiontext@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actiontext@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +76 -40
  68. data/sorbet/rbi/gems/{actionview@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → actionview@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +206 -195
  69. data/sorbet/rbi/gems/{activejob@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → activejob@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +64 -75
  70. data/sorbet/rbi/gems/{activemodel@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → activemodel@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +103 -56
  71. data/sorbet/rbi/gems/{activerecord@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → activerecord@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +1250 -898
  72. data/sorbet/rbi/gems/{activestorage@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → activestorage@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +92 -120
  73. data/sorbet/rbi/gems/{activesupport@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → activesupport@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +292 -193
  74. data/sorbet/rbi/gems/{ast@2.4.1.rbi → ast@2.4.2.rbi} +2 -1
  75. data/sorbet/rbi/gems/{better_html@1.0.15.rbi → better_html@1.0.16.rbi} +2 -2
  76. data/sorbet/rbi/gems/{concurrent-ruby@1.1.6.rbi → concurrent-ruby@1.1.8.rbi} +12 -9
  77. data/sorbet/rbi/gems/{erubi@1.9.0.rbi → erubi@1.10.0.rbi} +3 -1
  78. data/sorbet/rbi/gems/{i18n@1.8.2.rbi → i18n@1.8.10.rbi} +19 -52
  79. data/sorbet/rbi/gems/{loofah@2.5.0.rbi → loofah@2.9.0.rbi} +3 -1
  80. data/sorbet/rbi/gems/marcel@1.0.0.rbi +70 -0
  81. data/sorbet/rbi/gems/{mini_mime@1.0.2.rbi → mini_mime@1.0.3.rbi} +6 -6
  82. data/sorbet/rbi/gems/{mini_portile2@2.4.0.rbi → minitest-focus@1.2.1.rbi} +2 -2
  83. data/sorbet/rbi/gems/{minitest@5.14.0.rbi → minitest@5.14.4.rbi} +31 -29
  84. data/sorbet/rbi/gems/{mocha@1.11.2.rbi → mocha@1.12.0.rbi} +25 -36
  85. data/sorbet/rbi/gems/{nio4r@2.5.2.rbi → nio4r@2.5.7.rbi} +21 -20
  86. data/sorbet/rbi/gems/{nokogiri@1.10.9.rbi → nokogiri@1.11.2.rbi} +193 -154
  87. data/sorbet/rbi/gems/parallel@1.20.1.rbi +117 -0
  88. data/sorbet/rbi/gems/parlour@6.0.0.rbi +1272 -0
  89. data/sorbet/rbi/gems/{parser@2.7.1.4.rbi → parser@3.0.0.0.rbi} +287 -174
  90. data/sorbet/rbi/gems/{pry@0.13.1.rbi → pry@0.14.0.rbi} +1 -1
  91. data/sorbet/rbi/gems/racc@1.5.2.rbi +57 -0
  92. data/sorbet/rbi/gems/{rack@2.2.2.rbi → rack@2.2.3.rbi} +23 -35
  93. data/sorbet/rbi/gems/{rails@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → rails@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +1 -1
  94. data/sorbet/rbi/gems/{railties@6.1.0.alpha-d80c18a391e33552ae2d943e68af56946f883f65.rbi → railties@7.0.0.alpha-d612542336d9a61381311c95a27d801bb4094779.rbi} +132 -121
  95. data/sorbet/rbi/gems/{rake@13.0.1.rbi → rake@13.0.3.rbi} +16 -20
  96. data/sorbet/rbi/gems/{parallel@1.19.1.rbi → regexp_parser@2.1.1.rbi} +2 -2
  97. data/sorbet/rbi/gems/rubocop-ast@1.4.1.rbi +8 -0
  98. data/sorbet/rbi/gems/{rubocop-performance@1.5.2.rbi → rubocop-performance@1.10.2.rbi} +1 -1
  99. data/sorbet/rbi/gems/{rubocop-shopify@1.0.2.rbi → rubocop-shopify@2.0.1.rbi} +1 -1
  100. data/sorbet/rbi/gems/{rubocop-sorbet@0.3.7.rbi → rubocop-sorbet@0.6.1.rbi} +1 -1
  101. data/sorbet/rbi/gems/{rubocop@0.82.0.rbi → rubocop@1.12.0.rbi} +1 -1
  102. data/sorbet/rbi/gems/{ruby-progressbar@1.10.1.rbi → ruby-progressbar@1.11.0.rbi} +1 -1
  103. data/sorbet/rbi/gems/spoom@1.1.0.rbi +1061 -0
  104. data/sorbet/rbi/gems/{spring@2.1.0.rbi → spring@2.1.1.rbi} +7 -7
  105. data/sorbet/rbi/gems/{sprockets-rails@3.2.1.rbi → sprockets-rails@3.2.2.rbi} +88 -68
  106. data/sorbet/rbi/gems/{sprockets@4.0.0.rbi → sprockets@4.0.2.rbi} +8 -7
  107. data/sorbet/rbi/gems/{tapioca@0.4.5.rbi → tapioca@0.4.19.rbi} +109 -24
  108. data/sorbet/rbi/gems/{thor@1.0.1.rbi → thor@1.1.0.rbi} +16 -15
  109. data/sorbet/rbi/gems/{tzinfo@2.0.2.rbi → tzinfo@2.0.4.rbi} +21 -2
  110. data/sorbet/rbi/gems/{unicode-display_width@1.7.0.rbi → unicode-display_width@2.0.0.rbi} +1 -1
  111. data/sorbet/rbi/gems/{websocket-driver@0.7.1.rbi → websocket-driver@0.7.3.rbi} +29 -29
  112. data/sorbet/rbi/gems/{websocket-extensions@0.1.4.rbi → websocket-extensions@0.1.5.rbi} +2 -2
  113. data/sorbet/rbi/gems/zeitwerk@2.4.2.rbi +177 -0
  114. data/sorbet/tapioca/require.rb +1 -0
  115. metadata +83 -65
  116. data/lib/packwerk/cache_deprecated_references.rb +0 -47
  117. data/lib/packwerk/checking_deprecated_references.rb +0 -40
  118. data/lib/packwerk/commands/detect_stale_violations_command.rb +0 -63
  119. data/lib/packwerk/commands/offense_progress_marker.rb +0 -24
  120. data/lib/packwerk/detect_stale_deprecated_references.rb +0 -14
  121. data/lib/packwerk/generators/application_validation.rb +0 -62
  122. data/lib/packwerk/generators/templates/packwerk +0 -23
  123. data/lib/packwerk/generators/templates/packwerk_validator_test.rb +0 -11
  124. data/lib/packwerk/output_styles.rb +0 -41
  125. data/lib/packwerk/reference_lister.rb +0 -23
  126. data/lib/packwerk/spring_command.rb +0 -28
  127. data/lib/packwerk/updating_deprecated_references.rb +0 -14
  128. data/sorbet/rbi/gems/jaro_winkler@1.5.4.rbi +0 -8
  129. data/sorbet/rbi/gems/marcel@0.3.3.rbi +0 -30
  130. data/sorbet/rbi/gems/mimemagic@0.3.5.rbi +0 -47
  131. data/sorbet/rbi/gems/parlour@4.0.1.rbi +0 -561
  132. data/sorbet/rbi/gems/spoom@1.0.4.rbi +0 -418
  133. data/sorbet/rbi/gems/zeitwerk@2.3.0.rbi +0 -8
@@ -1,15 +1,11 @@
1
1
  # typed: strict
2
2
  # frozen_string_literal: true
3
3
 
4
- require "packwerk/constant_name_inspector"
5
- require "packwerk/checker"
6
-
7
4
  module Packwerk
8
5
  class NodeProcessorFactory < T::Struct
9
6
  extend T::Sig
10
7
 
11
8
  const :root_path, String
12
- const :reference_lister, ReferenceLister
13
9
  const :context_provider, Packwerk::ConstantDiscovery
14
10
  const :constant_name_inspectors, T::Array[ConstantNameInspector]
15
11
  const :checkers, T::Array[Checker]
@@ -18,7 +14,6 @@ module Packwerk
18
14
  def for(filename:, node:)
19
15
  ::Packwerk::NodeProcessor.new(
20
16
  reference_extractor: reference_extractor(node: node),
21
- reference_lister: reference_lister,
22
17
  filename: filename,
23
18
  checkers: checkers,
24
19
  )
@@ -1,8 +1,6 @@
1
1
  # typed: false
2
2
  # frozen_string_literal: true
3
3
 
4
- require "packwerk/node"
5
-
6
4
  module Packwerk
7
5
  class NodeVisitor
8
6
  def initialize(node_processor:)
@@ -10,8 +8,7 @@ module Packwerk
10
8
  end
11
9
 
12
10
  def visit(node, ancestors:, result:)
13
- offense = @node_processor.call(node, ancestors: ancestors)
14
- result << offense if offense
11
+ result.concat(@node_processor.call(node, ancestors))
15
12
 
16
13
  child_ancestors = [node] + ancestors
17
14
  Node.each_child(node) do |child|
@@ -2,9 +2,6 @@
2
2
  # frozen_string_literal: true
3
3
 
4
4
  require "parser/source/map"
5
- require "sorbet-runtime"
6
-
7
- require "packwerk/output_styles"
8
5
 
9
6
  module Packwerk
10
7
  class Offense
@@ -23,11 +20,8 @@ module Packwerk
23
20
  @message = message
24
21
  end
25
22
 
26
- sig do
27
- params(style: T.any(T.class_of(OutputStyles::Plain), T.class_of(OutputStyles::Coloured)))
28
- .returns(String)
29
- end
30
- def to_s(style = OutputStyles::Plain)
23
+ sig { params(style: OutputStyle).returns(String) }
24
+ def to_s(style = OutputStyles::Plain.new)
31
25
  if location
32
26
  <<~EOS
33
27
  #{style.filename}#{file}#{style.reset}:#{location.line}:#{location.column}
@@ -0,0 +1,84 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module Packwerk
5
+ class OffenseCollection
6
+ extend T::Sig
7
+ extend T::Helpers
8
+
9
+ sig do
10
+ params(
11
+ root_path: String,
12
+ deprecated_references: T::Hash[Packwerk::Package, Packwerk::DeprecatedReferences]
13
+ ).void
14
+ end
15
+ def initialize(root_path, deprecated_references = {})
16
+ @root_path = root_path
17
+ @deprecated_references = T.let(deprecated_references, T::Hash[Packwerk::Package, Packwerk::DeprecatedReferences])
18
+ @new_violations = T.let([], T::Array[Packwerk::ReferenceOffense])
19
+ @errors = T.let([], T::Array[Packwerk::Offense])
20
+ end
21
+
22
+ sig { returns(T::Array[Packwerk::ReferenceOffense]) }
23
+ attr_reader :new_violations
24
+
25
+ sig { returns(T::Array[Packwerk::Offense]) }
26
+ attr_reader :errors
27
+
28
+ sig do
29
+ params(offense: Packwerk::Offense)
30
+ .returns(T::Boolean)
31
+ end
32
+ def listed?(offense)
33
+ return false unless offense.is_a?(ReferenceOffense)
34
+ reference = offense.reference
35
+ deprecated_references_for(reference.source_package).listed?(reference, violation_type: offense.violation_type)
36
+ end
37
+
38
+ sig do
39
+ params(offense: Packwerk::Offense).void
40
+ end
41
+ def add_offense(offense)
42
+ unless offense.is_a?(ReferenceOffense)
43
+ @errors << offense
44
+ return
45
+ end
46
+ deprecated_references = deprecated_references_for(offense.reference.source_package)
47
+ unless deprecated_references.add_entries(offense.reference, offense.violation_type)
48
+ new_violations << offense
49
+ end
50
+ end
51
+
52
+ sig { returns(T::Boolean) }
53
+ def stale_violations?
54
+ @deprecated_references.values.any?(&:stale_violations?)
55
+ end
56
+
57
+ sig { void }
58
+ def dump_deprecated_references_files
59
+ @deprecated_references.each do |_, deprecated_references_file|
60
+ deprecated_references_file.dump
61
+ end
62
+ end
63
+
64
+ sig { returns(T::Array[Packwerk::Offense]) }
65
+ def outstanding_offenses
66
+ errors + new_violations
67
+ end
68
+
69
+ private
70
+
71
+ sig { params(package: Packwerk::Package).returns(Packwerk::DeprecatedReferences) }
72
+ def deprecated_references_for(package)
73
+ @deprecated_references[package] ||= Packwerk::DeprecatedReferences.new(
74
+ package,
75
+ deprecated_references_file_for(package),
76
+ )
77
+ end
78
+
79
+ sig { params(package: Packwerk::Package).returns(String) }
80
+ def deprecated_references_file_for(package)
81
+ File.join(@root_path, package.name, "deprecated_references.yml")
82
+ end
83
+ end
84
+ end
@@ -0,0 +1,19 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module Packwerk
5
+ module OffensesFormatter
6
+ extend T::Sig
7
+ extend T::Helpers
8
+
9
+ interface!
10
+
11
+ sig { abstract.params(offenses: T::Array[T.nilable(Offense)]).returns(String) }
12
+ def show_offenses(offenses)
13
+ end
14
+
15
+ sig { abstract.params(offense_collection: Packwerk::OffenseCollection).returns(String) }
16
+ def show_stale_violations(offense_collection)
17
+ end
18
+ end
19
+ end
@@ -0,0 +1,20 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module Packwerk
5
+ module OutputStyle
6
+ extend T::Sig
7
+ extend T::Helpers
8
+
9
+ interface!
10
+
11
+ sig { abstract.returns(String) }
12
+ def reset; end
13
+
14
+ sig { abstract.returns(String) }
15
+ def filename; end
16
+
17
+ sig { abstract.returns(String) }
18
+ def error; end
19
+ end
20
+ end
@@ -0,0 +1,29 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module Packwerk
5
+ module OutputStyles
6
+ # See https://en.wikipedia.org/wiki/ANSI_escape_code#3/4_bit for ANSI escape colour codes
7
+ class Coloured
8
+ extend T::Sig
9
+ include OutputStyle
10
+
11
+ sig { override.returns(String) }
12
+ def reset
13
+ "\033[m"
14
+ end
15
+
16
+ sig { override.returns(String) }
17
+ def filename
18
+ # 36 is foreground cyan
19
+ "\033[36m"
20
+ end
21
+
22
+ sig { override.returns(String) }
23
+ def error
24
+ # 31 is foreground red
25
+ "\033[31m"
26
+ end
27
+ end
28
+ end
29
+ end
@@ -0,0 +1,26 @@
1
+ # typed: strict
2
+ # frozen_string_literal: true
3
+
4
+ module Packwerk
5
+ module OutputStyles
6
+ class Plain
7
+ extend T::Sig
8
+ include OutputStyle
9
+
10
+ sig { override.returns(String) }
11
+ def reset
12
+ ""
13
+ end
14
+
15
+ sig { override.returns(String) }
16
+ def filename
17
+ ""
18
+ end
19
+
20
+ sig { override.returns(String) }
21
+ def error
22
+ ""
23
+ end
24
+ end
25
+ end
26
+ end
@@ -33,7 +33,15 @@ module Packwerk
33
33
  end
34
34
 
35
35
  def public_path
36
- @public_path ||= File.join(@name, user_defined_public_path || "app/public/")
36
+ @public_path ||= begin
37
+ unprefixed_public_path = user_defined_public_path || "app/public/"
38
+
39
+ if root?
40
+ unprefixed_public_path
41
+ else
42
+ File.join(@name, unprefixed_public_path)
43
+ end
44
+ end
37
45
  end
38
46
 
39
47
  def public_path?(path)
@@ -52,6 +60,14 @@ module Packwerk
52
60
  name <=> other.name
53
61
  end
54
62
 
63
+ def eql?(other)
64
+ self == other
65
+ end
66
+
67
+ def hash
68
+ name.hash
69
+ end
70
+
55
71
  def to_s
56
72
  name
57
73
  end
@@ -3,8 +3,6 @@
3
3
 
4
4
  require "pathname"
5
5
 
6
- require "packwerk/package"
7
-
8
6
  module Packwerk
9
7
  class PackageSet
10
8
  include Enumerable
@@ -60,7 +58,8 @@ module Packwerk
60
58
  end
61
59
 
62
60
  def package_from_path(file_path)
63
- @packages.values.find { |package| package.package_path?(file_path) }
61
+ path_string = file_path.to_s
62
+ @packages.values.find { |package| package.package_path?(path_string) }
64
63
  end
65
64
  end
66
65
  end
@@ -0,0 +1,106 @@
1
+ # typed: true
2
+ # frozen_string_literal: true
3
+
4
+ require "benchmark"
5
+ require "parallel"
6
+
7
+ module Packwerk
8
+ class ParseRun
9
+ extend T::Sig
10
+
11
+ def initialize(
12
+ files:,
13
+ configuration:,
14
+ progress_formatter: Formatters::ProgressFormatter.new(StringIO.new),
15
+ offenses_formatter: Formatters::OffensesFormatter.new
16
+ )
17
+ @configuration = configuration
18
+ @progress_formatter = progress_formatter
19
+ @offenses_formatter = offenses_formatter
20
+ @files = files
21
+ end
22
+
23
+ def detect_stale_violations
24
+ offense_collection = find_offenses
25
+
26
+ result_status = !offense_collection.stale_violations?
27
+ message = @offenses_formatter.show_stale_violations(offense_collection)
28
+
29
+ Result.new(message: message, status: result_status)
30
+ end
31
+
32
+ def update_deprecations
33
+ offense_collection = find_offenses
34
+ offense_collection.dump_deprecated_references_files
35
+
36
+ message = <<~EOS
37
+ #{@offenses_formatter.show_offenses(offense_collection.errors)}
38
+ ✅ `deprecated_references.yml` has been updated.
39
+ EOS
40
+
41
+ Result.new(message: message, status: offense_collection.errors.empty?)
42
+ end
43
+
44
+ def check
45
+ offense_collection = find_offenses(show_errors: true)
46
+
47
+ messages = [
48
+ @offenses_formatter.show_offenses(offense_collection.outstanding_offenses),
49
+ @offenses_formatter.show_stale_violations(offense_collection),
50
+ ]
51
+ result_status = offense_collection.outstanding_offenses.empty? && !offense_collection.stale_violations?
52
+
53
+ Result.new(message: messages.join("\n") + "\n", status: result_status)
54
+ end
55
+
56
+ private
57
+
58
+ def find_offenses(show_errors: false)
59
+ offense_collection = OffenseCollection.new(@configuration.root_path)
60
+ @progress_formatter.started(@files)
61
+
62
+ run_context = Packwerk::RunContext.from_configuration(@configuration)
63
+ all_offenses = T.let([], T.untyped)
64
+
65
+ process_file = -> (path) do
66
+ run_context.process_file(file: path).tap do |offenses|
67
+ failed = show_errors && offenses.any? { |offense| !offense_collection.listed?(offense) }
68
+ update_progress(failed: failed)
69
+ end
70
+ end
71
+
72
+ execution_time = Benchmark.realtime do
73
+ all_offenses = if @configuration.parallel?
74
+ Parallel.flat_map(@files, &process_file)
75
+ else
76
+ serial_find_offenses(&process_file)
77
+ end
78
+ end
79
+
80
+ @progress_formatter.finished(execution_time)
81
+
82
+ all_offenses.each { |offense| offense_collection.add_offense(offense) }
83
+ offense_collection
84
+ end
85
+
86
+ def serial_find_offenses
87
+ all_offenses = T.let([], T.untyped)
88
+ @files.each do |path|
89
+ offenses = yield path
90
+ all_offenses.concat(offenses)
91
+ end
92
+ all_offenses
93
+ rescue Interrupt
94
+ @progress_formatter.interrupted
95
+ all_offenses
96
+ end
97
+
98
+ def update_progress(failed: false)
99
+ if failed
100
+ @progress_formatter.mark_as_failed
101
+ else
102
+ @progress_formatter.mark_as_inspected
103
+ end
104
+ end
105
+ end
106
+ end
@@ -3,8 +3,6 @@
3
3
 
4
4
  require "ast/node"
5
5
 
6
- require "packwerk/node"
7
-
8
6
  module Packwerk
9
7
  class ParsedConstantDefinitions
10
8
  def initialize(root_node:)
@@ -28,8 +26,8 @@ module Packwerk
28
26
 
29
27
  fully_qualified_constant_name = "::#{constant_name}"
30
28
 
31
- possible_namespaces = namespace_path.reduce([""]) do |acc, current|
32
- acc << acc.last + "::" + current
29
+ possible_namespaces = namespace_path.each_with_object([""]) do |current, acc|
30
+ acc << "#{acc.last}::#{current}" if acc.last && current
33
31
  end
34
32
 
35
33
  possible_namespaces.map { |namespace| namespace + fully_qualified_constant_name }
@@ -1,8 +1,6 @@
1
1
  # typed: true
2
2
  # frozen_string_literal: true
3
3
 
4
- require "packwerk/offense"
5
-
6
4
  module Packwerk
7
5
  module Parsers
8
6
  autoload :Erb, "packwerk/parsers/erb"
@@ -6,8 +6,6 @@ require "better_html"
6
6
  require "better_html/parser"
7
7
  require "parser/source/buffer"
8
8
 
9
- require "packwerk/parsers"
10
-
11
9
  module Packwerk
12
10
  module Parsers
13
11
  class Erb