Files
certmanager/internal/server/server.go
Олег Бородин e9d4d1ef07 import sources
2024-07-30 09:49:53 +02:00

295 lines
6.1 KiB
Go

package server
import (
"os"
"os/signal"
"syscall"
"os/user"
"path/filepath"
"strconv"
"certmanager/internal/config"
"certmanager/internal/database"
"certmanager/internal/logic"
"certmanager/pkg/aux509"
"certmanager/pkg/logger"
ghandler "certmanager/internal/grpc/handler"
gservice "certmanager/internal/grpc/service"
whandler "certmanager/internal/wrpc/handler"
wservice "certmanager/internal/wrpc/service"
)
type Server struct {
conf *config.Config
lg *logic.Logic
wsvc *wservice.Service
whand *whandler.Handler
gsvc *gservice.Service
ghand *ghandler.Handler
log *logger.Logger
db *database.Database
x509cert []byte
x509key []byte
x509caCert []byte
x509caKey []byte
}
func NewServer() (*Server, error) {
var err error
srv := &Server{}
srv.log = logger.NewLogger("server")
return srv, err
}
func (srv *Server) Configure() error {
var err error
srv.conf = config.NewConfig()
err = srv.conf.ReadFile()
if err != nil {
return err
}
err = srv.conf.ReadEnv()
if err != nil {
return err
}
err = srv.conf.ReadOpts()
if err != nil {
return err
}
return err
}
func (srv *Server) Build() error {
var err error
srv.log.Infof("Build server")
// Create X509 certs
srv.x509caCert, srv.x509caKey, err = aux509.CreateX509CACert(srv.conf.Hostname)
if err != nil {
return err
}
srv.x509cert, srv.x509key, err = aux509.CreateX509Cert(srv.conf.Hostname, srv.x509caKey)
if err != nil {
return err
}
// Create database
db, err := database.NewDatabase(srv.conf.DataDir)
if err != nil {
return err
}
srv.db = db
// Create logic
logicConfig := &logic.LogicConfig{
Auths: srv.conf.Auths,
Database: srv.db,
}
srv.lg, err = logic.NewLogic(logicConfig)
if err != nil {
return err
}
// Create whandler
whandlerConfig := &whandler.HandlerConfig{
Logic: srv.lg,
}
srv.whand, err = whandler.NewHandler(whandlerConfig)
if err != nil {
return err
}
// Create wservice
wserviceConfig := &wservice.ServiceConfig{
PortNum: srv.conf.Wservice.PortNum,
Hostname: srv.conf.Hostname,
Handler: srv.whand,
X509Cert: srv.x509cert,
X509Key: srv.x509key,
}
srv.wsvc, err = wservice.NewService(wserviceConfig)
if err != nil {
return err
}
err = srv.wsvc.Build()
if err != nil {
return err
}
// Create ghandler
ghandlerConfig := &ghandler.HandlerConfig{
Logic: srv.lg,
}
srv.ghand = ghandler.NewHandler(ghandlerConfig)
if err != nil {
return err
}
// Create gservice
gserviceConfig := &gservice.ServiceConfig{
PortNum: srv.conf.Gservice.PortNum,
Hostname: srv.conf.Hostname,
Handler: srv.ghand,
Logic: srv.lg,
X509Cert: srv.x509cert,
X509Key: srv.x509key,
}
srv.gsvc = gservice.NewService(gserviceConfig)
if err != nil {
return err
}
return err
}
func (srv *Server) Run() error {
var err error
// Log configuration
yamlConfig, err := srv.conf.Yaml()
if err != nil {
return err
}
srv.log.Debugf("Server configuration:\n%s\n", yamlConfig)
// Show current user
currUser, err := user.Current()
if err != nil {
return err
}
srv.log.Infof("Running server as user %s", currUser.Username)
sigs := make(chan os.Signal, 1)
gdone := make(chan error, 1)
wdone := make(chan error, 1)
// Run wservice
startWservice := func(svc *wservice.Service, done chan error) {
err = svc.Run()
if err != nil {
srv.log.Errorf("Service error: %v", err)
done <- err
}
}
go startWservice(srv.wsvc, wdone)
// Run gservice
startGservice := func(svc *gservice.Service, done chan error) {
err = svc.Run()
if err != nil {
srv.log.Errorf("Service error: %v", err)
done <- err
}
}
go startGservice(srv.gsvc, gdone)
signal.Notify(sigs, syscall.SIGINT, syscall.SIGTERM)
var signal os.Signal
select {
case signal = <-sigs:
srv.log.Infof("Services stopped by signal: %v", signal)
srv.wsvc.Stop()
srv.gsvc.Stop()
case err = <-gdone:
srv.log.Infof("Service stopped by gservice error: %v", err)
srv.wsvc.Stop()
case err = <-wdone:
srv.log.Infof("Service stopped by wservice error: %v", err)
srv.gsvc.Stop()
}
return err
}
func (srv *Server) PseudoFork() error {
const successExit int = 0
var keyEnv string = "IMX0LTSELMRF8KASWER"
var err error
_, isChild := os.LookupEnv(keyEnv)
switch {
case !isChild:
os.Setenv(keyEnv, "TRUE")
procAttr := syscall.ProcAttr{}
cwd, err := os.Getwd()
if err != nil {
return err
}
var sysFiles = make([]uintptr, 3)
sysFiles[0] = uintptr(syscall.Stdin)
sysFiles[1] = uintptr(syscall.Stdout)
sysFiles[2] = uintptr(syscall.Stderr)
procAttr.Files = sysFiles
procAttr.Env = os.Environ()
procAttr.Dir = cwd
_, err = syscall.ForkExec(os.Args[0], os.Args, &procAttr)
if err != nil {
return err
}
os.Exit(successExit)
case isChild:
_, err = syscall.Setsid()
if err != nil {
return err
}
}
os.Unsetenv(keyEnv)
return err
}
func (srv *Server) Daemonize() error {
var err error
if srv.conf.Daemon {
// Restart process process
err = srv.PseudoFork()
if err != nil {
return err
}
// Redirect stdin
nullFile, err := os.OpenFile("/dev/null", os.O_RDWR, 0)
if err != nil {
return err
}
err = syscall.Dup2(int(nullFile.Fd()), int(os.Stdin.Fd()))
if err != nil {
return err
}
// Redirect stderr and stout
logdir := filepath.Dir(srv.conf.LogPath)
err = os.MkdirAll(logdir, 0750)
if err != nil {
return err
}
logFile, err := os.OpenFile(srv.conf.LogPath, os.O_WRONLY|os.O_CREATE|os.O_APPEND, 0640)
if err != nil {
return err
}
err = syscall.Dup2(int(logFile.Fd()), int(os.Stdout.Fd()))
if err != nil {
return err
}
err = syscall.Dup2(int(logFile.Fd()), int(os.Stderr.Fd()))
if err != nil {
return err
}
// Write process ID
rundir := filepath.Dir(srv.conf.RunPath)
err = os.MkdirAll(rundir, 0750)
if err != nil {
return err
}
pidFile, err := os.OpenFile(srv.conf.RunPath, os.O_WRONLY|os.O_CREATE|os.O_TRUNC, 0640)
if err != nil {
return err
}
defer pidFile.Close()
currPid := os.Getpid()
_, err = pidFile.WriteString(strconv.Itoa(currPid))
if err != nil {
return err
}
}
return err
}