Skip to content

Commit 9a9bd38

Browse files
authored
Merge branch 'github:main' into master
2 parents 8842cf6 + 0f5e5a1 commit 9a9bd38

File tree

13 files changed

+193
-13
lines changed

13 files changed

+193
-13
lines changed

.github/workflows/test.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -296,7 +296,7 @@ jobs:
296296
steps:
297297
- uses: actions/checkout@v3
298298
- name: Setup dotnet
299-
uses: actions/setup-dotnet@v3.0.3
299+
uses: actions/setup-dotnet@v3.1.0
300300
with:
301301
dotnet-version: ${{ matrix.dotnet }}
302302
- name: Set up Ruby

CHANGELOG.md

+7-1
Original file line numberDiff line numberDiff line change
@@ -6,6 +6,12 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.
66

77
## [Unreleased]
88

9+
## 4.4.0
10+
11+
### Added
12+
13+
- Licensed status command will alert on stale cached dependency records (https://github.com/github/licensed/pull/657)
14+
915
## 4.3.1
1016

1117
### Changed
@@ -735,4 +741,4 @@ and this project adheres to [Semantic Versioning](http://semver.org/spec/v2.0.0.
735741

736742
Initial release :tada:
737743

738-
[Unreleased]: https://github.com/github/licensed/compare/4.3.1...HEAD
744+
[Unreleased]: https://github.com/github/licensed/compare/4.4.0...HEAD

Gemfile.lock

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
PATH
22
remote: .
33
specs:
4-
licensed (4.3.1)
4+
licensed (4.4.0)
55
json (~> 2.6)
66
licensee (~> 9.16)
77
parallel (~> 1.22)

docs/commands/status.md

+15
Original file line numberDiff line numberDiff line change
@@ -31,6 +31,21 @@ A dependency will fail the status checks if:
3131
- If `license: other` is specified and all of the `licenses` entries match an `allowed` license a failure will not be logged
3232
- A `reviewed` entry must reference a specific version of the depdency, e.g. `<name>@<version>`. The version identifier must specify a specific dependency version, ranges are not allowed.
3333

34+
## Detect and alert on stale cached metadata files
35+
36+
Licensed can alert on any metadata files that don't correlate to a currently used dependency when `licensed status` is run. To configure this behavior, set a root-level `stale_records_action` value in your [licensed configuration file](./../configuration.md).
37+
38+
Available values are:
39+
40+
1. `'error'`: Treat stale cached records as errors. Licensed will output errors for any stale metadata files and will cause `licensed status` to fail.
41+
1. `'warn'`, `''`, or unset (default): Treat stale cached records as warnings. Licensed will output warnings for any stale metadata files but will not cause `licensed status` to fail.
42+
1. `'ignore'`, any other value: Ignore stale cached records. Licensed will not output any notifications about stale metadata files.
43+
44+
```yaml
45+
# in the licensed configuration file
46+
stale_records_action: 'warn'
47+
```
48+
3449
## Options
3550
3651
- `--config`/`-c`: the path to the licensed configuration file

docs/configuration.md

+9
Original file line numberDiff line numberDiff line change
@@ -26,6 +26,15 @@ cache_path: 'relative/path/to/cache'
2626
# Defaults to current directory when running `licensed`
2727
source_path: 'relative/path/to/source'
2828

29+
# Whether to take any action when records are detected in the cache paths that don't map to evaluated
30+
# dependencies.
31+
# Available values are:
32+
# - 'error': treat stale cached records as errors. Notify the user and fail status checks
33+
# - 'warn', '', unset: treat stale cached records as warnings. Notify the user but do not fail status checks
34+
# - 'ignore': Ignore stale cached records. Do not notify the user and do not fail status checks
35+
# Optional, when not set this defaults to 'warn' behavior
36+
stale_records_action: 'warn'
37+
2938
# Sources of metadata
3039
sources:
3140
bower: true

lib/licensed/commands/cache.rb

+1
Original file line numberDiff line numberDiff line change
@@ -23,6 +23,7 @@ def default_reporter(options)
2323
def run_command(report)
2424
super do |result|
2525
clear_stale_cached_records if result
26+
result
2627
end
2728
ensure
2829
cache_paths.clear

lib/licensed/commands/command.rb

+4-4
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ def run_command(report)
6969

7070
result = results.all?
7171

72-
yield(result) if block_given?
72+
result = yield(result) if block_given?
7373

7474
result
7575
ensure
@@ -103,7 +103,7 @@ def run_app(app, report)
103103

104104
result = results.all?
105105

106-
yield(result) if block_given?
106+
result = yield(result) if block_given?
107107

108108
result
109109
end
@@ -142,7 +142,7 @@ def run_source(app, source, report)
142142

143143
result = results.all?
144144

145-
yield(result) if block_given?
145+
result = yield(result) if block_given?
146146

147147
result
148148
rescue Licensed::Shell::Error => err
@@ -175,7 +175,7 @@ def run_dependency(app, source, dependency, report)
175175

176176
result = evaluate_dependency(app, source, dependency, report)
177177

178-
yield(result) if block_given?
178+
result = yield(result) if block_given?
179179

180180
result
181181
rescue Licensed::DependencyRecord::Error, Licensed::Shell::Error => err

lib/licensed/commands/status.rb

+62-1
Original file line numberDiff line numberDiff line change
@@ -23,10 +23,48 @@ def default_reporter(options)
2323
# Returns whether the command succeeded based on the call to super
2424
def run_command(report)
2525
super do |result|
26-
next if result
26+
stale_records = stale_cached_records
27+
if stale_records.any?
28+
messages = stale_records.map { |f| "Stale dependency record found: #{f}" }
29+
messages << "Please run the licensed cache command to clean up stale records"
30+
31+
case config["stale_records_action"].to_s
32+
when "error"
33+
report.errors.concat messages
34+
result = false
35+
when "warn", ""
36+
report.warnings.concat messages
37+
end
38+
end
39+
40+
next result if result
2741

2842
report.errors << "Licensed found errors during source enumeration. Please see https://github.com/github/licensed/tree/master/docs/commands/status.md#status-errors-and-resolutions for possible resolutions."
43+
44+
result
2945
end
46+
ensure
47+
cache_paths.clear
48+
files.clear
49+
end
50+
51+
# Run the command for all enumerated dependencies found in a dependency source,
52+
# recording results in a report.
53+
# Enumerating dependencies in the source is skipped if a :sources option
54+
# is provided and the evaluated `source.class.type` is not in the :sources values
55+
#
56+
# app - The application configuration for the source
57+
# source - A dependency source enumerator
58+
#
59+
# Returns whether the command succeeded for the dependency source enumerator
60+
def run_source(app, source, report)
61+
result = super
62+
63+
# add the full cache path to the list of cache paths
64+
# that should be checked for extra files after the command run
65+
cache_paths << app.cache_path.join(source.class.type) unless result == :skipped
66+
67+
result
3068
end
3169

3270
# Evaluates a dependency for any compliance errors.
@@ -49,6 +87,9 @@ def evaluate_dependency(app, source, dependency, report)
4987
filename = app.cache_path.join(source.class.type, "#{dependency.name}.#{DependencyRecord::EXTENSION}")
5088
report["filename"] = filename
5189
record = cached_record(filename)
90+
91+
# add the absolute dependency file path to the list of files seen during this licensed run
92+
files << filename.to_s
5293
end
5394

5495
if record.nil?
@@ -133,6 +174,26 @@ def license_from_text(text)
133174

134175
licenses.sort_by { |license| license != "other" ? 0 : 1 }.first
135176
end
177+
178+
# Check for cached files that don't match current dependencies
179+
#
180+
# Returns an array of any cached records that do not match a currently used dependency
181+
def stale_cached_records
182+
cache_paths.flat_map do |cache_path|
183+
record_search_glob_pattern = cache_path.join("**/*.#{DependencyRecord::EXTENSION}")
184+
Dir.glob(record_search_glob_pattern).select { |file| !files.include?(file) }
185+
end.uniq
186+
end
187+
188+
# Set of unique cache paths that are evaluted during the run
189+
def cache_paths
190+
@cache_paths ||= Set.new
191+
end
192+
193+
# Set of unique absolute file paths of cached records evaluted during the run
194+
def files
195+
@files ||= Set.new
196+
end
136197
end
137198
end
138199
end

lib/licensed/configuration.rb

+5
Original file line numberDiff line numberDiff line change
@@ -274,6 +274,7 @@ def self.load_from(path)
274274
end
275275

276276
def initialize(options = {})
277+
@options = options
277278
apps = options.delete("apps") || []
278279
apps << default_options.merge(options) if apps.empty?
279280

@@ -285,6 +286,10 @@ def initialize(options = {})
285286
@apps = apps.map { |app| AppConfiguration.new(app, options) }
286287
end
287288

289+
def [](key)
290+
@options&.fetch(key, nil)
291+
end
292+
288293
private
289294

290295
def self.expand_app_source_path(app_config)

lib/licensed/reporters/status_reporter.rb

+5
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,11 @@ class StatusReporter < Reporter
88
# command - The command being run
99
# report - A report object containing information about the command run
1010
def end_report_command(command, report)
11+
if report.warnings.any?
12+
shell.newline
13+
report.warnings.each { |e| shell.warn e }
14+
end
15+
1116
if report.errors.any?
1217
shell.newline
1318
report.errors.each { |e| shell.error e }

lib/licensed/version.rb

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
# frozen_string_literal: true
22
module Licensed
3-
VERSION = "4.3.1".freeze
3+
VERSION = "4.4.0".freeze
44

55
def self.previous_major_versions
66
major_version = Gem::Version.new(Licensed::VERSION).segments.first

test/commands/status_test.rb

+70-4
Original file line numberDiff line numberDiff line change
@@ -9,7 +9,8 @@
99
let(:reporter) { TestReporter.new }
1010
let(:apps) { [] }
1111
let(:source_config) { {} }
12-
let(:config) { Licensed::Configuration.new("apps" => apps, "cache_path" => cache_path, "sources" => { "test" => true }, "test" => source_config) }
12+
let(:command_config) { { "apps" => apps, "cache_path" => cache_path, "sources" => { "test" => true }, "test" => source_config } }
13+
let(:config) { Licensed::Configuration.new(command_config) }
1314
let(:fixtures) { File.expand_path("../../fixtures", __FILE__) }
1415
let(:command) { Licensed::Commands::Status.new(config: config) }
1516

@@ -24,11 +25,15 @@ def dependency_errors(app, source, dependency_name = "dependency")
2425
dependency_report&.errors || []
2526
end
2627

28+
def generate_metadata_files
29+
generator_config = Marshal.load(Marshal.dump(config))
30+
generator = Licensed::Commands::Cache.new(config: generator_config)
31+
generator.run(force: true, reporter: TestReporter.new)
32+
end
33+
2734
describe "with cached metadata data source" do
2835
before do
29-
generator_config = Marshal.load(Marshal.dump(config))
30-
generator = Licensed::Commands::Cache.new(config: generator_config)
31-
generator.run(force: true, reporter: TestReporter.new)
36+
generate_metadata_files
3237
end
3338

3439
after do
@@ -776,4 +781,65 @@ def run_command(**opts)
776781
end
777782
end
778783
end
784+
785+
describe "with stale cached records" do
786+
let(:unused_record_file_path) do
787+
app = config.apps.first
788+
source = app.sources.first
789+
File.join(app.cache_path, source.class.type, "unused.#{Licensed::DependencyRecord::EXTENSION}")
790+
end
791+
792+
before do
793+
# generate artifacts needed for the status command to normally pass
794+
# in order to validate that the command passes or fails depending on
795+
# the stale_records_action config setting
796+
generate_metadata_files
797+
config.apps.each do |app|
798+
app.allow "mit"
799+
end
800+
801+
FileUtils.mkdir_p File.dirname(unused_record_file_path)
802+
File.write(unused_record_file_path, "")
803+
end
804+
805+
after do
806+
config.apps.each do |app|
807+
FileUtils.rm_rf app.cache_path
808+
end
809+
end
810+
811+
it "reports an error on stale cached records when configured" do
812+
command_config["stale_records_action"] = "error"
813+
814+
refute run_command
815+
816+
assert reporter.report.errors.include?("Stale dependency record found: #{unused_record_file_path}")
817+
refute reporter.report.warnings.include?("Stale dependency record found: #{unused_record_file_path}")
818+
end
819+
820+
it "reports a warning on stale cached records when unconfigured" do
821+
assert run_command
822+
823+
refute reporter.report.errors.include?("Stale dependency record found: #{unused_record_file_path}")
824+
assert reporter.report.warnings.include?("Stale dependency record found: #{unused_record_file_path}")
825+
end
826+
827+
it "reports a warning on stale cached records when configured" do
828+
command_config["stale_records_action"] = "warn"
829+
830+
assert run_command
831+
832+
refute reporter.report.errors.include?("Stale dependency record found: #{unused_record_file_path}")
833+
assert reporter.report.warnings.include?("Stale dependency record found: #{unused_record_file_path}")
834+
end
835+
836+
it "ignores stale cached records when configured" do
837+
command_config["stale_records_action"] = "ignore"
838+
839+
assert run_command
840+
841+
refute reporter.report.errors.include?("Stale dependency record found: #{unused_record_file_path}")
842+
refute reporter.report.warnings.include?("Stale dependency record found: #{unused_record_file_path}")
843+
end
844+
end
779845
end

test/reporters/status_reporter_test.rb

+12
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,18 @@
2424
style: :error
2525
}
2626
end
27+
28+
it "reports any warnings specified on the report object" do
29+
report.warnings << "command warning"
30+
reporter.end_report_command(command, report)
31+
32+
assert_includes shell.messages,
33+
{
34+
message: "command warning",
35+
newline: true,
36+
style: :warn
37+
}
38+
end
2739
end
2840

2941
describe "#begin_report_app" do

0 commit comments

Comments
 (0)