mirror of
https://github.com/prometheus/prometheus
synced 2024-12-25 16:02:28 +00:00
8fdfa8abea
i) Uses the more idiomatic Wrap and Wrapf methods for creating nested errors. ii) Fixes some incorrect usages of fmt.Errorf where the error messages don't have any formatting directives. iii) Does away with the use of fmt package for errors in favour of pkg/errors Signed-off-by: tariqibrahim <tariq181290@gmail.com>
172 lines
5.8 KiB
Go
172 lines
5.8 KiB
Go
// Copyright 2017 The Prometheus Authors
|
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
|
// you may not use this file except in compliance with the License.
|
|
// You may obtain a copy of the License at
|
|
//
|
|
// http://www.apache.org/licenses/LICENSE-2.0
|
|
//
|
|
// Unless required by applicable law or agreed to in writing, software
|
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
// See the License for the specific language governing permissions and
|
|
// limitations under the License.
|
|
|
|
package openstack
|
|
|
|
import (
|
|
"context"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/go-kit/kit/log"
|
|
"github.com/gophercloud/gophercloud"
|
|
"github.com/gophercloud/gophercloud/openstack"
|
|
conntrack "github.com/mwitkow/go-conntrack"
|
|
"github.com/pkg/errors"
|
|
config_util "github.com/prometheus/common/config"
|
|
"github.com/prometheus/common/model"
|
|
|
|
"github.com/prometheus/prometheus/discovery/refresh"
|
|
"github.com/prometheus/prometheus/discovery/targetgroup"
|
|
)
|
|
|
|
// DefaultSDConfig is the default OpenStack SD configuration.
|
|
var DefaultSDConfig = SDConfig{
|
|
Port: 80,
|
|
RefreshInterval: model.Duration(60 * time.Second),
|
|
}
|
|
|
|
// SDConfig is the configuration for OpenStack based service discovery.
|
|
type SDConfig struct {
|
|
IdentityEndpoint string `yaml:"identity_endpoint"`
|
|
Username string `yaml:"username"`
|
|
UserID string `yaml:"userid"`
|
|
Password config_util.Secret `yaml:"password"`
|
|
ProjectName string `yaml:"project_name"`
|
|
ProjectID string `yaml:"project_id"`
|
|
DomainName string `yaml:"domain_name"`
|
|
DomainID string `yaml:"domain_id"`
|
|
ApplicationCredentialName string `yaml:"application_credential_name"`
|
|
ApplicationCredentialID string `yaml:"application_credential_id"`
|
|
ApplicationCredentialSecret config_util.Secret `yaml:"application_credential_secret"`
|
|
Role Role `yaml:"role"`
|
|
Region string `yaml:"region"`
|
|
RefreshInterval model.Duration `yaml:"refresh_interval,omitempty"`
|
|
Port int `yaml:"port"`
|
|
AllTenants bool `yaml:"all_tenants,omitempty"`
|
|
TLSConfig config_util.TLSConfig `yaml:"tls_config,omitempty"`
|
|
}
|
|
|
|
// Role is the role of the target in OpenStack.
|
|
type Role string
|
|
|
|
// The valid options for OpenStackRole.
|
|
const (
|
|
// OpenStack document reference
|
|
// https://docs.openstack.org/nova/pike/admin/arch.html#hypervisors
|
|
OpenStackRoleHypervisor Role = "hypervisor"
|
|
// OpenStack document reference
|
|
// https://docs.openstack.org/horizon/pike/user/launch-instances.html
|
|
OpenStackRoleInstance Role = "instance"
|
|
)
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
func (c *Role) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
if err := unmarshal((*string)(c)); err != nil {
|
|
return err
|
|
}
|
|
switch *c {
|
|
case OpenStackRoleHypervisor, OpenStackRoleInstance:
|
|
return nil
|
|
default:
|
|
return errors.Errorf("unknown OpenStack SD role %q", *c)
|
|
}
|
|
}
|
|
|
|
// UnmarshalYAML implements the yaml.Unmarshaler interface.
|
|
func (c *SDConfig) UnmarshalYAML(unmarshal func(interface{}) error) error {
|
|
*c = DefaultSDConfig
|
|
type plain SDConfig
|
|
err := unmarshal((*plain)(c))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if c.Role == "" {
|
|
return errors.New("role missing (one of: instance, hypervisor)")
|
|
}
|
|
if c.Region == "" {
|
|
return errors.New("openstack SD configuration requires a region")
|
|
}
|
|
return nil
|
|
}
|
|
|
|
type refresher interface {
|
|
refresh(context.Context) ([]*targetgroup.Group, error)
|
|
}
|
|
|
|
// NewDiscovery returns a new OpenStack Discoverer which periodically refreshes its targets.
|
|
func NewDiscovery(conf *SDConfig, l log.Logger) (*refresh.Discovery, error) {
|
|
r, err := newRefresher(conf, l)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
return refresh.NewDiscovery(
|
|
l,
|
|
"openstack",
|
|
time.Duration(conf.RefreshInterval),
|
|
r.refresh,
|
|
), nil
|
|
|
|
}
|
|
|
|
func newRefresher(conf *SDConfig, l log.Logger) (refresher, error) {
|
|
var opts gophercloud.AuthOptions
|
|
if conf.IdentityEndpoint == "" {
|
|
var err error
|
|
opts, err = openstack.AuthOptionsFromEnv()
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
} else {
|
|
opts = gophercloud.AuthOptions{
|
|
IdentityEndpoint: conf.IdentityEndpoint,
|
|
Username: conf.Username,
|
|
UserID: conf.UserID,
|
|
Password: string(conf.Password),
|
|
TenantName: conf.ProjectName,
|
|
TenantID: conf.ProjectID,
|
|
DomainName: conf.DomainName,
|
|
DomainID: conf.DomainID,
|
|
ApplicationCredentialID: conf.ApplicationCredentialID,
|
|
ApplicationCredentialName: conf.ApplicationCredentialName,
|
|
ApplicationCredentialSecret: string(conf.ApplicationCredentialSecret),
|
|
}
|
|
}
|
|
client, err := openstack.NewClient(opts.IdentityEndpoint)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
tls, err := config_util.NewTLSConfig(&conf.TLSConfig)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
client.HTTPClient = http.Client{
|
|
Transport: &http.Transport{
|
|
IdleConnTimeout: 5 * time.Duration(conf.RefreshInterval),
|
|
TLSClientConfig: tls,
|
|
DialContext: conntrack.NewDialContextFunc(
|
|
conntrack.DialWithTracing(),
|
|
conntrack.DialWithName("openstack_sd"),
|
|
),
|
|
},
|
|
Timeout: 5 * time.Duration(conf.RefreshInterval),
|
|
}
|
|
switch conf.Role {
|
|
case OpenStackRoleHypervisor:
|
|
return newHypervisorDiscovery(client, &opts, conf.Port, conf.Region, l), nil
|
|
case OpenStackRoleInstance:
|
|
return newInstanceDiscovery(client, &opts, conf.Port, conf.Region, conf.AllTenants, l), nil
|
|
}
|
|
return nil, errors.New("unknown OpenStack discovery role")
|
|
}
|