* Copyright (c) 2024 Huawei Technologies Co., Ltd.
* openFuyao is licensed under Mulan PSL v2.
* You can use this software according to the terms and conditions of the Mulan PSL v2.
* You may obtain a copy of Mulan PSL v2 at:
* http://license.coscl.org.cn/MulanPSL2
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
* See the Mulan PSL v2 for more details.
*/
package server
import (
"context"
"crypto/tls"
"fmt"
"net/http"
"github.com/emicklei/go-restful/v3"
"k8s.io/apimachinery/pkg/runtime/schema"
urlruntime "k8s.io/apimachinery/pkg/util/runtime"
"k8s.io/apimachinery/pkg/util/sets"
k8srequest "k8s.io/apiserver/pkg/endpoints/request"
"volcano-config-service/pkg/client/k8s"
"volcano-config-service/pkg/cmd/config"
"volcano-config-service/pkg/constant"
"volcano-config-service/pkg/server/filters"
"volcano-config-service/pkg/server/request"
"volcano-config-service/pkg/volcano"
)
type CServer struct {
Server *http.Server
container *restful.Container
KubernetesClient k8s.BaseClient
}
func NewServer(cfg *config.RunConfig, ctx context.Context) (*CServer, error) {
server := &CServer{}
httpServer, err := initServer(cfg)
if err != nil {
return nil, err
}
server.Server = httpServer
server.container = restful.NewContainer()
server.container.Router(restful.CurlyRouter{})
kubernetesClient, err := k8s.NewKubernetesClient(cfg.KubernetesCfg)
if err != nil {
return nil, err
}
server.KubernetesClient = kubernetesClient
return server, nil
}
func initServer(cfg *config.RunConfig) (*http.Server, error) {
httpServer := &http.Server{
Addr: fmt.Sprintf(":%d", cfg.Server.InsecurePort),
}
if cfg.Server.SecurePort != 0 {
certificate, err := tls.LoadX509KeyPair(cfg.Server.CertFile, cfg.Server.PrivateKey)
if err != nil {
return nil, err
}
httpServer.TLSConfig = &tls.Config{
Certificates: []tls.Certificate{certificate},
}
httpServer.Addr = fmt.Sprintf(":%d", cfg.Server.SecurePort)
}
return httpServer, nil
}
func (s *CServer) Run(ctx context.Context) error {
s.registerAPI()
s.Server.Handler = s.container
s.buildHandlerChain()
shutdownCtx, cancel := context.WithCancel(context.Background())
defer cancel()
go func() {
<-ctx.Done()
err := s.Server.Shutdown(shutdownCtx)
if err != nil {
return
}
}()
if s.Server.TLSConfig != nil {
return s.Server.ListenAndServeTLS("", "")
} else {
return s.Server.ListenAndServe()
}
}
func (s *CServer) registerAPI() {
urlruntime.Must(volcano.ConfigVolcano(s.container, s.KubernetesClient.ConfigClient()))
}
const (
groupName = "resources.fuyao.io"
version = "v1alpha1"
)
var groupVersion = schema.GroupVersion{Group: groupName, Version: version}
func resource(resource string) schema.GroupResource {
return groupVersion.WithResource(resource).GroupResource()
}
func (s *CServer) buildHandlerChain() {
handler := s.Server.Handler
handler = filters.ProxyAPIServer(handler, s.KubernetesClient.ConfigClient())
handler = filters.DispatchCluster(handler)
requestInfoResolver := &request.RequestInfoFactory{
RequestInfoFactory: &k8srequest.RequestInfoFactory{
APIPrefixes: sets.NewString("api", "apis", "fuyao"),
GrouplessAPIPrefixes: sets.NewString("api"),
},
GlobalResources: []schema.GroupResource{
resource(constant.ResourcesPluralCluster),
},
}
handler = filters.BuildRequestInfo(handler, requestInfoResolver)
s.Server.Handler = handler
}