142 lines
4.7 KiB
Go
142 lines
4.7 KiB
Go
// Copyright © 2018 Enrico Stahn <enrico.stahn@gmail.com>
|
|
// 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 cmd
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
"os"
|
|
"os/signal"
|
|
"time"
|
|
|
|
"github.com/hipages/php-fpm_exporter/phpfpm"
|
|
"github.com/prometheus/client_golang/prometheus"
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
"github.com/spf13/cobra"
|
|
)
|
|
|
|
// Configuration variables
|
|
var (
|
|
listeningAddress string
|
|
metricsEndpoint string
|
|
scrapeURIs []string
|
|
fixProcessCount bool
|
|
)
|
|
|
|
// serverCmd represents the server command
|
|
var serverCmd = &cobra.Command{
|
|
Use: "server",
|
|
Short: "A brief description of your command",
|
|
Long: `A longer description that spans multiple lines and likely contains examples
|
|
and usage of using your command. For example:
|
|
|
|
Cobra is a CLI library for Go that empowers applications.
|
|
This application is a tool to generate the needed files
|
|
to quickly create a Cobra application.`,
|
|
Run: func(cmd *cobra.Command, args []string) {
|
|
log.Infof("Starting server on %v with path %v", listeningAddress, metricsEndpoint)
|
|
|
|
pm := phpfpm.PoolManager{}
|
|
|
|
for _, uri := range scrapeURIs {
|
|
pm.Add(uri)
|
|
}
|
|
|
|
exporter := phpfpm.NewExporter(pm)
|
|
|
|
if fixProcessCount {
|
|
log.Info("Idle/Active/Total Processes will be calculated by php-fpm_exporter.")
|
|
exporter.CalculateProcessScoreboard = true
|
|
}
|
|
|
|
prometheus.MustRegister(exporter)
|
|
|
|
srv := &http.Server{
|
|
Addr: listeningAddress,
|
|
// Good practice to set timeouts to avoid Slowloris attacks.
|
|
WriteTimeout: time.Second * 15,
|
|
ReadTimeout: time.Second * 15,
|
|
IdleTimeout: time.Second * 60,
|
|
}
|
|
|
|
http.Handle(metricsEndpoint, promhttp.Handler())
|
|
http.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
|
w.Write([]byte(`<html>
|
|
<head><title>php-fpm_exporter</title></head>
|
|
<body>
|
|
<h1>php-fpm_exporter</h1>
|
|
<p><a href='` + metricsEndpoint + `'>Metrics</a></p>
|
|
</body>
|
|
</html>`))
|
|
})
|
|
|
|
// Run our server in a goroutine so that it doesn't block.
|
|
go func() {
|
|
if err := srv.ListenAndServe(); err != nil {
|
|
log.Error(err)
|
|
}
|
|
}()
|
|
|
|
c := make(chan os.Signal, 1)
|
|
// We'll accept graceful shutdowns when quit via SIGINT (Ctrl+C)
|
|
// SIGKILL, SIGQUIT or SIGTERM (Ctrl+/) will not be caught.
|
|
signal.Notify(c, os.Interrupt)
|
|
|
|
// Block until we receive our signal.
|
|
<-c
|
|
|
|
// Create a deadline to wait for.
|
|
var wait time.Duration
|
|
ctx, cancel := context.WithTimeout(context.Background(), wait)
|
|
defer cancel()
|
|
// Doesn't block if no connections, but will otherwise wait
|
|
// until the timeout deadline.
|
|
srv.Shutdown(ctx)
|
|
// Optionally, you could run srv.Shutdown in a goroutine and block on
|
|
// <-ctx.Done() if your application should wait for other services
|
|
// to finalize based on context cancellation.
|
|
log.Info("Shutting down")
|
|
os.Exit(0)
|
|
},
|
|
}
|
|
|
|
func init() {
|
|
RootCmd.AddCommand(serverCmd)
|
|
|
|
serverCmd.Flags().StringVar(&listeningAddress, "web.listen-address", ":9253", "Address on which to expose metrics and web interface.")
|
|
serverCmd.Flags().StringVar(&metricsEndpoint, "web.telemetry-path", "/metrics", "Path under which to expose metrics.")
|
|
serverCmd.Flags().StringSliceVar(&scrapeURIs, "phpfpm.scrape-uri", []string{"tcp://127.0.0.1:9000/status"}, "FastCGI address, e.g. unix:///tmp/php.sock;/status or tcp://127.0.0.1:9000/status")
|
|
serverCmd.Flags().BoolVar(&fixProcessCount, "phpfpm.fix-process-count", false, "Enable to calculate process numbers via php-fpm_exporter since PHP-FPM sporadically reports wrong active/idle/total process numbers.")
|
|
|
|
//viper.BindEnv("web.listen-address", "PHP_FPM_WEB_LISTEN_ADDRESS")
|
|
//viper.BindPFlag("web.listen-address", serverCmd.Flags().Lookup("web.listen-address"))
|
|
|
|
// Workaround since vipers BindEnv is currently not working as expected (see https://github.com/spf13/viper/issues/461)
|
|
|
|
envs := map[string]string{
|
|
"PHP_FPM_WEB_LISTEN_ADDRESS": "web.listen-address",
|
|
"PHP_FPM_WEB_TELEMETRY_PATH": "web.telemetry-path",
|
|
"PHP_FPM_SCRAPE_URI": "phpfpm.scrape-uri",
|
|
"PHP_FPM_FIX_PROCESS_COUNT": "phpfpm.fix-process-count",
|
|
}
|
|
|
|
for env, flag := range envs {
|
|
flag := serverCmd.Flags().Lookup(flag)
|
|
flag.Usage = fmt.Sprintf("%v [env %v]", flag.Usage, env)
|
|
if value := os.Getenv(env); value != "" {
|
|
flag.Value.Set(value)
|
|
}
|
|
}
|
|
}
|