@ -165,18 +165,29 @@ func HTTP(ctx *context.Context) {
}
}
}
}
callback := func ( rpc string , input [ ] byt e) {
callback := func ( rpc string , input * os . Fil e) {
if rpc != "receive-pack" || isWiki {
if rpc != "receive-pack" || isWiki {
return
return
}
}
var lastLine int64 = 0
var (
head = make ( [ ] byte , 4 ) // 00+size
n int
err error
)
for {
for {
head := input [ lastLine : lastLine + 2 ]
n , err = input . Read ( head )
if err != nil && err != io . EOF {
log . Error ( 4 , "read head: %v" , err )
return
} else if n < 4 {
break
}
if head [ 0 ] == '0' && head [ 1 ] == '0' {
if head [ 0 ] == '0' && head [ 1 ] == '0' {
size , err := strconv . ParseInt ( string ( input [ lastLine + 2 : lastLine + 4 ] ) , 16 , 32 )
size , err := strconv . ParseInt ( string ( head [ 2 : 4 ] ) , 16 , 32 )
if err != nil {
if err != nil {
log . Error ( 4 , "%v" , err )
log . Error ( 4 , "parse size: %v" , err )
return
return
}
}
@ -185,7 +196,16 @@ func HTTP(ctx *context.Context) {
break
break
}
}
line := input [ lastLine : lastLine + size ]
line := make ( [ ] byte , size )
n , err = input . Read ( line )
if err != nil {
log . Error ( 4 , "read line: %v" , err )
return
} else if n < int ( size ) {
log . Error ( 4 , "didn't read enough bytes: expect %d got %d" , size , n )
break
}
idx := bytes . IndexRune ( line , '\000' )
idx := bytes . IndexRune ( line , '\000' )
if idx > - 1 {
if idx > - 1 {
line = line [ : idx ]
line = line [ : idx ]
@ -193,7 +213,7 @@ func HTTP(ctx *context.Context) {
fields := strings . Fields ( string ( line ) )
fields := strings . Fields ( string ( line ) )
if len ( fields ) >= 3 {
if len ( fields ) >= 3 {
oldCommitId := fields [ 0 ] [ 4 : ]
oldCommitId := fields [ 0 ]
newCommitId := fields [ 1 ]
newCommitId := fields [ 1 ]
refFullName := fields [ 2 ]
refFullName := fields [ 2 ]
@ -211,7 +231,6 @@ func HTTP(ctx *context.Context) {
}
}
}
}
lastLine = lastLine + size
} else {
} else {
break
break
}
}
@ -230,7 +249,7 @@ func HTTP(ctx *context.Context) {
type serviceConfig struct {
type serviceConfig struct {
UploadPack bool
UploadPack bool
ReceivePack bool
ReceivePack bool
OnSucceed func ( rpc string , input [ ] byt e)
OnSucceed func ( rpc string , input * os . Fil e)
}
}
type serviceHandler struct {
type serviceHandler struct {
@ -348,7 +367,7 @@ func serviceRPC(h serviceHandler, service string) {
var (
var (
reqBody = h . r . Body
reqBody = h . r . Body
input [ ] byte
tmpFilename string
br io . Reader
br io . Reader
err error
err error
)
)
@ -371,7 +390,6 @@ func serviceRPC(h serviceHandler, service string) {
return
return
}
}
defer os . Remove ( tmpfile . Name ( ) )
defer os . Remove ( tmpfile . Name ( ) )
defer tmpfile . Close ( )
_ , err = io . Copy ( tmpfile , reqBody )
_ , err = io . Copy ( tmpfile , reqBody )
if err != nil {
if err != nil {
@ -379,23 +397,42 @@ func serviceRPC(h serviceHandler, service string) {
h . w . WriteHeader ( http . StatusInternalServerError )
h . w . WriteHeader ( http . StatusInternalServerError )
return
return
}
}
tmpfile . Close ( )
tmpFilename = tmpfile . Name ( )
tmpfile , err = os . Open ( tmpFilename )
if err != nil {
log . GitLogger . Error ( 2 , "fail to open temporary file: %v" , err )
h . w . WriteHeader ( http . StatusInternalServerError )
return
}
defer tmpfile . Close ( )
br = tmpfile
br = tmpfile
} else {
} else {
br = reqBody
br = reqBody
}
}
var stderr bytes . Buffer
cmd := exec . Command ( "git" , service , "--stateless-rpc" , h . dir )
cmd := exec . Command ( "git" , service , "--stateless-rpc" , h . dir )
cmd . Dir = h . dir
cmd . Dir = h . dir
cmd . Stdout = h . w
cmd . Stdout = h . w
cmd . Stderr = & stderr
cmd . Stdin = br
cmd . Stdin = br
if err := cmd . Run ( ) ; err != nil {
if err := cmd . Run ( ) ; err != nil {
log . GitLogger . Error ( 2 , "fail to serve RPC(%s): %v" , service , err )
log . GitLogger . Error ( 2 , "fail to serve RPC(%s): %v - %s " , service , err , std err)
h . w . WriteHeader ( http . StatusInternalServerError )
h . w . WriteHeader ( http . StatusInternalServerError )
return
return
}
}
if h . cfg . OnSucceed != nil {
if h . cfg . OnSucceed != nil {
input , err := os . Open ( tmpFilename )
if err != nil {
log . GitLogger . Error ( 2 , "fail to open temporary file: %v" , err )
h . w . WriteHeader ( http . StatusInternalServerError )
return
}
defer input . Close ( )
h . cfg . OnSucceed ( service , input )
h . cfg . OnSucceed ( service , input )
}
}
}
}