mirror of
https://github.com/danielmiessler/fabric
synced 2024-11-10 07:10:31 +00:00
107 lines
2.3 KiB
Go
107 lines
2.3 KiB
Go
package core
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"github.com/danielmiessler/fabric/vendors"
|
|
"sync"
|
|
)
|
|
|
|
func NewVendorsManager() *VendorsManager {
|
|
return &VendorsManager{
|
|
Vendors: map[string]vendors.Vendor{},
|
|
}
|
|
}
|
|
|
|
type VendorsManager struct {
|
|
Vendors map[string]vendors.Vendor
|
|
Models *VendorsModels
|
|
}
|
|
|
|
func (o *VendorsManager) AddVendors(vendors ...vendors.Vendor) {
|
|
for _, vendor := range vendors {
|
|
o.Vendors[vendor.GetName()] = vendor
|
|
}
|
|
}
|
|
|
|
func (o *VendorsManager) GetModels() *VendorsModels {
|
|
if o.Models == nil {
|
|
o.readModels()
|
|
}
|
|
return o.Models
|
|
}
|
|
|
|
func (o *VendorsManager) HasVendors() bool {
|
|
return len(o.Vendors) > 0
|
|
}
|
|
|
|
func (o *VendorsManager) FindByName(name string) vendors.Vendor {
|
|
return o.Vendors[name]
|
|
}
|
|
|
|
func (o *VendorsManager) readModels() {
|
|
o.Models = NewVendorsModels()
|
|
|
|
var wg sync.WaitGroup
|
|
resultsChan := make(chan modelResult, len(o.Vendors))
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
defer cancel()
|
|
|
|
for _, vendor := range o.Vendors {
|
|
wg.Add(1)
|
|
go o.fetchVendorModels(ctx, &wg, vendor, resultsChan)
|
|
}
|
|
|
|
// Wait for all goroutines to finish
|
|
go func() {
|
|
wg.Wait()
|
|
close(resultsChan)
|
|
}()
|
|
|
|
// Collect results
|
|
for result := range resultsChan {
|
|
if result.err != nil {
|
|
fmt.Println(result.vendorName, result.err)
|
|
o.Models.AddError(result.err)
|
|
cancel() // Cancel remaining goroutines if needed
|
|
} else {
|
|
o.Models.AddVendorModels(result.vendorName, result.models)
|
|
}
|
|
}
|
|
}
|
|
|
|
func (o *VendorsManager) fetchVendorModels(
|
|
ctx context.Context, wg *sync.WaitGroup, vendor vendors.Vendor, resultsChan chan<- modelResult) {
|
|
|
|
defer wg.Done()
|
|
|
|
models, err := vendor.ListModels()
|
|
select {
|
|
case <-ctx.Done():
|
|
// Context canceled, don't send the result
|
|
return
|
|
case resultsChan <- modelResult{vendorName: vendor.GetName(), models: models, err: err}:
|
|
// Result sent
|
|
}
|
|
}
|
|
|
|
func (o *VendorsManager) Setup() (ret map[string]vendors.Vendor, err error) {
|
|
ret = map[string]vendors.Vendor{}
|
|
for _, vendor := range o.Vendors {
|
|
fmt.Println()
|
|
if vendorErr := vendor.Setup(); vendorErr == nil {
|
|
fmt.Printf("[%v] configured\n", vendor.GetName())
|
|
ret[vendor.GetName()] = vendor
|
|
} else {
|
|
fmt.Printf("[%v] skipped\n", vendor.GetName())
|
|
}
|
|
}
|
|
return
|
|
}
|
|
|
|
type modelResult struct {
|
|
vendorName string
|
|
models []string
|
|
err error
|
|
}
|