http2https/main.go
2016-06-30 15:42:09 +02:00

181 lines
4.1 KiB
Go

// main
package main
import (
"flag"
"fmt"
"github.com/kardianos/service"
_ "golang.org/x/tools/go/ssa"
"log"
"os"
"runtime"
"sync"
)
var (
logger service.Logger
loggerLock sync.Mutex
verFlag *bool
svcFlag *string
listenPortFlag *int
destHostFlag *string
destPortFlag *int
destCertFlag *string
destKeyFlag *string
destKeyPassFlag *string
confServiceName string
confServiceDispName string
)
func logInfof(format string, a ...interface{}) error {
loggerLock.Lock()
defer loggerLock.Unlock()
if logger != nil {
return logger.Infof(format, a...)
}
return nil
}
func logError(v ...interface{}) error {
loggerLock.Lock()
defer loggerLock.Unlock()
if logger != nil {
return logger.Error(v)
}
return nil
}
// Program structures.
// Define Start and Stop methods.
type program struct {
exit chan struct{}
}
func (p *program) Start(s service.Service) error {
if service.Interactive() {
logInfof("Service \"%s\" is running in terminal.", confServiceDispName)
} else {
logInfof("Service \"%s\" is running under service manager.", confServiceDispName)
}
p.exit = make(chan struct{})
// Start should not block. Do the actual work async.
go p.run()
return nil
}
func (p *program) run() {
startServer()
logInfof("Service \"%s\" is started.", confServiceDispName)
for {
select {
case <-p.exit:
return
}
}
}
func (p *program) Stop(s service.Service) error {
// Any work in Stop should be quick, usually a few seconds at most.
logInfof("Service \"%s\" is stopping.", confServiceDispName)
stopServer()
logInfof("Service \"%s\" is stopped.", confServiceDispName)
close(p.exit)
return nil
}
// Service setup.
// Define service config.
// Create the service.
// Setup the logger.
// Handle service controls (optional).
// Run the service.
func main() {
runtime.GOMAXPROCS(runtime.NumCPU())
flag.Usage = usage
verFlag = flag.Bool("version", false, "Show version")
svcFlag = flag.String("service", "", fmt.Sprintf("Control the system service. Valid actions: %q\n", service.ControlAction))
listenPortFlag = flag.Int("listen_port", 13777, "Listening port")
destHostFlag = flag.String("dest_host", "", "Destination host name")
destPortFlag = flag.Int("dest_port", 443, "Destination port")
destCertFlag = flag.String("dest_cert", "", "Destination certificate file name")
destKeyFlag = flag.String("dest_key", "", "Destination key file name")
destKeyPassFlag = flag.String("dest_key_pass", "", "Destination key password")
flag.Parse()
if *verFlag == true {
fmt.Println("Version: ", VERSION)
fmt.Println("Build: ", BUILD_DATE)
os.Exit(0)
}
if *destHostFlag == "" {
usage()
os.Exit(2)
}
confServiceName = fmt.Sprintf("http2https_%v", *listenPortFlag)
confServiceDispName = fmt.Sprintf("%s for \"%s:%v\"", serviceDisplayName, *destHostFlag, *destPortFlag)
svcConfig := &service.Config{
Name: confServiceName,
DisplayName: confServiceDispName,
Description: confServiceDispName,
Arguments: []string{
fmt.Sprintf("-listen_port=%v", *listenPortFlag),
fmt.Sprintf("-dest_host=%s", *destHostFlag),
fmt.Sprintf("-dest_port=%v", *destPortFlag),
fmt.Sprintf("-dest_cert=%s", *destCertFlag),
fmt.Sprintf("-dest_key=%s", *destKeyFlag),
fmt.Sprintf("-dest_key_pass=%s", *destKeyPassFlag),
},
}
prg := &program{}
s, err := service.New(prg, svcConfig)
if err != nil {
log.Fatal(err)
}
errs := make(chan error, 5)
func() {
loggerLock.Lock()
defer loggerLock.Unlock()
logger, err = s.Logger(errs)
if err != nil {
log.Fatal(err)
}
}()
go func() {
for {
err := <-errs
if err != nil {
log.Print(err)
}
}
}()
if len(*svcFlag) != 0 {
err := service.Control(s, *svcFlag)
if err != nil {
log.Printf("Valid actions: %q\n", service.ControlAction)
log.Fatal(err)
}
return
}
err = s.Run()
if err != nil {
logError(err)
}
}
const serviceDisplayName = `HTTP to HTTPS tunneling service`
const usageTemplate = `Http2HttpS is ` + serviceDisplayName + `
Usage: http2https commands
The commands are:
`
func usage() {
fmt.Fprintln(os.Stderr, usageTemplate)
flag.PrintDefaults()
}