Files
steampipe/pkg/dashboard/dashboardexecute/resolved_runtime_dependency.go

72 lines
2.0 KiB
Go

package dashboardexecute
import (
"fmt"
"log"
"sync"
"github.com/turbot/go-kit/helpers"
"github.com/turbot/steampipe/pkg/dashboard/dashboardtypes"
"github.com/turbot/steampipe/pkg/steampipeconfig/modconfig"
"github.com/turbot/steampipe/pkg/type_conversion"
)
// ResolvedRuntimeDependency is a wrapper for RuntimeDependency which contains the resolved value
// we must wrap it so that we do not mutate the underlying workspace data when resolving dependency values
type ResolvedRuntimeDependency struct {
dependency *modconfig.RuntimeDependency
valueLock sync.Mutex
value any
valueChannel chan *dashboardtypes.ResolvedRuntimeDependencyValue
}
func NewResolvedRuntimeDependency(dep *modconfig.RuntimeDependency, valueChannel chan *dashboardtypes.ResolvedRuntimeDependencyValue) *ResolvedRuntimeDependency {
var wg sync.WaitGroup
wg.Add(1)
return &ResolvedRuntimeDependency{
dependency: dep,
valueChannel: valueChannel,
}
}
func (d *ResolvedRuntimeDependency) Resolve() error {
d.valueLock.Lock()
defer d.valueLock.Unlock()
log.Printf("[TRACE] ResolvedRuntimeDependency Resolve dep %s chan %p", d.dependency.PropertyPath, d.valueChannel)
// if we are already resolved, do nothing
if d.hasValue() {
return nil
}
// dependency must have a source resource - if not this is a bug
if d.dependency.SourceResource == nil {
return fmt.Errorf("runtime dependency '%s' Resolve() called but it does not have a source resource", d.dependency.String())
}
// wait for value
val := <-d.valueChannel
d.value = val.Value
// TACTICAL if the desired value is an array, wrap in an array
if d.dependency.IsArray {
d.value = type_conversion.AnySliceToTypedSlice([]any{d.value})
}
if val.Error != nil {
return val.Error
}
// we should have a non nil value now
if !d.hasValue() {
return fmt.Errorf("nil value recevied for runtime dependency %s", d.dependency.String())
}
return nil
}
func (d *ResolvedRuntimeDependency) hasValue() bool {
return !helpers.IsNil(d.value)
}