2018-03-13 22:30:51 +01:00
package rclone
import (
2018-03-14 21:29:54 +01:00
"bufio"
2018-03-13 22:30:51 +01:00
"context"
"crypto/tls"
2018-03-14 21:29:54 +01:00
"fmt"
2018-05-22 20:48:17 +02:00
"io"
2018-03-18 20:30:02 +01:00
"math/rand"
2018-03-13 22:30:51 +01:00
"net"
"net/http"
"net/url"
"os"
"os/exec"
2018-03-15 21:22:14 +01:00
"sync"
2022-09-22 21:52:18 +02:00
"syscall"
2018-03-13 22:30:51 +01:00
"time"
2022-08-23 22:05:04 +02:00
"github.com/cenkalti/backoff/v4"
2018-03-13 22:30:51 +01:00
"github.com/restic/restic/internal/backend"
2022-06-12 14:38:19 +02:00
"github.com/restic/restic/internal/backend/limiter"
2023-06-08 13:04:34 +02:00
"github.com/restic/restic/internal/backend/location"
2018-03-13 22:30:51 +01:00
"github.com/restic/restic/internal/backend/rest"
"github.com/restic/restic/internal/debug"
"github.com/restic/restic/internal/errors"
"golang.org/x/net/http2"
)
// Backend is used to access data stored somewhere via rclone.
type Backend struct {
* rest . Backend
tr * http2 . Transport
cmd * exec . Cmd
waitCh <- chan struct { }
waitResult error
2018-03-15 21:22:14 +01:00
wg * sync . WaitGroup
conn * StdioConn
2018-03-13 22:30:51 +01:00
}
2023-06-08 13:04:34 +02:00
func NewFactory ( ) location . Factory {
2023-06-08 17:32:43 +02:00
return location . NewLimitedBackendFactory ( "rclone" , ParseConfig , location . NoPassword , Create , Open )
2023-06-08 13:04:34 +02:00
}
2018-03-13 22:30:51 +01:00
// run starts command with args and initializes the StdioConn.
2022-09-25 12:48:59 +02:00
func run ( command string , args ... string ) ( * StdioConn , * sync . WaitGroup , chan struct { } , func ( ) error , error ) {
2018-03-13 22:30:51 +01:00
cmd := exec . Command ( command , args ... )
2018-03-15 21:22:14 +01:00
2018-03-14 21:29:54 +01:00
p , err := cmd . StderrPipe ( )
if err != nil {
2022-09-25 12:48:59 +02:00
return nil , nil , nil , nil , err
2018-03-14 21:29:54 +01:00
}
2018-03-15 21:22:14 +01:00
var wg sync . WaitGroup
2022-09-25 12:48:59 +02:00
waitCh := make ( chan struct { } )
2018-03-15 21:22:14 +01:00
2018-03-14 21:29:54 +01:00
// start goroutine to add a prefix to all messages printed by to stderr by rclone
2018-03-15 21:22:14 +01:00
wg . Add ( 1 )
2018-03-14 21:29:54 +01:00
go func ( ) {
2018-03-15 21:22:14 +01:00
defer wg . Done ( )
2022-09-25 12:48:59 +02:00
defer close ( waitCh )
2018-03-14 21:29:54 +01:00
sc := bufio . NewScanner ( p )
for sc . Scan ( ) {
fmt . Fprintf ( os . Stderr , "rclone: %v\n" , sc . Text ( ) )
}
2022-09-25 12:48:59 +02:00
debug . Log ( "command has exited, closing waitCh" )
2018-03-14 21:29:54 +01:00
} ( )
2018-03-13 22:30:51 +01:00
r , stdin , err := os . Pipe ( )
if err != nil {
2022-09-25 12:48:59 +02:00
return nil , nil , nil , nil , err
2018-03-13 22:30:51 +01:00
}
stdout , w , err := os . Pipe ( )
if err != nil {
2021-01-30 16:46:34 +01:00
// close first pipe and ignore subsequent errors
_ = r . Close ( )
_ = stdin . Close ( )
2022-09-25 12:48:59 +02:00
return nil , nil , nil , nil , err
2018-03-13 22:30:51 +01:00
}
cmd . Stdin = r
cmd . Stdout = w
bg , err := backend . StartForeground ( cmd )
2020-07-24 22:29:37 +02:00
// close rclone side of pipes
errR := r . Close ( )
errW := w . Close ( )
// return first error
if err == nil {
err = errR
}
if err == nil {
err = errW
}
2018-03-13 22:30:51 +01:00
if err != nil {
2022-09-20 21:26:01 +02:00
if backend . IsErrDot ( err ) {
2022-09-25 12:48:59 +02:00
return nil , nil , nil , nil , errors . Errorf ( "cannot implicitly run relative executable %v found in current directory, use -o rclone.program=./<program> to override" , cmd . Path )
2022-09-20 21:26:01 +02:00
}
2022-09-25 12:48:59 +02:00
return nil , nil , nil , nil , err
2018-03-13 22:30:51 +01:00
}
c := & StdioConn {
2020-07-25 23:51:02 +02:00
receive : stdout ,
send : stdin ,
cmd : cmd ,
2018-03-13 22:30:51 +01:00
}
2022-09-25 12:48:59 +02:00
return c , & wg , waitCh , bg , nil
2018-03-13 22:30:51 +01:00
}
2018-05-22 20:48:17 +02:00
// wrappedConn adds bandwidth limiting capabilities to the StdioConn by
// wrapping the Read/Write methods.
type wrappedConn struct {
* StdioConn
io . Reader
io . Writer
}
2021-08-01 09:11:50 +02:00
func ( c * wrappedConn ) Read ( p [ ] byte ) ( int , error ) {
2018-05-22 20:48:17 +02:00
return c . Reader . Read ( p )
}
2021-08-01 09:11:50 +02:00
func ( c * wrappedConn ) Write ( p [ ] byte ) ( int , error ) {
2018-05-22 20:48:17 +02:00
return c . Writer . Write ( p )
}
2021-08-01 09:11:50 +02:00
func wrapConn ( c * StdioConn , lim limiter . Limiter ) * wrappedConn {
wc := & wrappedConn {
2018-05-22 20:48:17 +02:00
StdioConn : c ,
Reader : c ,
Writer : c ,
}
if lim != nil {
wc . Reader = lim . Downstream ( c )
wc . Writer = lim . UpstreamWriter ( c )
}
return wc
}
2018-03-13 22:30:51 +01:00
// New initializes a Backend and starts the process.
2023-06-08 13:11:34 +02:00
func newBackend ( ctx context . Context , cfg Config , lim limiter . Limiter ) ( * Backend , error ) {
2018-03-13 22:30:51 +01:00
var (
args [ ] string
err error
)
// build program args, start with the program
if cfg . Program != "" {
a , err := backend . SplitShellStrings ( cfg . Program )
if err != nil {
return nil , err
}
args = append ( args , a ... )
}
// then add the arguments
if cfg . Args != "" {
a , err := backend . SplitShellStrings ( cfg . Args )
if err != nil {
return nil , err
}
args = append ( args , a ... )
}
// finally, add the remote
args = append ( args , cfg . Remote )
arg0 , args := args [ 0 ] , args [ 1 : ]
debug . Log ( "running command: %v %v" , arg0 , args )
2022-09-25 12:48:59 +02:00
stdioConn , wg , waitCh , bg , err := run ( arg0 , args ... )
2018-03-13 22:30:51 +01:00
if err != nil {
return nil , err
}
2018-05-22 20:48:17 +02:00
var conn net . Conn = stdioConn
if lim != nil {
conn = wrapConn ( stdioConn , lim )
}
2018-03-15 21:22:14 +01:00
dialCount := 0
2018-03-13 22:30:51 +01:00
tr := & http2 . Transport {
AllowHTTP : true , // this is not really HTTP, just stdin/stdout
DialTLS : func ( network , address string , cfg * tls . Config ) ( net . Conn , error ) {
debug . Log ( "new connection requested, %v %v" , network , address )
2018-03-15 21:22:14 +01:00
if dialCount > 0 {
2020-08-01 12:06:30 +02:00
// the connection to the child process is already closed
2022-08-23 22:05:04 +02:00
return nil , backoff . Permanent ( errors . New ( "rclone stdio connection already closed" ) )
2018-03-15 21:22:14 +01:00
}
dialCount ++
2018-03-13 22:30:51 +01:00
return conn , nil
} ,
}
2021-08-26 18:12:08 +02:00
cmd := stdioConn . cmd
2018-03-13 22:30:51 +01:00
be := & Backend {
tr : tr ,
cmd : cmd ,
waitCh : waitCh ,
2018-05-22 20:48:17 +02:00
conn : stdioConn ,
2018-03-15 21:22:14 +01:00
wg : wg ,
2018-03-13 22:30:51 +01:00
}
2023-06-08 13:11:34 +02:00
ctx , cancel := context . WithCancel ( ctx )
2022-09-25 12:48:59 +02:00
defer cancel ( )
2018-03-15 21:22:14 +01:00
wg . Add ( 1 )
2018-03-13 22:30:51 +01:00
go func ( ) {
2018-03-15 21:22:14 +01:00
defer wg . Done ( )
2022-09-25 12:48:59 +02:00
<- waitCh
cancel ( )
// according to the documentation of StdErrPipe, Wait() must only be called after the former has completed
2018-03-13 22:30:51 +01:00
err := cmd . Wait ( )
debug . Log ( "Wait returned %v" , err )
be . waitResult = err
2021-01-30 16:46:34 +01:00
// close our side of the pipes to rclone, ignore errors
_ = stdioConn . CloseAll ( )
2018-03-13 22:30:51 +01:00
} ( )
// send an HTTP request to the base URL, see if the server is there
2022-10-09 10:21:30 +02:00
client := http . Client {
2018-10-21 19:58:40 +02:00
Transport : debug . RoundTripper ( tr ) ,
2021-11-07 17:47:18 +01:00
Timeout : cfg . Timeout ,
2018-03-13 22:30:51 +01:00
}
2018-03-18 20:30:02 +01:00
// request a random file which does not exist. we just want to test when
// rclone is able to accept HTTP requests.
url := fmt . Sprintf ( "http://localhost/file-%d" , rand . Uint64 ( ) )
2021-01-30 20:43:53 +01:00
req , err := http . NewRequestWithContext ( ctx , http . MethodGet , url , nil )
2018-03-13 22:30:51 +01:00
if err != nil {
return nil , err
}
req . Header . Set ( "Accept" , rest . ContentTypeV2 )
2022-10-09 10:21:30 +02:00
res , err := client . Do ( req )
2018-03-13 22:30:51 +01:00
if err != nil {
2021-01-30 16:46:34 +01:00
// ignore subsequent errors
_ = bg ( )
2018-03-13 22:30:51 +01:00
_ = cmd . Process . Kill ( )
2022-09-22 21:52:18 +02:00
// wait for rclone to exit
wg . Wait ( )
// try to return the program exit code if communication with rclone has failed
2022-12-02 20:46:02 +01:00
if be . waitResult != nil && ( errors . Is ( err , context . Canceled ) || errors . Is ( err , io . ErrUnexpectedEOF ) || errors . Is ( err , syscall . EPIPE ) || errors . Is ( err , os . ErrClosed ) ) {
2022-09-22 21:52:18 +02:00
err = be . waitResult
}
return nil , fmt . Errorf ( "error talking HTTP to rclone: %w" , err )
2018-03-13 22:30:51 +01:00
}
debug . Log ( "HTTP status %q returned, moving instance to background" , res . Status )
2021-01-30 16:46:34 +01:00
err = bg ( )
if err != nil {
return nil , fmt . Errorf ( "error moving process to background: %w" , err )
}
2018-03-13 22:30:51 +01:00
return be , nil
}
// Open starts an rclone process with the given config.
2023-06-08 13:11:34 +02:00
func Open ( ctx context . Context , cfg Config , lim limiter . Limiter ) ( * Backend , error ) {
be , err := newBackend ( ctx , cfg , lim )
2018-03-13 22:30:51 +01:00
if err != nil {
return nil , err
}
url , err := url . Parse ( "http://localhost/" )
if err != nil {
return nil , err
}
restConfig := rest . Config {
2018-03-24 18:37:36 +01:00
Connections : cfg . Connections ,
2018-03-13 22:30:51 +01:00
URL : url ,
}
2023-06-08 13:11:34 +02:00
restBackend , err := rest . Open ( ctx , restConfig , debug . RoundTripper ( be . tr ) )
2018-03-13 22:30:51 +01:00
if err != nil {
2020-04-10 12:13:12 +02:00
_ = be . Close ( )
2018-03-13 22:30:51 +01:00
return nil , err
}
be . Backend = restBackend
return be , nil
}
2020-04-10 12:08:52 +02:00
// Create initializes a new restic repo with rclone.
2023-06-08 13:04:34 +02:00
func Create ( ctx context . Context , cfg Config , lim limiter . Limiter ) ( * Backend , error ) {
be , err := newBackend ( ctx , cfg , lim )
2018-03-13 22:30:51 +01:00
if err != nil {
return nil , err
}
debug . Log ( "new backend created" )
url , err := url . Parse ( "http://localhost/" )
if err != nil {
return nil , err
}
restConfig := rest . Config {
2020-09-19 19:11:28 +02:00
Connections : cfg . Connections ,
2018-03-13 22:30:51 +01:00
URL : url ,
}
2020-04-10 12:08:52 +02:00
restBackend , err := rest . Create ( ctx , restConfig , debug . RoundTripper ( be . tr ) )
2018-03-13 22:30:51 +01:00
if err != nil {
2018-03-14 22:28:12 +01:00
_ = be . Close ( )
2018-03-13 22:30:51 +01:00
return nil , err
}
be . Backend = restBackend
return be , nil
}
2018-03-15 21:22:14 +01:00
const waitForExit = 5 * time . Second
2018-03-13 22:30:51 +01:00
// Close terminates the backend.
func ( be * Backend ) Close ( ) error {
2018-03-15 19:00:25 +01:00
debug . Log ( "exiting rclone" )
2018-03-13 22:30:51 +01:00
be . tr . CloseIdleConnections ( )
2018-03-15 21:22:14 +01:00
select {
case <- be . waitCh :
debug . Log ( "rclone exited" )
case <- time . After ( waitForExit ) :
debug . Log ( "timeout, closing file descriptors" )
2020-07-24 23:27:47 +02:00
err := be . conn . CloseAll ( )
2018-03-15 21:22:14 +01:00
if err != nil {
return err
}
}
be . wg . Wait ( )
2018-03-13 22:30:51 +01:00
debug . Log ( "wait for rclone returned: %v" , be . waitResult )
return be . waitResult
}