docxTemplater/yadisk/api.go

478 lines
9.8 KiB
Go
Raw Normal View History

2022-07-28 15:00:43 +00:00
package YaDisk
import (
"context"
2022-07-28 15:00:43 +00:00
"encoding/json"
"errors"
2022-07-28 15:00:43 +00:00
"fmt"
2022-08-10 13:53:34 +00:00
"golang.org/x/oauth2"
"io"
2022-07-28 15:00:43 +00:00
"net/http"
"net/url"
"os"
2022-12-31 14:46:28 +00:00
"penahub.gitlab.yandexcloud.net/backend/templategen/tools"
"strconv"
"strings"
"time"
2022-07-28 15:00:43 +00:00
)
const (
BASE_URL = "https://cloud-api.yandex.net"
OAUTH_URL = "https://oauth.yandex.ru"
V1_DISK_API = BASE_URL + "/v1/disk"
DEFAULT_FOLDER = "disk:/templategen"
DEFAULT_TEMPLATE_FOLDER = DEFAULT_FOLDER + "/templates"
DEFAULT_SAVE_FOLDER = DEFAULT_FOLDER + "/saved"
2022-07-28 15:00:43 +00:00
)
type Client struct {
2022-08-10 13:53:34 +00:00
App *ClientApp
HTTPClient *http.Client
Token *oauth2.Token
2022-07-28 15:00:43 +00:00
}
type ClientApp struct {
Config *oauth2.Config
ServiceUrl string
2022-07-28 15:00:43 +00:00
}
func NewClientApp(clientID, clientSecret, redirectUri, serviceUrl string) *ClientApp {
2022-07-28 15:00:43 +00:00
return &ClientApp{
Config: &oauth2.Config{
ClientID: clientID,
ClientSecret: clientSecret,
Endpoint: oauth2.Endpoint{
AuthURL: OAUTH_URL + "/authorize",
TokenURL: OAUTH_URL + "/token",
},
RedirectURL: redirectUri,
Scopes: nil,
},
ServiceUrl: serviceUrl,
2022-07-28 15:00:43 +00:00
}
}
func (ca *ClientApp) GenerateOAuthUrl(userId, redirectUrl string) (string, error) {
state, err := tools.EncryptTokenRC4(tools.StateToken{
UserID: userId,
Service: "yandex",
RedirectUrl: redirectUrl,
})
if err != nil {
return "", err
}
return ca.Config.AuthCodeURL(
state,
oauth2.AccessTypeOffline,
oauth2.SetAuthURLParam("display", "popup"),
oauth2.SetAuthURLParam("force_confirm", "true"),
), nil
2022-07-28 15:00:43 +00:00
}
func (ca *ClientApp) NewClient(ctx context.Context, token *oauth2.Token, code string) (*Client, error) {
var err error
if code != "" {
token, err = ca.Config.Exchange(ctx, code)
}
// Этот костыль нужен, т.к. Яндекс принимает токены только типа OAuth, хоть и отправляет типа bearer
token.TokenType = "OAuth"
2022-07-28 15:00:43 +00:00
return &Client{
2022-08-10 13:53:34 +00:00
App: ca,
HTTPClient: ca.Config.Client(ctx, token),
2022-08-10 13:53:34 +00:00
Token: token,
}, err
2022-07-28 15:00:43 +00:00
}
func (ca *ClientApp) RefreshToken(ctx context.Context, oldToken *oauth2.Token) (*oauth2.Token, error) {
token, err := ca.Config.TokenSource(ctx, oldToken).Token()
if err != nil {
return nil, err
}
// Этот костыль нужен, т.к. Яндекс принимает токены только типа OAuth, хоть и отправляет типа bearer
token.TokenType = "OAuth"
return token, nil
2022-07-28 15:00:43 +00:00
}
func (c *Client) GetDisk() (*Disk, error) {
req, err := http.NewRequest("GET", V1_DISK_API, nil)
2022-07-28 15:00:43 +00:00
if err != nil {
return nil, err
}
2022-07-28 15:00:43 +00:00
resp, err := c.HTTPClient.Do(req)
2022-07-28 15:00:43 +00:00
if err != nil {
return nil, err
}
err = checkError(resp)
if err != nil {
return nil, err
}
2022-07-28 15:00:43 +00:00
r := Disk{}
err = json.NewDecoder(resp.Body).Decode(&r)
if err != nil {
return nil, err
}
return &r, nil
}
func (c *Client) GetUser() (*User, error) {
disk, err := c.GetDisk()
if err != nil {
return nil, err
}
return &disk.User, nil
}
func (c *Client) GetResources(path string, limit, offset int) (*Resource, error) {
if path == "" {
path = "disk:/"
}
2022-07-28 15:00:43 +00:00
data := url.Values{}
data.Set("path", path)
if limit > 0 {
data.Set("limit", strconv.Itoa(limit))
}
if offset > 0 {
data.Set("offset", strconv.Itoa(offset))
}
data.Encode()
req, err := http.NewRequest("GET", V1_DISK_API+"/resources?"+data.Encode(), nil)
2022-07-28 15:00:43 +00:00
if err != nil {
return nil, err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return nil, err
}
if resp.StatusCode == 404 {
return nil, nil
}
err = checkError(resp)
if err != nil {
return nil, err
2022-07-28 15:00:43 +00:00
}
r := Resource{}
err = json.NewDecoder(resp.Body).Decode(&r)
2022-07-28 15:00:43 +00:00
return &r, err
}
func (c *Client) DownloadResource(path, downloadPath string) error {
res, err := c.GetResources(path, 0, 0)
2022-11-21 20:27:11 +00:00
if res == nil {
return errors.New("file not found")
}
if err != nil {
return err
}
resp, err := http.Get(res.File)
if err != nil {
return err
}
2022-11-21 20:27:11 +00:00
defer resp.Body.Close()
2022-11-21 20:27:11 +00:00
if checkError(resp) != nil {
return err
}
// Create the file
out, err := os.Create(downloadPath)
if err != nil {
return err
}
defer out.Close()
// Write the body to file
_, err = io.Copy(out, resp.Body)
return err
}
// PutResources - создание папки
func (c *Client) PutResources(path string) (*RespPutResources, error) {
2022-07-28 15:00:43 +00:00
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequest("PUT", V1_DISK_API+"/resources?"+data.Encode(), nil)
2022-07-28 15:00:43 +00:00
if err != nil {
return nil, err
2022-07-28 15:00:43 +00:00
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return nil, err
2022-07-28 15:00:43 +00:00
}
if resp.StatusCode == 409 {
return nil, nil
}
err = checkError(resp)
if err != nil {
return nil, err
}
2022-07-28 15:00:43 +00:00
var r RespPutResources
2022-07-28 15:00:43 +00:00
err = json.NewDecoder(resp.Body).Decode(&r)
return &r, err
}
func (c *Client) DeleteResources(path string) error {
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequest("DELETE", V1_DISK_API+"/resources?"+data.Encode(), nil)
if err != nil {
return err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return err
}
err = checkError(resp)
2022-07-28 15:00:43 +00:00
return err
}
// UploadResourcesUrl - Загрузить файл на Яндекс диск по URL. Нерекомендуемый
func (c *Client) UploadResourcesUrl(path, downloadPath string) (string, error) {
downloadPath = strings.TrimLeft(downloadPath, "..")
downloadPath = strings.TrimLeft(downloadPath, ".")
fmt.Println("dp:", c.App.ServiceUrl+downloadPath)
fmt.Println("path:", path)
2022-07-28 15:00:43 +00:00
data := url.Values{}
data.Set("path", path)
data.Set("url", c.App.ServiceUrl+downloadPath)
req, err := http.NewRequest("POST", V1_DISK_API+"/resources/upload?"+data.Encode(), nil)
fmt.Println("req:", req.URL)
2022-07-28 15:00:43 +00:00
if err != nil {
return "", err
2022-07-28 15:00:43 +00:00
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return "", err
2022-07-28 15:00:43 +00:00
}
if checkError(resp) != nil {
return "", err
}
2022-07-28 15:00:43 +00:00
var r RespUploadResources
2022-07-28 15:00:43 +00:00
err = json.NewDecoder(resp.Body).Decode(&r)
if err != nil {
return "", err
}
2022-07-28 15:00:43 +00:00
status, err := c.GetOperationsByUrl(r.Href)
if err != nil {
return "", err
2022-11-21 20:27:11 +00:00
}
// Ожидаем пока загрузится файл на Я.Диск. Timeout: 5 sec; tick: 100 ms;
timer := time.NewTimer(5 * time.Second)
for status == "in-progress" {
select {
case <-timer.C:
status = "timeout"
case <-time.Tick(100 * time.Millisecond):
status, err = c.GetOperationsByUrl(r.Href)
if err != nil {
timer.Stop()
return "", err
}
}
}
timer.Stop()
2022-07-28 15:00:43 +00:00
if status != "success" {
return "", errors.New(fmt.Sprintf("bad upload status: %v", status))
}
resource, err := c.GetResources(path, 0, 0)
if err != nil {
return "", err
}
var exportUrl string
if resource != nil {
exportUrl = resource.File
} else {
return "", errors.New("resource not found")
}
return exportUrl, err
}
// UploadResources - Загрузить файл на Яндекс диск
2022-12-19 12:01:38 +00:00
func (c *Client) UploadResources(path string, file io.Reader) (string, string, error) {
// Get url for request
target, err := c.getUploadResourcesUrl(path)
if err != nil {
2022-12-19 12:01:38 +00:00
return "", "", err
}
if target == nil {
2022-12-19 12:01:38 +00:00
return "", "", errors.New("got empty url for upload")
}
req, err := http.NewRequest(target.Method, target.Href, file)
if err != nil {
2022-12-19 12:01:38 +00:00
return "", "", err
}
resp, err := c.HTTPClient.Do(req)
if err := checkError(resp); err != nil {
2022-12-19 12:01:38 +00:00
return "", "", err
}
// Get uploaded data
resource, err := c.GetResources(path, 0, 0)
if err != nil {
2022-12-19 12:01:38 +00:00
return "", "", err
}
var exportUrl string
if resource != nil {
exportUrl = resource.File
} else {
2022-12-19 12:01:38 +00:00
return "", "", errors.New("resource not found")
}
2022-12-19 12:01:38 +00:00
return resource.Path, exportUrl, err
}
// getUploadResourcesUrl - получить ссылку для отправки файла по TLS
func (c *Client) getUploadResourcesUrl(path string) (*RespUploadResources, error) {
data := url.Values{}
data.Set("path", path)
data.Set("overwrite", "true")
req, err := http.NewRequest("GET", V1_DISK_API+"/resources/upload?"+data.Encode(), nil)
if err != nil {
return nil, err
}
resp, err := c.HTTPClient.Do(req)
err = checkError(resp)
if err != nil {
return nil, err
}
r := RespUploadResources{}
err = json.NewDecoder(resp.Body).Decode(&r)
return &r, err
}
// GetOperations - возвращает статус асинхронной операции
func (c *Client) GetOperations(operationID string) (string, error) {
req, err := http.NewRequest("GET", V1_DISK_API+"/operations/"+operationID, nil)
if err != nil {
return "", err
}
resp, err := c.HTTPClient.Do(req)
err = checkError(resp)
if err != nil {
return "", err
}
r := Operation{}
err = json.NewDecoder(resp.Body).Decode(&r)
return r.Status, err
}
// GetOperationsByUrl - возвращает статус асинхронной операции
func (c *Client) GetOperationsByUrl(url string) (string, error) {
req, err := http.NewRequest("GET", url, nil)
if err != nil {
return "", err
}
resp, err := c.HTTPClient.Do(req)
err = checkError(resp)
if err != nil {
return "", err
}
r := Operation{}
err = json.NewDecoder(resp.Body).Decode(&r)
return r.Status, err
2022-07-28 15:00:43 +00:00
}
// PublishResource - публикует ресурс
func (c *Client) PublishResource(path string) error {
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequest("PUT", V1_DISK_API+"/resources/publish?"+data.Encode(), nil)
if err != nil {
return err
}
resp, err := c.HTTPClient.Do(req)
err = checkError(resp)
if err != nil {
return err
}
return nil
}
// checkError - если статус не в диапазоне 200-х вернет расшифровку ошибки
func checkError(resp *http.Response) error {
switch resp.StatusCode {
case 200, 201, 202, 203, 204, 205, 206, 207, 208, 226:
return nil
}
r := Error{}
err := json.NewDecoder(resp.Body).Decode(&r)
if err != nil {
return err
}
return errors.New(fmt.Sprintf("api/yaDisk (%v) err: %v | %v (%v)", resp.StatusCode, r.Message, r.Description,
r.Error))
}