mirror of
https://github.com/status-im/status-go.git
synced 2025-01-20 19:52:42 +00:00
322 lines
9.0 KiB
Go
322 lines
9.0 KiB
Go
// Copyright (c) 2021 Uber Technologies, Inc.
|
|
//
|
|
// Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
// of this software and associated documentation files (the "Software"), to deal
|
|
// in the Software without restriction, including without limitation the rights
|
|
// to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
// copies of the Software, and to permit persons to whom the Software is
|
|
// furnished to do so, subject to the following conditions:
|
|
//
|
|
// The above copyright notice and this permission notice shall be included in
|
|
// all copies or substantial portions of the Software.
|
|
//
|
|
// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
|
// THE SOFTWARE.
|
|
|
|
package dig
|
|
|
|
import (
|
|
"bytes"
|
|
"fmt"
|
|
"math/rand"
|
|
"reflect"
|
|
"sort"
|
|
"time"
|
|
)
|
|
|
|
// A ScopeOption modifies the default behavior of Scope; currently,
|
|
// there are no implementations.
|
|
type ScopeOption interface {
|
|
noScopeOption() //yet
|
|
}
|
|
|
|
// Scope is a scoped DAG of types and their dependencies.
|
|
// A Scope may also have one or more child Scopes that inherit
|
|
// from it.
|
|
type Scope struct {
|
|
// This implements containerStore interface.
|
|
|
|
// Name of the Scope
|
|
name string
|
|
// Mapping from key to all the constructor node that can provide a value for that
|
|
// key.
|
|
providers map[key][]*constructorNode
|
|
|
|
// Mapping from key to the decorator that decorates a value for that key.
|
|
decorators map[key]*decoratorNode
|
|
|
|
// constructorNodes provided directly to this Scope. i.e. it does not include
|
|
// any nodes that were provided to the parent Scope this inherited from.
|
|
nodes []*constructorNode
|
|
|
|
// Values that generated via decorators in the Scope.
|
|
decoratedValues map[key]reflect.Value
|
|
|
|
// Values that generated directly in the Scope.
|
|
values map[key]reflect.Value
|
|
|
|
// Values groups that generated directly in the Scope.
|
|
groups map[key][]reflect.Value
|
|
|
|
// Values groups that generated via decoraters in the Scope.
|
|
decoratedGroups map[key]reflect.Value
|
|
|
|
// Source of randomness.
|
|
rand *rand.Rand
|
|
|
|
// Flag indicating whether the graph has been checked for cycles.
|
|
isVerifiedAcyclic bool
|
|
|
|
// Defer acyclic check on provide until Invoke.
|
|
deferAcyclicVerification bool
|
|
|
|
// Recover from panics in user-provided code and wrap in an exported error type.
|
|
recoverFromPanics bool
|
|
|
|
// invokerFn calls a function with arguments provided to Provide or Invoke.
|
|
invokerFn invokerFn
|
|
|
|
// graph of this Scope. Note that this holds the dependency graph of all the
|
|
// nodes that affect this Scope, not just the ones provided directly to this Scope.
|
|
gh *graphHolder
|
|
|
|
// Parent of this Scope.
|
|
parentScope *Scope
|
|
|
|
// All the child scopes of this Scope.
|
|
childScopes []*Scope
|
|
}
|
|
|
|
func newScope() *Scope {
|
|
s := &Scope{
|
|
providers: make(map[key][]*constructorNode),
|
|
decorators: make(map[key]*decoratorNode),
|
|
values: make(map[key]reflect.Value),
|
|
decoratedValues: make(map[key]reflect.Value),
|
|
groups: make(map[key][]reflect.Value),
|
|
decoratedGroups: make(map[key]reflect.Value),
|
|
invokerFn: defaultInvoker,
|
|
rand: rand.New(rand.NewSource(time.Now().UnixNano())),
|
|
}
|
|
s.gh = newGraphHolder(s)
|
|
return s
|
|
}
|
|
|
|
// Scope creates a new Scope with the given name and options from current Scope.
|
|
// Any constructors that the current Scope knows about, as well as any modifications
|
|
// made to it in the future will be propagated to the child scope.
|
|
// However, no modifications made to the child scope being created will be propagated
|
|
// to the parent Scope.
|
|
func (s *Scope) Scope(name string, opts ...ScopeOption) *Scope {
|
|
child := newScope()
|
|
child.name = name
|
|
child.parentScope = s
|
|
child.invokerFn = s.invokerFn
|
|
child.deferAcyclicVerification = s.deferAcyclicVerification
|
|
child.recoverFromPanics = s.recoverFromPanics
|
|
|
|
// child copies the parent's graph nodes.
|
|
child.gh.nodes = append(child.gh.nodes, s.gh.nodes...)
|
|
|
|
for _, opt := range opts {
|
|
opt.noScopeOption()
|
|
}
|
|
|
|
s.childScopes = append(s.childScopes, child)
|
|
return child
|
|
}
|
|
|
|
// ancestors returns a list of scopes of ancestors of this scope up to the
|
|
// root. The scope at at index 0 is this scope itself.
|
|
func (s *Scope) ancestors() []*Scope {
|
|
var scopes []*Scope
|
|
for s := s; s != nil; s = s.parentScope {
|
|
scopes = append(scopes, s)
|
|
}
|
|
return scopes
|
|
}
|
|
|
|
func (s *Scope) appendSubscopes(dest []*Scope) []*Scope {
|
|
dest = append(dest, s)
|
|
for _, cs := range s.childScopes {
|
|
dest = cs.appendSubscopes(dest)
|
|
}
|
|
return dest
|
|
}
|
|
|
|
func (s *Scope) storesToRoot() []containerStore {
|
|
scopes := s.ancestors()
|
|
stores := make([]containerStore, len(scopes))
|
|
for i, s := range scopes {
|
|
stores[i] = s
|
|
}
|
|
return stores
|
|
}
|
|
|
|
func (s *Scope) knownTypes() []reflect.Type {
|
|
typeSet := make(map[reflect.Type]struct{}, len(s.providers))
|
|
for k := range s.providers {
|
|
typeSet[k.t] = struct{}{}
|
|
}
|
|
|
|
types := make([]reflect.Type, 0, len(typeSet))
|
|
for t := range typeSet {
|
|
types = append(types, t)
|
|
}
|
|
sort.Sort(byTypeName(types))
|
|
return types
|
|
}
|
|
|
|
func (s *Scope) getValue(name string, t reflect.Type) (v reflect.Value, ok bool) {
|
|
v, ok = s.values[key{name: name, t: t}]
|
|
return
|
|
}
|
|
|
|
func (s *Scope) getDecoratedValue(name string, t reflect.Type) (v reflect.Value, ok bool) {
|
|
v, ok = s.decoratedValues[key{name: name, t: t}]
|
|
return
|
|
}
|
|
|
|
func (s *Scope) setValue(name string, t reflect.Type, v reflect.Value) {
|
|
s.values[key{name: name, t: t}] = v
|
|
}
|
|
|
|
func (s *Scope) setDecoratedValue(name string, t reflect.Type, v reflect.Value) {
|
|
s.decoratedValues[key{name: name, t: t}] = v
|
|
}
|
|
|
|
func (s *Scope) getValueGroup(name string, t reflect.Type) []reflect.Value {
|
|
items := s.groups[key{group: name, t: t}]
|
|
// shuffle the list so users don't rely on the ordering of grouped values
|
|
return shuffledCopy(s.rand, items)
|
|
}
|
|
|
|
func (s *Scope) getDecoratedValueGroup(name string, t reflect.Type) (reflect.Value, bool) {
|
|
items, ok := s.decoratedGroups[key{group: name, t: t}]
|
|
return items, ok
|
|
}
|
|
|
|
func (s *Scope) submitGroupedValue(name string, t reflect.Type, v reflect.Value) {
|
|
k := key{group: name, t: t}
|
|
s.groups[k] = append(s.groups[k], v)
|
|
}
|
|
|
|
func (s *Scope) submitDecoratedGroupedValue(name string, t reflect.Type, v reflect.Value) {
|
|
k := key{group: name, t: t}
|
|
s.decoratedGroups[k] = v
|
|
}
|
|
|
|
func (s *Scope) getValueProviders(name string, t reflect.Type) []provider {
|
|
return s.getProviders(key{name: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getGroupProviders(name string, t reflect.Type) []provider {
|
|
return s.getProviders(key{group: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getValueDecorator(name string, t reflect.Type) (decorator, bool) {
|
|
return s.getDecorators(key{name: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getGroupDecorator(name string, t reflect.Type) (decorator, bool) {
|
|
return s.getDecorators(key{group: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getDecorators(k key) (decorator, bool) {
|
|
d, found := s.decorators[k]
|
|
return d, found
|
|
}
|
|
|
|
func (s *Scope) getProviders(k key) []provider {
|
|
nodes := s.providers[k]
|
|
providers := make([]provider, len(nodes))
|
|
for i, n := range nodes {
|
|
providers[i] = n
|
|
}
|
|
return providers
|
|
}
|
|
|
|
func (s *Scope) getAllGroupProviders(name string, t reflect.Type) []provider {
|
|
return s.getAllProviders(key{group: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getAllValueProviders(name string, t reflect.Type) []provider {
|
|
return s.getAllProviders(key{name: name, t: t})
|
|
}
|
|
|
|
func (s *Scope) getAllProviders(k key) []provider {
|
|
allScopes := s.ancestors()
|
|
var providers []provider
|
|
for _, scope := range allScopes {
|
|
providers = append(providers, scope.getProviders(k)...)
|
|
}
|
|
return providers
|
|
}
|
|
|
|
func (s *Scope) invoker() invokerFn {
|
|
return s.invokerFn
|
|
}
|
|
|
|
// adds a new graphNode to this Scope and all of its descendent
|
|
// scope.
|
|
func (s *Scope) newGraphNode(wrapped interface{}, orders map[*Scope]int) {
|
|
orders[s] = s.gh.NewNode(wrapped)
|
|
for _, cs := range s.childScopes {
|
|
cs.newGraphNode(wrapped, orders)
|
|
}
|
|
}
|
|
|
|
func (s *Scope) cycleDetectedError(cycle []int) error {
|
|
var path []cycleErrPathEntry
|
|
for _, n := range cycle {
|
|
if n, ok := s.gh.Lookup(n).(*constructorNode); ok {
|
|
path = append(path, cycleErrPathEntry{
|
|
Key: key{
|
|
t: n.CType(),
|
|
},
|
|
Func: n.Location(),
|
|
})
|
|
}
|
|
}
|
|
return errCycleDetected{Path: path, scope: s}
|
|
}
|
|
|
|
// Returns the root Scope that can be reached from this Scope.
|
|
func (s *Scope) rootScope() *Scope {
|
|
curr := s
|
|
for curr.parentScope != nil {
|
|
curr = curr.parentScope
|
|
}
|
|
return curr
|
|
}
|
|
|
|
// String representation of the entire Scope
|
|
func (s *Scope) String() string {
|
|
b := &bytes.Buffer{}
|
|
fmt.Fprintln(b, "nodes: {")
|
|
for k, vs := range s.providers {
|
|
for _, v := range vs {
|
|
fmt.Fprintln(b, "\t", k, "->", v)
|
|
}
|
|
}
|
|
fmt.Fprintln(b, "}")
|
|
|
|
fmt.Fprintln(b, "values: {")
|
|
for k, v := range s.values {
|
|
fmt.Fprintln(b, "\t", k, "=>", v)
|
|
}
|
|
for k, vs := range s.groups {
|
|
for _, v := range vs {
|
|
fmt.Fprintln(b, "\t", k, "=>", v)
|
|
}
|
|
}
|
|
fmt.Fprintln(b, "}")
|
|
|
|
return b.String()
|
|
}
|