Skip to content

Commit a9bade5

Browse files
committed
add rsync flag option to copy files using rsync
Signed-off-by: olalekan odukoya <[email protected]>
1 parent ccd3c0c commit a9bade5

File tree

2 files changed

+164
-22
lines changed

2 files changed

+164
-22
lines changed

cmd/limactl/copy.go

+124-22
Original file line numberDiff line numberDiff line change
@@ -21,6 +21,13 @@ Prefix guest filenames with the instance name and a colon.
2121
Example: limactl copy default:/etc/os-release .
2222
`
2323

24+
type copyTool string
25+
26+
const (
27+
rsync copyTool = "rsync"
28+
scp copyTool = "scp"
29+
)
30+
2431
func newCopyCommand() *cobra.Command {
2532
copyCommand := &cobra.Command{
2633
Use: "copy SOURCE ... TARGET",
@@ -49,13 +56,6 @@ func copyAction(cmd *cobra.Command, args []string) error {
4956
return err
5057
}
5158

52-
arg0, err := exec.LookPath("scp")
53-
if err != nil {
54-
return err
55-
}
56-
instances := make(map[string]*store.Instance)
57-
scpFlags := []string{}
58-
scpArgs := []string{}
5959
debug, err := cmd.Flags().GetBool("debug")
6060
if err != nil {
6161
return err
@@ -65,6 +65,48 @@ func copyAction(cmd *cobra.Command, args []string) error {
6565
verbose = true
6666
}
6767

68+
cpTool := rsync
69+
arg0, err := exec.LookPath(string(cpTool))
70+
if err != nil {
71+
arg0, err = exec.LookPath(string(cpTool))
72+
if err != nil {
73+
return err
74+
}
75+
}
76+
logrus.Infof("using copy tool %q", arg0)
77+
78+
var sshArgs, toolArgs []string
79+
80+
switch cpTool {
81+
case scp:
82+
sshArgs, toolArgs, err = useScp(args, verbose, recursive)
83+
if err != nil {
84+
return err
85+
}
86+
case rsync:
87+
toolArgs, err = useRsync(args, verbose, recursive)
88+
if err != nil {
89+
return err
90+
}
91+
default:
92+
return fmt.Errorf("invalid copy tool %q", cpTool)
93+
}
94+
95+
sshCmd := exec.Command(arg0, append(sshArgs, toolArgs...)...)
96+
sshCmd.Stdin = cmd.InOrStdin()
97+
sshCmd.Stdout = cmd.OutOrStdout()
98+
sshCmd.Stderr = cmd.ErrOrStderr()
99+
logrus.Debugf("executing scp (may take a long time): %+v", sshCmd.Args)
100+
101+
// TODO: use syscall.Exec directly (results in losing tty?)
102+
return sshCmd.Run()
103+
}
104+
105+
func useScp(args []string, verbose, recursive bool) (sshArgs, scpArgs []string, err error) {
106+
instances := make(map[string]*store.Instance)
107+
108+
scpFlags := []string{}
109+
68110
if verbose {
69111
scpFlags = append(scpFlags, "-v")
70112
} else {
@@ -74,6 +116,7 @@ func copyAction(cmd *cobra.Command, args []string) error {
74116
if recursive {
75117
scpFlags = append(scpFlags, "-r")
76118
}
119+
77120
// this assumes that ssh and scp come from the same place, but scp has no -V
78121
legacySSH := sshutil.DetectOpenSSHVersion("ssh").LessThan(*semver.New("8.0.0"))
79122
for _, arg := range args {
@@ -86,12 +129,12 @@ func copyAction(cmd *cobra.Command, args []string) error {
86129
inst, err := store.Inspect(instName)
87130
if err != nil {
88131
if errors.Is(err, os.ErrNotExist) {
89-
return fmt.Errorf("instance %q does not exist, run `limactl create %s` to create a new instance", instName, instName)
132+
return nil, nil, fmt.Errorf("instance %q does not exist, run `limactl create %s` to create a new instance", instName, instName)
90133
}
91-
return err
134+
return nil, nil, err
92135
}
93136
if inst.Status == store.StatusStopped {
94-
return fmt.Errorf("instance %q is stopped, run `limactl start %s` to start the instance", instName, instName)
137+
return nil, nil, fmt.Errorf("instance %q is stopped, run `limactl start %s` to start the instance", instName, instName)
95138
}
96139
if legacySSH {
97140
scpFlags = append(scpFlags, "-P", fmt.Sprintf("%d", inst.SSHLocalPort))
@@ -101,11 +144,11 @@ func copyAction(cmd *cobra.Command, args []string) error {
101144
}
102145
instances[instName] = inst
103146
default:
104-
return fmt.Errorf("path %q contains multiple colons", arg)
147+
return nil, nil, fmt.Errorf("path %q contains multiple colons", arg)
105148
}
106149
}
107150
if legacySSH && len(instances) > 1 {
108-
return errors.New("more than one (instance) host is involved in this command, this is only supported for openSSH v8.0 or higher")
151+
return nil, nil, errors.New("more than one (instance) host is involved in this command, this is only supported for openSSH v8.0 or higher")
109152
}
110153
scpFlags = append(scpFlags, "-3", "--")
111154
scpArgs = append(scpFlags, scpArgs...)
@@ -118,24 +161,83 @@ func copyAction(cmd *cobra.Command, args []string) error {
118161
for _, inst := range instances {
119162
sshOpts, err = sshutil.SSHOpts("ssh", inst.Dir, *inst.Config.User.Name, false, false, false, false)
120163
if err != nil {
121-
return err
164+
return nil, nil, err
122165
}
123166
}
124167
} else {
125168
// Copying among multiple hosts; we can't pass in host-specific options.
126169
sshOpts, err = sshutil.CommonOpts("ssh", false)
127170
if err != nil {
128-
return err
171+
return nil, nil, err
129172
}
130173
}
131-
sshArgs := sshutil.SSHArgsFromOpts(sshOpts)
174+
sshArgs = sshutil.SSHArgsFromOpts(sshOpts)
132175

133-
sshCmd := exec.Command(arg0, append(sshArgs, scpArgs...)...)
134-
sshCmd.Stdin = cmd.InOrStdin()
135-
sshCmd.Stdout = cmd.OutOrStdout()
136-
sshCmd.Stderr = cmd.ErrOrStderr()
137-
logrus.Debugf("executing scp (may take a long time): %+v", sshCmd.Args)
176+
return sshArgs, scpArgs, nil
177+
}
138178

139-
// TODO: use syscall.Exec directly (results in losing tty?)
140-
return sshCmd.Run()
179+
func useRsync(args []string, verbose, recursive bool) ([]string, error) {
180+
instances := make(map[string]*store.Instance)
181+
182+
var instName string
183+
184+
rsyncFlags := []string{}
185+
rsyncArgs := []string{}
186+
187+
if verbose {
188+
rsyncFlags = append(rsyncFlags, "-v", "--progress")
189+
} else {
190+
rsyncFlags = append(rsyncFlags, "-q")
191+
}
192+
193+
if recursive {
194+
rsyncFlags = append(rsyncFlags, "-r")
195+
}
196+
197+
for _, arg := range args {
198+
path := strings.Split(arg, ":")
199+
switch len(path) {
200+
case 1:
201+
inst, ok := instances[instName]
202+
if !ok {
203+
return nil, fmt.Errorf("instance %q does not exist, run `limactl create %s` to create a new instance", instName, instName)
204+
}
205+
guestVM := fmt.Sprintf("%[email protected]:%s", *inst.Config.User.Name, path[0])
206+
rsyncArgs = append(rsyncArgs, guestVM)
207+
case 2:
208+
instName = path[0]
209+
inst, err := store.Inspect(instName)
210+
if err != nil {
211+
if errors.Is(err, os.ErrNotExist) {
212+
return nil, fmt.Errorf("instance %q does not exist, run `limactl create %s` to create a new instance", instName, instName)
213+
}
214+
return nil, err
215+
}
216+
sshOpts, err := sshutil.SSHOpts("ssh", inst.Dir, *inst.Config.User.Name, false, false, false, false)
217+
if err != nil {
218+
return nil, err
219+
}
220+
221+
sshStr := fmt.Sprintf("ssh -p %s -i %s", fmt.Sprintf("%d", inst.SSHLocalPort), extractSSHOptionField(sshOpts, "IdentityFile"))
222+
rsyncArgs = append(rsyncArgs, "-avz", "-e", sshStr, path[1])
223+
instances[instName] = inst
224+
default:
225+
return nil, fmt.Errorf("path %q contains multiple colons", arg)
226+
}
227+
}
228+
229+
rsyncArgs = append(rsyncFlags, rsyncArgs...)
230+
231+
return rsyncArgs, nil
232+
}
233+
234+
func extractSSHOptionField(sshOpts []string, optName string) string {
235+
for _, opt := range sshOpts {
236+
optField := fmt.Sprintf("%s=", optName)
237+
if strings.HasPrefix(opt, optField) {
238+
identityFile := strings.TrimPrefix(opt, optField)
239+
return strings.Trim(identityFile, `"`)
240+
}
241+
}
242+
return ""
141243
}

