autobrr/pkg/sabnzbd/sabnzbd.go
Kyle Sanderson 3234f0d919
refactor(http): implement shared transport and clients (#1288)
* fix(http): flip to a shared transport and clients

* nice threads

* that is terrible

* fake uri for magnet

* lazy locking

* why bother with r's

* flip magic params to struct

* refactor(http-clients): use separate clients with shared transport

* refactor(http-clients): add missing license header

* refactor(http-clients): defer and fix errors

---------

Co-authored-by: ze0s <ze0s@riseup.net>
2023-12-29 23:49:22 +01:00

179 lines
2.9 KiB
Go

// Copyright (c) 2021 - 2023, Ludvig Lundgren and the autobrr contributors.
// SPDX-License-Identifier: GPL-2.0-or-later
package sabnzbd
import (
"context"
"encoding/json"
"fmt"
"io"
"log"
"net/http"
"net/url"
"time"
"github.com/autobrr/autobrr/pkg/sharedhttp"
)
type Client struct {
addr string
apiKey string
basicUser string
basicPass string
log *log.Logger
http *http.Client
}
type Options struct {
Addr string
ApiKey string
BasicUser string
BasicPass string
Log *log.Logger
}
func New(opts Options) *Client {
c := &Client{
addr: opts.Addr,
apiKey: opts.ApiKey,
basicUser: opts.BasicUser,
basicPass: opts.BasicPass,
log: log.New(io.Discard, "", log.LstdFlags),
http: &http.Client{
Timeout: time.Second * 60,
Transport: sharedhttp.Transport,
},
}
if opts.Log != nil {
c.log = opts.Log
}
return c
}
func (c *Client) AddFromUrl(ctx context.Context, r AddNzbRequest) (*AddFileResponse, error) {
v := url.Values{}
v.Set("mode", "addurl")
v.Set("name", r.Url)
v.Set("output", "json")
v.Set("apikey", c.apiKey)
v.Set("cat", "*")
if r.Category != "" {
v.Set("cat", r.Category)
}
addr, err := url.JoinPath(c.addr, "/api")
if err != nil {
return nil, err
}
u, err := url.Parse(addr)
if err != nil {
return nil, err
}
u.RawQuery = v.Encode()
req, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), nil)
if err != nil {
return nil, err
}
if c.basicUser != "" && c.basicPass != "" {
req.SetBasicAuth(c.basicUser, c.basicPass)
}
res, err := c.http.Do(req)
if err != nil {
return nil, err
}
defer res.Body.Close()
body, err := io.ReadAll(res.Body)
if err != nil {
return nil, err
}
fmt.Print(body)
var data AddFileResponse
if err := json.Unmarshal(body, &data); err != nil {
return nil, err
}
return &data, nil
}
func (c *Client) Version(ctx context.Context) (*VersionResponse, error) {
v := url.Values{}
v.Set("mode", "version")
v.Set("output", "json")
v.Set("apikey", c.apiKey)
addr, err := url.JoinPath(c.addr, "/api")
if err != nil {
return nil, err
}
u, err := url.Parse(addr)
if err != nil {
return nil, err
}
u.RawQuery = v.Encode()
req, err := http.NewRequestWithContext(ctx, http.MethodGet, u.String(), nil)
if err != nil {
return nil, err
}
if c.basicUser != "" && c.basicPass != "" {
req.SetBasicAuth(c.basicUser, c.basicPass)
}
res, err := c.http.Do(req)
if err != nil {
return nil, err
}
defer res.Body.Close()
body, err := io.ReadAll(res.Body)
if err != nil {
return nil, err
}
var data VersionResponse
if err := json.Unmarshal(body, &data); err != nil {
return nil, err
}
return &data, nil
}
type VersionResponse struct {
Version string `json:"version"`
}
type AddFileResponse struct {
NzoIDs []string `json:"nzo_ids"`
ApiError
}
type ApiError struct {
ErrorMsg string `json:"error,omitempty"`
}
type AddNzbRequest struct {
Url string
Category string
}