docxTemplater/yadisk/api.go

587 lines
12 KiB
Go
Raw Normal View History

package yadisk
2022-07-28 15:00:43 +00:00
import (
"context"
2022-07-28 15:00:43 +00:00
"encoding/json"
"errors"
2022-07-28 15:00:43 +00:00
"fmt"
"io"
2022-07-28 15:00:43 +00:00
"net/http"
"net/url"
"os"
"strconv"
"strings"
"time"
"golang.org/x/oauth2"
2024-12-16 09:17:48 +00:00
"gitea.pena/PenaSide/docxTemplater/tools"
2022-07-28 15:00:43 +00:00
)
const (
BaseURL = "https://cloud-api.yandex.net"
OauthURL = "https://oauth.yandex.ru"
V1DiskAPI = BaseURL + "/v1/disk"
DefaultFolder = "disk:/templategen"
DefaultTemplateFolder = DefaultFolder + "/templates"
DefaultSaveFolder = DefaultFolder + "/saved"
2023-08-17 08:12:22 +00:00
TokenTypeOAuth = "OAuth"
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: OauthURL + "/authorize",
TokenURL: OauthURL + "/token",
},
RedirectURL: redirectURI,
Scopes: nil,
},
ServiceURL: serviceURL,
2022-07-28 15:00:43 +00:00
}
}
2023-09-06 10:13:15 +00:00
func (ca *ClientApp) GenerateOAuthURL(amoID, penaID, redirectURL string) (string, error) {
state, err := tools.EncryptTokenAES(tools.StateToken{
2023-09-06 13:01:46 +00:00
AmoID: amoID,
PenaID: penaID,
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
2023-08-17 08:12:22 +00:00
token.TokenType = TokenTypeOAuth
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
2023-08-17 08:12:22 +00:00
token.TokenType = TokenTypeOAuth
return token, nil
2022-07-28 15:00:43 +00:00
}
func (c *Client) GetDisk(ctx context.Context) (*Disk, error) {
req, err := http.NewRequestWithContext(ctx, http.MethodGet, V1DiskAPI, http.NoBody)
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
}
2023-08-15 02:20:33 +00:00
defer resp.Body.Close()
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(ctx context.Context) (*User, error) {
disk, err := c.GetDisk(ctx)
if err != nil {
return nil, err
}
return &disk.User, nil
}
type DResp struct {
Href string `json:"href"`
Method string `json:"method"`
Templated bool `json:"templated"`
}
func (c *Client) DownloadFile(path string) (*DResp, error) {
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequest(http.MethodGet, V1DiskAPI+"/resources/download?"+data.Encode(), http.NoBody)
if err != nil {
return nil, err
}
res, err := c.HTTPClient.Do(req)
if err != nil {
return nil, err
}
2023-08-15 19:32:50 +00:00
defer res.Body.Close()
2023-04-03 18:47:47 +00:00
if err := checkError(res); err != nil {
return nil, err
}
var resp DResp
if err := json.NewDecoder(res.Body).Decode(&resp); err != nil {
return nil, err
}
return &resp, nil
}
func (c *Client) GetResources(ctx context.Context, 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.NewRequestWithContext(ctx, http.MethodGet, V1DiskAPI+"/resources?"+data.Encode(), http.NoBody)
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
}
2023-08-15 02:20:33 +00:00
defer resp.Body.Close()
if resp.StatusCode == http.StatusNotFound {
2022-07-28 15:00:43 +00:00
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(ctx context.Context, path, downloadPath string) error {
res, err := c.GetResources(ctx, path, 0, 0)
if err != nil {
return err
}
2022-11-21 20:27:11 +00:00
if res == nil {
return errors.New("file not found")
}
req, err := http.NewRequestWithContext(ctx, http.MethodGet, res.File, http.NoBody)
if err != nil {
return err
}
2022-11-21 20:27:11 +00:00
resp, err := http.DefaultClient.Do(req)
2022-11-21 20:27:11 +00:00
if checkError(resp) != nil {
return err
}
defer resp.Body.Close()
// Create the file
out, err := os.Create(downloadPath)
if err != nil {
return err
}
defer out.Close() //nolint
// Write the body to file
_, err = io.Copy(out, resp.Body)
return err
}
func (c *Client) DownloadResourcesBytes(ctx context.Context, path string) ([]byte, error) {
res, err := c.GetResources(ctx, path, 0, 0)
if err != nil {
return nil, err
}
if res == nil {
return nil, errors.New("file not found")
}
req, err := http.NewRequestWithContext(ctx, http.MethodGet, res.File, http.NoBody)
if err != nil {
return nil, err
}
resp, err := http.DefaultClient.Do(req)
if err != nil {
return nil, err
}
defer resp.Body.Close()
if err = checkError(resp); err != nil {
return nil, err
}
return io.ReadAll(resp.Body)
}
// 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(http.MethodPut, V1DiskAPI+"/resources?"+data.Encode(), http.NoBody)
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
}
2023-08-15 02:20:33 +00:00
defer resp.Body.Close()
if resp.StatusCode == http.StatusConflict {
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(ctx context.Context, path string) error {
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequestWithContext(ctx, http.MethodDelete, V1DiskAPI+"/resources?"+data.Encode(), http.NoBody)
if err != nil {
return err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
err = checkError(resp)
2022-07-28 15:00:43 +00:00
return err
}
// UploadResourcesURL - Загрузить файл на Яндекс диск по URL. Нерекомендуемый.
func (c *Client) UploadResourcesURL(ctx context.Context, path, downloadPath string) (string, error) {
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.NewRequestWithContext(ctx, http.MethodPost, V1DiskAPI+"/resources/upload?"+data.Encode(), http.NoBody)
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
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
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(ctx, 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)
ticker := time.NewTicker(100 * time.Millisecond)
defer ticker.Stop()
for status == "in-progress" {
select {
case <-timer.C:
status = "timeout"
case <-ticker.C:
status, err = c.GetOperationsByURL(ctx, r.Href)
if err != nil {
timer.Stop()
return "", err
}
}
}
timer.Stop()
2022-07-28 15:00:43 +00:00
if status != "success" {
return "", fmt.Errorf("bad upload status: %v", status)
}
resource, err := c.GetResources(ctx, 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 - Загрузить файл на Яндекс диск.
func (c *Client) UploadResources(ctx context.Context, path string, file io.Reader) (string, string, error) {
// Get url for request
target, err := c.getUploadResourcesURL(ctx, 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.NewRequestWithContext(ctx, 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 != nil {
return "", "", err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
err = checkError(resp)
if err != nil {
2022-12-19 12:01:38 +00:00
return "", "", err
}
// Get uploaded data
resource, err := c.GetResources(ctx, 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")
}
return resource.Path, exportURL, err
}
// getUploadResourcesURL - получить ссылку для отправки файла по TLS.
func (c *Client) getUploadResourcesURL(ctx context.Context, path string) (*RespUploadResources, error) {
data := url.Values{}
data.Set("path", path)
data.Set("overwrite", "true")
req, err := http.NewRequestWithContext(ctx, http.MethodGet, V1DiskAPI+"/resources/upload?"+data.Encode(), http.NoBody)
if err != nil {
return nil, err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return nil, err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
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(ctx context.Context, operationID string) (string, error) {
req, err := http.NewRequestWithContext(ctx, http.MethodGet, V1DiskAPI+"/operations/"+operationID, http.NoBody)
if err != nil {
return "", err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return "", err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
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(ctx context.Context, url string) (string, error) {
req, err := http.NewRequestWithContext(ctx, http.MethodGet, url, http.NoBody)
if err != nil {
return "", err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return "", err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
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(ctx context.Context, path string) error {
data := url.Values{}
data.Set("path", path)
req, err := http.NewRequestWithContext(ctx, http.MethodPut, V1DiskAPI+"/resources/publish?"+data.Encode(), http.NoBody)
if err != nil {
return err
}
resp, err := c.HTTPClient.Do(req)
if err != nil {
return err
}
2023-08-15 19:32:50 +00:00
defer resp.Body.Close()
err = checkError(resp)
if err != nil {
return err
}
return nil
}
// checkError - если статус не в диапазоне 200-х вернет расшифровку ошибки.
func checkError(resp *http.Response) error {
switch resp.StatusCode {
case http.StatusOK, http.StatusCreated, http.StatusAccepted, http.StatusNonAuthoritativeInfo,
http.StatusNoContent, http.StatusResetContent, http.StatusPartialContent,
http.StatusMultiStatus, http.StatusAlreadyReported, http.StatusIMUsed:
return nil
}
r := Error{}
err := json.NewDecoder(resp.Body).Decode(&r)
if err != nil {
return err
}
return fmt.Errorf("api/yaDisk (%v) err: %v | %v (%v)", resp.StatusCode, r.Message, r.Description, r.Error)
}