@@ -18,7 +18,7 @@ use env;
18
18
use ffi:: { OsString , OsStr } ;
19
19
use fmt;
20
20
use fs;
21
- use io:: { self , Error } ;
21
+ use io:: { self , Error , ErrorKind } ;
22
22
use libc:: c_void;
23
23
use mem;
24
24
use os:: windows:: ffi:: OsStrExt ;
@@ -43,13 +43,21 @@ fn mk_key(s: &OsStr) -> OsString {
43
43
} )
44
44
}
45
45
46
+ fn ensure_no_nuls < T : AsRef < OsStr > > ( str : T ) -> io:: Result < T > {
47
+ if str. as_ref ( ) . encode_wide ( ) . any ( |b| b == 0 ) {
48
+ Err ( io:: Error :: new ( ErrorKind :: InvalidInput , "nul byte found in provided data" ) )
49
+ } else {
50
+ Ok ( str)
51
+ }
52
+ }
53
+
46
54
#[ derive( Clone ) ]
47
55
pub struct Command {
48
- pub program : OsString ,
49
- pub args : Vec < OsString > ,
50
- pub env : Option < HashMap < OsString , OsString > > ,
51
- pub cwd : Option < OsString > ,
52
- pub detach : bool , // not currently exposed in std::process
56
+ program : OsString ,
57
+ args : Vec < OsString > ,
58
+ env : Option < HashMap < OsString , OsString > > ,
59
+ cwd : Option < OsString > ,
60
+ detach : bool , // not currently exposed in std::process
53
61
}
54
62
55
63
impl Command {
@@ -92,6 +100,16 @@ impl Command {
92
100
}
93
101
}
94
102
103
+ impl fmt:: Debug for Command {
104
+ fn fmt ( & self , f : & mut fmt:: Formatter ) -> fmt:: Result {
105
+ try!( write ! ( f, "{:?}" , self . program) ) ;
106
+ for arg in & self . args {
107
+ try!( write ! ( f, " {:?}" , arg) ) ;
108
+ }
109
+ Ok ( ( ) )
110
+ }
111
+ }
112
+
95
113
////////////////////////////////////////////////////////////////////////////////
96
114
// Processes
97
115
////////////////////////////////////////////////////////////////////////////////
@@ -153,7 +171,7 @@ impl Process {
153
171
si. hStdError = stderr. raw ( ) ;
154
172
155
173
let program = program. as_ref ( ) . unwrap_or ( & cfg. program ) ;
156
- let mut cmd_str = make_command_line ( program, & cfg. args ) ;
174
+ let mut cmd_str = try! ( make_command_line ( program, & cfg. args ) ) ;
157
175
cmd_str. push ( 0 ) ; // add null terminator
158
176
159
177
// stolen from the libuv code.
@@ -162,8 +180,8 @@ impl Process {
162
180
flags |= c:: DETACHED_PROCESS | c:: CREATE_NEW_PROCESS_GROUP ;
163
181
}
164
182
165
- let ( envp, _data) = make_envp ( cfg. env . as_ref ( ) ) ;
166
- let ( dirp, _data) = make_dirp ( cfg. cwd . as_ref ( ) ) ;
183
+ let ( envp, _data) = try! ( make_envp ( cfg. env . as_ref ( ) ) ) ;
184
+ let ( dirp, _data) = try! ( make_dirp ( cfg. cwd . as_ref ( ) ) ) ;
167
185
let mut pi = zeroed_process_information ( ) ;
168
186
try!( unsafe {
169
187
// `CreateProcess` is racy!
@@ -265,22 +283,24 @@ fn zeroed_process_information() -> c::PROCESS_INFORMATION {
265
283
}
266
284
}
267
285
268
- // Produces a wide string *without terminating null*
269
- fn make_command_line ( prog : & OsStr , args : & [ OsString ] ) -> Vec < u16 > {
286
+ // Produces a wide string *without terminating null*; returns an error if
287
+ // `prog` or any of the `args` contain a nul.
288
+ fn make_command_line ( prog : & OsStr , args : & [ OsString ] ) -> io:: Result < Vec < u16 > > {
270
289
// Encode the command and arguments in a command line string such
271
290
// that the spawned process may recover them using CommandLineToArgvW.
272
291
let mut cmd: Vec < u16 > = Vec :: new ( ) ;
273
- append_arg ( & mut cmd, prog) ;
292
+ try! ( append_arg ( & mut cmd, prog) ) ;
274
293
for arg in args {
275
294
cmd. push ( ' ' as u16 ) ;
276
- append_arg ( & mut cmd, arg) ;
295
+ try! ( append_arg ( & mut cmd, arg) ) ;
277
296
}
278
- return cmd;
297
+ return Ok ( cmd) ;
279
298
280
- fn append_arg ( cmd : & mut Vec < u16 > , arg : & OsStr ) {
299
+ fn append_arg ( cmd : & mut Vec < u16 > , arg : & OsStr ) -> io :: Result < ( ) > {
281
300
// If an argument has 0 characters then we need to quote it to ensure
282
301
// that it actually gets passed through on the command line or otherwise
283
302
// it will be dropped entirely when parsed on the other end.
303
+ try!( ensure_no_nuls ( arg) ) ;
284
304
let arg_bytes = & arg. as_inner ( ) . inner . as_inner ( ) ;
285
305
let quote = arg_bytes. iter ( ) . any ( |c| * c == b' ' || * c == b'\t' )
286
306
|| arg_bytes. is_empty ( ) ;
@@ -312,11 +332,12 @@ fn make_command_line(prog: &OsStr, args: &[OsString]) -> Vec<u16> {
312
332
}
313
333
cmd. push ( '"' as u16 ) ;
314
334
}
335
+ Ok ( ( ) )
315
336
}
316
337
}
317
338
318
339
fn make_envp ( env : Option < & collections:: HashMap < OsString , OsString > > )
319
- -> ( * mut c_void , Vec < u16 > ) {
340
+ -> io :: Result < ( * mut c_void , Vec < u16 > ) > {
320
341
// On Windows we pass an "environment block" which is not a char**, but
321
342
// rather a concatenation of null-terminated k=v\0 sequences, with a final
322
343
// \0 to terminate.
@@ -325,26 +346,27 @@ fn make_envp(env: Option<&collections::HashMap<OsString, OsString>>)
325
346
let mut blk = Vec :: new ( ) ;
326
347
327
348
for pair in env {
328
- blk. extend ( pair. 0 . encode_wide ( ) ) ;
349
+ blk. extend ( try! ( ensure_no_nuls ( pair. 0 ) ) . encode_wide ( ) ) ;
329
350
blk. push ( '=' as u16 ) ;
330
- blk. extend ( pair. 1 . encode_wide ( ) ) ;
351
+ blk. extend ( try! ( ensure_no_nuls ( pair. 1 ) ) . encode_wide ( ) ) ;
331
352
blk. push ( 0 ) ;
332
353
}
333
354
blk. push ( 0 ) ;
334
- ( blk. as_mut_ptr ( ) as * mut c_void , blk)
355
+ Ok ( ( blk. as_mut_ptr ( ) as * mut c_void , blk) )
335
356
}
336
- _ => ( ptr:: null_mut ( ) , Vec :: new ( ) )
357
+ _ => Ok ( ( ptr:: null_mut ( ) , Vec :: new ( ) ) )
337
358
}
338
359
}
339
360
340
- fn make_dirp ( d : Option < & OsString > ) -> ( * const u16 , Vec < u16 > ) {
361
+ fn make_dirp ( d : Option < & OsString > ) -> io:: Result < ( * const u16 , Vec < u16 > ) > {
362
+
341
363
match d {
342
364
Some ( dir) => {
343
- let mut dir_str: Vec < u16 > = dir. encode_wide ( ) . collect ( ) ;
365
+ let mut dir_str: Vec < u16 > = try! ( ensure_no_nuls ( dir) ) . encode_wide ( ) . collect ( ) ;
344
366
dir_str. push ( 0 ) ;
345
- ( dir_str. as_ptr ( ) , dir_str)
367
+ Ok ( ( dir_str. as_ptr ( ) , dir_str) )
346
368
} ,
347
- None => ( ptr:: null ( ) , Vec :: new ( ) )
369
+ None => Ok ( ( ptr:: null ( ) , Vec :: new ( ) ) )
348
370
}
349
371
}
350
372
@@ -397,11 +419,12 @@ mod tests {
397
419
#[ test]
398
420
fn test_make_command_line ( ) {
399
421
fn test_wrapper ( prog : & str , args : & [ & str ] ) -> String {
400
- String :: from_utf16 (
401
- & make_command_line ( OsStr :: new ( prog) ,
402
- & args. iter ( )
403
- . map ( |a| OsString :: from ( a) )
404
- . collect :: < Vec < OsString > > ( ) ) ) . unwrap ( )
422
+ let command_line = & make_command_line ( OsStr :: new ( prog) ,
423
+ & args. iter ( )
424
+ . map ( |a| OsString :: from ( a) )
425
+ . collect :: < Vec < OsString > > ( ) )
426
+ . unwrap ( ) ;
427
+ String :: from_utf16 ( command_line) . unwrap ( )
405
428
}
406
429
407
430
assert_eq ! (
0 commit comments