2015-06-15 03:36:32 -07:00
// Copyright 2015 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.
// The main package for the Prometheus server executeable.
package main
import (
2015-06-15 03:23:02 -07:00
"bytes"
2015-06-15 03:36:32 -07:00
"flag"
2015-06-15 03:23:02 -07:00
"fmt"
2015-06-15 03:36:32 -07:00
_ "net/http/pprof" // Comment this line to disable pprof endpoint.
"os"
"os/signal"
2015-06-18 03:33:39 -07:00
"strings"
2015-06-15 03:36:32 -07:00
"syscall"
2015-06-15 03:23:02 -07:00
"text/template"
2015-06-15 03:36:32 -07:00
"time"
"github.com/prometheus/log"
2015-06-23 09:04:04 -07:00
"github.com/prometheus/client_golang/prometheus"
2015-06-15 03:36:32 -07:00
"github.com/prometheus/prometheus/config"
"github.com/prometheus/prometheus/notification"
"github.com/prometheus/prometheus/promql"
"github.com/prometheus/prometheus/retrieval"
"github.com/prometheus/prometheus/rules"
"github.com/prometheus/prometheus/storage"
"github.com/prometheus/prometheus/storage/local"
"github.com/prometheus/prometheus/storage/remote"
2015-06-15 03:23:02 -07:00
"github.com/prometheus/prometheus/version"
2015-06-15 03:36:32 -07:00
"github.com/prometheus/prometheus/web"
)
func main ( ) {
os . Exit ( Main ( ) )
}
func Main ( ) int {
if err := parse ( os . Args [ 1 : ] ) ; err != nil {
return 2
}
2015-06-15 03:23:02 -07:00
printVersion ( )
2015-06-15 03:36:32 -07:00
if cfg . printVersion {
return 0
}
var (
2015-06-25 16:32:44 -07:00
memStorage = local . NewMemorySeriesStorage ( & cfg . storage )
remoteStorage = remote . New ( & cfg . remote )
sampleAppender = storage . Fanout { memStorage }
2015-06-15 03:36:32 -07:00
)
2015-06-23 09:04:04 -07:00
if remoteStorage != nil {
sampleAppender = append ( sampleAppender , remoteStorage )
}
2015-06-15 03:36:32 -07:00
2015-06-25 16:32:44 -07:00
var (
notificationHandler = notification . NewNotificationHandler ( & cfg . notification )
targetManager = retrieval . NewTargetManager ( sampleAppender )
queryEngine = promql . NewEngine ( memStorage , & cfg . queryEngine )
)
2015-06-15 03:36:32 -07:00
ruleManager := rules . NewManager ( & rules . ManagerOptions {
SampleAppender : sampleAppender ,
NotificationHandler : notificationHandler ,
QueryEngine : queryEngine ,
2015-06-30 05:38:01 -07:00
ExternalURL : cfg . web . ExternalURL ,
2015-06-15 03:36:32 -07:00
} )
flags := map [ string ] string { }
cfg . fs . VisitAll ( func ( f * flag . Flag ) {
flags [ f . Name ] = f . Value . String ( )
} )
status := & web . PrometheusStatus {
TargetPools : targetManager . Pools ,
Rules : ruleManager . Rules ,
Flags : flags ,
Birth : time . Now ( ) ,
}
webHandler := web . New ( memStorage , queryEngine , ruleManager , status , & cfg . web )
if ! reloadConfig ( cfg . configFile , status , targetManager , ruleManager ) {
2015-06-23 09:04:04 -07:00
return 1
2015-06-15 03:36:32 -07:00
}
// Wait for reload or termination signals. Start the handler for SIGHUP as
// early as possible, but ignore it until we are ready to handle reloading
// our config.
hup := make ( chan os . Signal )
hupReady := make ( chan bool )
signal . Notify ( hup , syscall . SIGHUP )
go func ( ) {
<- hupReady
2015-08-11 00:08:17 -07:00
for {
select {
case <- hup :
case <- webHandler . Reload ( ) :
}
2015-06-15 03:36:32 -07:00
reloadConfig ( cfg . configFile , status , targetManager , ruleManager )
}
} ( )
// Start all components.
if err := memStorage . Start ( ) ; err != nil {
log . Errorln ( "Error opening memory series storage:" , err )
return 1
}
defer func ( ) {
if err := memStorage . Stop ( ) ; err != nil {
log . Errorln ( "Error stopping storage:" , err )
}
} ( )
2015-06-23 09:04:04 -07:00
if remoteStorage != nil {
prometheus . MustRegister ( remoteStorage )
2015-06-15 03:36:32 -07:00
2015-06-23 09:04:04 -07:00
go remoteStorage . Run ( )
defer remoteStorage . Stop ( )
2015-06-15 03:36:32 -07:00
}
2015-06-23 09:04:04 -07:00
// The storage has to be fully initialized before registering.
prometheus . MustRegister ( memStorage )
prometheus . MustRegister ( notificationHandler )
2015-06-15 03:36:32 -07:00
go ruleManager . Run ( )
defer ruleManager . Stop ( )
go notificationHandler . Run ( )
defer notificationHandler . Stop ( )
go targetManager . Run ( )
defer targetManager . Stop ( )
defer queryEngine . Stop ( )
go webHandler . Run ( )
// Wait for reload or termination signals.
close ( hupReady ) // Unblock SIGHUP handler.
term := make ( chan os . Signal )
signal . Notify ( term , os . Interrupt , syscall . SIGTERM )
select {
case <- term :
log . Warn ( "Received SIGTERM, exiting gracefully..." )
case <- webHandler . Quit ( ) :
log . Warn ( "Received termination request via web service, exiting gracefully..." )
}
log . Info ( "See you next time!" )
return 0
}
// Reloadable things can change their internal state to match a new config
// and handle failure gracefully.
type Reloadable interface {
ApplyConfig ( * config . Config ) bool
}
func reloadConfig ( filename string , rls ... Reloadable ) bool {
log . Infof ( "Loading configuration file %s" , filename )
2015-08-05 09:30:37 -07:00
conf , err := config . LoadFile ( filename )
2015-06-15 03:36:32 -07:00
if err != nil {
log . Errorf ( "Couldn't load configuration (-config.file=%s): %v" , filename , err )
log . Errorf ( "Note: The configuration format has changed with version 0.14. Please see the documentation (http://prometheus.io/docs/operating/configuration/) and the provided configuration migration tool (https://github.com/prometheus/migrate)." )
return false
}
success := true
for _ , rl := range rls {
success = success && rl . ApplyConfig ( conf )
}
return success
}
2015-06-15 03:23:02 -07:00
2015-06-18 03:33:39 -07:00
var versionInfoTmpl = `
prometheus , version { { . version } } ( branch : { { . branch } } , revision : { { . revision } } )
2015-06-15 03:23:02 -07:00
build user : { { . buildUser } }
2015-06-18 03:33:39 -07:00
build date : { { . buildDate } }
2015-06-15 03:23:02 -07:00
go version : { { . goVersion } }
`
func printVersion ( ) {
t := template . Must ( template . New ( "version" ) . Parse ( versionInfoTmpl ) )
var buf bytes . Buffer
if err := t . ExecuteTemplate ( & buf , "version" , version . Map ) ; err != nil {
panic ( err )
}
2015-06-18 03:33:39 -07:00
fmt . Fprintln ( os . Stdout , strings . TrimSpace ( buf . String ( ) ) )
2015-06-15 03:23:02 -07:00
}