738 lines
19 KiB
Go
738 lines
19 KiB
Go
|
// Copyright (c) 2015 Serge Gebhardt. All rights reserved.
|
||
|
//
|
||
|
// Use of this source code is governed by the ISC
|
||
|
// license that can be found in the LICENSE file.
|
||
|
|
||
|
package acd
|
||
|
|
||
|
import (
|
||
|
"bytes"
|
||
|
"encoding/json"
|
||
|
"errors"
|
||
|
"fmt"
|
||
|
"io"
|
||
|
"mime/multipart"
|
||
|
"net/http"
|
||
|
"net/url"
|
||
|
"os"
|
||
|
"reflect"
|
||
|
"regexp"
|
||
|
|
||
|
"github.com/google/go-querystring/query"
|
||
|
)
|
||
|
|
||
|
var (
|
||
|
ErrorNodeNotFound = errors.New("Node not found")
|
||
|
)
|
||
|
|
||
|
// NodesService provides access to the nodes in the Amazon Cloud Drive API.
|
||
|
//
|
||
|
// See: https://developer.amazon.com/public/apis/experience/cloud-drive/content/nodes
|
||
|
type NodesService struct {
|
||
|
client *Client
|
||
|
}
|
||
|
|
||
|
// Gets the root folder of the Amazon Cloud Drive.
|
||
|
func (s *NodesService) GetRoot() (*Folder, *http.Response, error) {
|
||
|
opts := &NodeListOptions{Filters: "kind:FOLDER AND isRoot:true"}
|
||
|
|
||
|
roots, resp, err := s.GetNodes(opts)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
if len(roots) < 1 {
|
||
|
return nil, resp, errors.New("No root found")
|
||
|
}
|
||
|
|
||
|
return &Folder{roots[0]}, resp, nil
|
||
|
}
|
||
|
|
||
|
// Gets the list of all nodes.
|
||
|
func (s *NodesService) GetAllNodes(opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
return s.listAllNodes("nodes", opts)
|
||
|
}
|
||
|
|
||
|
// Gets a list of nodes, up until the limit (either default or the one set in opts).
|
||
|
func (s *NodesService) GetNodes(opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
nodes, res, err := s.listNodes("nodes", opts)
|
||
|
return nodes, res, err
|
||
|
}
|
||
|
|
||
|
func (s *NodesService) listAllNodes(url string, opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
// Need opts to maintain state (NodeListOptions.reachedEnd)
|
||
|
if opts == nil {
|
||
|
opts = &NodeListOptions{}
|
||
|
}
|
||
|
|
||
|
result := make([]*Node, 0, 200)
|
||
|
|
||
|
for {
|
||
|
nodes, resp, err := s.listNodes(url, opts)
|
||
|
if err != nil {
|
||
|
return result, resp, err
|
||
|
}
|
||
|
if nodes == nil {
|
||
|
break
|
||
|
}
|
||
|
|
||
|
result = append(result, nodes...)
|
||
|
}
|
||
|
|
||
|
return result, nil, nil
|
||
|
}
|
||
|
|
||
|
func (s *NodesService) listNodes(url string, opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
if opts != nil && opts.reachedEnd {
|
||
|
return nil, nil, nil
|
||
|
}
|
||
|
|
||
|
url, err := addOptions(url, opts)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
req, err := s.client.NewMetadataRequest("GET", url, nil)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
nodeList := &nodeListInternal{}
|
||
|
resp, err := s.client.Do(req, nodeList)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
if opts != nil {
|
||
|
if nodeList.NextToken != nil {
|
||
|
opts.StartToken = *nodeList.NextToken
|
||
|
} else {
|
||
|
opts.reachedEnd = true
|
||
|
}
|
||
|
}
|
||
|
|
||
|
nodes := nodeList.Data
|
||
|
for _, node := range nodes {
|
||
|
node.service = s
|
||
|
}
|
||
|
|
||
|
return nodes, resp, nil
|
||
|
}
|
||
|
|
||
|
type nodeListInternal struct {
|
||
|
Count *uint64 `json:"count"`
|
||
|
NextToken *string `json:"nextToken"`
|
||
|
Data []*Node `json:"data"`
|
||
|
}
|
||
|
|
||
|
// Node represents a digital asset on the Amazon Cloud Drive, including files
|
||
|
// and folders, in a parent-child relationship. A node contains only metadata
|
||
|
// (e.g. folder) or it contains metadata and content (e.g. file).
|
||
|
type Node struct {
|
||
|
Id *string `json:"id"`
|
||
|
Name *string `json:"name"`
|
||
|
Kind *string `json:"kind"`
|
||
|
ModifiedDate *string `json:"modifiedDate"`
|
||
|
Parents []string `json:"parents"`
|
||
|
Status *string `json:"status"`
|
||
|
ContentProperties *struct {
|
||
|
Size *uint64 `json:"size"`
|
||
|
Md5 *string `json:"md5"`
|
||
|
ContentType *string `json:"contentType"`
|
||
|
} `json:"contentProperties"`
|
||
|
TempURL string `json:"tempLink"`
|
||
|
|
||
|
service *NodesService
|
||
|
}
|
||
|
|
||
|
// NodeFromId constructs a skeleton Node from an Id and a NodeService
|
||
|
func NodeFromId(Id string, service *NodesService) *Node {
|
||
|
return &Node{
|
||
|
Id: &Id,
|
||
|
service: service,
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// IsFile returns whether the node represents a file.
|
||
|
func (n *Node) IsFile() bool {
|
||
|
return n.Kind != nil && *n.Kind == "FILE"
|
||
|
}
|
||
|
|
||
|
// IsFolder returns whether the node represents a folder.
|
||
|
func (n *Node) IsFolder() bool {
|
||
|
return n.Kind != nil && *n.Kind == "FOLDER"
|
||
|
}
|
||
|
|
||
|
// Typed returns the Node typed as either File or Folder.
|
||
|
func (n *Node) Typed() interface{} {
|
||
|
if n.IsFile() {
|
||
|
return &File{n}
|
||
|
}
|
||
|
|
||
|
if n.IsFolder() {
|
||
|
return &Folder{n}
|
||
|
}
|
||
|
|
||
|
return n
|
||
|
}
|
||
|
|
||
|
// GetTempURL sets the TempURL for the node passed in if it isn't already set
|
||
|
func (n *Node) GetTempURL() (*http.Response, error) {
|
||
|
if n.TempURL != "" {
|
||
|
return nil, nil
|
||
|
}
|
||
|
url := fmt.Sprintf("nodes/%s?tempLink=true", *n.Id)
|
||
|
req, err := n.service.client.NewMetadataRequest("GET", url, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
node := &Node{}
|
||
|
resp, err := n.service.client.Do(req, node)
|
||
|
if err != nil {
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
if node.TempURL == "" {
|
||
|
return resp, fmt.Errorf("Couldn't read TempURL")
|
||
|
}
|
||
|
|
||
|
// Set the TempURL in the node
|
||
|
n.TempURL = node.TempURL
|
||
|
return resp, nil
|
||
|
}
|
||
|
|
||
|
// GetMetadata return a pretty-printed JSON string of the node's metadata
|
||
|
func (n *Node) GetMetadata() (string, error) {
|
||
|
url := fmt.Sprintf("nodes/%s?tempLink=true", *n.Id)
|
||
|
req, err := n.service.client.NewMetadataRequest("GET", url, nil)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
buf := &bytes.Buffer{}
|
||
|
_, err = n.service.client.Do(req, buf)
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
md := &bytes.Buffer{}
|
||
|
err = json.Indent(md, buf.Bytes(), "", " ")
|
||
|
if err != nil {
|
||
|
return "", err
|
||
|
}
|
||
|
|
||
|
return md.String(), nil
|
||
|
}
|
||
|
|
||
|
// Move node
|
||
|
func (n *Node) Move(newParent string) (*Node, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/children", EscapeForFilter(newParent))
|
||
|
metadata := moveNode{
|
||
|
Id: *n.Id,
|
||
|
ParentId: n.Parents[0],
|
||
|
}
|
||
|
|
||
|
req, err := n.service.client.NewMetadataRequest("POST", url, &metadata)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
node := &Node{service: n.service}
|
||
|
resp, err := n.service.client.Do(req, node)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
return node, resp, nil
|
||
|
}
|
||
|
|
||
|
// Rename node
|
||
|
func (n *Node) Rename(newName string) (*Node, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s", *n.Id)
|
||
|
metadata := renameNode{
|
||
|
Name: newName,
|
||
|
}
|
||
|
|
||
|
req, err := n.service.client.NewMetadataRequest("PATCH", url, &metadata)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
node := &Node{service: n.service}
|
||
|
resp, err := n.service.client.Do(req, node)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
return node, resp, nil
|
||
|
}
|
||
|
|
||
|
// Trash places Node n into the trash. If the node is a directory it
|
||
|
// places it and all of its contents into the trash.
|
||
|
func (n *Node) Trash() (*http.Response, error) {
|
||
|
url := fmt.Sprintf("trash/%s", *n.Id)
|
||
|
req, err := n.service.client.NewMetadataRequest("PUT", url, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
resp, err := n.service.client.Do(req, nil)
|
||
|
if err != nil {
|
||
|
return resp, err
|
||
|
}
|
||
|
err = resp.Body.Close()
|
||
|
if err != nil {
|
||
|
return resp, err
|
||
|
}
|
||
|
return resp, nil
|
||
|
|
||
|
}
|
||
|
|
||
|
// File represents a file on the Amazon Cloud Drive.
|
||
|
type File struct {
|
||
|
*Node
|
||
|
}
|
||
|
|
||
|
// Open the content of the file f for read
|
||
|
//
|
||
|
// Extra headers for the GET can be passed in in headers
|
||
|
//
|
||
|
// You must call in.Close() when finished
|
||
|
func (f *File) OpenHeaders(headers map[string]string) (in io.ReadCloser, resp *http.Response, err error) {
|
||
|
url := fmt.Sprintf("nodes/%s/content", *f.Id)
|
||
|
req, err := f.service.client.NewContentRequest("GET", url, nil)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
for k, v := range headers {
|
||
|
req.Header.Add(k, v)
|
||
|
}
|
||
|
resp, err = f.service.client.Do(req, nil)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
return resp.Body, resp, nil
|
||
|
}
|
||
|
|
||
|
// Open the content of the file f for read
|
||
|
//
|
||
|
// You must call in.Close() when finished
|
||
|
func (f *File) Open() (in io.ReadCloser, resp *http.Response, err error) {
|
||
|
return f.OpenHeaders(nil)
|
||
|
}
|
||
|
|
||
|
// OpenTempURL opens the content of the file f for read from the TempURL
|
||
|
//
|
||
|
// Pass in an http Client (without authorization) for the download.
|
||
|
//
|
||
|
// You must call in.Close() when finished
|
||
|
func (f *File) OpenTempURLHeaders(client *http.Client, headers map[string]string) (in io.ReadCloser, resp *http.Response, err error) {
|
||
|
resp, err = f.GetTempURL()
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
req, err := http.NewRequest("GET", f.TempURL, nil)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
if f.service.client.UserAgent != "" {
|
||
|
req.Header.Add("User-Agent", f.service.client.UserAgent)
|
||
|
}
|
||
|
for k, v := range headers {
|
||
|
req.Header.Add(k, v)
|
||
|
}
|
||
|
resp, err = client.Do(req)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
return resp.Body, resp, nil
|
||
|
}
|
||
|
|
||
|
// OpenTempURL opens the content of the file f for read from the TempURL
|
||
|
//
|
||
|
// Pass in an http Client (without authorization) for the download.
|
||
|
//
|
||
|
// You must call in.Close() when finished
|
||
|
func (f *File) OpenTempURL(client *http.Client) (in io.ReadCloser, resp *http.Response, err error) {
|
||
|
return f.OpenTempURLHeaders(client, nil)
|
||
|
}
|
||
|
|
||
|
// Download fetches the content of file f and stores it into the file pointed
|
||
|
// to by path. Errors if the file at path already exists. Does not create the
|
||
|
// intermediate directories in path.
|
||
|
func (f *File) Download(path string) (*http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/content", *f.Id)
|
||
|
req, err := f.service.client.NewContentRequest("GET", url, nil)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
out, err := os.OpenFile(path, os.O_WRONLY|os.O_CREATE|os.O_EXCL, 0666)
|
||
|
if err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
defer out.Close()
|
||
|
|
||
|
resp, err := f.service.client.Do(req, out)
|
||
|
return resp, err
|
||
|
}
|
||
|
|
||
|
// Folder represents a folder on the Amazon Cloud Drive.
|
||
|
type Folder struct {
|
||
|
*Node
|
||
|
}
|
||
|
|
||
|
// FolderFromId constructs a skeleton Folder from an Id and a NodeService
|
||
|
func FolderFromId(Id string, service *NodesService) *Folder {
|
||
|
return &Folder{
|
||
|
Node: NodeFromId(Id, service),
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// Gets the list of all children.
|
||
|
func (f *Folder) GetAllChildren(opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/children", *f.Id)
|
||
|
return f.service.listAllNodes(url, opts)
|
||
|
}
|
||
|
|
||
|
// Gets a list of children, up until the limit (either default or the one set in opts).
|
||
|
func (f *Folder) GetChildren(opts *NodeListOptions) ([]*Node, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/children", *f.Id)
|
||
|
return f.service.listNodes(url, opts)
|
||
|
}
|
||
|
|
||
|
// Gets the subfolder by name. It is an error if not exactly one subfolder is found.
|
||
|
//
|
||
|
// If it isn't found then it returns the error ErrorNodeNotFound
|
||
|
func (f *Folder) GetFolder(name string) (*Folder, *http.Response, error) {
|
||
|
n, resp, err := f.GetNode(name)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
res, ok := n.Typed().(*Folder)
|
||
|
if !ok {
|
||
|
err := errors.New(fmt.Sprintf("Node '%s' is not a folder", name))
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
return res, resp, nil
|
||
|
}
|
||
|
|
||
|
// createNode is a cut down set of parameters for creating nodes
|
||
|
type createNode struct {
|
||
|
Name string `json:"name"`
|
||
|
Kind string `json:"kind"`
|
||
|
Parents []string `json:"parents"`
|
||
|
}
|
||
|
|
||
|
// moveNode is a cut down set of parameters for moving nodes
|
||
|
type moveNode struct {
|
||
|
ParentId string `json:"fromParent"`
|
||
|
Id string `json:"childId"`
|
||
|
}
|
||
|
|
||
|
// renameNode is a cut down set of parameters for renaming nodes
|
||
|
type renameNode struct {
|
||
|
Name string `json:"name"`
|
||
|
}
|
||
|
|
||
|
// CreateFolder makes a new folder with the given name.
|
||
|
//
|
||
|
// The new Folder is returned
|
||
|
func (f *Folder) CreateFolder(name string) (*Folder, *http.Response, error) {
|
||
|
createFolder := createNode{
|
||
|
Name: name,
|
||
|
Kind: "FOLDER",
|
||
|
Parents: []string{*f.Id},
|
||
|
}
|
||
|
req, err := f.service.client.NewMetadataRequest("POST", "nodes", &createFolder)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
folder := &Folder{&Node{service: f.service}}
|
||
|
resp, err := f.service.client.Do(req, folder)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
return folder, resp, nil
|
||
|
|
||
|
}
|
||
|
|
||
|
// Gets the file by name. It is an error if not exactly one file is found.
|
||
|
//
|
||
|
// If it isn't found then it returns the error ErrorNodeNotFound
|
||
|
func (f *Folder) GetFile(name string) (*File, *http.Response, error) {
|
||
|
n, resp, err := f.GetNode(name)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
res, ok := n.Typed().(*File)
|
||
|
if !ok {
|
||
|
err := errors.New(fmt.Sprintf("Node '%s' is not a file", name))
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
return res, resp, nil
|
||
|
}
|
||
|
|
||
|
var escapeForFilterRe = regexp.MustCompile(`([+\-&|!(){}\[\]^'"~*?:\\ ])`)
|
||
|
|
||
|
// EscapeForFilter escapes an abitrary string for use as a filter
|
||
|
// query parameter.
|
||
|
//
|
||
|
// Special characters that are part of the query syntax will be
|
||
|
// escaped. The list of special characters are:
|
||
|
//
|
||
|
// + - & | ! ( ) { } [ ] ^ ' " ~ * ? : \
|
||
|
//
|
||
|
// Additionally, space will be escaped. Characters are escaped by
|
||
|
// using \ before the character.
|
||
|
func EscapeForFilter(s string) string {
|
||
|
return escapeForFilterRe.ReplaceAllString(s, `\$1`)
|
||
|
}
|
||
|
|
||
|
// Gets the node by name. It is an error if not exactly one node is found.
|
||
|
//
|
||
|
// If it isn't found then it returns the error ErrorNodeNotFound
|
||
|
func (f *Folder) GetNode(name string) (*Node, *http.Response, error) {
|
||
|
filter := fmt.Sprintf(`parents:"%v" AND name:"%s"`, *f.Id, EscapeForFilter(name))
|
||
|
opts := &NodeListOptions{Filters: filter}
|
||
|
|
||
|
nodes, resp, err := f.service.GetNodes(opts)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
if len(nodes) < 1 {
|
||
|
return nil, resp, ErrorNodeNotFound
|
||
|
}
|
||
|
if len(nodes) > 1 {
|
||
|
err := errors.New(fmt.Sprintf("Too many nodes '%s' found (%v)", name, len(nodes)))
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
return nodes[0], resp, nil
|
||
|
}
|
||
|
|
||
|
// WalkNodes walks the given node hierarchy, getting each node along the way, and returns
|
||
|
// the deepest node. If an error occurs, returns the furthest successful node and the list
|
||
|
// of HTTP responses.
|
||
|
func (f *Folder) WalkNodes(names ...string) (*Node, []*http.Response, error) {
|
||
|
resps := make([]*http.Response, 0, len(names))
|
||
|
|
||
|
if len(names) == 0 {
|
||
|
return f.Node, resps, nil
|
||
|
}
|
||
|
|
||
|
// process each node except the last one
|
||
|
fp := f
|
||
|
for _, name := range names[:len(names)-1] {
|
||
|
fn, resp, err := fp.GetFolder(name)
|
||
|
resps = append(resps, resp)
|
||
|
if err != nil {
|
||
|
return fp.Node, resps, err
|
||
|
}
|
||
|
|
||
|
fp = fn
|
||
|
}
|
||
|
|
||
|
// process the last node
|
||
|
nl, resp, err := fp.GetNode(names[len(names)-1])
|
||
|
resps = append(resps, resp)
|
||
|
if err != nil {
|
||
|
return fp.Node, resps, err
|
||
|
}
|
||
|
|
||
|
return nl, resps, nil
|
||
|
}
|
||
|
|
||
|
// Put stores the data read from in at path as name on the Amazon Cloud Drive.
|
||
|
// Errors if the file already exists on the drive.
|
||
|
func (service *NodesService) putOrOverwrite(in io.Reader, httpVerb, url, name, metadata string) (*File, *http.Response, error) {
|
||
|
bodyReader, bodyWriter := io.Pipe()
|
||
|
writer := multipart.NewWriter(bodyWriter)
|
||
|
contentType := writer.FormDataContentType()
|
||
|
|
||
|
errChan := make(chan error, 1)
|
||
|
go func() {
|
||
|
defer bodyWriter.Close()
|
||
|
var err error
|
||
|
|
||
|
if metadata != "" {
|
||
|
err = writer.WriteField("metadata", string(metadata))
|
||
|
if err != nil {
|
||
|
errChan <- err
|
||
|
return
|
||
|
}
|
||
|
}
|
||
|
|
||
|
part, err := writer.CreateFormFile("content", name)
|
||
|
if err != nil {
|
||
|
errChan <- err
|
||
|
return
|
||
|
}
|
||
|
if _, err := io.Copy(part, in); err != nil {
|
||
|
errChan <- err
|
||
|
return
|
||
|
}
|
||
|
errChan <- writer.Close()
|
||
|
}()
|
||
|
|
||
|
req, err := service.client.NewContentRequest(httpVerb, url, bodyReader)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
req.Header.Add("Content-Type", contentType)
|
||
|
|
||
|
file := &File{&Node{service: service}}
|
||
|
resp, err := service.client.Do(req, file)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
err = <-errChan
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
return file, resp, err
|
||
|
}
|
||
|
|
||
|
// Put stores the data read from in at path as name on the Amazon Cloud Drive.
|
||
|
// Errors if the file already exists on the drive.
|
||
|
func (service *NodesService) putOrOverwriteSized(in io.Reader, fileSize int64, httpVerb, url, name, metadata string) (*File, *http.Response, error) {
|
||
|
var err error
|
||
|
bodyBuf := bytes.NewBufferString("")
|
||
|
bodyWriter := multipart.NewWriter(bodyBuf)
|
||
|
|
||
|
// use the bodyWriter to write the Part headers to the buffer
|
||
|
if metadata != "" {
|
||
|
err = bodyWriter.WriteField("metadata", string(metadata))
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
}
|
||
|
_, err = bodyWriter.CreateFormFile("content", name)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
// need to know the boundary to properly close the part myself.
|
||
|
boundary := bodyWriter.Boundary()
|
||
|
close_buf := bytes.NewBufferString(fmt.Sprintf("\r\n--%s--\r\n", boundary))
|
||
|
|
||
|
// use multi-reader to defer the reading of the file data
|
||
|
// until writing to the socket buffer.
|
||
|
request_reader := io.MultiReader(bodyBuf, in, close_buf)
|
||
|
|
||
|
req, err := service.client.NewContentRequest(httpVerb, url, request_reader)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
|
||
|
// Set headers for multipart, and Content Length
|
||
|
req.Header.Add("Content-Type", "multipart/form-data; boundary="+boundary)
|
||
|
req.ContentLength = fileSize + int64(bodyBuf.Len()) + int64(close_buf.Len())
|
||
|
|
||
|
file := &File{&Node{service: service}}
|
||
|
resp, err := service.client.Do(req, file)
|
||
|
if err != nil {
|
||
|
return nil, resp, err
|
||
|
}
|
||
|
|
||
|
return file, resp, err
|
||
|
}
|
||
|
|
||
|
// Put stores the data read from in at path as name on the Amazon Cloud Drive.
|
||
|
// Errors if the file already exists on the drive.
|
||
|
//
|
||
|
// Can't put file with 0 length file (works sometimes)
|
||
|
func (f *Folder) Put(in io.Reader, name string) (*File, *http.Response, error) {
|
||
|
metadata := createNode{
|
||
|
Name: name,
|
||
|
Kind: "FILE",
|
||
|
Parents: []string{*f.Id},
|
||
|
}
|
||
|
metadataJson, err := json.Marshal(&metadata)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
return f.service.putOrOverwrite(in, "POST", "nodes?suppress=deduplication", name, string(metadataJson))
|
||
|
}
|
||
|
|
||
|
// Overwrite updates the file contents from in
|
||
|
//
|
||
|
// Can't overwrite with 0 length file (works sometimes)
|
||
|
func (f *File) Overwrite(in io.Reader) (*File, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/content", *f.Id)
|
||
|
return f.service.putOrOverwrite(in, "PUT", url, *f.Name, "")
|
||
|
}
|
||
|
|
||
|
// Put stores the data read from in at path as name on the Amazon Cloud Drive.
|
||
|
// Errors if the file already exists on the drive.
|
||
|
func (f *Folder) PutSized(in io.Reader, size int64, name string) (*File, *http.Response, error) {
|
||
|
metadata := createNode{
|
||
|
Name: name,
|
||
|
Kind: "FILE",
|
||
|
Parents: []string{*f.Id},
|
||
|
}
|
||
|
metadataJson, err := json.Marshal(&metadata)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
return f.service.putOrOverwriteSized(in, size, "POST", "nodes?suppress=deduplication", name, string(metadataJson))
|
||
|
}
|
||
|
|
||
|
// Overwrite updates the file contents from in
|
||
|
func (f *File) OverwriteSized(in io.Reader, size int64) (*File, *http.Response, error) {
|
||
|
url := fmt.Sprintf("nodes/%s/content", *f.Id)
|
||
|
return f.service.putOrOverwriteSized(in, size, "PUT", url, *f.Name, "")
|
||
|
}
|
||
|
|
||
|
// Upload stores the content of file at path as name on the Amazon Cloud Drive.
|
||
|
// Errors if the file already exists on the drive.
|
||
|
func (f *Folder) Upload(path, name string) (*File, *http.Response, error) {
|
||
|
in, err := os.Open(path)
|
||
|
if err != nil {
|
||
|
return nil, nil, err
|
||
|
}
|
||
|
defer in.Close()
|
||
|
return f.Put(in, name)
|
||
|
}
|
||
|
|
||
|
// NodeListOptions holds the options when getting a list of nodes, such as the filter,
|
||
|
// sorting and pagination.
|
||
|
type NodeListOptions struct {
|
||
|
Limit uint `url:"limit,omitempty"`
|
||
|
Filters string `url:"filters,omitempty"`
|
||
|
Sort string `url:"sort,omitempty"`
|
||
|
|
||
|
// Token where to start for next page (internal)
|
||
|
StartToken string `url:"startToken,omitempty"`
|
||
|
reachedEnd bool
|
||
|
}
|
||
|
|
||
|
// addOptions adds the parameters in opts as URL query parameters to s. opts
|
||
|
// must be a struct whose fields may contain "url" tags.
|
||
|
func addOptions(s string, opts interface{}) (string, error) {
|
||
|
v := reflect.ValueOf(opts)
|
||
|
if v.Kind() == reflect.Ptr && v.IsNil() {
|
||
|
return s, nil
|
||
|
}
|
||
|
|
||
|
u, err := url.Parse(s)
|
||
|
if err != nil {
|
||
|
return s, err
|
||
|
}
|
||
|
|
||
|
qs, err := query.Values(opts)
|
||
|
if err != nil {
|
||
|
return s, err
|
||
|
}
|
||
|
|
||
|
u.RawQuery = qs.Encode()
|
||
|
return u.String(), nil
|
||
|
}
|