mirror of
https://github.com/kubesphere/kubekey.git
synced 2025-12-30 15:12:54 +00:00
Signed-off-by: joyceliu <joyceliu@yunify.com> Co-authored-by: joyceliu <joyceliu@yunify.com>
290 lines
7.9 KiB
Go
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 := ©Args{}
|
|
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
|
|
}
|