blob: c4010f03f3023707eb1577b75228d7fc41fe8f92 [file] [log] [blame] [edit]
# Copyright 2022 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""License compliance checking."""
load(
"@rules_license//rules:gather_licenses_info.bzl",
"gather_licenses_info",
"gather_licenses_info_and_write",
"write_licenses_info",
)
load(
"@rules_license//rules_gathering:gathering_providers.bzl",
"TransitiveLicensesInfo",
)
# Forward licenses used until users migrate. Delete at 0.0.7 or 0.1.0.
load(
"@rules_license//sample_reports:licenses_used.bzl",
_licenses_used = "licenses_used",
)
licenses_used = _licenses_used
# This rule is proof of concept, and may not represent the final
# form of a rule for compliance validation.
def _check_license_impl(ctx):
# Gather all licenses and write information to one place
licenses_file = ctx.actions.declare_file("_%s_licenses_info.json" % ctx.label.name)
write_licenses_info(ctx, ctx.attr.deps, licenses_file)
license_files = []
if ctx.outputs.license_texts:
license_files = get_licenses_mapping(ctx.attr.deps).keys()
# Now run the checker on it
inputs = [licenses_file]
outputs = [ctx.outputs.report]
args = ctx.actions.args()
args.add("--licenses_info", licenses_file.path)
args.add("--report", ctx.outputs.report.path)
if ctx.attr.check_conditions:
args.add("--check_conditions")
if ctx.outputs.copyright_notices:
args.add("--copyright_notices", ctx.outputs.copyright_notices.path)
outputs.append(ctx.outputs.copyright_notices)
if ctx.outputs.license_texts:
args.add("--license_texts", ctx.outputs.license_texts.path)
outputs.append(ctx.outputs.license_texts)
inputs.extend(license_files)
ctx.actions.run(
mnemonic = "CheckLicenses",
progress_message = "Checking license compliance for %s" % ctx.label,
inputs = inputs,
outputs = outputs,
executable = ctx.executable._checker,
arguments = [args],
)
return [
DefaultInfo(files = depset(outputs)),
OutputGroupInfo(licenses_file = depset([licenses_file])),
]
_check_license = rule(
implementation = _check_license_impl,
attrs = {
"deps": attr.label_list(
aspects = [gather_licenses_info],
),
"check_conditions": attr.bool(default = True, mandatory = False),
"copyright_notices": attr.output(mandatory = False),
"license_texts": attr.output(mandatory = False),
"report": attr.output(mandatory = True),
"_checker": attr.label(
default = Label("@rules_license//tools:checker_demo"),
executable = True,
allow_files = True,
cfg = "exec",
),
},
)
# TODO(b/152546336): Update the check to take a pointer to a condition list.
def check_license(**kwargs):
_check_license(**kwargs)
def _manifest_impl(ctx):
# Gather all licenses and make it available as deps for downstream rules
# Additionally write the list of license filenames to a file that can
# also be used as an input to downstream rules.
licenses_file = ctx.actions.declare_file(ctx.attr.out.name)
mappings = get_licenses_mapping(ctx.attr.deps, ctx.attr.warn_on_legacy_licenses)
ctx.actions.write(
output = licenses_file,
content = "\n".join([",".join([f.path, p]) for (f, p) in mappings.items()]),
)
return [DefaultInfo(files = depset(mappings.keys()))]
_manifest = rule(
implementation = _manifest_impl,
doc = """Internal tmplementation method for manifest().""",
attrs = {
"deps": attr.label_list(
doc = """List of targets to collect license files for.""",
aspects = [gather_licenses_info],
),
"out": attr.output(
doc = """Output file.""",
mandatory = True,
),
"warn_on_legacy_licenses": attr.bool(default = False),
},
)
def manifest(name, deps, out = None, **kwargs):
if not out:
out = name + ".manifest"
_manifest(name = name, deps = deps, out = out, **kwargs)
def get_licenses_mapping(deps, warn = False):
"""Creates list of entries representing all licenses for the deps.
Args:
deps: a list of deps which should have TransitiveLicensesInfo providers.
This requires that you have run the gather_licenses_info
aspect over them
warn: boolean, if true, display output about legacy targets that need
update
Returns:
{File:package_name}
"""
tls = []
for dep in deps:
lds = dep[TransitiveLicensesInfo].licenses
tls.append(lds)
ds = depset(transitive = tls)
# Ignore any legacy licenses that may be in the report
mappings = {}
for lic in ds.to_list():
if type(lic.license_text) == "File":
mappings[lic.license_text] = lic.package_name
elif warn:
print("Legacy license %s not included, rule needs updating" % lic.license_text)
return mappings