pkg/hostagent/hostagent.go

+40
Original file line numberDiff line numberDiff line change
@@ -424,6 +424,41 @@ func (a *HostAgent) Info(_ context.Context) (*hostagentapi.Info, error) {
424424
return info, nil
425425
}
426426

427+
func (a *HostAgent) installPackage() error {
428+
logrus.Debugf("installing packages")
429+
430+
faScript := `#!/bin/bash
431+
if ! output=$(type rsync 2>&1); then
432+
echo "rsync is not installed. Attempting to install..."
433+
434+
# Try to install rsync based on the OS
435+
if [ -f /etc/debian_version ]; then
436+
sudo apt-get update && sudo apt-get install -y rsync
437+
elif [ -f /etc/alpine-release ]; then
438+
sudo apk add rsync
439+
elif [ -f /etc/redhat-release ]; then
440+
sudo yum install -y rsync
441+
elif [ -f /etc/arch-release ]; then
442+
sudo pacman -S --noconfirm rsync
443+
else
444+
echo "Unsupported Linux distribution. Please install rsync manually."
445+
fi
446+
447+
echo "rsync installation complete."
448+
else
449+
echo "rsync is already installed."
450+
fi`
451+
faDesc := "installing rsync"
452+
stdout, stderr, err := ssh.ExecuteScript(a.instSSHAddress, a.sshLocalPort, a.sshConfig, faScript, faDesc)
453+
logrus.Debugf("stdout=%q, stderr=%q, err=%v", stdout, stderr, err)
454+
if err != nil {
455+
err = fmt.Errorf("stdout=%q, stderr=%q: %w", stdout, stderr, err)
456+
return err
457+
}
458+
459+
return nil
460+
}
461+
427462
func (a *HostAgent) startHostAgentRoutines(ctx context.Context) error {
428463
if *a.instConfig.Plain {
429464
logrus.Info("Running in plain mode. Mounts, port forwarding, containerd, etc. will be ignored. Guest agent will not be running.")
@@ -439,6 +474,11 @@ func (a *HostAgent) startHostAgentRoutines(ctx context.Context) error {
439474
if err := a.waitForRequirements("essential", a.essentialRequirements()); err != nil {
440475
errs = append(errs, err)
441476
}
477+
478+
if err := a.installPackage(); err != nil {
479+
errs = append(errs, err)
480+
}
481+
442482
if *a.instConfig.SSH.ForwardAgent {
443483
faScript := `#!/bin/bash
444484
set -eux -o pipefail

0 commit comments

Comments
 (0)