Skip to content

Commit

Permalink
Merge pull request #6 from promhippie/multi-credentials
Browse files Browse the repository at this point in the history
Add support for multiple credentials
  • Loading branch information
tboerger authored Mar 26, 2019
2 parents 3f59d75 + 55e29eb commit 746668c
Show file tree
Hide file tree
Showing 13 changed files with 304 additions and 120 deletions.
1 change: 1 addition & 0 deletions .gitignore
Original file line number Diff line number Diff line change
Expand Up @@ -32,3 +32,4 @@ coverage.out

.envrc
scw.json
config.json
8 changes: 8 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,15 @@
### Changed

* Switch to cloud.drone.io for CI

### Added

* Add support for server tags
* Support multiple credentials and config file

### Removed

* Region is not a requirement anymore

## [0.2.0] - 2019-01-12

Expand Down
7 changes: 4 additions & 3 deletions Gopkg.lock

Some generated files are not rendered by default. Learn more about how customized files appear on GitHub.

4 changes: 4 additions & 0 deletions Gopkg.toml
Original file line number Diff line number Diff line change
Expand Up @@ -53,3 +53,7 @@
[prune]
go-tests = true
unused-packages = true

[[constraint]]
branch = "v2"
name = "gopkg.in/yaml.v2"
88 changes: 58 additions & 30 deletions cmd/prometheus-scw-sd/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -22,8 +22,8 @@ var (
// ErrMissingScwOrg defines the error if scw.org is empty.
ErrMissingScwOrg = errors.New("Missing required scw.org")

// ErrMissingScwRegion defines the error if scw.region is empty.
ErrMissingScwRegion = errors.New("Missing required scw.region")
// ErrMissingAnyCredentials defines the error if no credentials are provided.
ErrMissingAnyCredentials = errors.New("Missing any credentials")
)

