pub mod cfg;
mod diags;
mod gather;
use crate::{
diag::{CfgCoord, Check, Diagnostic, Label, Pack, Severity},
LintLevel,
};
use cfg::BlanketAgreement;
pub use gather::{Gatherer, LicenseInfo, LicenseStore};
use gather::{KrateLicense, LicenseExprInfo, LicenseExprSource, Summary};
pub use cfg::{Config, ValidConfig};
pub use diags::Code;
use bitvec::prelude::*;
struct Hits {
allowed: BitVec<usize, LocalBits>,
exceptions: BitVec<usize, LocalBits>,
}
fn evaluate_expression(
cfg: &ValidConfig,
krate_lic_nfo: &KrateLicense<'_>,
expr: &spdx::Expression,
nfo: &LicenseExprInfo,
hits: &mut Hits,
) -> Diagnostic {
#[derive(Debug)]
enum Reason {
Denied,
IsFsfFree,
IsOsiApproved,
IsBothFreeAndOsi,
ExplicitAllowance,
ExplicitException,
IsCopyleft,
Default,
}
let mut reasons = smallvec::SmallVec::<[(Reason, bool); 8]>::new();
macro_rules! deny {
($reason:ident) => {
reasons.push((Reason::$reason, false));
return false;
};
}
macro_rules! allow {
($reason:ident) => {
reasons.push((Reason::$reason, true));
return true;
};
}
let mut warnings = 0;
let exception_ind = cfg.exceptions.iter().position(|exc| {
exc.name.as_ref() == &krate_lic_nfo.krate.name
&& crate::match_req(&krate_lic_nfo.krate.version, exc.version.as_ref())
});
let eval_res = expr.evaluate_with_failures(|req| {
if let Some(ind) = exception_ind {
let exception = &cfg.exceptions[ind];
for allow in &exception.allowed {
if allow.value.satisfies(req) {
hits.exceptions.as_mut_bitslice().set(ind, true);
allow!(ExplicitException);
}
}
}
for deny in &cfg.denied {
if deny.value.satisfies(req) {
deny!(Denied);
}
}
for (i, allow) in cfg.allowed.iter().enumerate() {
if allow.value.satisfies(req) {
hits.allowed.as_mut_bitslice().set(i, true);
allow!(ExplicitAllowance);
}
}
if let spdx::LicenseItem::Spdx { id, .. } = req.license {
if id.is_copyleft() {
match cfg.copyleft {
LintLevel::Allow => {
allow!(IsCopyleft);
}
LintLevel::Warn => {
warnings += 1;
allow!(IsCopyleft);
}
LintLevel::Deny => {
deny!(IsCopyleft);
}
}
}
match cfg.allow_osi_fsf_free {
BlanketAgreement::Neither => {}
BlanketAgreement::Either => {
if id.is_osi_approved() {
allow!(IsOsiApproved);
} else if id.is_fsf_free_libre() {
allow!(IsFsfFree);
}
}
BlanketAgreement::Both => {
if id.is_fsf_free_libre() && id.is_osi_approved() {
allow!(IsBothFreeAndOsi);
}
}
BlanketAgreement::Osi => {
if id.is_osi_approved() {
allow!(IsOsiApproved);
}
}
BlanketAgreement::Fsf => {
if id.is_fsf_free_libre() {
allow!(IsFsfFree);
}
}
BlanketAgreement::OsiOnly => {
if id.is_osi_approved() {
if id.is_fsf_free_libre() {
deny!(IsFsfFree);
} else {
allow!(IsOsiApproved);
}
}
}
BlanketAgreement::FsfOnly => {
if id.is_fsf_free_libre() {
if id.is_osi_approved() {
deny!(IsOsiApproved);
} else {
allow!(IsFsfFree);
}
}
}
}
}
match cfg.default {
LintLevel::Deny => {
deny!(Default);
}
LintLevel::Warn => {
warnings += 1;
allow!(Default);
}
LintLevel::Allow => {
allow!(Default);
}
}
});
let (message, severity) = match eval_res {
Err(_) => ("failed to satisfy license requirements", Severity::Error),
Ok(_) => (
"license requirements satisfied",
if warnings > 0 {
Severity::Warning
} else {
Severity::Help
},
),
};
let mut labels = Vec::with_capacity(reasons.len() + 1);
labels.extend(krate_lic_nfo.labels.clone());
labels.push(
Label::secondary(nfo.file_id, nfo.offset..nfo.offset + expr.as_ref().len()).with_message(
format!(
"license expression retrieved via {}",
match &nfo.source {
LicenseExprSource::Metadata => "Cargo.toml `license`".to_owned(),
LicenseExprSource::UserOverride => "user override".to_owned(),
LicenseExprSource::LicenseFiles(lfs) => lfs.join(", "),
LicenseExprSource::OverlayOverride => unreachable!(),
}
),
),
);
for (reason, failed_req) in reasons.into_iter().zip(expr.requirements()) {
labels.push(
Label::primary(
nfo.file_id,
nfo.offset + failed_req.span.start as usize
..nfo.offset + failed_req.span.end as usize,
)
.with_message(format!(
"{}: {}",
if reason.1 { "accepted" } else { "rejected" },
match reason.0 {
Reason::Denied => "explicitly denied",
Reason::IsFsfFree =>
"license is FSF approved https://www.gnu.org/licenses/license-list.en.html",
Reason::IsOsiApproved =>
"license is OSI approved https://opensource.org/licenses",
Reason::ExplicitAllowance => "license is explicitly allowed",
Reason::ExplicitException => "license is explicitly allowed via an exception",
Reason::IsBothFreeAndOsi => "license is FSF AND OSI approved",
Reason::IsCopyleft => "license is considered copyleft",
Reason::Default => {
match cfg.default {
LintLevel::Deny => "not explicitly allowed",
LintLevel::Warn => "warned by default",
LintLevel::Allow => "allowed by default",
}
}
}
)),
);
}
Diagnostic::new(severity)
.with_message(message)
.with_code(if severity != Severity::Error {
diags::Code::Accepted
} else {
diags::Code::Rejected
})
.with_labels(labels)
}
pub fn check(
ctx: crate::CheckCtx<'_, ValidConfig>,
summary: Summary<'_>,
mut sink: crate::diag::ErrorSink,
) {
let mut hits = Hits {
allowed: BitVec::repeat(false, ctx.cfg.allowed.len()),
exceptions: BitVec::repeat(false, ctx.cfg.exceptions.len()),
};
let private_registries: Vec<_> = ctx
.cfg
.private
.registries
.iter()
.map(|s| s.as_str())
.collect();
for krate_lic_nfo in summary.nfos {
let mut pack = Pack::with_kid(Check::Licenses, krate_lic_nfo.krate.id.clone());
if ctx.cfg.private.ignore
&& (krate_lic_nfo.krate.is_private(&private_registries)
|| ctx
.cfg
.ignore_sources
.iter()
.any(|url| krate_lic_nfo.krate.matches_url(url, true)))
{
pack.push(diags::SkippedPrivateWorkspaceCrate {
krate: krate_lic_nfo.krate,
});
sink.push(pack);
continue;
}
match &krate_lic_nfo.lic_info {
LicenseInfo::SpdxExpression { expr, nfo } => {
pack.push(evaluate_expression(
&ctx.cfg,
&krate_lic_nfo,
expr,
nfo,
&mut hits,
));
}
LicenseInfo::Unlicensed => {
let severity = match ctx.cfg.unlicensed {
LintLevel::Allow => Severity::Note,
LintLevel::Warn => Severity::Warning,
LintLevel::Deny => Severity::Error,
};
pack.push(diags::Unlicensed {
krate: krate_lic_nfo.krate,
severity,
breadcrumbs: krate_lic_nfo.labels.into_iter().collect(),
});
}
}
if !pack.is_empty() {
sink.push(pack);
}
}
{
let mut pack = Pack::new(Check::Licenses);
for exc in hits
.exceptions
.into_iter()
.zip(ctx.cfg.exceptions.into_iter())
.filter_map(|(hit, exc)| if !hit { Some(exc) } else { None })
{
if exc.file_id != ctx.cfg.file_id {
continue;
}
pack.push(diags::UnmatchedLicenseException {
license_exc_cfg: CfgCoord {
file: exc.file_id,
span: exc.name.span,
},
});
}
if !pack.is_empty() {
sink.push(pack);
}
}
{
let mut pack = Pack::new(Check::Licenses);
for allowed in hits
.allowed
.into_iter()
.zip(ctx.cfg.allowed.into_iter())
.filter_map(|(hit, allowed)| if !hit { Some(allowed) } else { None })
{
pack.push(diags::UnmatchedLicenseAllowance {
severity: ctx.cfg.unused_allowed_license.into(),
allowed_license_cfg: CfgCoord {
file: ctx.cfg.file_id,
span: allowed.span,
},
});
}
if !pack.is_empty() {
sink.push(pack);
}
}
}