mirror of
https://github.com/OliveTin/OliveTin
synced 2025-12-12 17:15:37 +00:00
Single endpoint for web+ui (Issue #4). Migrated to modern protobuf+grpc.
This commit is contained in:
6
Makefile
6
Makefile
@@ -23,8 +23,9 @@ daemon-unittests:
|
|||||||
go tool cover -html=reports/unittests.out -o reports/unittests.html
|
go tool cover -html=reports/unittests.out -o reports/unittests.html
|
||||||
|
|
||||||
grpc:
|
grpc:
|
||||||
protoc -I.:/usr/share/gocode/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis/ --go_out=plugins=grpc:gen/grpc/ OliveTin.proto
|
protoc -I.:$(GOPATH)/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis/ --go_out=. --go-grpc_out=. --grpc-gateway_out=. OliveTin.proto
|
||||||
protoc -I.:/usr/share/gocode/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis/ --grpc-gateway_out=gen/grpc --grpc-gateway_opt paths=source_relative OliveTin.proto
|
# protoc --go-grpc_out=grpc:gen/grpc/ OliveTin.proto
|
||||||
|
# protoc -I.:$(GOPATH)/src/github.com/grpc-ecosystem/grpc-gateway/third_party/googleapis/ --grpc-gateway_out=gen/grpc --grpc-gateway_opt paths=source_relative OliveTin.proto
|
||||||
|
|
||||||
podman-image:
|
podman-image:
|
||||||
buildah bud -t olivetin
|
buildah bud -t olivetin
|
||||||
@@ -46,6 +47,7 @@ webui-codestyle:
|
|||||||
cd webui && stylelint style.css
|
cd webui && stylelint style.css
|
||||||
|
|
||||||
release-common:
|
release-common:
|
||||||
|
rm -rf webui/node_modules/
|
||||||
rm -rf releases/
|
rm -rf releases/
|
||||||
mkdir -p releases/common/
|
mkdir -p releases/common/
|
||||||
cp -r webui releases/common/
|
cp -r webui releases/common/
|
||||||
|
|||||||
@@ -1,5 +1,7 @@
|
|||||||
syntax = "proto3";
|
syntax = "proto3";
|
||||||
|
|
||||||
|
option go_package = "gen/grpc";
|
||||||
|
|
||||||
import "google/api/annotations.proto";
|
import "google/api/annotations.proto";
|
||||||
|
|
||||||
message ActionButton {
|
message ActionButton {
|
||||||
@@ -28,13 +30,13 @@ message StartActionResponse {
|
|||||||
service OliveTinApi {
|
service OliveTinApi {
|
||||||
rpc GetButtons(GetButtonsRequest) returns (GetButtonsResponse) {
|
rpc GetButtons(GetButtonsRequest) returns (GetButtonsResponse) {
|
||||||
option (google.api.http) = {
|
option (google.api.http) = {
|
||||||
get: "/GetButtons"
|
get: "/api/GetButtons"
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
rpc StartAction(StartActionRequest) returns (StartActionResponse) {
|
rpc StartAction(StartActionRequest) returns (StartActionResponse) {
|
||||||
option (google.api.http) = {
|
option (google.api.http) = {
|
||||||
get: "/StartAction"
|
get: "/api/StartAction"
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,8 @@ import (
|
|||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
|
|
||||||
grpcapi "github.com/jamesread/OliveTin/internal/grpcapi"
|
grpcapi "github.com/jamesread/OliveTin/internal/grpcapi"
|
||||||
restapi "github.com/jamesread/OliveTin/internal/restapi"
|
|
||||||
webuiServer "github.com/jamesread/OliveTin/internal/webuiServer"
|
"github.com/jamesread/OliveTin/internal/httpservers"
|
||||||
|
|
||||||
config "github.com/jamesread/OliveTin/internal/config"
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
"github.com/spf13/viper"
|
"github.com/spf13/viper"
|
||||||
@@ -39,22 +39,19 @@ func init() {
|
|||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
log.SetLevel(cfg.GetLogLevel())
|
if logLevel, err := log.ParseLevel(cfg.LogLevel); err == nil {
|
||||||
|
log.SetLevel(logLevel)
|
||||||
|
}
|
||||||
|
|
||||||
viper.WatchConfig()
|
viper.WatchConfig()
|
||||||
}
|
}
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
log.WithFields(log.Fields{
|
log.Info("OliveTin started")
|
||||||
"listenAddressGrpcActions": cfg.ListenAddressGrpcActions,
|
|
||||||
"listenAddressRestActions": cfg.ListenAddressRestActions,
|
|
||||||
"listenAddressWebUI": cfg.ListenAddressWebUI,
|
|
||||||
}).Info("OliveTin started")
|
|
||||||
|
|
||||||
log.Debugf("%+v", cfg)
|
log.Debugf("%+v", cfg)
|
||||||
|
|
||||||
go grpcapi.Start(cfg.ListenAddressGrpcActions, cfg)
|
go grpcapi.Start(cfg)
|
||||||
go restapi.Start(cfg.ListenAddressRestActions, cfg.ListenAddressGrpcActions, cfg)
|
|
||||||
|
|
||||||
webuiServer.Start(cfg.ListenAddressWebUI, cfg.ListenAddressRestActions)
|
httpservers.StartServers(cfg)
|
||||||
}
|
}
|
||||||
|
|||||||
18
go.mod
18
go.mod
@@ -3,18 +3,14 @@ module github.com/jamesread/OliveTin
|
|||||||
go 1.15
|
go 1.15
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/golang/protobuf v1.5.2
|
github.com/grpc-ecosystem/grpc-gateway/v2 v2.4.0
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0
|
|
||||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.3.0
|
|
||||||
github.com/sirupsen/logrus v1.8.1
|
github.com/sirupsen/logrus v1.8.1
|
||||||
github.com/spf13/pflag v1.0.5 // indirect
|
|
||||||
github.com/spf13/viper v1.7.1
|
github.com/spf13/viper v1.7.1
|
||||||
github.com/stretchr/testify v1.7.0 // indirect
|
github.com/stretchr/testify v1.7.0
|
||||||
golang.org/x/text v0.3.5 // indirect
|
golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4 // indirect
|
||||||
golang.org/x/tools v0.1.1 // indirect
|
golang.org/x/sys v0.0.0-20210510120138-977fb7262007 // indirect
|
||||||
golang.org/x/xerrors v0.0.0-20200804184101-5ec99f83aff1 // indirect
|
google.golang.org/genproto v0.0.0-20210520160233-290a1ae68a05
|
||||||
google.golang.org/genproto v0.0.0-20210224155714-063164c882e6
|
|
||||||
google.golang.org/grpc v1.37.0
|
google.golang.org/grpc v1.37.0
|
||||||
gopkg.in/yaml.v2 v2.3.0 // indirect
|
google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.0.0
|
||||||
gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect
|
google.golang.org/protobuf v1.26.0
|
||||||
)
|
)
|
||||||
|
|||||||
@@ -1,10 +1,8 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import ()
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"strings"
|
|
||||||
)
|
|
||||||
|
|
||||||
|
// ActionButton represents a button that is shown in the webui.
|
||||||
type ActionButton struct {
|
type ActionButton struct {
|
||||||
Title string
|
Title string
|
||||||
Icon string
|
Icon string
|
||||||
@@ -13,6 +11,8 @@ type ActionButton struct {
|
|||||||
Timeout int
|
Timeout int
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Entity represents a "thing" that can have multiple actions associated with it.
|
||||||
|
// for example, a media player with a start and stop action.
|
||||||
type Entity struct {
|
type Entity struct {
|
||||||
Title string
|
Title string
|
||||||
Icon string
|
Icon string
|
||||||
@@ -20,34 +20,28 @@ type Entity struct {
|
|||||||
CSS map[string]string
|
CSS map[string]string
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Config is the global config used through the whole app.
|
||||||
type Config struct {
|
type Config struct {
|
||||||
|
UseSingleHTTPFrontend bool
|
||||||
|
ListenAddressSingleHTTPFrontend string
|
||||||
ListenAddressWebUI string
|
ListenAddressWebUI string
|
||||||
ListenAddressRestActions string
|
ListenAddressRestActions string
|
||||||
ListenAddressGrpcActions string
|
ListenAddressGrpcActions string
|
||||||
|
ExternalRestAddress string
|
||||||
LogLevel string
|
LogLevel string
|
||||||
ActionButtons []ActionButton `mapstructure:"actions"`
|
ActionButtons []ActionButton `mapstructure:"actions"`
|
||||||
Entities []Entity `mapstructure:"omitempty"`
|
Entities []Entity `mapstructure:"omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// DefaultConfig gets a new Config structure with sensible default values.
|
||||||
func DefaultConfig() *Config {
|
func DefaultConfig() *Config {
|
||||||
config := Config{}
|
config := Config{}
|
||||||
config.ListenAddressWebUI = "0.0.0.0:1337"
|
config.UseSingleHTTPFrontend = true
|
||||||
config.ListenAddressRestActions = "0.0.0.0:1338"
|
config.ListenAddressSingleHTTPFrontend = "0.0.0.0:1337"
|
||||||
config.ListenAddressGrpcActions = "0.0.0.0:1339"
|
config.ListenAddressRestActions = "localhost:1338"
|
||||||
|
config.ListenAddressGrpcActions = "localhost:1339"
|
||||||
|
config.ListenAddressWebUI = "localhost:1340"
|
||||||
config.LogLevel = "INFO"
|
config.LogLevel = "INFO"
|
||||||
|
|
||||||
return &config
|
return &config
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) GetLogLevel() log.Level {
|
|
||||||
switch strings.ToUpper(cfg.LogLevel) {
|
|
||||||
case "INFO":
|
|
||||||
return log.InfoLevel
|
|
||||||
case "WARN":
|
|
||||||
return log.WarnLevel
|
|
||||||
case "DEBUG":
|
|
||||||
return log.DebugLevel
|
|
||||||
default:
|
|
||||||
return log.InfoLevel
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -1,27 +1,10 @@
|
|||||||
package config
|
package config
|
||||||
|
|
||||||
import (
|
import (
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"testing"
|
"testing"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TestGetLog(t *testing.T) {
|
|
||||||
c := Config{}
|
|
||||||
c.LogLevel = ""
|
|
||||||
|
|
||||||
assert.Equal(t, c.GetLogLevel(), log.InfoLevel, "Info log level should be default")
|
|
||||||
|
|
||||||
c.LogLevel = "INFO"
|
|
||||||
assert.Equal(t, c.GetLogLevel(), log.InfoLevel, "set info log level")
|
|
||||||
|
|
||||||
c.LogLevel = "WARN"
|
|
||||||
assert.Equal(t, c.GetLogLevel(), log.WarnLevel, "set warn log level")
|
|
||||||
|
|
||||||
c.LogLevel = "DEBUG"
|
|
||||||
assert.Equal(t, c.GetLogLevel(), log.DebugLevel, "set debug log level")
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestCreateDefaultConfig(t *testing.T) {
|
func TestCreateDefaultConfig(t *testing.T) {
|
||||||
c := DefaultConfig()
|
c := DefaultConfig()
|
||||||
|
|
||||||
|
|||||||
@@ -11,6 +11,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// ExecAction executes an action.
|
||||||
func ExecAction(cfg *config.Config, action string) *pb.StartActionResponse {
|
func ExecAction(cfg *config.Config, action string) *pb.StartActionResponse {
|
||||||
res := &pb.StartActionResponse{}
|
res := &pb.StartActionResponse{}
|
||||||
res.TimedOut = false
|
res.TimedOut = false
|
||||||
|
|||||||
@@ -16,6 +16,7 @@ var (
|
|||||||
)
|
)
|
||||||
|
|
||||||
type oliveTinAPI struct {
|
type oliveTinAPI struct {
|
||||||
|
pb.UnimplementedOliveTinApiServer
|
||||||
}
|
}
|
||||||
|
|
||||||
func (api *oliveTinAPI) StartAction(ctx ctx.Context, req *pb.StartActionRequest) (*pb.StartActionResponse, error) {
|
func (api *oliveTinAPI) StartAction(ctx ctx.Context, req *pb.StartActionRequest) (*pb.StartActionResponse, error) {
|
||||||
@@ -39,10 +40,11 @@ func (api *oliveTinAPI) GetButtons(ctx ctx.Context, req *pb.GetButtonsRequest) (
|
|||||||
return res, nil
|
return res, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func Start(listenAddress string, globalConfig *config.Config) {
|
// Start will start the GRPC API.
|
||||||
|
func Start(globalConfig *config.Config) {
|
||||||
cfg = globalConfig
|
cfg = globalConfig
|
||||||
|
|
||||||
lis, err := net.Listen("tcp", listenAddress)
|
lis, err := net.Listen("tcp", cfg.ListenAddressGrpcActions)
|
||||||
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Fatalf("Failed to listen - %v", err)
|
log.Fatalf("Failed to listen - %v", err)
|
||||||
|
|||||||
@@ -3,17 +3,17 @@ package grpcapi
|
|||||||
// Thank you: https://stackoverflow.com/questions/42102496/testing-a-grpc-service
|
// Thank you: https://stackoverflow.com/questions/42102496/testing-a-grpc-service
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"net"
|
|
||||||
"context"
|
"context"
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
"testing"
|
|
||||||
"google.golang.org/grpc/test/bufconn"
|
|
||||||
"google.golang.org/grpc"
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/grpc/test/bufconn"
|
||||||
|
"net"
|
||||||
|
"testing"
|
||||||
|
|
||||||
log "github.com/sirupsen/logrus"
|
log "github.com/sirupsen/logrus"
|
||||||
|
|
||||||
config "github.com/jamesread/OliveTin/internal/config"
|
|
||||||
pb "github.com/jamesread/OliveTin/gen/grpc"
|
pb "github.com/jamesread/OliveTin/gen/grpc"
|
||||||
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
)
|
)
|
||||||
|
|
||||||
const bufSize = 1024 * 1024
|
const bufSize = 1024 * 1024
|
||||||
@@ -53,11 +53,11 @@ func getNewTestServerAndClient(t *testing.T, injectedConfig *config.Config) (*gr
|
|||||||
}
|
}
|
||||||
|
|
||||||
func TestGetButtonsAndStart(t *testing.T) {
|
func TestGetButtonsAndStart(t *testing.T) {
|
||||||
cfg = config.DefaultConfig();
|
cfg = config.DefaultConfig()
|
||||||
btn1 := config.ActionButton{}
|
btn1 := config.ActionButton{}
|
||||||
btn1.Title = "blat"
|
btn1.Title = "blat"
|
||||||
btn1.Shell = "echo 'test'"
|
btn1.Shell = "echo 'test'"
|
||||||
cfg.ActionButtons = append(cfg.ActionButtons, btn1);
|
cfg.ActionButtons = append(cfg.ActionButtons, btn1)
|
||||||
|
|
||||||
conn, client := getNewTestServerAndClient(t, cfg)
|
conn, client := getNewTestServerAndClient(t, cfg)
|
||||||
|
|
||||||
|
|||||||
16
internal/httpservers/httpServer.go
Normal file
16
internal/httpservers/httpServer.go
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
package httpservers
|
||||||
|
|
||||||
|
import (
|
||||||
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
// StartServers will start 3 HTTP servers. The WebUI, the Rest API, and a proxy
|
||||||
|
// for both of them.
|
||||||
|
func StartServers(cfg *config.Config) {
|
||||||
|
go startWebUIServer(cfg)
|
||||||
|
go startRestAPIServer(cfg)
|
||||||
|
|
||||||
|
if cfg.UseSingleHTTPFrontend {
|
||||||
|
StartSingleHTTPFrontend(cfg)
|
||||||
|
}
|
||||||
|
}
|
||||||
54
internal/httpservers/restapi.go
Normal file
54
internal/httpservers/restapi.go
Normal file
@@ -0,0 +1,54 @@
|
|||||||
|
package httpservers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"context"
|
||||||
|
"github.com/grpc-ecosystem/grpc-gateway/v2/runtime"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"google.golang.org/grpc"
|
||||||
|
"google.golang.org/protobuf/encoding/protojson"
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
gw "github.com/jamesread/OliveTin/gen/grpc"
|
||||||
|
|
||||||
|
cors "github.com/jamesread/OliveTin/internal/cors"
|
||||||
|
|
||||||
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
cfg *config.Config
|
||||||
|
)
|
||||||
|
|
||||||
|
func startRestAPIServer(globalConfig *config.Config) error {
|
||||||
|
cfg = globalConfig
|
||||||
|
|
||||||
|
log.WithFields(log.Fields{
|
||||||
|
"address": cfg.ListenAddressGrpcActions,
|
||||||
|
}).Info("Starting REST API")
|
||||||
|
|
||||||
|
ctx := context.Background()
|
||||||
|
ctx, cancel := context.WithCancel(ctx)
|
||||||
|
defer cancel()
|
||||||
|
|
||||||
|
// The JSONPb.EmitDefaults is necssary, so "empty" fields are returned in JSON.
|
||||||
|
//mux := runtime.NewServeMux(runtime.WithMarshalerOption(runtime.MIMEWildcard, &runtime.JSONPb{OrigName: true, EmitDefaults: true}))
|
||||||
|
mux := runtime.NewServeMux(
|
||||||
|
runtime.WithMarshalerOption(runtime.MIMEWildcard, &runtime.HTTPBodyMarshaler{
|
||||||
|
Marshaler: &runtime.JSONPb{
|
||||||
|
MarshalOptions: protojson.MarshalOptions{
|
||||||
|
UseProtoNames: true,
|
||||||
|
EmitUnpopulated: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
opts := []grpc.DialOption{grpc.WithInsecure()}
|
||||||
|
|
||||||
|
err := gw.RegisterOliveTinApiHandlerFromEndpoint(ctx, mux, cfg.ListenAddressGrpcActions, opts)
|
||||||
|
|
||||||
|
if err != nil {
|
||||||
|
log.Fatalf("gw error %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return http.ListenAndServe(cfg.ListenAddressRestActions, cors.AllowCors(mux))
|
||||||
|
}
|
||||||
50
internal/httpservers/singleFrontend.go
Normal file
50
internal/httpservers/singleFrontend.go
Normal file
@@ -0,0 +1,50 @@
|
|||||||
|
package httpservers
|
||||||
|
|
||||||
|
/*
|
||||||
|
This file implements a very simple, lightweight reverse proxy so that REST and
|
||||||
|
the webui can be accessed from a single endpoint.
|
||||||
|
|
||||||
|
This makes external reverse proxies (treafik, haproxy, etc) easier, CORS goes
|
||||||
|
away, and several other issues.
|
||||||
|
*/
|
||||||
|
|
||||||
|
import (
|
||||||
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httputil"
|
||||||
|
"net/url"
|
||||||
|
)
|
||||||
|
|
||||||
|
// StartSingleHTTPFrontend will create a reverse proxy that proxies the API
|
||||||
|
// and webui internally.
|
||||||
|
func StartSingleHTTPFrontend(cfg *config.Config) {
|
||||||
|
log.WithFields(log.Fields{
|
||||||
|
"address": cfg.ListenAddressSingleHTTPFrontend,
|
||||||
|
}).Info("Starting single HTTP frontend")
|
||||||
|
|
||||||
|
apiURL, _ := url.Parse("http://" + cfg.ListenAddressRestActions)
|
||||||
|
apiProxy := httputil.NewSingleHostReverseProxy(apiURL)
|
||||||
|
|
||||||
|
webuiURL, _ := url.Parse("http://" + cfg.ListenAddressWebUI)
|
||||||
|
webuiProxy := httputil.NewSingleHostReverseProxy(webuiURL)
|
||||||
|
|
||||||
|
mux := http.NewServeMux()
|
||||||
|
|
||||||
|
mux.HandleFunc("/api/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
log.Debugf("api req: %v", r.URL)
|
||||||
|
apiProxy.ServeHTTP(w, r)
|
||||||
|
})
|
||||||
|
|
||||||
|
mux.HandleFunc("/", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
log.Debugf("ui req: %v", r.URL)
|
||||||
|
webuiProxy.ServeHTTP(w, r)
|
||||||
|
})
|
||||||
|
|
||||||
|
srv := &http.Server{
|
||||||
|
Addr: cfg.ListenAddressSingleHTTPFrontend,
|
||||||
|
Handler: mux,
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Fatal(srv.ListenAndServe())
|
||||||
|
}
|
||||||
63
internal/httpservers/webuiServer.go
Normal file
63
internal/httpservers/webuiServer.go
Normal file
@@ -0,0 +1,63 @@
|
|||||||
|
package httpservers
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
// cors "github.com/jamesread/OliveTin/internal/cors"
|
||||||
|
log "github.com/sirupsen/logrus"
|
||||||
|
"net/http"
|
||||||
|
"os"
|
||||||
|
|
||||||
|
config "github.com/jamesread/OliveTin/internal/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
type webUISettings struct {
|
||||||
|
Rest string
|
||||||
|
}
|
||||||
|
|
||||||
|
func findWebuiDir() string {
|
||||||
|
directoriesToSearch := []string{
|
||||||
|
"./webui",
|
||||||
|
"/var/www/olivetin/",
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, dir := range directoriesToSearch {
|
||||||
|
if _, err := os.Stat(dir); !os.IsNotExist(err) {
|
||||||
|
log.Infof("Found the webui directory here: %v", dir)
|
||||||
|
|
||||||
|
return dir
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Warnf("Did not find the webui directory, you will probably get 404 errors.")
|
||||||
|
|
||||||
|
return "./webui" // Should not exist
|
||||||
|
}
|
||||||
|
|
||||||
|
func startWebUIServer(cfg *config.Config) {
|
||||||
|
log.WithFields(log.Fields{
|
||||||
|
"address": cfg.ListenAddressWebUI,
|
||||||
|
}).Info("Starting WebUI server")
|
||||||
|
|
||||||
|
mux := http.NewServeMux()
|
||||||
|
mux.Handle("/", http.FileServer(http.Dir(findWebuiDir())))
|
||||||
|
mux.HandleFunc("/webUiSettings.json", func(w http.ResponseWriter, r *http.Request) {
|
||||||
|
restAddress := ""
|
||||||
|
|
||||||
|
if !cfg.UseSingleHTTPFrontend {
|
||||||
|
restAddress = cfg.ExternalRestAddress
|
||||||
|
}
|
||||||
|
|
||||||
|
jsonRet, _ := json.Marshal(webUISettings{
|
||||||
|
Rest: restAddress + "/api/",
|
||||||
|
})
|
||||||
|
|
||||||
|
w.Write([]byte(jsonRet))
|
||||||
|
})
|
||||||
|
|
||||||
|
srv := &http.Server{
|
||||||
|
Addr: cfg.ListenAddressWebUI,
|
||||||
|
Handler: mux,
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Fatal(srv.ListenAndServe())
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
package webuiServer
|
package httpservers
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"github.com/stretchr/testify/assert"
|
"github.com/stretchr/testify/assert"
|
||||||
@@ -1,39 +0,0 @@
|
|||||||
package restapi
|
|
||||||
|
|
||||||
import (
|
|
||||||
"context"
|
|
||||||
"github.com/grpc-ecosystem/grpc-gateway/runtime"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"google.golang.org/grpc"
|
|
||||||
"net/http"
|
|
||||||
|
|
||||||
gw "github.com/jamesread/OliveTin/gen/grpc"
|
|
||||||
|
|
||||||
cors "github.com/jamesread/OliveTin/internal/cors"
|
|
||||||
|
|
||||||
config "github.com/jamesread/OliveTin/internal/config"
|
|
||||||
)
|
|
||||||
|
|
||||||
var (
|
|
||||||
cfg *config.Config
|
|
||||||
)
|
|
||||||
|
|
||||||
func Start(listenAddressRest string, listenAddressGrpc string, globalConfig *config.Config) error {
|
|
||||||
cfg = globalConfig
|
|
||||||
|
|
||||||
ctx := context.Background()
|
|
||||||
ctx, cancel := context.WithCancel(ctx)
|
|
||||||
defer cancel()
|
|
||||||
|
|
||||||
// The JSONPb.EmitDefaults is necssary, so "empty" fields are returned in JSON.
|
|
||||||
mux := runtime.NewServeMux(runtime.WithMarshalerOption(runtime.MIMEWildcard, &runtime.JSONPb{OrigName: true, EmitDefaults: true}))
|
|
||||||
opts := []grpc.DialOption{grpc.WithInsecure()}
|
|
||||||
|
|
||||||
err := gw.RegisterOliveTinApiHandlerFromEndpoint(ctx, mux, listenAddressGrpc, opts)
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
log.Fatalf("gw error %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return http.ListenAndServe(listenAddressRest, cors.AllowCors(mux))
|
|
||||||
}
|
|
||||||
@@ -1,48 +0,0 @@
|
|||||||
package webuiServer
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
cors "github.com/jamesread/OliveTin/internal/cors"
|
|
||||||
log "github.com/sirupsen/logrus"
|
|
||||||
"net/http"
|
|
||||||
"os"
|
|
||||||
)
|
|
||||||
|
|
||||||
type WebUISettings struct {
|
|
||||||
Rest string
|
|
||||||
}
|
|
||||||
|
|
||||||
func findWebuiDir() string {
|
|
||||||
directoriesToSearch := []string{
|
|
||||||
"./webui",
|
|
||||||
"/var/www/olivetin/",
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, dir := range directoriesToSearch {
|
|
||||||
if _, err := os.Stat(dir); !os.IsNotExist(err) {
|
|
||||||
log.Infof("Found the webui directory here: %v", dir)
|
|
||||||
|
|
||||||
return dir
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
log.Warnf("Did not find the webui directory, you will probably get 404 errors.")
|
|
||||||
|
|
||||||
return "./webui" // Should not exist
|
|
||||||
}
|
|
||||||
|
|
||||||
func Start(listenAddress string, listenAddressRest string) {
|
|
||||||
http.Handle("/", cors.AllowCors(http.FileServer(http.Dir(findWebuiDir()))))
|
|
||||||
|
|
||||||
http.HandleFunc("/webUiSettings.json", func(w http.ResponseWriter, r *http.Request) {
|
|
||||||
ret := WebUISettings{
|
|
||||||
Rest: "http://" + listenAddressRest + "/",
|
|
||||||
}
|
|
||||||
|
|
||||||
jsonRet, _ := json.Marshal(ret)
|
|
||||||
|
|
||||||
w.Write([]byte(jsonRet))
|
|
||||||
})
|
|
||||||
|
|
||||||
log.Fatal(http.ListenAndServe(listenAddress, nil))
|
|
||||||
}
|
|
||||||
Reference in New Issue
Block a user