* add devops client interface Signed-off-by: runzexia <runzexia@yunify.com> * direct return jenkins Signed-off-by: runzexia <runzexia@yunify.com> * add some interface Signed-off-by: runzexia <runzexia@yunify.com> * update Signed-off-by: runzexia <runzexia@yunify.com> * update interface Signed-off-by: runzexia <runzexia@yunify.com> * update Signed-off-by: runzexia <runzexia@yunify.com> * credential op structs Signed-off-by: runzexia <runzexia@yunify.com> * status Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update interface Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * credential handler Signed-off-by: runzexia <runzexia@yunify.com> * update devopsoperator func Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * get build sonar Signed-off-by: runzexia <runzexia@yunify.com> * sonar handler * mv code to cilent Signed-off-by: runzexia <runzexia@yunify.com> * update Signed-off-by: runzexia <runzexia@yunify.com> * project member handler Signed-off-by: runzexia <runzexia@yunify.com> * update pipeline operator interface Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add tenant devops handler Signed-off-by: runzexia <runzexia@yunify.com> * update merge Signed-off-by: runzexia <runzexia@yunify.com> * clean Signed-off-by: runzexia <runzexia@yunify.com> * fmt Signed-off-by: runzexia <runzexia@yunify.com> * update ListPipelineRuns Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * complate pipelineOperator interface Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update HttpParameters Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add pipeline steps interface Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update pipeline GetNodesDetail Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add s2i api Signed-off-by: runzexia <runzexia@yunify.com> * add branch pipeline interface and update handler Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add scan branch interface and update handler Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add common interface and update handler Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add SCM interface and update handler Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add handler Signed-off-by: runzexia <runzexia@yunify.com> * add fake s3 Signed-off-by: runzexia <runzexia@yunify.com> * add webhook&check interface and update handler Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * clean Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * clean Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * format Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add some func Signed-off-by: runzexia <runzexia@yunify.com> * clean code Signed-off-by: runzexia <runzexia@yunify.com> * implement interface Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * fix interface GetBranchArtifacts Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add s2ibinary upload test Signed-off-by: runzexia <runzexia@yunify.com> * tenant devops Signed-off-by: runzexia <runzexia@yunify.com> * update tenant Signed-off-by: runzexia <runzexia@yunify.com> * fake Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add some unit test Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * add devops tenant handler Signed-off-by: runzexia <runzexia@yunify.com> * status Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * status Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * status Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update fake test Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update unit test and fake data Signed-off-by: zhuxiaoyang <sunzhu@yunify.com> * update Co-authored-by: Xiaoyang Zhu <sunzhu@yunify.com>
187 lines
5.3 KiB
Go
187 lines
5.3 KiB
Go
package devops
|
|
|
|
import (
|
|
"code.cloudfoundry.org/bytefmt"
|
|
"fmt"
|
|
"github.com/aws/aws-sdk-go/aws/awserr"
|
|
awsS3 "github.com/aws/aws-sdk-go/service/s3"
|
|
"github.com/emicklei/go-restful"
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
"k8s.io/client-go/util/retry"
|
|
"k8s.io/klog"
|
|
"kubesphere.io/kubesphere/pkg/apis/devops/v1alpha1"
|
|
"kubesphere.io/kubesphere/pkg/client/clientset/versioned"
|
|
"kubesphere.io/kubesphere/pkg/client/informers/externalversions"
|
|
"kubesphere.io/kubesphere/pkg/simple/client/s3"
|
|
"mime/multipart"
|
|
"net/http"
|
|
"reflect"
|
|
)
|
|
|
|
const (
|
|
GetS2iBinaryURL = "http://ks-apiserver.kubesphere-system.svc/kapis/devops.kubesphere.io/v1alpha2/namespaces/%s/s2ibinaries/%s/file/%s"
|
|
)
|
|
|
|
type S2iBinaryUploader interface {
|
|
UploadS2iBinary(namespace, name, md5 string, header *multipart.FileHeader) (*v1alpha1.S2iBinary, error)
|
|
|
|
DownloadS2iBinary(namespace, name, fileName string) (string, error)
|
|
}
|
|
|
|
type s2iBinaryUploader struct {
|
|
client versioned.Interface
|
|
informers externalversions.SharedInformerFactory
|
|
s3Client s3.Interface
|
|
}
|
|
|
|
func NewS2iBinaryUploader(client versioned.Interface, informers externalversions.SharedInformerFactory, s3Client s3.Interface) S2iBinaryUploader {
|
|
return &s2iBinaryUploader{
|
|
client: client,
|
|
informers: informers,
|
|
s3Client: s3Client,
|
|
}
|
|
}
|
|
|
|
func (s *s2iBinaryUploader) UploadS2iBinary(namespace, name, md5 string, fileHeader *multipart.FileHeader) (*v1alpha1.S2iBinary, error) {
|
|
binFile, err := fileHeader.Open()
|
|
if err != nil {
|
|
klog.Errorf("%+v", err)
|
|
return nil, err
|
|
}
|
|
defer binFile.Close()
|
|
|
|
origin, err := s.informers.Devops().V1alpha1().S2iBinaries().Lister().S2iBinaries(namespace).Get(name)
|
|
if err != nil {
|
|
klog.Errorf("%+v", err)
|
|
return nil, err
|
|
}
|
|
//Check file is uploading
|
|
if origin.Status.Phase == v1alpha1.StatusUploading {
|
|
err := restful.NewError(http.StatusConflict, "file is uploading, please try later")
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
copy := origin.DeepCopy()
|
|
copy.Spec.MD5 = md5
|
|
copy.Spec.Size = bytefmt.ByteSize(uint64(fileHeader.Size))
|
|
copy.Spec.FileName = fileHeader.Filename
|
|
copy.Spec.DownloadURL = fmt.Sprintf(GetS2iBinaryURL, namespace, name, copy.Spec.FileName)
|
|
if origin.Status.Phase == v1alpha1.StatusReady && reflect.DeepEqual(origin, copy) {
|
|
return origin, nil
|
|
}
|
|
|
|
//Set status Uploading to lock resource
|
|
uploading, err := s.SetS2iBinaryStatus(copy, v1alpha1.StatusUploading)
|
|
if err != nil {
|
|
err := restful.NewError(http.StatusConflict, fmt.Sprintf("could not set status: %+v", err))
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
|
|
copy = uploading.DeepCopy()
|
|
copy.Spec.MD5 = md5
|
|
copy.Spec.Size = bytefmt.ByteSize(uint64(fileHeader.Size))
|
|
copy.Spec.FileName = fileHeader.Filename
|
|
copy.Spec.DownloadURL = fmt.Sprintf(GetS2iBinaryURL, namespace, name, copy.Spec.FileName)
|
|
|
|
err = s.s3Client.Upload(fmt.Sprintf("%s-%s", namespace, name), copy.Spec.FileName, binFile)
|
|
if err != nil {
|
|
if aerr, ok := err.(awserr.Error); ok {
|
|
switch aerr.Code() {
|
|
case awsS3.ErrCodeNoSuchBucket:
|
|
klog.Error(err)
|
|
_, serr := s.SetS2iBinaryStatusWithRetry(copy, origin.Status.Phase)
|
|
if serr != nil {
|
|
klog.Error(serr)
|
|
}
|
|
return nil, err
|
|
default:
|
|
klog.Error(err)
|
|
_, serr := s.SetS2iBinaryStatusWithRetry(copy, v1alpha1.StatusUploadFailed)
|
|
if serr != nil {
|
|
klog.Error(serr)
|
|
}
|
|
return nil, err
|
|
}
|
|
}
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
|
|
if copy.Spec.UploadTimeStamp == nil {
|
|
copy.Spec.UploadTimeStamp = new(metav1.Time)
|
|
}
|
|
*copy.Spec.UploadTimeStamp = metav1.Now()
|
|
copy, err = s.client.DevopsV1alpha1().S2iBinaries(namespace).Update(copy)
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
|
|
copy, err = s.SetS2iBinaryStatusWithRetry(copy, v1alpha1.StatusReady)
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
return copy, nil
|
|
}
|
|
|
|
func (s *s2iBinaryUploader) DownloadS2iBinary(namespace, name, fileName string) (string, error) {
|
|
|
|
origin, err := s.informers.Devops().V1alpha1().S2iBinaries().Lister().S2iBinaries(namespace).Get(name)
|
|
if err != nil {
|
|
klog.Errorf("%+v", err)
|
|
return "", err
|
|
}
|
|
|
|
if origin.Spec.FileName != fileName {
|
|
err := fmt.Errorf("could not fould file %s", fileName)
|
|
klog.Error(err)
|
|
return "", err
|
|
}
|
|
|
|
if origin.Status.Phase != v1alpha1.StatusReady {
|
|
err := restful.NewError(http.StatusBadRequest, "file is not ready, please try later")
|
|
klog.Error(err)
|
|
return "", err
|
|
}
|
|
return s.s3Client.GetDownloadURL(fmt.Sprintf("%s-%s", namespace, name), fileName)
|
|
}
|
|
|
|
func (s *s2iBinaryUploader) SetS2iBinaryStatus(s2ibin *v1alpha1.S2iBinary, status string) (*v1alpha1.S2iBinary, error) {
|
|
copy := s2ibin.DeepCopy()
|
|
copy.Status.Phase = status
|
|
copy, err := s.client.DevopsV1alpha1().S2iBinaries(s2ibin.Namespace).Update(copy)
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
return copy, nil
|
|
}
|
|
|
|
func (s *s2iBinaryUploader) SetS2iBinaryStatusWithRetry(s2ibin *v1alpha1.S2iBinary, status string) (*v1alpha1.S2iBinary, error) {
|
|
|
|
var bin *v1alpha1.S2iBinary
|
|
var err error
|
|
err = retry.RetryOnConflict(retry.DefaultRetry, func() error {
|
|
bin, err = s.client.DevopsV1alpha1().S2iBinaries(s2ibin.Namespace).Get(s2ibin.Name, metav1.GetOptions{})
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return err
|
|
}
|
|
bin.Status.Phase = status
|
|
bin, err = s.client.DevopsV1alpha1().S2iBinaries(s2ibin.Namespace).Update(bin)
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return err
|
|
}
|
|
return nil
|
|
})
|
|
if err != nil {
|
|
klog.Error(err)
|
|
return nil, err
|
|
}
|
|
|
|
return bin, nil
|
|
}
|