2017-11-20 07:28:49 +01:00
|
|
|
package operations
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
2017-11-20 22:31:20 +01:00
|
|
|
"sort"
|
2017-11-20 07:28:49 +01:00
|
|
|
|
|
|
|
"github.com/pulumi/pulumi/pkg/resource"
|
|
|
|
"github.com/pulumi/pulumi/pkg/tokens"
|
|
|
|
"github.com/pulumi/pulumi/pkg/util/contract"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Resource is a tree representation of a resource/component hierarchy
|
|
|
|
type Resource struct {
|
|
|
|
ns tokens.QName
|
|
|
|
alloc tokens.PackageName
|
|
|
|
state *resource.State
|
|
|
|
parent *Resource
|
|
|
|
children map[resource.URN]*Resource
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewResource constructs a tree representation of a resource/component hierarchy
|
|
|
|
func NewResource(source []*resource.State) *Resource {
|
|
|
|
treeNodes := map[resource.URN]*Resource{}
|
|
|
|
var ns tokens.QName
|
|
|
|
var alloc tokens.PackageName
|
|
|
|
|
|
|
|
// Walk the ordered resource list and build tree nodes based on child relationships
|
|
|
|
for _, state := range source {
|
|
|
|
ns = state.URN.Namespace()
|
|
|
|
alloc = state.URN.Alloc()
|
|
|
|
newTree := &Resource{
|
|
|
|
ns: ns,
|
|
|
|
alloc: alloc,
|
|
|
|
state: state,
|
|
|
|
parent: nil,
|
|
|
|
children: map[resource.URN]*Resource{},
|
|
|
|
}
|
|
|
|
for _, childURN := range state.Children {
|
|
|
|
childTree, ok := treeNodes[childURN]
|
|
|
|
contract.Assertf(ok, "Expected children to be before parents in resource checkpoint")
|
|
|
|
childTree.parent = newTree
|
|
|
|
newTree.children[childTree.state.URN] = childTree
|
|
|
|
}
|
|
|
|
treeNodes[state.URN] = newTree
|
|
|
|
}
|
|
|
|
|
|
|
|
// Create a single root node which is the parent of all unparented nodes
|
|
|
|
root := &Resource{
|
|
|
|
ns: ns,
|
|
|
|
alloc: alloc,
|
|
|
|
state: nil,
|
|
|
|
parent: nil,
|
|
|
|
children: map[resource.URN]*Resource{},
|
|
|
|
}
|
|
|
|
for _, node := range treeNodes {
|
|
|
|
if node.parent == nil {
|
|
|
|
root.children[node.state.URN] = node
|
|
|
|
node.parent = root
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Return the root node
|
|
|
|
return root
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetChild find a child with the given type and name or returns `nil`.
|
|
|
|
func (r *Resource) GetChild(typ string, name string) *Resource {
|
|
|
|
childURN := resource.NewURN(r.ns, r.alloc, tokens.Type(typ), tokens.QName(name))
|
|
|
|
return r.children[childURN]
|
|
|
|
}
|
|
|
|
|
|
|
|
// OperationsProvider gets an OperationsProvider for this resource.
|
|
|
|
func (r *Resource) OperationsProvider(config map[tokens.ModuleMember]string) Provider {
|
|
|
|
return &resourceOperations{
|
|
|
|
resource: r,
|
|
|
|
config: config,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResourceOperations is an OperationsProvider for Resources
|
|
|
|
type resourceOperations struct {
|
|
|
|
resource *Resource
|
|
|
|
config map[tokens.ModuleMember]string
|
|
|
|
}
|
|
|
|
|
|
|
|
var _ Provider = (*resourceOperations)(nil)
|
|
|
|
|
|
|
|
// GetLogs gets logs for a Resource
|
|
|
|
func (ops *resourceOperations) GetLogs(query LogQuery) (*[]LogEntry, error) {
|
2017-11-23 05:58:46 +01:00
|
|
|
// Only get logs for this resource if it matches the resource filter query
|
2017-11-23 06:33:36 +01:00
|
|
|
if ops.matchesResourceFilter(query.ResourceFilter) {
|
|
|
|
// Set query to be a new query with `ResourceFilter` nil so that we don't filter out logs from any children of
|
|
|
|
// this resource since this resource did match the resource filter.
|
2017-11-23 06:09:13 +01:00
|
|
|
query = LogQuery{
|
2017-11-23 06:33:36 +01:00
|
|
|
StartTime: query.StartTime,
|
|
|
|
EndTime: query.EndTime,
|
|
|
|
Query: query.Query,
|
|
|
|
ResourceFilter: nil,
|
2017-11-23 06:09:13 +01:00
|
|
|
}
|
2017-11-23 05:58:46 +01:00
|
|
|
// Try to get an operations provider for this resource, it may be `nil`
|
|
|
|
opsProvider, err := ops.getOperationsProvider()
|
2017-11-20 07:28:49 +01:00
|
|
|
if err != nil {
|
2017-11-23 05:58:46 +01:00
|
|
|
return nil, err
|
2017-11-20 07:28:49 +01:00
|
|
|
}
|
2017-11-23 05:58:46 +01:00
|
|
|
if opsProvider != nil {
|
|
|
|
// If this resource has an operations provider - use it and don't recur into children. It is the
|
|
|
|
// responsibility of it's GetLogs implementation to aggregate all logs from children, either by passing them
|
|
|
|
// through or by filtering specific content out.
|
|
|
|
logsResult, err := opsProvider.GetLogs(query)
|
|
|
|
if err != nil {
|
|
|
|
return logsResult, err
|
|
|
|
}
|
|
|
|
if logsResult != nil {
|
|
|
|
return logsResult, nil
|
|
|
|
}
|
2017-11-20 07:28:49 +01:00
|
|
|
}
|
|
|
|
}
|
2017-11-23 05:58:46 +01:00
|
|
|
// If this resource did not choose to provide it's own logs, recur into children and collect + aggregate their logs.
|
2017-11-20 07:28:49 +01:00
|
|
|
var logs []LogEntry
|
2017-11-23 06:33:36 +01:00
|
|
|
// Kick off GetLogs on all children in parallel, writing results to shared channels
|
|
|
|
ch := make(chan *[]LogEntry)
|
|
|
|
errch := make(chan error)
|
2017-11-20 07:28:49 +01:00
|
|
|
for _, child := range ops.resource.children {
|
|
|
|
childOps := &resourceOperations{
|
|
|
|
resource: child,
|
|
|
|
config: ops.config,
|
|
|
|
}
|
2017-11-23 06:33:36 +01:00
|
|
|
go func() {
|
|
|
|
childLogs, err := childOps.GetLogs(query)
|
|
|
|
ch <- childLogs
|
|
|
|
errch <- err
|
|
|
|
}()
|
|
|
|
}
|
|
|
|
// Handle results from GetLogs calls as they complete
|
|
|
|
for range ops.resource.children {
|
|
|
|
childLogs := <-ch
|
|
|
|
err := <-errch
|
2017-11-20 07:28:49 +01:00
|
|
|
if err != nil {
|
|
|
|
return &logs, err
|
|
|
|
}
|
|
|
|
if childLogs != nil {
|
|
|
|
logs = append(logs, *childLogs...)
|
|
|
|
}
|
|
|
|
}
|
2017-11-20 22:31:20 +01:00
|
|
|
// Sort
|
|
|
|
sort.SliceStable(logs, func(i, j int) bool { return logs[i].Timestamp < logs[j].Timestamp })
|
|
|
|
// Remove duplicates
|
|
|
|
var retLogs []LogEntry
|
2017-11-20 23:55:09 +01:00
|
|
|
var lastLogTimestamp int64
|
|
|
|
var lastLogs []LogEntry
|
2017-11-20 22:31:20 +01:00
|
|
|
for _, log := range logs {
|
2017-11-20 23:55:09 +01:00
|
|
|
shouldContinue := false
|
|
|
|
if log.Timestamp == lastLogTimestamp {
|
|
|
|
for _, lastLog := range lastLogs {
|
|
|
|
if log.Message == lastLog.Message {
|
|
|
|
shouldContinue = true
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
lastLogs = nil
|
|
|
|
}
|
|
|
|
if shouldContinue {
|
2017-11-20 22:31:20 +01:00
|
|
|
continue
|
|
|
|
}
|
2017-11-20 23:55:09 +01:00
|
|
|
lastLogs = append(lastLogs, log)
|
|
|
|
lastLogTimestamp = log.Timestamp
|
2017-11-20 22:31:20 +01:00
|
|
|
retLogs = append(retLogs, log)
|
|
|
|
}
|
|
|
|
return &retLogs, nil
|
2017-11-20 07:28:49 +01:00
|
|
|
}
|
|
|
|
|
2017-11-23 05:58:46 +01:00
|
|
|
// matchesResourceFilter determines whether this resource matches the provided resource filter.
|
|
|
|
func (ops *resourceOperations) matchesResourceFilter(filter *ResourceFilter) bool {
|
|
|
|
if filter == nil {
|
|
|
|
// No filter, all resources match it.
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if ops.resource == nil || ops.resource.state == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
urn := ops.resource.state.URN
|
|
|
|
if resource.URN(*filter) == urn {
|
|
|
|
// The filter matched the full URN
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if string(*filter) == string(urn.Type())+"::"+string(urn.Name()) {
|
|
|
|
// The filter matched the '<type>::<name>' part of the URN
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
if tokens.QName(*filter) == urn.Name() {
|
|
|
|
// The filter matched the '<name>' part of the URN
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
|
2017-11-20 07:28:49 +01:00
|
|
|
// ListMetrics lists metrics for a Resource
|
|
|
|
func (ops *resourceOperations) ListMetrics() []MetricName {
|
|
|
|
return []MetricName{}
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMetricStatistics gets metric statistics for a Resource
|
|
|
|
func (ops *resourceOperations) GetMetricStatistics(metric MetricRequest) ([]MetricDataPoint, error) {
|
|
|
|
return nil, fmt.Errorf("not yet implemented")
|
|
|
|
}
|
2017-11-20 22:31:20 +01:00
|
|
|
|
|
|
|
func (ops *resourceOperations) getOperationsProvider() (Provider, error) {
|
|
|
|
if ops.resource == nil || ops.resource.state == nil {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
switch ops.resource.state.Type.Package() {
|
|
|
|
case "cloud":
|
|
|
|
return CloudOperationsProvider(ops.config, ops.resource)
|
|
|
|
case "aws":
|
|
|
|
return AWSOperationsProvider(ops.config, ops.resource)
|
|
|
|
default:
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
}
|