mirror of
https://github.com/Oxalide/vsphere-influxdb-go.git
synced 2023-10-10 13:36:51 +02:00
general cleanup; extra checks to avoid segfaults; channels and goroutines for concurrency
This commit is contained in:
parent
65b5bd801f
commit
01483a5b97
@ -68,6 +68,7 @@ type VCenter struct {
|
|||||||
Username string
|
Username string
|
||||||
Password string
|
Password string
|
||||||
MetricGroups []*MetricGroup
|
MetricGroups []*MetricGroup
|
||||||
|
client *govmomi.Client
|
||||||
}
|
}
|
||||||
|
|
||||||
// MetricDef metric definition
|
// MetricDef metric definition
|
||||||
@ -101,8 +102,7 @@ var debug bool
|
|||||||
var stdlog, errlog *log.Logger
|
var stdlog, errlog *log.Logger
|
||||||
|
|
||||||
// Connect to the actual vCenter connection used to query data
|
// Connect to the actual vCenter connection used to query data
|
||||||
func (vcenter *VCenter) Connect() (*govmomi.Client, error) {
|
func (vcenter *VCenter) Connect() error {
|
||||||
// Prepare vCenter Connections
|
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
@ -111,38 +111,49 @@ func (vcenter *VCenter) Connect() (*govmomi.Client, error) {
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
errlog.Println("Could not parse vcenter url: ", vcenter.Hostname)
|
errlog.Println("Could not parse vcenter url: ", vcenter.Hostname)
|
||||||
errlog.Println("Error: ", err)
|
errlog.Println("Error: ", err)
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
client, err := govmomi.NewClient(ctx, u, true)
|
client, err := govmomi.NewClient(ctx, u, true)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errlog.Println("Could not connect to vcenter: ", vcenter.Hostname)
|
errlog.Println("Could not connect to vcenter: ", vcenter.Hostname)
|
||||||
errlog.Println("Error: ", err)
|
errlog.Println("Error: ", err)
|
||||||
return nil, err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
return client, nil
|
vcenter.client = client
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Init the VCenter connection
|
// Disconnect from the vCenter
|
||||||
func (vcenter *VCenter) Init(config Configuration) {
|
func (vcenter *VCenter) Disconnect() error {
|
||||||
ctx, cancel := context.WithCancel(context.Background())
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
client, err := vcenter.Connect()
|
if vcenter.client != nil {
|
||||||
if err != nil {
|
if err := vcenter.client.Logout(ctx); err != nil {
|
||||||
errlog.Println("Could not connect to vcenter: ", vcenter.Hostname)
|
errlog.Println("Could not disconnect properly from vcenter", vcenter.Hostname, err)
|
||||||
errlog.Println("Error: ", err)
|
return err
|
||||||
return
|
|
||||||
}
|
}
|
||||||
defer client.Logout(ctx)
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// Init the VCenter connection
|
||||||
|
func (vcenter *VCenter) Init(config Configuration) error {
|
||||||
|
ctx, cancel := context.WithCancel(context.Background())
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
client := vcenter.client
|
||||||
|
|
||||||
var perfmanager mo.PerformanceManager
|
var perfmanager mo.PerformanceManager
|
||||||
err = client.RetrieveOne(ctx, *client.ServiceContent.PerfManager, nil, &perfmanager)
|
err := client.RetrieveOne(ctx, *client.ServiceContent.PerfManager, nil, &perfmanager)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errlog.Println("Could not get performance manager")
|
errlog.Println("Could not get performance manager")
|
||||||
errlog.Println("Error: ", err)
|
errlog.Println("Error: ", err)
|
||||||
return
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, perf := range perfmanager.PerfCounter {
|
for _, perf := range perfmanager.PerfCounter {
|
||||||
@ -171,10 +182,11 @@ func (vcenter *VCenter) Init(config Configuration) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
// Query a vcenter
|
// Query a vcenter
|
||||||
func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.Client) {
|
func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.Client, nowTime time.Time) {
|
||||||
stdlog.Println("Setting up query inventory of vcenter: ", vcenter.Hostname)
|
stdlog.Println("Setting up query inventory of vcenter: ", vcenter.Hostname)
|
||||||
|
|
||||||
// Create the contect
|
// Create the contect
|
||||||
@ -182,17 +194,11 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
defer cancel()
|
defer cancel()
|
||||||
|
|
||||||
// Get the client
|
// Get the client
|
||||||
client, err := vcenter.Connect()
|
client := vcenter.client
|
||||||
if err != nil {
|
|
||||||
errlog.Println("Could not connect to vcenter: ", vcenter.Hostname)
|
|
||||||
errlog.Println("Error: ", err)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer client.Logout(ctx)
|
|
||||||
|
|
||||||
// Create the view manager
|
// Create the view manager
|
||||||
var viewManager mo.ViewManager
|
var viewManager mo.ViewManager
|
||||||
err = client.RetrieveOne(ctx, *client.ServiceContent.ViewManager, nil, &viewManager)
|
err := client.RetrieveOne(ctx, *client.ServiceContent.ViewManager, nil, &viewManager)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
errlog.Println("Could not get view manager from vcenter: " + vcenter.Hostname)
|
errlog.Println("Could not get view manager from vcenter: " + vcenter.Hostname)
|
||||||
errlog.Println("Error: ", err)
|
errlog.Println("Error: ", err)
|
||||||
@ -255,7 +261,7 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
|
|
||||||
newMors := []types.ManagedObjectReference{}
|
newMors := []types.ManagedObjectReference{}
|
||||||
|
|
||||||
if debug == true {
|
if debug {
|
||||||
spew.Dump(mors)
|
spew.Dump(mors)
|
||||||
}
|
}
|
||||||
// Assign each MORS type to a specific array
|
// Assign each MORS type to a specific array
|
||||||
@ -279,21 +285,27 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
mors = newMors
|
mors = newMors
|
||||||
pc := property.DefaultCollector(client.Client)
|
pc := property.DefaultCollector(client.Client)
|
||||||
|
|
||||||
|
// govmomi segfaults when the list objects to retrieve is empty, so check everything
|
||||||
|
|
||||||
// Retrieve properties for all vms
|
// Retrieve properties for all vms
|
||||||
var vmmo []mo.VirtualMachine
|
var vmmo []mo.VirtualMachine
|
||||||
|
if len(vmRefs) > 0 {
|
||||||
err = pc.Retrieve(ctx, vmRefs, []string{"summary"}, &vmmo)
|
err = pc.Retrieve(ctx, vmRefs, []string{"summary"}, &vmmo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Retrieve properties for hosts
|
// Retrieve properties for hosts
|
||||||
var hsmo []mo.HostSystem
|
var hsmo []mo.HostSystem
|
||||||
|
if len(hostRefs) > 0 {
|
||||||
err = pc.Retrieve(ctx, hostRefs, []string{"parent", "summary"}, &hsmo)
|
err = pc.Retrieve(ctx, hostRefs, []string{"parent", "summary"}, &hsmo)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
//Retrieve properties for Cluster(s)
|
//Retrieve properties for Cluster(s)
|
||||||
var clmo []mo.ClusterComputeResource
|
var clmo []mo.ClusterComputeResource
|
||||||
@ -317,11 +329,13 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
|
|
||||||
// Retrieve summary property for all datastores
|
// Retrieve summary property for all datastores
|
||||||
var dss []mo.Datastore
|
var dss []mo.Datastore
|
||||||
|
if len(datastoreRefs) > 0 {
|
||||||
err = pc.Retrieve(ctx, datastoreRefs, []string{"summary"}, &dss)
|
err = pc.Retrieve(ctx, datastoreRefs, []string{"summary"}, &dss)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatal(err)
|
log.Fatal(err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// Initialize the map that will hold the VM MOR to ResourcePool reference
|
// Initialize the map that will hold the VM MOR to ResourcePool reference
|
||||||
vmToPool := make(map[types.ManagedObjectReference]string)
|
vmToPool := make(map[types.ManagedObjectReference]string)
|
||||||
@ -329,7 +343,7 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
var respool []mo.ResourcePool
|
var respool []mo.ResourcePool
|
||||||
// Retrieve properties for ResourcePools
|
// Retrieve properties for ResourcePools
|
||||||
if len(respoolRefs) > 0 {
|
if len(respoolRefs) > 0 {
|
||||||
if debug == true {
|
if debug {
|
||||||
stdlog.Println("going inside ResourcePools")
|
stdlog.Println("going inside ResourcePools")
|
||||||
}
|
}
|
||||||
err = pc.Retrieve(ctx, respoolRefs, []string{"name", "config", "vm"}, &respool)
|
err = pc.Retrieve(ctx, respoolRefs, []string{"name", "config", "vm"}, &respool)
|
||||||
@ -340,12 +354,12 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
for _, pool := range respool {
|
for _, pool := range respool {
|
||||||
stdlog.Println(pool.Config.MemoryAllocation.GetResourceAllocationInfo().Limit)
|
stdlog.Println(pool.Config.MemoryAllocation.GetResourceAllocationInfo().Limit)
|
||||||
stdlog.Println(pool.Config.CpuAllocation.GetResourceAllocationInfo().Limit)
|
stdlog.Println(pool.Config.CpuAllocation.GetResourceAllocationInfo().Limit)
|
||||||
if debug == true {
|
if debug {
|
||||||
stdlog.Println("---resourcepool name - you should see every resourcepool here (+VMs inside)----")
|
stdlog.Println("---resourcepool name - you should see every resourcepool here (+VMs inside)----")
|
||||||
stdlog.Println(pool.Name)
|
stdlog.Println(pool.Name)
|
||||||
}
|
}
|
||||||
for _, vm := range pool.Vm {
|
for _, vm := range pool.Vm {
|
||||||
if debug == true {
|
if debug {
|
||||||
stdlog.Println("--VM ID - you should see every VM ID here--")
|
stdlog.Println("--VM ID - you should see every VM ID here--")
|
||||||
stdlog.Println(vm)
|
stdlog.Println(vm)
|
||||||
}
|
}
|
||||||
@ -357,12 +371,12 @@ func (vcenter *VCenter) Query(config Configuration, InfluxDBClient influxclient.
|
|||||||
// Initialize the map that will hold the VM MOR to cluster reference
|
// Initialize the map that will hold the VM MOR to cluster reference
|
||||||
vmToCluster := make(map[types.ManagedObjectReference]string)
|
vmToCluster := make(map[types.ManagedObjectReference]string)
|
||||||
|
|
||||||
// Initialize the map that will hold the VM MOR to cluster reference
|
// Initialize the map that will hold the host MOR to cluster reference
|
||||||
hostToCluster := make(map[types.ManagedObjectReference]string)
|
hostToCluster := make(map[types.ManagedObjectReference]string)
|
||||||
|
|
||||||
// Retrieve properties for clusters, if any
|
// Retrieve properties for clusters, if any
|
||||||
if len(clusterRefs) > 0 {
|
if len(clusterRefs) > 0 {
|
||||||
if debug == true {
|
if debug {
|
||||||
stdlog.Println("going inside clusters")
|
stdlog.Println("going inside clusters")
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -746,15 +760,38 @@ func queryVCenter(vcenter VCenter, config Configuration, InfluxDBClient influxcl
|
|||||||
vcenter.Query(config, InfluxDBClient)
|
vcenter.Query(config, InfluxDBClient)
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func worker(id int, config Configuration, influxDBClient influxclient.Client, nowTime time.Time, vcenters <-chan *VCenter, results chan<- bool) {
|
||||||
flag.BoolVar(&debug, "debug", false, "Debug mode")
|
for vcenter := range vcenters {
|
||||||
var cfgFile = flag.String("config", "/etc/"+path.Base(os.Args[0])+".json", "Config file to use. Default is /etc/"+path.Base(os.Args[0])+".json")
|
if debug {
|
||||||
flag.Parse()
|
stdlog.Println("Worker", id, "received vcenter", vcenter.Hostname)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := vcenter.Connect(); err != nil {
|
||||||
|
errlog.Println("Could not initialize connection to vcenter", vcenter.Hostname, err)
|
||||||
|
results <- true
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := vcenter.Init(config); err == nil {
|
||||||
|
vcenter.Query(config, influxDBClient, nowTime)
|
||||||
|
}
|
||||||
|
|
||||||
|
vcenter.Disconnect()
|
||||||
|
results <- true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func main() {
|
||||||
|
baseName := path.Base(os.Args[0])
|
||||||
stdlog = log.New(os.Stdout, "", log.Ldate|log.Ltime)
|
stdlog = log.New(os.Stdout, "", log.Ldate|log.Ltime)
|
||||||
errlog = log.New(os.Stderr, "", log.Ldate|log.Ltime)
|
errlog = log.New(os.Stderr, "", log.Ldate|log.Ltime)
|
||||||
|
|
||||||
stdlog.Println("Starting :", path.Base(os.Args[0]))
|
flag.BoolVar(&debug, "debug", false, "Debug mode")
|
||||||
|
workerCount := flag.Int("workers", 4, "Number of concurrent workers to query vcenters")
|
||||||
|
cfgFile := flag.String("config", "/etc/"+baseName+".json", "Config file to use. Default is /etc/"+baseName+".json")
|
||||||
|
flag.Parse()
|
||||||
|
|
||||||
|
stdlog.Println("Starting", baseName, "with config file", *cfgFile)
|
||||||
|
|
||||||
// read the configuration
|
// read the configuration
|
||||||
file, err := os.Open(*cfgFile)
|
file, err := os.Open(*cfgFile)
|
||||||
@ -791,28 +828,48 @@ func main() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// Print configuration in debug mode
|
// Print configuration in debug mode
|
||||||
if debug == true {
|
if debug {
|
||||||
stdlog.Println("---Configuration - you should see the config here---")
|
stdlog.Println("---Configuration - you should see the config here---")
|
||||||
spew.Dump(config)
|
spew.Dump(config)
|
||||||
}
|
}
|
||||||
|
|
||||||
for _, vcenter := range config.VCenters {
|
// Initialize InfluxDB and connect to database
|
||||||
vcenter.Init(config)
|
|
||||||
}
|
|
||||||
|
|
||||||
InfluxDBClient, err := influxclient.NewHTTPClient(influxclient.HTTPConfig{
|
InfluxDBClient, err := influxclient.NewHTTPClient(influxclient.HTTPConfig{
|
||||||
Addr: config.InfluxDB.Hostname,
|
Addr: config.InfluxDB.Hostname,
|
||||||
Username: config.InfluxDB.Username,
|
Username: config.InfluxDB.Username,
|
||||||
Password: config.InfluxDB.Password,
|
Password: config.InfluxDB.Password,
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
errlog.Println("Could not initialize InfluxDB client")
|
||||||
|
errlog.Fatalln(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, _, err := InfluxDBClient.Ping(0); err != nil {
|
||||||
errlog.Println("Could not connect to InfluxDB")
|
errlog.Println("Could not connect to InfluxDB")
|
||||||
errlog.Fatalln(err)
|
errlog.Fatalln(err)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
defer InfluxDBClient.Close()
|
||||||
|
|
||||||
stdlog.Println("Successfully connected to Influx")
|
stdlog.Println("Successfully connected to Influx")
|
||||||
|
|
||||||
|
// make the channels, get the time, launch the goroutines
|
||||||
|
vcenterCount := len(config.VCenters)
|
||||||
|
vcenters := make(chan *VCenter, vcenterCount)
|
||||||
|
results := make(chan bool, vcenterCount)
|
||||||
|
nowTime := time.Now()
|
||||||
|
|
||||||
|
for i := 0; i < *workerCount; i++ {
|
||||||
|
go worker(i, config, InfluxDBClient, nowTime, vcenters, results)
|
||||||
|
}
|
||||||
|
|
||||||
for _, vcenter := range config.VCenters {
|
for _, vcenter := range config.VCenters {
|
||||||
queryVCenter(*vcenter, config, InfluxDBClient)
|
vcenters <- vcenter
|
||||||
}
|
}
|
||||||
|
close(vcenters)
|
||||||
|
|
||||||
|
for i := 0; i < vcenterCount; i++ {
|
||||||
|
<-results
|
||||||
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user