@@ -12,6 +12,7 @@ import (
12
12
"io"
13
13
"os"
14
14
"os/exec"
15
+ "path/filepath"
15
16
"runtime"
16
17
"strings"
17
18
"time"
@@ -43,18 +44,24 @@ type Command struct {
43
44
prog string
44
45
args []string
45
46
parentContext context.Context
46
- desc string
47
47
globalArgsLength int
48
48
brokenArgs []string
49
49
}
50
50
51
- func (c * Command ) String () string {
52
- return c .toString (false )
51
+ func logArgSanitize (arg string ) string {
52
+ if strings .Contains (arg , "://" ) && strings .Contains (arg , "@" ) {
53
+ return util .SanitizeCredentialURLs (arg )
54
+ } else if filepath .IsAbs (arg ) {
55
+ base := filepath .Base (arg )
56
+ dir := filepath .Dir (arg )
57
+ return filepath .Join (filepath .Base (dir ), base )
58
+ }
59
+ return arg
53
60
}
54
61
55
- func (c * Command ) toString ( sanitizing bool ) string {
62
+ func (c * Command ) LogString ( ) string {
56
63
// WARNING: this function is for debugging purposes only. It's much better than old code (which only joins args with space),
57
- // It's impossible to make a simple and 100% correct implementation of argument quoting for different platforms.
64
+ // It's impossible to make a simple and 100% correct implementation of argument quoting for different platforms here .
58
65
debugQuote := func (s string ) string {
59
66
if strings .ContainsAny (s , " `'\" \t \r \n " ) {
60
67
return fmt .Sprintf ("%q" , s )
@@ -63,12 +70,11 @@ func (c *Command) toString(sanitizing bool) string {
63
70
}
64
71
a := make ([]string , 0 , len (c .args )+ 1 )
65
72
a = append (a , debugQuote (c .prog ))
66
- for _ , arg := range c .args {
67
- if sanitizing && (strings .Contains (arg , "://" ) && strings .Contains (arg , "@" )) {
68
- a = append (a , debugQuote (util .SanitizeCredentialURLs (arg )))
69
- } else {
70
- a = append (a , debugQuote (arg ))
71
- }
73
+ if c .globalArgsLength > 0 {
74
+ a = append (a , "...global..." )
75
+ }
76
+ for i := c .globalArgsLength ; i < len (c .args ); i ++ {
77
+ a = append (a , debugQuote (logArgSanitize (c .args [i ])))
72
78
}
73
79
return strings .Join (a , " " )
74
80
}
@@ -112,12 +118,6 @@ func (c *Command) SetParentContext(ctx context.Context) *Command {
112
118
return c
113
119
}
114
120
115
- // SetDescription sets the description for this command which be returned on c.String()
116
- func (c * Command ) SetDescription (desc string ) * Command {
117
- c .desc = desc
118
- return c
119
- }
120
-
121
121
// isSafeArgumentValue checks if the argument is safe to be used as a value (not an option)
122
122
func isSafeArgumentValue (s string ) bool {
123
123
return s == "" || s [0 ] != '-'
@@ -271,8 +271,12 @@ var ErrBrokenCommand = errors.New("git command is broken")
271
271
272
272
// Run runs the command with the RunOpts
273
273
func (c * Command ) Run (opts * RunOpts ) error {
274
+ return c .run (1 , opts )
275
+ }
276
+
277
+ func (c * Command ) run (skip int , opts * RunOpts ) error {
274
278
if len (c .brokenArgs ) != 0 {
275
- log .Error ("git command is broken: %s, broken args: %s" , c .String (), strings .Join (c .brokenArgs , " " ))
279
+ log .Error ("git command is broken: %s, broken args: %s" , c .LogString (), strings .Join (c .brokenArgs , " " ))
276
280
return ErrBrokenCommand
277
281
}
278
282
if opts == nil {
@@ -285,20 +289,14 @@ func (c *Command) Run(opts *RunOpts) error {
285
289
timeout = defaultCommandExecutionTimeout
286
290
}
287
291
288
- if len (opts .Dir ) == 0 {
289
- log .Debug ("git.Command.Run: %s" , c )
290
- } else {
291
- log .Debug ("git.Command.RunDir(%s): %s" , opts .Dir , c )
292
- }
293
-
294
- desc := c .desc
295
- if desc == "" {
296
- if opts .Dir == "" {
297
- desc = fmt .Sprintf ("git: %s" , c .toString (true ))
298
- } else {
299
- desc = fmt .Sprintf ("git(dir:%s): %s" , opts .Dir , c .toString (true ))
300
- }
292
+ var desc string
293
+ callerInfo := util .CallerFuncName (1 /* util */ + 1 /* this */ + skip /* parent */ )
294
+ if pos := strings .LastIndex (callerInfo , "/" ); pos >= 0 {
295
+ callerInfo = callerInfo [pos + 1 :]
301
296
}
297
+ // these logs are for debugging purposes only, so no guarantee of correctness or stability
298
+ desc = fmt .Sprintf ("git.Run(by:%s, repo:%s): %s" , callerInfo , logArgSanitize (opts .Dir ), c .LogString ())
299
+ log .Debug ("git.Command: %s" , desc )
302
300
303
301
var ctx context.Context
304
302
var cancel context.CancelFunc
@@ -401,7 +399,7 @@ func IsErrorExitCode(err error, code int) bool {
401
399
402
400
// RunStdString runs the command with options and returns stdout/stderr as string. and store stderr to returned error (err combined with stderr).
403
401
func (c * Command ) RunStdString (opts * RunOpts ) (stdout , stderr string , runErr RunStdError ) {
404
- stdoutBytes , stderrBytes , err := c .RunStdBytes (opts )
402
+ stdoutBytes , stderrBytes , err := c .runStdBytes (opts )
405
403
stdout = util .UnsafeBytesToString (stdoutBytes )
406
404
stderr = util .UnsafeBytesToString (stderrBytes )
407
405
if err != nil {
@@ -413,6 +411,10 @@ func (c *Command) RunStdString(opts *RunOpts) (stdout, stderr string, runErr Run
413
411
414
412
// RunStdBytes runs the command with options and returns stdout/stderr as bytes. and store stderr to returned error (err combined with stderr).
415
413
func (c * Command ) RunStdBytes (opts * RunOpts ) (stdout , stderr []byte , runErr RunStdError ) {
414
+ return c .runStdBytes (opts )
415
+ }
416
+
417
+ func (c * Command ) runStdBytes (opts * RunOpts ) (stdout , stderr []byte , runErr RunStdError ) {
416
418
if opts == nil {
417
419
opts = & RunOpts {}
418
420
}
@@ -435,7 +437,7 @@ func (c *Command) RunStdBytes(opts *RunOpts) (stdout, stderr []byte, runErr RunS
435
437
PipelineFunc : opts .PipelineFunc ,
436
438
}
437
439
438
- err := c .Run ( newOpts )
440
+ err := c .run ( 2 , newOpts )
439
441
stderr = stderrBuf .Bytes ()
440
442
if err != nil {
441
443
return nil , stderr , & runStdError {err : err , stderr : util .UnsafeBytesToString (stderr )}
0 commit comments