2016-11-04 01:16:01 +03:00
// Copyright 2015 The Gogs Authors. All rights reserved.
// Use of this source code is governed by a MIT-style
// license that can be found in the LICENSE file.
package git
import (
"bytes"
2017-04-08 05:23:39 +03:00
"context"
2016-11-04 01:16:01 +03:00
"fmt"
"io"
2019-11-10 11:42:51 +03:00
"os"
2016-11-04 01:16:01 +03:00
"os/exec"
"strings"
"time"
2019-06-26 21:15:26 +03:00
"code.gitea.io/gitea/modules/process"
2016-11-04 01:16:01 +03:00
)
2016-12-22 12:30:52 +03:00
var (
// GlobalCommandArgs global command args for external package setting
GlobalCommandArgs [ ] string
2018-01-07 16:10:20 +03:00
// DefaultCommandExecutionTimeout default command execution timeout duration
DefaultCommandExecutionTimeout = 60 * time . Second
2016-12-22 12:30:52 +03:00
)
2019-11-10 11:42:51 +03:00
// DefaultLocale is the default LC_ALL to run git commands in.
const DefaultLocale = "C"
2016-11-04 01:16:01 +03:00
// Command represents a command with its subcommands or arguments.
type Command struct {
name string
args [ ] string
}
func ( c * Command ) String ( ) string {
if len ( c . args ) == 0 {
return c . name
}
return fmt . Sprintf ( "%s %s" , c . name , strings . Join ( c . args , " " ) )
}
// NewCommand creates and returns a new Git Command based on given command and arguments.
func NewCommand ( args ... string ) * Command {
2018-11-28 10:00:25 +03:00
// Make an explicit copy of GlobalCommandArgs, otherwise append might overwrite it
cargs := make ( [ ] string , len ( GlobalCommandArgs ) )
copy ( cargs , GlobalCommandArgs )
2016-11-04 01:16:01 +03:00
return & Command {
2019-04-17 14:11:37 +03:00
name : GitExecutable ,
2018-11-28 10:00:25 +03:00
args : append ( cargs , args ... ) ,
2016-11-04 01:16:01 +03:00
}
}
// AddArguments adds new argument(s) to the command.
func ( c * Command ) AddArguments ( args ... string ) * Command {
c . args = append ( c . args , args ... )
return c
}
2019-05-11 18:29:17 +03:00
// RunInDirTimeoutEnvPipeline executes the command in given directory with given timeout,
2016-11-04 01:16:01 +03:00
// it pipes stdout and stderr to given io.Writer.
2019-05-11 18:29:17 +03:00
func ( c * Command ) RunInDirTimeoutEnvPipeline ( env [ ] string , timeout time . Duration , dir string , stdout , stderr io . Writer ) error {
return c . RunInDirTimeoutEnvFullPipeline ( env , timeout , dir , stdout , stderr , nil )
}
// RunInDirTimeoutEnvFullPipeline executes the command in given directory with given timeout,
// it pipes stdout and stderr to given io.Writer and passes in an io.Reader as stdin.
func ( c * Command ) RunInDirTimeoutEnvFullPipeline ( env [ ] string , timeout time . Duration , dir string , stdout , stderr io . Writer , stdin io . Reader ) error {
2016-11-04 01:16:01 +03:00
if timeout == - 1 {
2018-01-07 16:10:20 +03:00
timeout = DefaultCommandExecutionTimeout
2016-11-04 01:16:01 +03:00
}
if len ( dir ) == 0 {
log ( c . String ( ) )
} else {
log ( "%s: %v" , dir , c )
}
2017-04-08 05:23:39 +03:00
ctx , cancel := context . WithTimeout ( context . Background ( ) , timeout )
defer cancel ( )
cmd := exec . CommandContext ( ctx , c . name , c . args ... )
2019-11-10 11:42:51 +03:00
if env == nil {
cmd . Env = append ( os . Environ ( ) , fmt . Sprintf ( "LC_ALL=%s" , DefaultLocale ) )
} else {
cmd . Env = env
cmd . Env = append ( cmd . Env , fmt . Sprintf ( "LC_ALL=%s" , DefaultLocale ) )
}
2016-11-04 01:16:01 +03:00
cmd . Dir = dir
cmd . Stdout = stdout
cmd . Stderr = stderr
2019-05-11 18:29:17 +03:00
cmd . Stdin = stdin
2016-11-04 01:16:01 +03:00
if err := cmd . Start ( ) ; err != nil {
return err
}
2019-06-26 21:15:26 +03:00
pid := process . GetManager ( ) . Add ( fmt . Sprintf ( "%s %s %s [repo_path: %s]" , GitExecutable , c . name , strings . Join ( c . args , " " ) , dir ) , cmd )
defer process . GetManager ( ) . Remove ( pid )
2017-05-30 12:32:01 +03:00
if err := cmd . Wait ( ) ; err != nil {
return err
}
return ctx . Err ( )
2016-11-04 01:16:01 +03:00
}
2019-05-11 18:29:17 +03:00
// RunInDirTimeoutPipeline executes the command in given directory with given timeout,
// it pipes stdout and stderr to given io.Writer.
func ( c * Command ) RunInDirTimeoutPipeline ( timeout time . Duration , dir string , stdout , stderr io . Writer ) error {
return c . RunInDirTimeoutEnvPipeline ( nil , timeout , dir , stdout , stderr )
}
// RunInDirTimeoutFullPipeline executes the command in given directory with given timeout,
// it pipes stdout and stderr to given io.Writer, and stdin from the given io.Reader
func ( c * Command ) RunInDirTimeoutFullPipeline ( timeout time . Duration , dir string , stdout , stderr io . Writer , stdin io . Reader ) error {
return c . RunInDirTimeoutEnvFullPipeline ( nil , timeout , dir , stdout , stderr , stdin )
}
2016-11-04 01:16:01 +03:00
// RunInDirTimeout executes the command in given directory with given timeout,
// and returns stdout in []byte and error (combined with stderr).
func ( c * Command ) RunInDirTimeout ( timeout time . Duration , dir string ) ( [ ] byte , error ) {
2019-05-11 18:29:17 +03:00
return c . RunInDirTimeoutEnv ( nil , timeout , dir )
}
// RunInDirTimeoutEnv executes the command in given directory with given timeout,
// and returns stdout in []byte and error (combined with stderr).
func ( c * Command ) RunInDirTimeoutEnv ( env [ ] string , timeout time . Duration , dir string ) ( [ ] byte , error ) {
2016-11-04 01:16:01 +03:00
stdout := new ( bytes . Buffer )
stderr := new ( bytes . Buffer )
2019-05-11 18:29:17 +03:00
if err := c . RunInDirTimeoutEnvPipeline ( env , timeout , dir , stdout , stderr ) ; err != nil {
2016-11-04 01:16:01 +03:00
return nil , concatenateError ( err , stderr . String ( ) )
}
if stdout . Len ( ) > 0 {
log ( "stdout:\n%s" , stdout . Bytes ( ) [ : 1024 ] )
}
return stdout . Bytes ( ) , nil
}
// RunInDirPipeline executes the command in given directory,
// it pipes stdout and stderr to given io.Writer.
func ( c * Command ) RunInDirPipeline ( dir string , stdout , stderr io . Writer ) error {
2019-05-11 18:29:17 +03:00
return c . RunInDirFullPipeline ( dir , stdout , stderr , nil )
}
// RunInDirFullPipeline executes the command in given directory,
// it pipes stdout and stderr to given io.Writer.
func ( c * Command ) RunInDirFullPipeline ( dir string , stdout , stderr io . Writer , stdin io . Reader ) error {
return c . RunInDirTimeoutFullPipeline ( - 1 , dir , stdout , stderr , stdin )
2016-11-04 01:16:01 +03:00
}
2016-12-22 12:30:52 +03:00
// RunInDirBytes executes the command in given directory
2016-11-04 01:16:01 +03:00
// and returns stdout in []byte and error (combined with stderr).
func ( c * Command ) RunInDirBytes ( dir string ) ( [ ] byte , error ) {
return c . RunInDirTimeout ( - 1 , dir )
}
// RunInDir executes the command in given directory
// and returns stdout in string and error (combined with stderr).
func ( c * Command ) RunInDir ( dir string ) ( string , error ) {
2019-05-11 18:29:17 +03:00
return c . RunInDirWithEnv ( dir , nil )
}
// RunInDirWithEnv executes the command in given directory
// and returns stdout in string and error (combined with stderr).
func ( c * Command ) RunInDirWithEnv ( dir string , env [ ] string ) ( string , error ) {
stdout , err := c . RunInDirTimeoutEnv ( env , - 1 , dir )
2016-11-04 01:16:01 +03:00
if err != nil {
return "" , err
}
return string ( stdout ) , nil
}
2019-03-27 12:33:00 +03:00
// RunTimeout executes the command in default working directory with given timeout,
2016-11-04 01:16:01 +03:00
// and returns stdout in string and error (combined with stderr).
func ( c * Command ) RunTimeout ( timeout time . Duration ) ( string , error ) {
stdout , err := c . RunInDirTimeout ( timeout , "" )
if err != nil {
return "" , err
}
return string ( stdout ) , nil
}
2019-03-27 12:33:00 +03:00
// Run executes the command in default working directory
2016-11-04 01:16:01 +03:00
// and returns stdout in string and error (combined with stderr).
func ( c * Command ) Run ( ) ( string , error ) {
return c . RunTimeout ( - 1 )
}