forked from TrueCloudLab/rclone
librclone: exports, errors, docs and examples #4891
- rename C exports to be namespaced with Rclone prefix - fix error handling in RcloneRPC - add more examples - add more docs - add README - simplify ctest Makefile
This commit is contained in:
parent
316e65589b
commit
5db88fed2b
4 changed files with 140 additions and 62 deletions
31
librclone/README.md
Normal file
31
librclone/README.md
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
# librclone
|
||||||
|
|
||||||
|
This directory contains code to build rclone as a C library and the
|
||||||
|
shims for accessing rclone from C.
|
||||||
|
|
||||||
|
The shims are a thin wrapper over the rclone RPC.
|
||||||
|
|
||||||
|
Build a shared library like this:
|
||||||
|
|
||||||
|
go build --buildmode=c-shared -o librclone.so github.com/rclone/rclone/librclone
|
||||||
|
|
||||||
|
Build a static library like this:
|
||||||
|
|
||||||
|
go build --buildmode=c-archive -o librclone.a github.com/rclone/rclone/librclone
|
||||||
|
|
||||||
|
Both the above commands will also generate `librclone.h` which should
|
||||||
|
be `#include`d in `C` programs wishing to use the library.
|
||||||
|
|
||||||
|
The library will depend on `libdl` and `libpthread`.
|
||||||
|
|
||||||
|
## Documentation
|
||||||
|
|
||||||
|
For documentation see the Go documentation for:
|
||||||
|
|
||||||
|
- [RcloneInitialize](https://pkg.go.dev/github.com/rclone/rclone/librclone#RcloneInitialize)
|
||||||
|
- [RcloneFinalize](https://pkg.go.dev/github.com/rclone/rclone/librclone#RcloneFinalize)
|
||||||
|
- [RcloneRPC](https://pkg.go.dev/github.com/rclone/rclone/librclone#RcloneRPC)
|
||||||
|
|
||||||
|
## C Example
|
||||||
|
|
||||||
|
There is an example program `ctest.c` with Makefile in the `ctest` subdirectory
|
|
@ -1,21 +1,13 @@
|
||||||
CFLAGS = -g -Wall
|
CFLAGS = -g -Wall
|
||||||
LDFLAGS = -L. -lrclone -lpthread -ldl
|
LDFLAGS = -L. -lrclone -lpthread -ldl
|
||||||
|
|
||||||
static: ctest
|
ctest: ctest.o librclone.a
|
||||||
|
|
||||||
shared:
|
|
||||||
go build --buildmode=c-shared -o librclone.so github.com/rclone/rclone/librclone
|
|
||||||
|
|
||||||
ctest: ctest.o librclone.h
|
|
||||||
$(CC) $(CFLAGS) -o $@ $< $(LDFLAGS)
|
$(CC) $(CFLAGS) -o $@ $< $(LDFLAGS)
|
||||||
|
|
||||||
ctest.o: ctest.c librclone.h
|
ctest.o: ctest.c librclone.h
|
||||||
$(CC) $(CFLAGS) -c $^ $(LDFLAGS)
|
$(CC) $(CFLAGS) -c $^ $(LDFLAGS)
|
||||||
|
|
||||||
build:
|
librclone.a librclone.h:
|
||||||
go build
|
|
||||||
|
|
||||||
librclone.h:
|
|
||||||
go build --buildmode=c-archive -o librclone.a github.com/rclone/rclone/librclone
|
go build --buildmode=c-archive -o librclone.a github.com/rclone/rclone/librclone
|
||||||
|
|
||||||
clean:
|
clean:
|
||||||
|
|
|
@ -4,29 +4,60 @@
|
||||||
#include <dlfcn.h>
|
#include <dlfcn.h>
|
||||||
#include "librclone.h"
|
#include "librclone.h"
|
||||||
|
|
||||||
// copy file using "operations/copyfile" command
|
void testRPC(char *method, char *in) {
|
||||||
void testCopyFile() {
|
struct RcloneRPC_return out = RcloneRPC(method, in);
|
||||||
struct CRPC_return res = CRPC("operations/copyfile", "{ \"srcFs\": \"/tmp\", \"srcRemote\": \"tmpfile\", \"dstFs\": \"/tmp\", \"dstRemote\": \"tmpfile2\" }");
|
printf("status: %d\n", out.r1);
|
||||||
printf("%d\n", res.r1); // status
|
printf("output: %s\n", out.r0);
|
||||||
printf("%s\n", res.r0); // output
|
free(out.r0);
|
||||||
free(res.r0);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// noop command
|
// noop command
|
||||||
void testNoOp() {
|
void testNoOp() {
|
||||||
struct CRPC_return res = CRPC("rc/noop", "{ \"p1\": [1,\"2\",null,4], \"p2\": { \"a\":1, \"b\":2 } }");
|
printf("test rc/noop\n");
|
||||||
printf("%d\n", res.r1); // status
|
testRPC("rc/noop",
|
||||||
printf("%s\n", res.r0); // output
|
"{"
|
||||||
free(res.r0);
|
" \"p1\": [1,\"2\",null,4],"
|
||||||
|
" \"p2\": { \"a\":1, \"b\":2 } "
|
||||||
|
"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
// error command
|
||||||
|
void testError() {
|
||||||
|
printf("test rc/error\n");
|
||||||
|
testRPC("rc/error",
|
||||||
|
"{"
|
||||||
|
" \"p1\": [1,\"2\",null,4],"
|
||||||
|
" \"p2\": { \"a\":1, \"b\":2 } "
|
||||||
|
"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
// copy file using "operations/copyfile" command
|
||||||
|
void testCopyFile() {
|
||||||
|
printf("test operations/copyfile\n");
|
||||||
|
testRPC("operations/copyfile",
|
||||||
|
"{"
|
||||||
|
"\"srcFs\": \"/tmp\","
|
||||||
|
"\"srcRemote\": \"tmpfile\","
|
||||||
|
"\"dstFs\": \"/tmp\","
|
||||||
|
"\"dstRemote\": \"tmpfile2\""
|
||||||
|
"}");
|
||||||
|
}
|
||||||
|
|
||||||
|
// list the remotes
|
||||||
|
void testListRemotes() {
|
||||||
|
printf("test operations/listremotes\n");
|
||||||
|
testRPC("config/listremotes", "{}");
|
||||||
}
|
}
|
||||||
|
|
||||||
int main(int argc, char** argv) {
|
int main(int argc, char** argv) {
|
||||||
printf("c main begin\n");
|
printf("c main begin\n");
|
||||||
Cinit();
|
RcloneInitialize();
|
||||||
|
|
||||||
//testNoOp();
|
testNoOp();
|
||||||
|
testError();
|
||||||
testCopyFile();
|
testCopyFile();
|
||||||
|
testListRemotes();
|
||||||
|
|
||||||
Cdestroy();
|
RcloneFinalize();
|
||||||
return EXIT_SUCCESS;
|
return EXIT_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,13 +1,30 @@
|
||||||
// Package exports exports function for c library
|
// Package librclone exports shims for C library use
|
||||||
|
//
|
||||||
|
// This directory contains code to build rclone as a C library and the
|
||||||
|
// shims for accessing rclone from C.
|
||||||
|
//
|
||||||
|
// The shims are a thin wrapper over the rclone RPC.
|
||||||
|
//
|
||||||
|
// Build a shared library like this:
|
||||||
|
//
|
||||||
|
// go build --buildmode=c-shared -o librclone.so github.com/rclone/rclone/librclone
|
||||||
|
//
|
||||||
|
// Build a static library like this:
|
||||||
|
//
|
||||||
|
// go build --buildmode=c-archive -o librclone.a github.com/rclone/rclone/librclone
|
||||||
|
//
|
||||||
|
// Both the above commands will also generate `librclone.h` which should
|
||||||
|
// be `#include`d in `C` programs wishing to use the library.
|
||||||
|
//
|
||||||
|
// The library will depend on `libdl` and `libpthread`.
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"C"
|
"C"
|
||||||
"bytes"
|
|
||||||
"context"
|
"context"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"io"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"runtime"
|
"runtime"
|
||||||
"strings"
|
"strings"
|
||||||
|
@ -18,40 +35,51 @@ import (
|
||||||
"github.com/rclone/rclone/fs/rc/jobs"
|
"github.com/rclone/rclone/fs/rc/jobs"
|
||||||
|
|
||||||
_ "github.com/rclone/rclone/backend/all" // import all backends
|
_ "github.com/rclone/rclone/backend/all" // import all backends
|
||||||
_ "github.com/rclone/rclone/cmd/all" // import all commands
|
|
||||||
_ "github.com/rclone/rclone/lib/plugin" // import plugins
|
_ "github.com/rclone/rclone/lib/plugin" // import plugins
|
||||||
)
|
)
|
||||||
|
|
||||||
func init() {
|
// RcloneInitialize initializes rclone as a library
|
||||||
// do nothing
|
//
|
||||||
}
|
//export RcloneInitialize
|
||||||
|
func RcloneInitialize() {
|
||||||
// call to init the library
|
|
||||||
//export Cinit
|
|
||||||
func Cinit() {
|
|
||||||
// TODO: what need to be initialized manually?
|
// TODO: what need to be initialized manually?
|
||||||
}
|
}
|
||||||
|
|
||||||
// call to destroy the whole thing
|
// RcloneFinalize finalizes the library
|
||||||
//export Cdestroy
|
//
|
||||||
func Cdestroy() {
|
//export RcloneFinalize
|
||||||
|
func RcloneFinalize() {
|
||||||
// TODO: how to clean up? what happens when rcserver terminates?
|
// TODO: how to clean up? what happens when rcserver terminates?
|
||||||
// what about unfinished async jobs?
|
// what about unfinished async jobs?
|
||||||
runtime.GC()
|
runtime.GC()
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// RcloneRPC does a single RPC call. The inputs are (method, input)
|
||||||
|
// and the output is (output, status). This is an exported interface
|
||||||
|
// to the rclone API as described in https://rclone.org/rc/
|
||||||
|
//
|
||||||
|
// method is a string, eg "operations/list"
|
||||||
|
// input should be a serialized JSON object
|
||||||
|
// output will be returned as a serialized JSON object
|
||||||
|
// status is a HTTP status return (200=OK anything else fail)
|
||||||
|
//
|
||||||
// Caller is responsible for freeing the memory for output
|
// Caller is responsible for freeing the memory for output
|
||||||
// TODO: how to specify config file?
|
//
|
||||||
//export CRPC
|
// Note that when calling from C output and status are returned in an
|
||||||
func CRPC(method *C.char, input *C.char) (output *C.char, status C.int) {
|
// RcloneRPC_return which has two members r0 which is output and r1
|
||||||
|
// which is status.
|
||||||
|
//
|
||||||
|
//export RcloneRPC
|
||||||
|
func RcloneRPC(method *C.char, input *C.char) (output *C.char, status C.int) { //nolint:golint
|
||||||
res, s := callFunctionJSON(C.GoString(method), C.GoString(input))
|
res, s := callFunctionJSON(C.GoString(method), C.GoString(input))
|
||||||
return C.CString(res), C.int(s)
|
return C.CString(res), C.int(s)
|
||||||
}
|
}
|
||||||
|
|
||||||
// copied from rcserver.go
|
// writeError returns a formatted error string and the status passed in
|
||||||
// writeError writes a formatted error to the output
|
func writeError(path string, in rc.Params, err error, status int) (string, int) {
|
||||||
func writeError(path string, in rc.Params, w io.Writer, err error, status int) {
|
|
||||||
fs.Errorf(nil, "rc: %q: error: %v", path, err)
|
fs.Errorf(nil, "rc: %q: error: %v", path, err)
|
||||||
|
var w strings.Builder
|
||||||
|
// FIXME should factor this
|
||||||
// Adjust the error return for some well known errors
|
// Adjust the error return for some well known errors
|
||||||
errOrig := errors.Cause(err)
|
errOrig := errors.Cause(err)
|
||||||
switch {
|
switch {
|
||||||
|
@ -61,7 +89,7 @@ func writeError(path string, in rc.Params, w io.Writer, err error, status int) {
|
||||||
status = http.StatusBadRequest
|
status = http.StatusBadRequest
|
||||||
}
|
}
|
||||||
// w.WriteHeader(status)
|
// w.WriteHeader(status)
|
||||||
err = rc.WriteJSON(w, rc.Params{
|
err = rc.WriteJSON(&w, rc.Params{
|
||||||
"status": status,
|
"status": status,
|
||||||
"error": err.Error(),
|
"error": err.Error(),
|
||||||
"input": in,
|
"input": in,
|
||||||
|
@ -69,8 +97,9 @@ func writeError(path string, in rc.Params, w io.Writer, err error, status int) {
|
||||||
})
|
})
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// can't return the error at this point
|
// can't return the error at this point
|
||||||
fs.Errorf(nil, "rc: failed to write JSON output: %v", err)
|
return fmt.Sprintf(`{"error": "rc: failed to write JSON output: %v"}`, err), status
|
||||||
}
|
}
|
||||||
|
return w.String(), status
|
||||||
}
|
}
|
||||||
|
|
||||||
// operations/uploadfile and core/command are not supported as they need request or response object
|
// operations/uploadfile and core/command are not supported as they need request or response object
|
||||||
|
@ -78,32 +107,27 @@ func writeError(path string, in rc.Params, w io.Writer, err error, status int) {
|
||||||
// call a rc function using JSON to input parameters and output the resulted JSON
|
// call a rc function using JSON to input parameters and output the resulted JSON
|
||||||
func callFunctionJSON(method string, input string) (output string, status int) {
|
func callFunctionJSON(method string, input string) (output string, status int) {
|
||||||
// create a buffer to capture the output
|
// create a buffer to capture the output
|
||||||
buf := new(bytes.Buffer)
|
|
||||||
in := make(rc.Params)
|
in := make(rc.Params)
|
||||||
err := json.NewDecoder(strings.NewReader(input)).Decode(&in)
|
err := json.NewDecoder(strings.NewReader(input)).Decode(&in)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// TODO: handle error
|
// TODO: handle error
|
||||||
writeError(method, in, buf, errors.Wrap(err, "failed to read input JSON"), http.StatusBadRequest)
|
return writeError(method, in, errors.Wrap(err, "failed to read input JSON"), http.StatusBadRequest)
|
||||||
return buf.String(), http.StatusBadRequest
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Find the call
|
// Find the call
|
||||||
call := rc.Calls.Get(method)
|
call := rc.Calls.Get(method)
|
||||||
if call == nil {
|
if call == nil {
|
||||||
writeError(method, in, buf, errors.Errorf("couldn't find method %q", method), http.StatusNotFound)
|
return writeError(method, in, errors.Errorf("couldn't find method %q", method), http.StatusNotFound)
|
||||||
return buf.String(), http.StatusNotFound
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO: handle these cases
|
// TODO: handle these cases
|
||||||
if call.NeedsRequest {
|
if call.NeedsRequest {
|
||||||
writeError(method, in, buf, errors.Errorf("method %q needs request, not supported", method), http.StatusNotFound)
|
return writeError(method, in, errors.Errorf("method %q needs request, not supported", method), http.StatusNotFound)
|
||||||
return buf.String(), http.StatusNotFound
|
|
||||||
// Add the request to RC
|
// Add the request to RC
|
||||||
//in["_request"] = r
|
//in["_request"] = r
|
||||||
}
|
}
|
||||||
if call.NeedsResponse {
|
if call.NeedsResponse {
|
||||||
writeError(method, in, buf, errors.Errorf("method %q need response, not supported", method), http.StatusNotFound)
|
return writeError(method, in, errors.Errorf("method %q need response, not supported", method), http.StatusNotFound)
|
||||||
return buf.String(), http.StatusNotFound
|
|
||||||
//in["_response"] = w
|
//in["_response"] = w
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -112,22 +136,22 @@ func callFunctionJSON(method string, input string) (output string, status int) {
|
||||||
_, out, err := jobs.NewJob(context.Background(), call.Fn, in)
|
_, out, err := jobs.NewJob(context.Background(), call.Fn, in)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
// handle error
|
// handle error
|
||||||
writeError(method, in, buf, err, http.StatusInternalServerError)
|
return writeError(method, in, err, http.StatusInternalServerError)
|
||||||
return buf.String(), http.StatusInternalServerError
|
|
||||||
}
|
}
|
||||||
if out == nil {
|
if out == nil {
|
||||||
out = make(rc.Params)
|
out = make(rc.Params)
|
||||||
}
|
}
|
||||||
|
|
||||||
fs.Debugf(nil, "rc: %q: reply %+v: %v", method, out, err)
|
fs.Debugf(nil, "rc: %q: reply %+v: %v", method, out, err)
|
||||||
err = rc.WriteJSON(buf, out)
|
|
||||||
|
var w strings.Builder
|
||||||
|
err = rc.WriteJSON(&w, out)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
writeError(method, in, buf, err, http.StatusInternalServerError)
|
|
||||||
return buf.String(), http.StatusInternalServerError
|
|
||||||
fs.Errorf(nil, "rc: failed to write JSON output: %v", err)
|
fs.Errorf(nil, "rc: failed to write JSON output: %v", err)
|
||||||
|
return writeError(method, in, err, http.StatusInternalServerError)
|
||||||
}
|
}
|
||||||
return buf.String(), http.StatusOK
|
return w.String(), http.StatusOK
|
||||||
}
|
}
|
||||||
|
|
||||||
// do nothing here
|
// do nothing here - necessary for building into a C library
|
||||||
func main() {}
|
func main() {}
|
||||||
|
|
Loading…
Reference in a new issue