feat: kubesphere 4.0 (#6115)
* feat: kubesphere 4.0 Signed-off-by: ci-bot <ci-bot@kubesphere.io> * feat: kubesphere 4.0 Signed-off-by: ci-bot <ci-bot@kubesphere.io> --------- Signed-off-by: ci-bot <ci-bot@kubesphere.io> Co-authored-by: ks-ci-bot <ks-ci-bot@example.com> Co-authored-by: joyceliu <joyceliu@yunify.com>
This commit is contained in:
committed by
GitHub
parent
b5015ec7b9
commit
447a51f08b
@@ -1,105 +1,43 @@
|
||||
/*
|
||||
Copyright 2020 KubeSphere Authors
|
||||
|
||||
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.
|
||||
*/
|
||||
* Please refer to the LICENSE file in the root directory of the project.
|
||||
* https://github.com/kubesphere/kubesphere/blob/master/LICENSE
|
||||
*/
|
||||
|
||||
package v1alpha2
|
||||
|
||||
import (
|
||||
"fmt"
|
||||
"net/http"
|
||||
"strconv"
|
||||
"strings"
|
||||
|
||||
"github.com/emicklei/go-restful/v3"
|
||||
v1 "k8s.io/api/core/v1"
|
||||
k8serr "k8s.io/apimachinery/pkg/api/errors"
|
||||
"k8s.io/client-go/kubernetes"
|
||||
"k8s.io/klog/v2"
|
||||
|
||||
"kubesphere.io/kubesphere/pkg/api"
|
||||
"kubesphere.io/kubesphere/pkg/informers"
|
||||
"kubesphere.io/kubesphere/pkg/apiserver/query"
|
||||
"kubesphere.io/kubesphere/pkg/models/components"
|
||||
"kubesphere.io/kubesphere/pkg/models/git"
|
||||
"kubesphere.io/kubesphere/pkg/models/kubeconfig"
|
||||
"kubesphere.io/kubesphere/pkg/models/kubectl"
|
||||
"kubesphere.io/kubesphere/pkg/models/quotas"
|
||||
"kubesphere.io/kubesphere/pkg/models/registries"
|
||||
"kubesphere.io/kubesphere/pkg/models/resources/v1alpha2"
|
||||
"kubesphere.io/kubesphere/pkg/models/resources/v1alpha2/resource"
|
||||
resourcev1alpha3 "kubesphere.io/kubesphere/pkg/models/resources/v1alpha3/resource"
|
||||
"kubesphere.io/kubesphere/pkg/models/revisions"
|
||||
"kubesphere.io/kubesphere/pkg/models/routers"
|
||||
"kubesphere.io/kubesphere/pkg/server/errors"
|
||||
"kubesphere.io/kubesphere/pkg/server/params"
|
||||
)
|
||||
|
||||
type resourceHandler struct {
|
||||
resourcesGetter *resource.ResourceGetter
|
||||
componentsGetter components.ComponentsGetter
|
||||
type handler struct {
|
||||
componentsGetter components.Getter
|
||||
resourceQuotaGetter quotas.ResourceQuotaGetter
|
||||
revisionGetter revisions.RevisionGetter
|
||||
routerOperator routers.RouterOperator
|
||||
gitVerifier git.GitVerifier
|
||||
registryGetter registries.RegistryGetter
|
||||
kubeconfigOperator kubeconfig.Interface
|
||||
kubectlOperator kubectl.Interface
|
||||
resourceGetter *resourcev1alpha3.Getter
|
||||
}
|
||||
|
||||
func newResourceHandler(k8sClient kubernetes.Interface, factory informers.InformerFactory, masterURL string) *resourceHandler {
|
||||
|
||||
return &resourceHandler{
|
||||
resourcesGetter: resource.NewResourceGetter(factory),
|
||||
componentsGetter: components.NewComponentsGetter(factory.KubernetesSharedInformerFactory()),
|
||||
resourceQuotaGetter: quotas.NewResourceQuotaGetter(factory.KubernetesSharedInformerFactory()),
|
||||
revisionGetter: revisions.NewRevisionGetter(factory.KubernetesSharedInformerFactory()),
|
||||
routerOperator: routers.NewRouterOperator(k8sClient, factory.KubernetesSharedInformerFactory()),
|
||||
gitVerifier: git.NewGitVerifier(factory.KubernetesSharedInformerFactory()),
|
||||
registryGetter: registries.NewRegistryGetter(factory.KubernetesSharedInformerFactory()),
|
||||
kubeconfigOperator: kubeconfig.NewReadOnlyOperator(factory.KubernetesSharedInformerFactory().Core().V1().ConfigMaps().Lister(), masterURL),
|
||||
kubectlOperator: kubectl.NewOperator(nil, factory.KubernetesSharedInformerFactory().Apps().V1().Deployments(),
|
||||
factory.KubernetesSharedInformerFactory().Core().V1().Pods(),
|
||||
factory.KubeSphereSharedInformerFactory().Iam().V1alpha2().Users(), ""),
|
||||
}
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleListNamespaceResources(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
resource := request.PathParameter("resources")
|
||||
orderBy := params.GetStringValueWithDefault(request, params.OrderByParam, v1alpha2.CreateTime)
|
||||
limit, offset := params.ParsePaging(request)
|
||||
reverse := params.GetBoolValueWithDefault(request, params.ReverseParam, false)
|
||||
conditions, err := params.ParseConditions(request)
|
||||
|
||||
if err != nil {
|
||||
klog.Error(err)
|
||||
api.HandleBadRequest(response, request, err)
|
||||
return
|
||||
}
|
||||
|
||||
result, err := r.resourcesGetter.ListResources(namespace, resource, conditions, orderBy, reverse, limit, offset)
|
||||
|
||||
if err != nil {
|
||||
klog.Error(err)
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteEntity(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetSystemHealthStatus(_ *restful.Request, response *restful.Response) {
|
||||
result, err := r.componentsGetter.GetSystemHealthStatus()
|
||||
func (h *handler) GetSystemHealthStatus(_ *restful.Request, response *restful.Response) {
|
||||
result, err := h.componentsGetter.GetSystemHealthStatus()
|
||||
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
@@ -109,9 +47,9 @@ func (r *resourceHandler) handleGetSystemHealthStatus(_ *restful.Request, respon
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetComponentStatus(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetComponentStatus(request *restful.Request, response *restful.Response) {
|
||||
component := request.PathParameter("component")
|
||||
result, err := r.componentsGetter.GetComponentStatus(component)
|
||||
result, err := h.componentsGetter.GetComponentStatus(component)
|
||||
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
@@ -121,8 +59,8 @@ func (r *resourceHandler) handleGetComponentStatus(request *restful.Request, res
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetComponents(_ *restful.Request, response *restful.Response) {
|
||||
result, err := r.componentsGetter.GetAllComponentsStatus()
|
||||
func (h *handler) GetComponents(_ *restful.Request, response *restful.Response) {
|
||||
result, err := h.componentsGetter.GetAllComponentsStatus()
|
||||
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
@@ -132,8 +70,8 @@ func (r *resourceHandler) handleGetComponents(_ *restful.Request, response *rest
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetClusterQuotas(_ *restful.Request, response *restful.Response) {
|
||||
result, err := r.resourceQuotaGetter.GetClusterQuota()
|
||||
func (h *handler) GetClusterQuotas(_ *restful.Request, response *restful.Response) {
|
||||
result, err := h.resourceQuotaGetter.GetClusterQuota()
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
@@ -142,9 +80,9 @@ func (r *resourceHandler) handleGetClusterQuotas(_ *restful.Request, response *r
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetNamespaceQuotas(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetNamespaceQuotas(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
quota, err := r.resourceQuotaGetter.GetNamespaceQuota(namespace)
|
||||
quota, err := h.resourceQuotaGetter.GetNamespaceQuota(namespace)
|
||||
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
@@ -154,7 +92,7 @@ func (r *resourceHandler) handleGetNamespaceQuotas(request *restful.Request, res
|
||||
response.WriteAsJson(quota)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetDaemonSetRevision(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetDaemonSetRevision(request *restful.Request, response *restful.Response) {
|
||||
daemonset := request.PathParameter("daemonset")
|
||||
namespace := request.PathParameter("namespace")
|
||||
revision, err := strconv.Atoi(request.PathParameter("revision"))
|
||||
@@ -164,7 +102,7 @@ func (r *resourceHandler) handleGetDaemonSetRevision(request *restful.Request, r
|
||||
return
|
||||
}
|
||||
|
||||
result, err := r.revisionGetter.GetDaemonSetRevision(namespace, daemonset, revision)
|
||||
result, err := h.revisionGetter.GetDaemonSetRevision(namespace, daemonset, revision)
|
||||
|
||||
if err != nil {
|
||||
response.WriteHeaderAndEntity(http.StatusInternalServerError, errors.Wrap(err))
|
||||
@@ -174,13 +112,12 @@ func (r *resourceHandler) handleGetDaemonSetRevision(request *restful.Request, r
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetDeploymentRevision(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetDeploymentRevision(request *restful.Request, response *restful.Response) {
|
||||
deploy := request.PathParameter("deployment")
|
||||
namespace := request.PathParameter("namespace")
|
||||
revision := request.PathParameter("revision")
|
||||
|
||||
result, err := r.revisionGetter.GetDeploymentRevision(namespace, deploy, revision)
|
||||
|
||||
result, err := h.revisionGetter.GetDeploymentRevision(namespace, deploy, revision)
|
||||
if err != nil {
|
||||
response.WriteHeaderAndEntity(http.StatusInternalServerError, errors.Wrap(err))
|
||||
return
|
||||
@@ -189,7 +126,7 @@ func (r *resourceHandler) handleGetDeploymentRevision(request *restful.Request,
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetStatefulSetRevision(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetStatefulSetRevision(request *restful.Request, response *restful.Response) {
|
||||
statefulset := request.PathParameter("statefulset")
|
||||
namespace := request.PathParameter("namespace")
|
||||
revision, err := strconv.Atoi(request.PathParameter("revision"))
|
||||
@@ -198,7 +135,7 @@ func (r *resourceHandler) handleGetStatefulSetRevision(request *restful.Request,
|
||||
return
|
||||
}
|
||||
|
||||
result, err := r.revisionGetter.GetStatefulSetRevision(namespace, statefulset, revision)
|
||||
result, err := h.revisionGetter.GetStatefulSetRevision(namespace, statefulset, revision)
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
@@ -206,83 +143,7 @@ func (r *resourceHandler) handleGetStatefulSetRevision(request *restful.Request,
|
||||
response.WriteAsJson(result)
|
||||
}
|
||||
|
||||
// Get ingress controller service for specified namespace
|
||||
func (r *resourceHandler) handleGetRouter(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
router, err := r.routerOperator.GetRouter(namespace)
|
||||
if err != nil {
|
||||
if k8serr.IsNotFound(err) {
|
||||
response.WriteHeaderAndEntity(http.StatusNotFound, errors.Wrap(err))
|
||||
} else {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteAsJson(router)
|
||||
}
|
||||
|
||||
// Create ingress controller and related services
|
||||
func (r *resourceHandler) handleCreateRouter(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
newRouter := api.Router{}
|
||||
err := request.ReadEntity(&newRouter)
|
||||
if err != nil {
|
||||
response.WriteHeaderAndEntity(http.StatusBadRequest, errors.Wrap(fmt.Errorf("wrong annotations, missing key or value")))
|
||||
return
|
||||
}
|
||||
|
||||
routerType := v1.ServiceTypeNodePort
|
||||
if strings.Compare(strings.ToLower(newRouter.RouterType), "loadbalancer") == 0 {
|
||||
routerType = v1.ServiceTypeLoadBalancer
|
||||
}
|
||||
|
||||
router, err := r.routerOperator.CreateRouter(namespace, routerType, newRouter.Annotations)
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteAsJson(router)
|
||||
}
|
||||
|
||||
// Delete ingress controller and services
|
||||
func (r *resourceHandler) handleDeleteRouter(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
|
||||
router, err := r.routerOperator.DeleteRouter(namespace)
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteAsJson(router)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleUpdateRouter(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
newRouter := api.Router{}
|
||||
err := request.ReadEntity(&newRouter)
|
||||
if err != nil {
|
||||
response.WriteHeaderAndEntity(http.StatusBadRequest, errors.Wrap(err))
|
||||
return
|
||||
}
|
||||
|
||||
var routerType = v1.ServiceTypeNodePort
|
||||
if strings.Compare(strings.ToLower(newRouter.RouterType), "loadbalancer") == 0 {
|
||||
routerType = v1.ServiceTypeLoadBalancer
|
||||
}
|
||||
router, err := r.routerOperator.UpdateRouter(namespace, routerType, newRouter.Annotations)
|
||||
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteAsJson(router)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleVerifyGitCredential(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) VerifyGitCredential(request *restful.Request, response *restful.Response) {
|
||||
var credential api.GitCredential
|
||||
err := request.ReadEntity(&credential)
|
||||
if err != nil {
|
||||
@@ -294,7 +155,7 @@ func (r *resourceHandler) handleVerifyGitCredential(request *restful.Request, re
|
||||
namespace = credential.SecretRef.Namespace
|
||||
secretName = credential.SecretRef.Name
|
||||
}
|
||||
err = r.gitVerifier.VerifyGitCredential(credential.RemoteUrl, namespace, secretName)
|
||||
err = h.gitVerifier.VerifyGitCredential(credential.RemoteUrl, namespace, secretName)
|
||||
if err != nil {
|
||||
response.WriteHeaderAndEntity(http.StatusInternalServerError, errors.Wrap(err))
|
||||
return
|
||||
@@ -302,7 +163,7 @@ func (r *resourceHandler) handleVerifyGitCredential(request *restful.Request, re
|
||||
response.WriteAsJson(errors.None)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleVerifyRegistryCredential(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) VerifyRegistryCredential(request *restful.Request, response *restful.Response) {
|
||||
var credential api.RegistryCredential
|
||||
err := request.ReadEntity(&credential)
|
||||
if err != nil {
|
||||
@@ -310,7 +171,7 @@ func (r *resourceHandler) handleVerifyRegistryCredential(request *restful.Reques
|
||||
return
|
||||
}
|
||||
|
||||
err = r.registryGetter.VerifyRegistryCredential(credential)
|
||||
err = h.registryGetter.VerifyRegistryCredential(credential)
|
||||
if err != nil {
|
||||
api.HandleBadRequest(response, nil, err)
|
||||
return
|
||||
@@ -319,13 +180,13 @@ func (r *resourceHandler) handleVerifyRegistryCredential(request *restful.Reques
|
||||
response.WriteHeader(http.StatusOK)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetRegistryEntry(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetRegistryEntry(request *restful.Request, response *restful.Response) {
|
||||
imageName := request.QueryParameter("image")
|
||||
namespace := request.QueryParameter("namespace")
|
||||
secretName := request.QueryParameter("secret")
|
||||
insecure := request.QueryParameter("insecure") == "true"
|
||||
|
||||
detail, err := r.registryGetter.GetEntry(namespace, secretName, imageName, insecure)
|
||||
detail, err := h.registryGetter.GetEntry(namespace, secretName, imageName, insecure)
|
||||
if err != nil {
|
||||
api.HandleBadRequest(response, nil, err)
|
||||
return
|
||||
@@ -334,7 +195,7 @@ func (r *resourceHandler) handleGetRegistryEntry(request *restful.Request, respo
|
||||
response.WriteAsJson(detail)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) handleGetNamespacedAbnormalWorkloads(request *restful.Request, response *restful.Response) {
|
||||
func (h *handler) GetNamespacedAbnormalWorkloads(request *restful.Request, response *restful.Response) {
|
||||
namespace := request.PathParameter("namespace")
|
||||
|
||||
result := api.Workloads{
|
||||
@@ -347,14 +208,17 @@ func (r *resourceHandler) handleGetNamespacedAbnormalWorkloads(request *restful.
|
||||
|
||||
switch workloadType {
|
||||
case api.ResourceKindPersistentVolumeClaim:
|
||||
notReadyStatus = strings.Join([]string{v1alpha2.StatusPending, v1alpha2.StatusLost}, "|")
|
||||
notReadyStatus = strings.Join([]string{"pending", "lost"}, "|")
|
||||
case api.ResourceKindJob:
|
||||
notReadyStatus = v1alpha2.StatusFailed
|
||||
notReadyStatus = "failed"
|
||||
default:
|
||||
notReadyStatus = v1alpha2.StatusUpdating
|
||||
notReadyStatus = "updating"
|
||||
}
|
||||
|
||||
res, err := r.resourcesGetter.ListResources(namespace, workloadType, ¶ms.Conditions{Match: map[string]string{v1alpha2.Status: notReadyStatus}}, "", false, -1, 0)
|
||||
q := query.New()
|
||||
q.Filters[query.FieldStatus] = query.Value(notReadyStatus)
|
||||
|
||||
res, err := h.resourceGetter.List(workloadType, namespace, q)
|
||||
if err != nil {
|
||||
api.HandleInternalError(response, nil, err)
|
||||
}
|
||||
@@ -363,27 +227,12 @@ func (r *resourceHandler) handleGetNamespacedAbnormalWorkloads(request *restful.
|
||||
}
|
||||
|
||||
response.WriteAsJson(result)
|
||||
|
||||
}
|
||||
|
||||
func (r *resourceHandler) GetKubectlPod(request *restful.Request, response *restful.Response) {
|
||||
user := request.PathParameter("user")
|
||||
func (h *handler) GetKubeconfig(request *restful.Request, response *restful.Response) {
|
||||
username := request.PathParameter("user")
|
||||
|
||||
kubectlPod, err := r.kubectlOperator.GetKubectlPod(user)
|
||||
|
||||
if err != nil {
|
||||
klog.Errorln(err)
|
||||
response.WriteHeaderAndEntity(http.StatusInternalServerError, errors.Wrap(err))
|
||||
return
|
||||
}
|
||||
|
||||
response.WriteEntity(kubectlPod)
|
||||
}
|
||||
|
||||
func (r *resourceHandler) GetKubeconfig(request *restful.Request, response *restful.Response) {
|
||||
user := request.PathParameter("user")
|
||||
|
||||
kubectlConfig, err := r.kubeconfigOperator.GetKubeConfig(user)
|
||||
kubectlConfig, err := h.kubeconfigOperator.GetKubeConfig(request.Request.Context(), username)
|
||||
|
||||
if err != nil {
|
||||
klog.Error(err)
|
||||
|
||||
Reference in New Issue
Block a user