android_build/tools/compliance/resolutionset.go

305 lines
9.1 KiB
Go

// Copyright 2021 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
//
// http://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.
package compliance
import (
"fmt"
"strings"
)
// JoinResolutionSets returns a new ResolutionSet combining the resolutions from
// multiple resolution sets. All sets must be derived from the same license
// graph.
//
// e.g. combine "restricted", "reciprocal", and "proprietary" resolutions.
func JoinResolutionSets(resolutions ...*ResolutionSet) *ResolutionSet {
if len(resolutions) < 1 {
panic(fmt.Errorf("attempt to join 0 resolution sets"))
}
rmap := make(map[*TargetNode]actionSet)
for _, r := range resolutions {
if len(r.resolutions) < 1 {
continue
}
for attachesTo, as := range r.resolutions {
if as.isEmpty() {
continue
}
if _, ok := rmap[attachesTo]; !ok {
rmap[attachesTo] = as.copy()
continue
}
rmap[attachesTo].addSet(as)
}
}
return &ResolutionSet{rmap}
}
// ResolutionSet describes an immutable set of targets and the license
// conditions each target must satisfy or "resolve" in a specific context.
//
// Ultimately, the purpose of recording the license metadata and building a
// license graph is to identify, describe, and verify the necessary actions or
// operations for compliance policy.
//
// i.e. What is the source-sharing policy? Has it been met? Meet it.
//
// i.e. Are there incompatible policy requirements? Such as a source-sharing
// policy applied to code that policy also says may not be shared? If so, stop
// and remove the dependencies that create the situation.
//
// The ResolutionSet is the base unit for mapping license conditions to the
// targets triggering some necessary action per policy. Different ResolutionSet
// values may be calculated for different contexts.
//
// e.g. Suppose an unencumbered binary links in a notice .a library.
//
// An "unencumbered" condition would originate from the binary, and a "notice"
// condition would originate from the .a library. A ResolutionSet for the
// context of the Notice policy might apply both conditions to the binary while
// preserving the origin of each condition. By applying the notice condition to
// the binary, the ResolutionSet stipulates the policy that the release of the
// unencumbered binary must provide suitable notice for the .a library.
//
// The resulting ResolutionSet could be used for building a notice file, for
// validating that a suitable notice has been built into the distribution, or
// for reporting what notices need to be given.
//
// Resolutions for different contexts may be combined in a new ResolutionSet
// using JoinResolutions(...).
//
// See: resolve.go for:
// * ResolveBottomUpConditions(...)
// * ResolveTopDownForCondition(...)
// See also: policy.go for:
// * ResolveSourceSharing(...)
// * ResolveSourcePrivacy(...)
type ResolutionSet struct {
// resolutions maps names of target with applicable conditions to the set of conditions that apply.
resolutions map[*TargetNode]actionSet
}
// String returns a string representation of the set.
func (rs *ResolutionSet) String() string {
var sb strings.Builder
fmt.Fprintf(&sb, "{")
sep := ""
for attachesTo, as := range rs.resolutions {
fmt.Fprintf(&sb, "%s%s -> %s", sep, attachesTo.name, as.String())
sep = ", "
}
fmt.Fprintf(&sb, "}")
return sb.String()
}
// AttachesTo identifies the list of targets triggering action to resolve
// conditions. (unordered)
func (rs *ResolutionSet) AttachesTo() TargetNodeList {
targets := make(TargetNodeList, 0, len(rs.resolutions))
for attachesTo := range rs.resolutions {
targets = append(targets, attachesTo)
}
return targets
}
// ActsOn identifies the list of targets to act on (share, give notice etc.)
// to resolve conditions. (unordered)
func (rs *ResolutionSet) ActsOn() TargetNodeList {
tset := make(map[*TargetNode]struct{})
for _, as := range rs.resolutions {
for actsOn := range as {
tset[actsOn] = struct{}{}
}
}
targets := make(TargetNodeList, 0, len(tset))
for target := range tset {
targets = append(targets, target)
}
return targets
}
// Origins identifies the list of targets originating conditions to resolve.
// (unordered)
func (rs *ResolutionSet) Origins() TargetNodeList {
tset := make(map[*TargetNode]struct{})
for _, as := range rs.resolutions {
for _, cs := range as {
for _, origins := range cs.conditions {
for origin := range origins {
tset[origin] = struct{}{}
}
}
}
}
targets := make(TargetNodeList, 0, len(tset))
for target := range tset {
targets = append(targets, target)
}
return targets
}
// Resolutions returns the list of resolutions that `attachedTo`
// target must resolve. Returns empty list if no conditions apply.
//
// Panics if `attachedTo` does not appear in the set.
func (rs *ResolutionSet) Resolutions(attachedTo *TargetNode) ResolutionList {
as, ok := rs.resolutions[attachedTo]
if !ok {
return ResolutionList{}
}
result := make(ResolutionList, 0, len(as))
for actsOn, cs := range as {
result = append(result, Resolution{attachedTo, actsOn, cs.Copy()})
}
return result
}
// ResolutionsByActsOn returns the list of resolutions that must `actOn` to
// resolvee. Returns empty list if no conditions apply.
//
// Panics if `actOn` does not appear in the set.
func (rs *ResolutionSet) ResolutionsByActsOn(actOn *TargetNode) ResolutionList {
c := 0
for _, as := range rs.resolutions {
if _, ok := as[actOn]; ok {
c++
}
}
result := make(ResolutionList, 0, c)
for attachedTo, as := range rs.resolutions {
if cs, ok := as[actOn]; ok {
result = append(result, Resolution{attachedTo, actOn, cs.Copy()})
}
}
return result
}
// AttachesToByOrigin identifies the list of targets requiring action to
// resolve conditions originating at `origin`. (unordered)
func (rs *ResolutionSet) AttachesToByOrigin(origin *TargetNode) TargetNodeList {
tset := make(map[*TargetNode]struct{})
for attachesTo, as := range rs.resolutions {
for _, cs := range as {
if cs.HasAnyByOrigin(origin) {
tset[attachesTo] = struct{}{}
break
}
}
}
targets := make(TargetNodeList, 0, len(tset))
for target := range tset {
targets = append(targets, target)
}
return targets
}
// AttachesToTarget returns true if the set contains conditions that
// are `attachedTo`.
func (rs *ResolutionSet) AttachesToTarget(attachedTo *TargetNode) bool {
_, isPresent := rs.resolutions[attachedTo]
return isPresent
}
// AnyByNameAttachToTarget returns true if the set contains conditions matching
// `names` that attach to `attachedTo`.
func (rs *ResolutionSet) AnyByNameAttachToTarget(attachedTo *TargetNode, names ...ConditionNames) bool {
as, isPresent := rs.resolutions[attachedTo]
if !isPresent {
return false
}
for _, cs := range as {
for _, cn := range names {
for _, name := range cn {
_, isPresent = cs.conditions[name]
if isPresent {
return true
}
}
}
}
return false
}
// AllByNameAttachTo returns true if the set contains at least one condition
// matching each element of `names` for `attachedTo`.
func (rs *ResolutionSet) AllByNameAttachToTarget(attachedTo *TargetNode, names ...ConditionNames) bool {
as, isPresent := rs.resolutions[attachedTo]
if !isPresent {
return false
}
for _, cn := range names {
found := false
asloop:
for _, cs := range as {
for _, name := range cn {
_, isPresent = cs.conditions[name]
if isPresent {
found = true
break asloop
}
}
}
if !found {
return false
}
}
return true
}
// IsEmpty returns true if the set contains no conditions to resolve.
func (rs *ResolutionSet) IsEmpty() bool {
for _, as := range rs.resolutions {
if !as.isEmpty() {
return false
}
}
return true
}
// compliance-only ResolutionSet methods
// newResolutionSet constructs a new, empty instance of resolutionSetImp for graph `lg`.
func newResolutionSet() *ResolutionSet {
return &ResolutionSet{make(map[*TargetNode]actionSet)}
}
// addConditions attaches all of the license conditions in `as` to `attachTo` to act on the originating node if not already applied.
func (rs *ResolutionSet) addConditions(attachTo *TargetNode, as actionSet) {
_, ok := rs.resolutions[attachTo]
if !ok {
rs.resolutions[attachTo] = as.copy()
return
}
rs.resolutions[attachTo].addSet(as)
}
// add attaches all of the license conditions in `as` to `attachTo` to act on `attachTo` if not already applied.
func (rs *ResolutionSet) addSelf(attachTo *TargetNode, as actionSet) {
for _, cs := range as {
if cs.IsEmpty() {
return
}
_, ok := rs.resolutions[attachTo]
if !ok {
rs.resolutions[attachTo] = make(actionSet)
}
_, ok = rs.resolutions[attachTo][attachTo]
if !ok {
rs.resolutions[attachTo][attachTo] = newLicenseConditionSet()
}
rs.resolutions[attachTo][attachTo].AddSet(cs)
}
}