perkeep/lib/go/camli/webserver/webserver.go

172 lines
4.1 KiB
Go
Raw Normal View History

2011-01-28 07:07:18 +00:00
/*
Copyright 2011 Google Inc.
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 webserver
import (
"bufio"
2011-03-18 01:56:55 +00:00
"crypto/rand"
"crypto/tls"
"flag"
"http"
"log"
"net"
"os"
"strconv"
"strings"
2011-03-18 01:56:55 +00:00
"time"
)
2011-03-18 01:56:55 +00:00
var Listen = flag.String("listen", "0.0.0.0:2856", "host:port to listen on, or :0 to auto-select")
type HandlerPicker func(req *http.Request) (http.HandlerFunc, bool)
type Server struct {
2011-03-18 01:56:55 +00:00
premux []HandlerPicker
mux *http.ServeMux
listener net.Listener
enableTLS bool
tlsCertFile, tlsKeyFile string
}
func New() *Server {
return &Server{
2011-03-18 01:56:55 +00:00
premux: make([]HandlerPicker, 0),
mux: http.NewServeMux(),
}
}
func (s *Server) SetTLS(certFile, keyFile string) {
s.enableTLS = true
s.tlsCertFile = certFile
s.tlsKeyFile = keyFile
}
func (s *Server) BaseURL() string {
scheme := "http"
if s.enableTLS {
scheme = "https"
}
if s.listener != nil {
return scheme + "://" + s.listener.Addr().String()
}
if strings.HasPrefix(*Listen, ":") {
return scheme + "://localhost" + *Listen
}
return scheme + "://" + strings.Replace(*Listen, "0.0.0.0:", "localhost:", 1)
}
// Register conditional handler-picker functions which get run before
// HandleFunc or Handle. The HandlerPicker should return false if
// it's not interested in a request.
func (s *Server) RegisterPreMux(hp HandlerPicker) {
s.premux = append(s.premux, hp)
}
func (s *Server) HandleFunc(pattern string, fn func(http.ResponseWriter, *http.Request)) {
s.mux.HandleFunc(pattern, fn)
}
func (s *Server) Handle(pattern string, handler http.Handler) {
s.mux.Handle(pattern, handler)
}
func (s *Server) ServeHTTP(rw http.ResponseWriter, req *http.Request) {
for _, hp := range s.premux {
handler, ok := hp(req)
if ok {
handler(rw, req)
return
}
}
s.mux.ServeHTTP(rw, req)
}
func (s *Server) Serve() {
doLog := os.Getenv("TESTING_PORT_WRITE_FD") == "" // Don't make noise during unit tests
base := s.BaseURL()
if doLog {
log.Printf("Starting to listen on %s\n", base)
}
var err os.Error
s.listener, err = net.Listen("tcp", *Listen)
if err != nil {
log.Fatalf("Failed to listen on %s: %v", *Listen, err)
}
2011-03-18 01:56:55 +00:00
if doLog && strings.HasSuffix(base, ":0") {
log.Printf("Now listening on %s\n", s.BaseURL())
}
if s.enableTLS {
2011-03-18 01:56:55 +00:00
config := &tls.Config{
Rand: rand.Reader,
Time: time.Seconds,
NextProtos: []string{"http/1.1"},
}
config.Certificates = make([]tls.Certificate, 1)
config.Certificates[0], err = tls.LoadX509KeyPair(s.tlsCertFile, s.tlsKeyFile)
2011-03-18 01:56:55 +00:00
if err != nil {
log.Fatalf("Failed to load TLS cert: %v", err)
}
s.listener = tls.NewListener(s.listener, config)
}
go runTestHarnessIntegration(s.listener)
err = http.Serve(s.listener, s)
if err != nil {
log.Printf("Error in http server: %v\n", err)
os.Exit(1)
}
}
func pipeFromEnvFd(env string) *os.File {
fdStr := os.Getenv(env)
if fdStr == "" {
return nil
}
fd, err := strconv.Atoi(fdStr)
if err != nil {
log.Fatalf("Bogus test harness fd '%s': %v", fdStr, err)
}
2011-03-18 01:56:55 +00:00
return os.NewFile(fd, "testingpipe-"+env)
}
// Signals the test harness that we've started listening.
// TODO: write back the port number that we randomly selected?
// For now just writes back a single byte.
func runTestHarnessIntegration(listener net.Listener) {
writePipe := pipeFromEnvFd("TESTING_PORT_WRITE_FD")
readPipe := pipeFromEnvFd("TESTING_CONTROL_READ_FD")
if writePipe != nil {
writePipe.Write([]byte(listener.Addr().String() + "\n"))
}
if readPipe != nil {
bufr := bufio.NewReader(readPipe)
for {
line, err := bufr.ReadString('\n')
if err == os.EOF || line == "EXIT\n" {
os.Exit(0)
}
return
}
}
}