@ -13,6 +13,9 @@ import (
"errors"
"fmt"
"io"
"maps"
"mime"
"mime/multipart"
"net"
"net/http"
"net/http/httputil"
@ -28,6 +31,7 @@ import (
"sync"
"time"
"github.com/google/uuid"
"tailscale.com/client/tailscale/apitype"
"tailscale.com/clientupdate"
"tailscale.com/envknob"
@ -57,6 +61,7 @@ import (
"tailscale.com/util/mak"
"tailscale.com/util/osdiag"
"tailscale.com/util/osuser"
"tailscale.com/util/progresstracking"
"tailscale.com/util/rands"
"tailscale.com/version"
"tailscale.com/wgengine/magicsock"
@ -1529,9 +1534,17 @@ func (h *Handler) serveFileTargets(w http.ResponseWriter, r *http.Request) {
// The Windows client currently (2021-11-30) uses the peerapi (/v0/put/)
// directly, as the Windows GUI always runs in tun mode anyway.
//
// In addition to single file PUTs, this endpoint accepts multipart file
// POSTS encoded as multipart/form-data. Each part must include a
// "Content-Length" in the MIME header indicating the size of the file.
// The first part should be an application/json file that contains a JSON map
// of filename -> length, which we can use for tracking progress even before
// reading the file parts.
//
// URL format:
//
// - PUT /localapi/v0/file-put/:stableID/:escaped-filename
// - POST /localapi/v0/file-put/:stableID
func ( h * Handler ) serveFilePut ( w http . ResponseWriter , r * http . Request ) {
metricFilePutCalls . Add ( 1 )
@ -1539,10 +1552,12 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
http . Error ( w , "file access denied" , http . StatusForbidden )
return
}
if r . Method != "PUT" {
if r . Method != "PUT" && r . Method != "POST" {
http . Error ( w , "want PUT to put file" , http . StatusBadRequest )
return
}
fts , err := h . b . FileTargets ( )
if err != nil {
http . Error ( w , err . Error ( ) , http . StatusInternalServerError )
@ -1554,16 +1569,22 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
http . Error ( w , "misconfigured" , http . StatusInternalServerError )
return
}
stableIDStr , filenameEscaped , ok := strings . Cut ( upath , "/" )
var peerIDStr , filenameEscaped string
if r . Method == "PUT" {
ok := false
peerIDStr , filenameEscaped , ok = strings . Cut ( upath , "/" )
if ! ok {
http . Error ( w , "bogus URL" , http . StatusBadRequest )
return
}
stableID := tailcfg . StableNodeID ( stableIDStr )
} else {
peerIDStr = upath
}
peerID := tailcfg . StableNodeID ( peerIDStr )
var ft * apitype . FileTarget
for _ , x := range fts {
if x . Node . StableID == stable ID {
if x . Node . StableID == peer ID {
ft = x
break
}
@ -1578,20 +1599,181 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
return
}
// Report progress on outgoing files every 5 seconds
outgoingFiles := make ( map [ string ] ipn . OutgoingFile )
t := time . NewTicker ( 5 * time . Second )
progressUpdates := make ( chan ipn . OutgoingFile )
defer close ( progressUpdates )
go func ( ) {
defer t . Stop ( )
defer h . b . UpdateOutgoingFiles ( outgoingFiles )
for {
select {
case u , ok := <- progressUpdates :
if ! ok {
return
}
outgoingFiles [ u . ID ] = u
case <- t . C :
h . b . UpdateOutgoingFiles ( outgoingFiles )
}
}
} ( )
switch r . Method {
case "PUT" :
file := ipn . OutgoingFile {
ID : uuid . Must ( uuid . NewRandom ( ) ) . String ( ) ,
PeerID : peerID ,
Name : filenameEscaped ,
DeclaredSize : r . ContentLength ,
}
h . singleFilePut ( r . Context ( ) , progressUpdates , w , r . Body , dstURL , file )
case "POST" :
h . multiFilePost ( progressUpdates , w , r , peerID , dstURL )
default :
http . Error ( w , "want PUT to put file" , http . StatusBadRequest )
return
}
}
func ( h * Handler ) multiFilePost ( progressUpdates chan ( ipn . OutgoingFile ) , w http . ResponseWriter , r * http . Request , peerID tailcfg . StableNodeID , dstURL * url . URL ) {
_ , params , err := mime . ParseMediaType ( r . Header . Get ( "Content-Type" ) )
if err != nil {
http . Error ( w , fmt . Sprintf ( "invalid Content-Type for multipart POST: %s" , err ) , http . StatusBadRequest )
return
}
ww := & multiFilePostResponseWriter { }
defer func ( ) {
if err := ww . Flush ( w ) ; err != nil {
h . logf ( "error: multiFilePostResponseWriter.Flush(): %s" , err )
}
} ( )
outgoingFilesByName := make ( map [ string ] ipn . OutgoingFile )
first := true
mr := multipart . NewReader ( r . Body , params [ "boundary" ] )
for {
part , err := mr . NextPart ( )
if err == io . EOF {
// No more parts.
return
} else if err != nil {
http . Error ( ww , fmt . Sprintf ( "failed to decode multipart/form-data: %s" , err ) , http . StatusBadRequest )
return
}
if first {
first = false
if part . Header . Get ( "Content-Type" ) != "application/json" {
http . Error ( ww , "first MIME part must be a JSON map of filename -> size" , http . StatusBadRequest )
return
}
var manifest map [ string ] int64
err := json . NewDecoder ( part ) . Decode ( & manifest )
if err != nil {
http . Error ( ww , fmt . Sprintf ( "invalid manifest: %s" , err ) , http . StatusBadRequest )
return
}
for filename , size := range manifest {
file := ipn . OutgoingFile {
ID : uuid . Must ( uuid . NewRandom ( ) ) . String ( ) ,
Name : filename ,
PeerID : peerID ,
DeclaredSize : size ,
}
outgoingFilesByName [ filename ] = file
progressUpdates <- file
}
continue
}
if ! h . singleFilePut ( r . Context ( ) , progressUpdates , ww , part , dstURL , outgoingFilesByName [ part . FileName ( ) ] ) {
return
}
if ww . statusCode >= 400 {
// put failed, stop immediately
h . logf ( "error: singleFilePut: failed with status %d" , ww . statusCode )
return
}
}
}
// multiFilePostResponseWriter is a buffering http.ResponseWriter that can be
// reused across multiple singleFilePut calls and then flushed to the client
// when all files have been PUT.
type multiFilePostResponseWriter struct {
header http . Header
statusCode int
body * bytes . Buffer
}
func ( ww * multiFilePostResponseWriter ) Header ( ) http . Header {
if ww . header == nil {
ww . header = make ( http . Header )
}
return ww . header
}
func ( ww * multiFilePostResponseWriter ) WriteHeader ( statusCode int ) {
ww . statusCode = statusCode
}
func ( ww * multiFilePostResponseWriter ) Write ( p [ ] byte ) ( int , error ) {
if ww . body == nil {
ww . body = bytes . NewBuffer ( nil )
}
return ww . body . Write ( p )
}
func ( ww * multiFilePostResponseWriter ) Flush ( w http . ResponseWriter ) error {
maps . Copy ( w . Header ( ) , ww . Header ( ) )
w . WriteHeader ( ww . statusCode )
_ , err := io . Copy ( w , ww . body )
return err
}
func ( h * Handler ) singleFilePut (
ctx context . Context ,
progressUpdates chan ( ipn . OutgoingFile ) ,
w http . ResponseWriter ,
body io . Reader ,
dstURL * url . URL ,
outgoingFile ipn . OutgoingFile ,
) bool {
outgoingFile . Started = time . Now ( )
body = progresstracking . NewReader ( body , 1 * time . Second , func ( n int , err error ) {
outgoingFile . Sent = int64 ( n )
progressUpdates <- outgoingFile
} )
fail := func ( ) {
outgoingFile . Finished = true
outgoingFile . Succeeded = false
progressUpdates <- outgoingFile
}
// Before we PUT a file we check to see if there are any existing partial file and if so,
// we resume the upload from where we left off by sending the remaining file instead of
// the full file.
var offset int64
var resumeDuration time . Duration
remainingBody := io . Reader ( r . Body )
remainingBody := io . Reader ( b ody)
client := & http . Client {
Transport : h . b . Dialer ( ) . PeerAPITransport ( ) ,
Timeout : 10 * time . Second ,
}
req , err := http . NewRequestWithContext ( r . Context ( ) , "GET" , dstURL . String ( ) + "/v0/put/" + filenameEscaped , nil )
req , err := http . NewRequestWithContext ( ctx , "GET" , dstURL . String ( ) + "/v0/put/" + outgoingFile. Name , nil )
if err != nil {
http . Error ( w , "bogus peer URL" , http . StatusInternalServerError )
return
fail ( )
return false
}
switch resp , err := client . Do ( req ) ; {
case err != nil :
@ -1603,7 +1785,7 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
default :
resumeStart := time . Now ( )
dec := json . NewDecoder ( resp . Body )
offset , remainingBody , err = taildrop . ResumeReader ( r. B ody, func ( ) ( out taildrop . BlockChecksum , err error ) {
offset , remainingBody , err = taildrop . ResumeReader ( b ody, func ( ) ( out taildrop . BlockChecksum , err error ) {
err = dec . Decode ( & out )
return out , err
} )
@ -1613,12 +1795,13 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
resumeDuration = time . Since ( resumeStart ) . Round ( time . Millisecond )
}
outReq , err := http . NewRequestWithContext ( r. Context ( ) , "PUT" , "http://peer/v0/put/" + filenameEscaped , remainingBody )
outReq , err := http . NewRequestWithContext ( ctx, "PUT" , "http://peer/v0/put/" + outgoingFile . Name , remainingBody )
if err != nil {
http . Error ( w , "bogus outreq" , http . StatusInternalServerError )
return
fail ( )
return false
}
outReq . ContentLength = r. ContentLength
outReq . ContentLength = outgoingFile. DeclaredSize
if offset > 0 {
h . logf ( "resuming put at offset %d after %v" , offset , resumeDuration )
rangeHdr , _ := httphdr . FormatRange ( [ ] httphdr . Range { { Start : offset , Length : 0 } } )
@ -1631,6 +1814,12 @@ func (h *Handler) serveFilePut(w http.ResponseWriter, r *http.Request) {
rp := httputil . NewSingleHostReverseProxy ( dstURL )
rp . Transport = h . b . Dialer ( ) . PeerAPITransport ( )
rp . ServeHTTP ( w , outReq )
outgoingFile . Finished = true
outgoingFile . Succeeded = true
progressUpdates <- outgoingFile
return true
}
func ( h * Handler ) serveSetDNS ( w http . ResponseWriter , r * http . Request ) {