@@ -21,6 +21,13 @@ Prefix guest filenames with the instance name and a colon.
21
21
Example: limactl copy default:/etc/os-release .
22
22
`
23
23
24
+ type copyTool string
25
+
26
+ const (
27
+ rsync copyTool = "rsync"
28
+ scp copyTool = "scp"
29
+ )
30
+
24
31
func newCopyCommand () * cobra.Command {
25
32
copyCommand := & cobra.Command {
26
33
Use : "copy SOURCE ... TARGET" ,
@@ -49,13 +56,6 @@ func copyAction(cmd *cobra.Command, args []string) error {
49
56
return err
50
57
}
51
58
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 {}
59
59
debug , err := cmd .Flags ().GetBool ("debug" )
60
60
if err != nil {
61
61
return err
@@ -65,6 +65,48 @@ func copyAction(cmd *cobra.Command, args []string) error {
65
65
verbose = true
66
66
}
67
67
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
+
68
110
if verbose {
69
111
scpFlags = append (scpFlags , "-v" )
70
112
} else {
@@ -74,6 +116,7 @@ func copyAction(cmd *cobra.Command, args []string) error {
74
116
if recursive {
75
117
scpFlags = append (scpFlags , "-r" )
76
118
}
119
+
77
120
// this assumes that ssh and scp come from the same place, but scp has no -V
78
121
legacySSH := sshutil .DetectOpenSSHVersion ("ssh" ).LessThan (* semver .New ("8.0.0" ))
79
122
for _ , arg := range args {
@@ -86,12 +129,12 @@ func copyAction(cmd *cobra.Command, args []string) error {
86
129
inst , err := store .Inspect (instName )
87
130
if err != nil {
88
131
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 )
90
133
}
91
- return err
134
+ return nil , nil , err
92
135
}
93
136
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 )
95
138
}
96
139
if legacySSH {
97
140
scpFlags = append (scpFlags , "-P" , fmt .Sprintf ("%d" , inst .SSHLocalPort ))
@@ -101,11 +144,11 @@ func copyAction(cmd *cobra.Command, args []string) error {
101
144
}
102
145
instances [instName ] = inst
103
146
default :
104
- return fmt .Errorf ("path %q contains multiple colons" , arg )
147
+ return nil , nil , fmt .Errorf ("path %q contains multiple colons" , arg )
105
148
}
106
149
}
107
150
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" )
109
152
}
110
153
scpFlags = append (scpFlags , "-3" , "--" )
111
154
scpArgs = append (scpFlags , scpArgs ... )
@@ -118,24 +161,83 @@ func copyAction(cmd *cobra.Command, args []string) error {
118
161
for _ , inst := range instances {
119
162
sshOpts , err = sshutil .SSHOpts ("ssh" , inst .Dir , * inst .Config .User .Name , false , false , false , false )
120
163
if err != nil {
121
- return err
164
+ return nil , nil , err
122
165
}
123
166
}
124
167
} else {
125
168
// Copying among multiple hosts; we can't pass in host-specific options.
126
169
sshOpts , err = sshutil .CommonOpts ("ssh" , false )
127
170
if err != nil {
128
- return err
171
+ return nil , nil , err
129
172
}
130
173
}
131
- sshArgs : = sshutil .SSHArgsFromOpts (sshOpts )
174
+ sshArgs = sshutil .SSHArgsFromOpts (sshOpts )
132
175
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
+ }
138
178
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 ""
141
243
}
0 commit comments