kubekey/pkg/modules/copy.go
II 865913fea9
feat: add more golangci lint rule. (#2366)
Signed-off-by: joyceliu <joyceliu@yunify.com>
Co-authored-by: joyceliu <joyceliu@yunify.com>
2024-08-21 16:33:36 +08:00

290 lines
7.9 KiB
Go

/*
Copyright 2023 The 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.
*/
package modules
import (
"context"
"errors"
"fmt"
"io/fs"
"os"
"path/filepath"
"strings"
"k8s.io/apimachinery/pkg/runtime"
"k8s.io/klog/v2"
ctrlclient "sigs.k8s.io/controller-runtime/pkg/client"
"github.com/kubesphere/kubekey/v4/pkg/connector"
kkcorev1alpha1 "github.com/kubesphere/kubekey/v4/pkg/apis/core/v1alpha1"
"github.com/kubesphere/kubekey/v4/pkg/project"
"github.com/kubesphere/kubekey/v4/pkg/variable"
)
type copyArgs struct {
src string
content string
dest string
mode *int
}
func newCopyArgs(_ context.Context, raw runtime.RawExtension, vars map[string]any) (*copyArgs, error) {
var err error
ca := &copyArgs{}
args := variable.Extension2Variables(raw)
ca.src, _ = variable.StringVar(vars, args, "src")
ca.content, _ = variable.StringVar(vars, args, "content")
ca.dest, err = variable.StringVar(vars, args, "dest")
if err != nil {
return nil, errors.New("\"dest\" in args should be string")
}
ca.mode, _ = variable.IntVar(vars, args, "mode")
return ca, nil
}
// ModuleCopy deal "copy" module
func ModuleCopy(ctx context.Context, options ExecOptions) (string, string) {
// get host variable
ha, err := options.getAllVariables()
if err != nil {
return "", err.Error()
}
ca, err := newCopyArgs(ctx, options.Args, ha)
if err != nil {
klog.V(4).ErrorS(err, "get copy args error", "task", ctrlclient.ObjectKeyFromObject(&options.Task))
return "", err.Error()
}
// get connector
conn, err := getConnector(ctx, options.Host, ha)
if err != nil {
return "", fmt.Sprintf("get connector error: %v", err)
}
defer conn.Close(ctx)
switch {
case ca.src != "": // copy local file to remote
return ca.copySrc(ctx, options, conn)
case ca.content != "":
return ca.copyContent(ctx, os.ModePerm, conn)
default:
return "", "either \"src\" or \"content\" must be provided."
}
}
// copySrc copy src file to dest
func (ca copyArgs) copySrc(ctx context.Context, options ExecOptions, conn connector.Connector) (string, string) {
if filepath.IsAbs(ca.src) { // if src is absolute path. find it in local path
fileInfo, err := os.Stat(ca.src)
if err != nil {
return "", fmt.Sprintf(" get src file %s in local path error: %v", ca.src, err)
}
if fileInfo.IsDir() { // src is dir
if err := ca.absDir(ctx, conn); err != nil {
return "", fmt.Sprintf("sync copy absolute dir error %s", err)
}
} else { // src is file
if err := ca.absFile(ctx, fileInfo.Mode(), conn); err != nil {
return "", fmt.Sprintf("sync copy absolute dir error %s", err)
}
}
} else { // if src is not absolute path. find file in project
pj, err := project.New(ctx, options.Pipeline, false)
if err != nil {
return "", fmt.Sprintf("get project error: %v", err)
}
fileInfo, err := pj.Stat(ca.src, project.GetFileOption{IsFile: true, Role: options.Task.Annotations[kkcorev1alpha1.TaskAnnotationRole]})
if err != nil {
return "", fmt.Sprintf("get file %s from project error %v", ca.src, err)
}
if fileInfo.IsDir() {
if err := ca.relDir(ctx, pj, options.Task.Annotations[kkcorev1alpha1.TaskAnnotationRole], conn); err != nil {
return "", fmt.Sprintf("sync copy relative dir error %s", err)
}
} else {
if err := ca.relFile(ctx, pj, options.Task.Annotations[kkcorev1alpha1.TaskAnnotationRole], fileInfo.Mode(), conn); err != nil {
return "", fmt.Sprintf("sync copy relative dir error %s", err)
}
}
}
return StdoutSuccess, ""
}
// copyContent convert content param and copy to dest
func (ca copyArgs) copyContent(ctx context.Context, mode fs.FileMode, conn connector.Connector) (string, string) {
if strings.HasSuffix(ca.dest, "/") {
return "", "\"content\" should copy to a file"
}
if ca.mode != nil {
mode = os.FileMode(*ca.mode)
}
if err := conn.PutFile(ctx, []byte(ca.content), ca.dest, mode); err != nil {
return "", fmt.Sprintf("copy file error: %v", err)
}
return StdoutSuccess, ""
}
// relFile when copy.src is relative dir, get all files from project, and copy to remote.
func (ca copyArgs) relFile(ctx context.Context, pj project.Project, role string, mode fs.FileMode, conn connector.Connector) any {
data, err := pj.ReadFile(ca.src, project.GetFileOption{IsFile: true, Role: role})
if err != nil {
return fmt.Errorf("read file error: %w", err)
}
dest := ca.dest
if strings.HasSuffix(ca.dest, "/") {
dest = filepath.Join(ca.dest, filepath.Base(ca.src))
}
if ca.mode != nil {
mode = os.FileMode(*ca.mode)
}
if err := conn.PutFile(ctx, data, dest, mode); err != nil {
return fmt.Errorf("copy file error: %w", err)
}
return nil
}
// relDir when copy.src is relative dir, get all files from project, and copy to remote.
func (ca copyArgs) relDir(ctx context.Context, pj project.Project, role string, conn connector.Connector) error {
if err := pj.WalkDir(ca.src, project.GetFileOption{IsFile: true, Role: role}, func(path string, d fs.DirEntry, err error) error {
if d.IsDir() { // only copy file
return nil
}
if err != nil {
return fmt.Errorf("walk dir %s error: %w", ca.src, err)
}
info, err := d.Info()
if err != nil {
return fmt.Errorf("get file info error: %w", err)
}
mode := info.Mode()
if ca.mode != nil {
mode = os.FileMode(*ca.mode)
}
data, err := pj.ReadFile(path, project.GetFileOption{Role: role})
if err != nil {
return fmt.Errorf("read file error: %w", err)
}
dest := ca.dest
if strings.HasSuffix(ca.dest, "/") {
rel, err := pj.Rel(ca.src, path, project.GetFileOption{Role: role})
if err != nil {
return fmt.Errorf("get relative file path error: %w", err)
}
dest = filepath.Join(ca.dest, rel)
}
if err := conn.PutFile(ctx, data, dest, mode); err != nil {
return fmt.Errorf("copy file error: %w", err)
}
return nil
}); err != nil {
return err
}
return nil
}
// absFile when copy.src is absolute file, get file from os, and copy to remote.
func (ca copyArgs) absFile(ctx context.Context, mode fs.FileMode, conn connector.Connector) error {
data, err := os.ReadFile(ca.src)
if err != nil {
return fmt.Errorf("read file error: %w", err)
}
dest := ca.dest
if strings.HasSuffix(ca.dest, "/") {
dest = filepath.Join(ca.dest, filepath.Base(ca.src))
}
if ca.mode != nil {
mode = os.FileMode(*ca.mode)
}
if err := conn.PutFile(ctx, data, dest, mode); err != nil {
return fmt.Errorf("copy file error: %w", err)
}
return nil
}
// absDir when copy.src is absolute dir, get all files from os, and copy to remote.
func (ca copyArgs) absDir(ctx context.Context, conn connector.Connector) error {
if err := filepath.WalkDir(ca.src, func(path string, d fs.DirEntry, err error) error {
if d.IsDir() { // only copy file
return nil
}
if err != nil {
return fmt.Errorf("walk dir %s error: %w", ca.src, err)
}
// get file old mode
info, err := d.Info()
if err != nil {
return fmt.Errorf("get file info error: %w", err)
}
mode := info.Mode()
if ca.mode != nil {
mode = os.FileMode(*ca.mode)
}
// read file
data, err := os.ReadFile(path)
if err != nil {
return fmt.Errorf("read file error: %w", err)
}
// copy file to remote
dest := ca.dest
if strings.HasSuffix(ca.dest, "/") {
rel, err := filepath.Rel(ca.src, path)
if err != nil {
return fmt.Errorf("get relative file path error: %w", err)
}
dest = filepath.Join(ca.dest, rel)
}
if err := conn.PutFile(ctx, data, dest, mode); err != nil {
return fmt.Errorf("copy file error: %w", err)
}
return nil
}); err != nil {
return err
}
return nil
}