forked from TrueCloudLab/rclone
vendor: add github.com/putdotio/go-putio for putio client
This commit is contained in:
parent
8159658e67
commit
566aa0fca7
15 changed files with 1315 additions and 0 deletions
429
vendor/github.com/putdotio/go-putio/putio/files.go
generated
vendored
Normal file
429
vendor/github.com/putdotio/go-putio/putio/files.go
generated
vendored
Normal file
|
@ -0,0 +1,429 @@
|
|||
package putio
|
||||
|
||||
import (
|
||||
"bytes"
|
||||
"context"
|
||||
"fmt"
|
||||
"io"
|
||||
"mime/multipart"
|
||||
"net/url"
|
||||
"strconv"
|
||||
"strings"
|
||||
)
|
||||
|
||||
// FilesService is a general service to gather information about user files,
|
||||
// such as listing, searching, creating new ones, or just fetching a single
|
||||
// file.
|
||||
type FilesService struct {
|
||||
client *Client
|
||||
}
|
||||
|
||||
// Get fetches file metadata for given file ID.
|
||||
func (f *FilesService) Get(ctx context.Context, id int64) (File, error) {
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id), nil)
|
||||
if err != nil {
|
||||
return File{}, err
|
||||
}
|
||||
|
||||
var r struct {
|
||||
File File `json:"file"`
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return File{}, err
|
||||
}
|
||||
return r.File, nil
|
||||
}
|
||||
|
||||
// List fetches children for given directory ID.
|
||||
func (f *FilesService) List(ctx context.Context, id int64) (children []File, parent File, err error) {
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/list?parent_id="+itoa(id)+"&per_page=1000", nil)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
var r struct {
|
||||
Files []File `json:"files"`
|
||||
Parent File `json:"parent"`
|
||||
Cursor string `json:"cursor"`
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
children = append(children, r.Files...)
|
||||
parent = r.Parent
|
||||
for r.Cursor != "" {
|
||||
body := strings.NewReader(`{"cursor": "` + r.Cursor + `"}`)
|
||||
req, err = f.client.NewRequest(ctx, "POST", "/v2/files/list/continue", body)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
req.Header.Set("content-type", "application/json")
|
||||
r.Files = nil
|
||||
r.Cursor = ""
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return
|
||||
}
|
||||
children = append(children, r.Files...)
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
// URL returns a URL of the file for downloading or streaming.
|
||||
func (f *FilesService) URL(ctx context.Context, id int64, useTunnel bool) (string, error) {
|
||||
notunnel := "notunnel=1"
|
||||
if useTunnel {
|
||||
notunnel = "notunnel=0"
|
||||
}
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id)+"/url?"+notunnel, nil)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
var r struct {
|
||||
URL string `json:"url"`
|
||||
}
|
||||
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return "", err
|
||||
}
|
||||
|
||||
return r.URL, nil
|
||||
}
|
||||
|
||||
// CreateFolder creates a new folder under parent.
|
||||
func (f *FilesService) CreateFolder(ctx context.Context, name string, parent int64) (File, error) {
|
||||
if name == "" {
|
||||
return File{}, fmt.Errorf("empty folder name")
|
||||
}
|
||||
|
||||
params := url.Values{}
|
||||
params.Set("name", name)
|
||||
params.Set("parent_id", itoa(parent))
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/create-folder", strings.NewReader(params.Encode()))
|
||||
if err != nil {
|
||||
return File{}, err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
var r struct {
|
||||
File File `json:"file"`
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return File{}, err
|
||||
}
|
||||
|
||||
return r.File, nil
|
||||
}
|
||||
|
||||
// Delete deletes given files.
|
||||
func (f *FilesService) Delete(ctx context.Context, files ...int64) error {
|
||||
if len(files) == 0 {
|
||||
return fmt.Errorf("no file id is given")
|
||||
}
|
||||
|
||||
var ids []string
|
||||
for _, id := range files {
|
||||
ids = append(ids, itoa(id))
|
||||
}
|
||||
|
||||
params := url.Values{}
|
||||
params.Set("file_ids", strings.Join(ids, ","))
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/delete", strings.NewReader(params.Encode()))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
_, err = f.client.Do(req, &struct{}{})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Rename change the name of the file to newname.
|
||||
func (f *FilesService) Rename(ctx context.Context, id int64, newname string) error {
|
||||
if newname == "" {
|
||||
return fmt.Errorf("new filename cannot be empty")
|
||||
}
|
||||
|
||||
params := url.Values{}
|
||||
params.Set("file_id", itoa(id))
|
||||
params.Set("name", newname)
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/rename", strings.NewReader(params.Encode()))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
_, err = f.client.Do(req, &struct{}{})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// Move moves files to the given destination.
|
||||
func (f *FilesService) Move(ctx context.Context, parent int64, files ...int64) error {
|
||||
if len(files) == 0 {
|
||||
return fmt.Errorf("no files given")
|
||||
}
|
||||
|
||||
var ids []string
|
||||
for _, file := range files {
|
||||
ids = append(ids, itoa(file))
|
||||
}
|
||||
|
||||
params := url.Values{}
|
||||
params.Set("file_ids", strings.Join(ids, ","))
|
||||
params.Set("parent_id", itoa(parent))
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/move", strings.NewReader(params.Encode()))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
_, err = f.client.Do(req, &struct{}{})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
return nil
|
||||
}
|
||||
|
||||
// Upload reads from given io.Reader and uploads the file contents to Put.io
|
||||
// servers under directory given by parent. If parent is negative, user's
|
||||
// prefered folder is used.
|
||||
//
|
||||
// If the uploaded file is a torrent file, Put.io will interpret it as a
|
||||
// transfer and Transfer field will be present to represent the status of the
|
||||
// tranfer. Likewise, if the uploaded file is a regular file, Transfer field
|
||||
// would be nil and the uploaded file will be represented by the File field.
|
||||
//
|
||||
// This method reads the file contents into the memory, so it should be used for
|
||||
// <150MB files.
|
||||
func (f *FilesService) Upload(ctx context.Context, r io.Reader, filename string, parent int64) (Upload, error) {
|
||||
if filename == "" {
|
||||
return Upload{}, fmt.Errorf("filename cannot be empty")
|
||||
}
|
||||
|
||||
var buf bytes.Buffer
|
||||
mw := multipart.NewWriter(&buf)
|
||||
|
||||
// negative parent means use user's prefered download folder.
|
||||
if parent >= 0 {
|
||||
err := mw.WriteField("parent_id", itoa(parent))
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
}
|
||||
|
||||
formfile, err := mw.CreateFormFile("file", filename)
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
|
||||
_, err = io.Copy(formfile, r)
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
|
||||
err = mw.Close()
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/upload", &buf)
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
req.Header.Set("Content-Type", mw.FormDataContentType())
|
||||
|
||||
var response struct {
|
||||
Upload
|
||||
}
|
||||
_, err = f.client.Do(req, &response)
|
||||
if err != nil {
|
||||
return Upload{}, err
|
||||
}
|
||||
return response.Upload, nil
|
||||
}
|
||||
|
||||
// Search makes a search request with the given query. Servers return 50
|
||||
// results at a time. The URL for the next 50 results are in Next field. If
|
||||
// page is -1, all results are returned.
|
||||
func (f *FilesService) Search(ctx context.Context, query string, page int64) (Search, error) {
|
||||
if page == 0 || page < -1 {
|
||||
return Search{}, fmt.Errorf("invalid page number")
|
||||
}
|
||||
if query == "" {
|
||||
return Search{}, fmt.Errorf("no query given")
|
||||
}
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/search/"+query+"/page/"+itoa(page), nil)
|
||||
if err != nil {
|
||||
return Search{}, err
|
||||
}
|
||||
|
||||
var r Search
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return Search{}, err
|
||||
}
|
||||
|
||||
return r, nil
|
||||
}
|
||||
|
||||
// Shared returns list of shared files and share information.
|
||||
func (f *FilesService) shared(ctx context.Context) ([]share, error) {
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/shared", nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var r struct {
|
||||
Shared []share
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return r.Shared, nil
|
||||
}
|
||||
|
||||
// SharedWith returns list of users the given file is shared with.
|
||||
func (f *FilesService) sharedWith(ctx context.Context, id int64) ([]share, error) {
|
||||
// FIXME: shared-with returns different json structure than /shared/
|
||||
// endpoint. so it's not an exported method until a common structure is
|
||||
// decided
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id)+"/shared-with", nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var r struct {
|
||||
Shared []share `json:"shared-with"`
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return r.Shared, nil
|
||||
}
|
||||
|
||||
// Subtitles lists available subtitles for the given file for user's prefered
|
||||
// subtitle language.
|
||||
func (f *FilesService) Subtitles(ctx context.Context, id int64) ([]Subtitle, error) {
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id)+"/subtitles", nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
var r struct {
|
||||
Subtitles []Subtitle
|
||||
Default string
|
||||
}
|
||||
_, err = f.client.Do(req, &r)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return r.Subtitles, nil
|
||||
}
|
||||
|
||||
// DownloadSubtitle sends the contents of the subtitle file. If the key is empty string,
|
||||
// `default` key is used. This key is used to search for a subtitle in the
|
||||
// following order and returns the first match:
|
||||
// - A subtitle file that has identical parent folder and name with the video.
|
||||
// - Subtitle file extracted from video if the format is MKV.
|
||||
// - First match from OpenSubtitles.org.
|
||||
func (f *FilesService) DownloadSubtitle(ctx context.Context, id int64, key string, format string) (io.ReadCloser, error) {
|
||||
if key == "" {
|
||||
key = "default"
|
||||
}
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id)+"/subtitles/"+key, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
resp, err := f.client.Do(req, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return resp.Body, nil
|
||||
}
|
||||
|
||||
// HLSPlaylist serves a HLS playlist for a video file. Use “all” as
|
||||
// subtitleKey to get available subtitles for user’s preferred languages.
|
||||
func (f *FilesService) HLSPlaylist(ctx context.Context, id int64, subtitleKey string) (io.ReadCloser, error) {
|
||||
if subtitleKey == "" {
|
||||
return nil, fmt.Errorf("empty subtitle key is given")
|
||||
}
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "GET", "/v2/files/"+itoa(id)+"/hls/media.m3u8?subtitle_key"+subtitleKey, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
resp, err := f.client.Do(req, nil)
|
||||
if err != nil {
|
||||
return nil, err
|
||||
}
|
||||
|
||||
return resp.Body, nil
|
||||
}
|
||||
|
||||
// SetVideoPosition sets default video position for a video file.
|
||||
func (f *FilesService) SetVideoPosition(ctx context.Context, id int64, t int) error {
|
||||
if t < 0 {
|
||||
return fmt.Errorf("time cannot be negative")
|
||||
}
|
||||
|
||||
params := url.Values{}
|
||||
params.Set("time", strconv.Itoa(t))
|
||||
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/"+itoa(id)+"/start-from", strings.NewReader(params.Encode()))
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
_, err = f.client.Do(req, &struct{}{})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
// DeleteVideoPosition deletes video position for a video file.
|
||||
func (f *FilesService) DeleteVideoPosition(ctx context.Context, id int64) error {
|
||||
req, err := f.client.NewRequest(ctx, "POST", "/v2/files/"+itoa(id)+"/start-from/delete", nil)
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||
|
||||
_, err = f.client.Do(req, &struct{}{})
|
||||
if err != nil {
|
||||
return err
|
||||
}
|
||||
|
||||
return nil
|
||||
}
|
||||
|
||||
func itoa(i int64) string {
|
||||
return strconv.FormatInt(i, 10)
|
||||
}
|
Loading…
Add table
Add a link
Reference in a new issue