-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathserver.go
182 lines (163 loc) · 5.71 KB
/
server.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
// Package goserv is a simple web application server bootstrap. It provides a common base
// for those wishing to focus on their application rather than on setting up
// flags, configuration files or logging. go-serv also comes with a simple
// status system for those who wish to expose structures via HTTP(s) for
// monitoring.
package goserv
import (
//"flag"
"fmt"
flag "github.com/ogier/pflag"
"io/ioutil"
"log"
"net/http"
"os"
"time"
"github.com/BurntSushi/toml"
"github.com/Sirupsen/logrus"
)
// Version of the library
const VERSION = "0.0.0"
var (
// Package level logger
Logger logrus.Logger
// Logger used with http.Server. This is an instance of Logger.Writer()
ServerErrorLogger log.Logger
)
// BaseConfiguration structure which can be filled out via
// defaults passed via flags, a configuration file or via
// the defaults set by the programmer (in that order of
// precedence)
type BaseConfiguration struct {
BindAddress string
BindPort int
ReadTimeout time.Duration
WriteTimeout time.Duration
MaxHeaderBytes int
LogLevel string
LogFile string
BindHttpsPort int
CertFile string
KeyFile string
}
// makeLogger configures the package level Logger instance
func makeLogger(conf *BaseConfiguration) {
levelName, err := logrus.ParseLevel(conf.LogLevel)
if err != nil {
defer Logger.Warnf("%s is not a valid level. Defaulting to info.", conf.LogLevel)
levelName = logrus.InfoLevel
}
logOut, err := os.OpenFile(conf.LogFile, os.O_CREATE|os.O_WRONLY|os.O_APPEND, 0644)
if err != nil {
defer Logger.Warnf("%s. Defaulting to Stderr.", err)
logOut = os.Stderr
}
formatter := new(logrus.TextFormatter)
formatter.DisableColors = true
Logger = logrus.Logger{
Out: logOut,
Formatter: formatter,
Level: levelName,
}
Logger.Infof("Initialized at level %s", Logger.Level)
}
// LogAccess logs Access requests that come in to the server
func LogAccess(handler http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
Logger.WithFields(logrus.Fields{
"RemoteAddr": r.RemoteAddr,
"UserAgent": r.UserAgent(),
"RequestURI": r.RequestURI,
"Method": r.Method,
"Proto": r.Proto,
"ConnectionLength": r.ContentLength,
}).Info("Access")
handler.ServeHTTP(w, r)
})
}
// BackgroundRunHttp runs an http server in a goroutine.
// Returns a send-only channel to watch for errors.
func BackgroundRunHttp(server *http.Server, conf *BaseConfiguration) chan error {
http := make(chan error)
go func() {
http <- server.ListenAndServe()
}()
return http
}
// BackgroundRunHttps runs an https server in a goroutine.
// Returns a send-only channel to watch for errors.
func BackgroundRunHttps(server *http.Server, conf *BaseConfiguration) chan error {
https := make(chan error)
go func() {
server.Addr = fmt.Sprintf("%s:%d", conf.BindAddress, int(conf.BindHttpsPort))
https <- server.ListenAndServeTLS(conf.CertFile, conf.KeyFile)
}()
return https
}
// Runs both Http and Https servers in their own goroutines. If one server
// exists the channels are closed and execution returns to the main goroutine.
func RunHttpAndHttps(server *http.Server, conf *BaseConfiguration) error {
http := BackgroundRunHttp(server, conf)
https := BackgroundRunHttps(server, conf)
var err error
LOOP:
for {
select {
case err = <-http:
Logger.Fatalf("HTTP server error: %s. Exiting ...", err)
break LOOP
case err = <-https:
Logger.Fatalf("HTTPS server error: %s. Exiting ...", err)
break LOOP
}
}
close(http)
close(https)
return err
}
// NewServer creates a new http.Server instance based off the BaseConfiguration.
// NewServer also handles reading the TOML configuration file and
// providing/reading the command line flags. Because of this
// NewServer should always be called after all flags have been defined.
func NewServer(conf *BaseConfiguration) http.Server {
// TOML configuration file can overwrite defaults
tomlData, err := ioutil.ReadFile(os.Args[len(os.Args)-1])
if err != nil {
defer Logger.Info("No conf. Skipping.")
} else {
if _, err := toml.Decode(string(tomlData), &conf); err != nil {
defer Logger.Errorf("Configuration file could not be decoded. %s. Exiting...", err)
}
}
// Flags can override config items
// Server flags
flag.StringVar(&conf.BindAddress, "BindAddress", conf.BindAddress, "Bind address.")
flag.IntVar(&conf.BindPort, "BindPort", conf.BindPort, "HTTP bind port.")
flag.DurationVar(&conf.ReadTimeout, "ReadTimeout", conf.ReadTimeout, "Read timeout.")
flag.DurationVar(&conf.WriteTimeout, "WriteTimeout", conf.WriteTimeout, "Write timeout.")
flag.IntVar(&conf.MaxHeaderBytes, "MaxHeaderBytes", conf.MaxHeaderBytes, "Max header bytes.")
// Server Logger flags
flag.StringVar(&conf.LogLevel, "LogLevel", conf.LogLevel, "Log level.")
flag.StringVar(&conf.LogFile, "LogFile", conf.LogFile, "Log file.")
// TLS related flags
flag.IntVar(&conf.BindHttpsPort, "BindHttpsPort", conf.BindHttpsPort, "HTTPS bind port.")
flag.StringVar(&conf.CertFile, "CertFile", conf.CertFile, "Cert file.")
flag.StringVar(&conf.KeyFile, "KeyFile", conf.KeyFile, "Key file.")
flag.Parse()
// Logging specific work also injecting the logrus log into the Servers errorlog
// BUG(ashcrow): This needs work!!!
makeLogger(conf)
Logger.Debugf("Final configuration: %+v", conf)
w := Logger.Writer()
defer w.Close()
ServerErrorLogger = *log.New(w, "ServerErrorLogger", log.Lshortfile)
// -------------
// Return the configured http.Server
return http.Server{
Addr: fmt.Sprintf("%s:%d", conf.BindAddress, conf.BindPort),
ReadTimeout: conf.ReadTimeout,
WriteTimeout: conf.WriteTimeout,
MaxHeaderBytes: conf.MaxHeaderBytes,
ErrorLog: &ServerErrorLogger,
}
}