func main() {
Expand Down Expand Up @@ -93,30 +93,44 @@ func main() {
Destination: &cfg.Target.Refresh,
},
&cli.StringFlag{
Name: "scw.token",
Value: "",
Usage: "Access token for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_TOKEN"},
Destination: &cfg.Target.Token,
Name: "scw.token",
Value: "",
Usage: "Access token for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_TOKEN"},
},
&cli.StringFlag{
Name: "scw.org",
Value: "",
Usage: "Organization for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_ORG"},
Destination: &cfg.Target.Org,
Name: "scw.org",
Value: "",
Usage: "Organization for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_ORG"},
},
&cli.StringFlag{
Name: "scw.region",
Value: "",
Usage: "Region for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_REGION"},
Destination: &cfg.Target.Region,
Name: "scw.region",
Value: "",
Usage: "Region for the Scaleway API",
EnvVars: []string{"PROMETHEUS_SCW_REGION"},
},
&cli.StringFlag{
Name: "scw.config",
Value: "",
Usage: "Path to Scaleway configuration file",
EnvVars: []string{"PROMETHEUS_SCW_CONFIG"},
},
},
Action: func(c *cli.Context) error {
logger := setupLogger(cfg)

if c.IsSet("scw.config") {
if err := readConfig(c.String("scw.config"), cfg); err != nil {
level.Error(logger).Log(
"msg", "Failed to read config",
"err", err,
)

return err
}
}

if cfg.Target.File == "" {
level.Error(logger).Log(
"msg", ErrMissingOutputFile,
Expand All @@ -125,28 +139,42 @@ func main() {
return ErrMissingOutputFile
}

if cfg.Target.Token == "" {
level.Error(logger).Log(
"msg", ErrMissingScwToken,
if c.IsSet("scw.token") && c.IsSet("scw.org") && c.IsSet("scw.region") {
credentials := config.Credential{
Project: "default",
Token: c.String("scw.token"),
Org: c.String("scw.org"),
Region: c.String("scw.region"),
}

cfg.Target.Credentials = append(
cfg.Target.Credentials,
credentials,
)

return ErrMissingScwToken
}
if credentials.Token == "" {
level.Error(logger).Log(
"msg", ErrMissingScwToken,
)

if cfg.Target.Org == "" {
level.Error(logger).Log(
"msg", ErrMissingScwOrg,
)
return ErrMissingScwToken
}

if credentials.Org == "" {
level.Error(logger).Log(
"msg", ErrMissingScwOrg,
)

return ErrMissingScwOrg
return ErrMissingScwOrg
}
}

if cfg.Target.Region == "" {
if len(cfg.Target.Credentials) == 0 {
level.Error(logger).Log(
"msg", ErrMissingScwRegion,
"msg", ErrMissingAnyCredentials,
)

return ErrMissingScwRegion
return ErrMissingAnyCredentials
}

return action.Server(cfg, logger)
Expand Down
37 changes: 37 additions & 0 deletions cmd/prometheus-scw-sd/setup.go
Original file line number Diff line number Diff line change
@@ -1,14 +1,24 @@
package main

import (
"encoding/json"
"errors"
"gopkg.in/yaml.v2"
"io/ioutil"
"os"
"path/filepath"
"strings"

"github.com/go-kit/kit/log"
"github.com/go-kit/kit/log/level"
"github.com/promhippie/prometheus-scw-sd/pkg/config"
)

var (
// ErrConfigFormatInvalid defines the error if ext is unsupported.
ErrConfigFormatInvalid = errors.New("Config extension is not supported")
)

func setupLogger(cfg *config.Config) log.Logger {
var logger log.Logger

Expand Down Expand Up @@ -40,3 +50,30 @@ func setupLogger(cfg *config.Config) log.Logger {
"ts", log.DefaultTimestampUTC,
)
}

func readConfig(file string, cfg *config.Config) error {
if file == "" {
return nil
}

content, err := ioutil.ReadFile(file)

if err != nil {
return err
}

switch strings.ToLower(filepath.Ext(file)) {
case ".yaml", ".yml":
if err = yaml.Unmarshal(content, cfg); err != nil {
return err
}
case ".json":
if err = json.Unmarshal(content, cfg); err != nil {
return err
}
default:
return ErrConfigFormatInvalid
}

return nil
}
34 changes: 34 additions & 0 deletions config/example.json
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
{
"server": {
"addr": "0.0.0.0:9000",
"path": "/metrics"
},
"logs": {
"level": "error",
"pretty": false
},
"target": {
"file": "/etc/prometheus/scw.json",
"refresh": 30,
"credentials": [
{
"project": "example1",
"token": "",
"org": "",
"region": "par1"
},
{
"project": "example2",
"token": "",
"org": "",
"region": "par1"
},
{
"project": "example3",
"token": "",
"org": "",
"region": "par1"
}
]
}
}
24 changes: 24 additions & 0 deletions config/example.yaml
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
server:
addr: 0.0.0.0:9000
path: /metrics

logs:
level: error
pretty: false

target:
file: /etc/prometheus/scw.json
refresh: 30
credentials:
- project: example1
token:
org:
region: par1
- project: example2
token:
org:
region: par1
- project: example3
token:
org:
region: par1
11 changes: 11 additions & 0 deletions docs/content/getting-started.md
Original file line number Diff line number Diff line change
Expand Up @@ -39,6 +39,13 @@ Currently we have not prepared a deployment for Kubernetes, but this is somethin

## Configuration

### Envrionment variables

If you prefer to configure the service with environment variables you can see the available variables below, in case you want to configure multiple accounts with a single service you are forced to use the configuration file as the environment variables are limited to a single account. As the service is pretty lightweight you can even start an instance per account and configure it entirely by the variables, it's up to you.

PROMETHEUS_SCW_CONFIG
: Path to Scaleway configuration file, optionally, required for muli credentials

PROMETHEUS_SCW_TOKEN
: Access token for the Scaleway API, required for authentication

Expand Down Expand Up @@ -66,6 +73,10 @@ PROMETHEUS_SCW_OUTPUT_FILE
PROMETHEUS_SCW_OUTPUT_REFRESH
: Discovery refresh interval in seconds, defaults to `30`

### Configuration file

Especially if you want to configure multiple accounts within a single service discovery you got to use the configuration file. So far we support the file formats `JSON` and `YAML`, if you want to get a full example configuration just take a look at [our repository](https://github.com/promhippie/prometheus-scw-sd/tree/master/config), there you can always see the latest configuration format. These example configurations include all available options, they also include the default values.

## Labels

* `__meta_scaleway_name`
Expand Down
Loading

0 comments on commit 746668c

Please sign in to comment.