parent
739fb68095
commit
81ba0df9ed
@ -3,6 +3,7 @@ package metainfo
|
|||||||
import (
|
import (
|
||||||
"encoding/base32"
|
"encoding/base32"
|
||||||
"encoding/hex"
|
"encoding/hex"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
"strings"
|
"strings"
|
||||||
@ -13,65 +14,98 @@ type Magnet struct {
|
|||||||
InfoHash Hash
|
InfoHash Hash
|
||||||
Trackers []string
|
Trackers []string
|
||||||
DisplayName string
|
DisplayName string
|
||||||
|
Params url.Values
|
||||||
}
|
}
|
||||||
|
|
||||||
const xtPrefix = "urn:btih:"
|
const xtPrefix = "urn:btih:"
|
||||||
|
|
||||||
func (m Magnet) String() string {
|
func (m Magnet) String() string {
|
||||||
// net.URL likes to assume //, and encodes ':' on us, so we do most of
|
// Deep-copy m.Params
|
||||||
// this manually.
|
vs := make(url.Values, len(m.Params)+len(m.Trackers)+2)
|
||||||
ret := "magnet:?xt="
|
for k, v := range m.Params {
|
||||||
ret += xtPrefix + hex.EncodeToString(m.InfoHash[:])
|
vs[k] = append([]string(nil), v...)
|
||||||
if m.DisplayName != "" {
|
|
||||||
ret += "&dn=" + url.QueryEscape(m.DisplayName)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
vs.Add("xt", xtPrefix+m.InfoHash.HexString())
|
||||||
for _, tr := range m.Trackers {
|
for _, tr := range m.Trackers {
|
||||||
ret += "&tr=" + url.QueryEscape(tr)
|
vs.Add("tr", tr)
|
||||||
}
|
}
|
||||||
return ret
|
if m.DisplayName != "" {
|
||||||
|
vs.Add("dn", m.DisplayName)
|
||||||
|
}
|
||||||
|
|
||||||
|
return (&url.URL{
|
||||||
|
Scheme: "magnet",
|
||||||
|
RawQuery: vs.Encode(),
|
||||||
|
}).String()
|
||||||
}
|
}
|
||||||
|
|
||||||
// ParseMagnetURI parses Magnet-formatted URIs into a Magnet instance
|
// ParseMagnetURI parses Magnet-formatted URIs into a Magnet instance
|
||||||
func ParseMagnetURI(uri string) (m Magnet, err error) {
|
func ParseMagnetURI(uri string) (m Magnet, err error) {
|
||||||
u, err := url.Parse(uri)
|
u, err := url.Parse(uri)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = fmt.Errorf("error parsing uri: %s", err)
|
err = fmt.Errorf("error parsing uri: %w", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if u.Scheme != "magnet" {
|
if u.Scheme != "magnet" {
|
||||||
err = fmt.Errorf("unexpected scheme: %q", u.Scheme)
|
err = fmt.Errorf("unexpected scheme %q", u.Scheme)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
xt := u.Query().Get("xt")
|
q := u.Query()
|
||||||
if !strings.HasPrefix(xt, xtPrefix) {
|
xt := q.Get("xt")
|
||||||
err = fmt.Errorf("bad xt parameter")
|
m.InfoHash, err = parseInfohash(q.Get("xt"))
|
||||||
return
|
|
||||||
}
|
|
||||||
infoHash := xt[len(xtPrefix):]
|
|
||||||
|
|
||||||
// BTIH hash can be in HEX or BASE32 encoding
|
|
||||||
// will assign appropriate func judging from symbol length
|
|
||||||
var decode func(dst, src []byte) (int, error)
|
|
||||||
switch len(infoHash) {
|
|
||||||
case 40:
|
|
||||||
decode = hex.Decode
|
|
||||||
case 32:
|
|
||||||
decode = base32.StdEncoding.Decode
|
|
||||||
}
|
|
||||||
|
|
||||||
if decode == nil {
|
|
||||||
err = fmt.Errorf("unhandled xt parameter encoding: encoded length %d", len(infoHash))
|
|
||||||
return
|
|
||||||
}
|
|
||||||
n, err := decode(m.InfoHash[:], []byte(infoHash))
|
|
||||||
if err != nil {
|
if err != nil {
|
||||||
err = fmt.Errorf("error decoding xt: %s", err)
|
err = fmt.Errorf("error parsing infohash %q: %w", xt, err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
dropFirst(q, "xt")
|
||||||
|
m.DisplayName = q.Get("dn")
|
||||||
|
dropFirst(q, "dn")
|
||||||
|
m.Trackers = q["tr"]
|
||||||
|
delete(q, "tr")
|
||||||
|
if len(q) == 0 {
|
||||||
|
q = nil
|
||||||
|
}
|
||||||
|
m.Params = q
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseInfohash(xt string) (ih Hash, err error) {
|
||||||
|
if !strings.HasPrefix(xt, xtPrefix) {
|
||||||
|
err = errors.New("bad xt parameter prefix")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
encoded := xt[len(xtPrefix):]
|
||||||
|
decode := func() func(dst, src []byte) (int, error) {
|
||||||
|
switch len(encoded) {
|
||||||
|
case 40:
|
||||||
|
return hex.Decode
|
||||||
|
case 32:
|
||||||
|
return base32.StdEncoding.Decode
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}()
|
||||||
|
if decode == nil {
|
||||||
|
err = fmt.Errorf("unhandled xt parameter encoding (encoded length %d)", len(encoded))
|
||||||
|
return
|
||||||
|
}
|
||||||
|
n, err := decode(ih[:], []byte(encoded))
|
||||||
|
if err != nil {
|
||||||
|
err = fmt.Errorf("error decoding xt: %w", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if n != 20 {
|
if n != 20 {
|
||||||
panic(n)
|
panic(n)
|
||||||
}
|
}
|
||||||
m.DisplayName = u.Query().Get("dn")
|
|
||||||
m.Trackers = u.Query()["tr"]
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func dropFirst(vs url.Values, key string) {
|
||||||
|
sl := vs[key]
|
||||||
|
switch len(sl) {
|
||||||
|
case 0, 1:
|
||||||
|
vs.Del(key)
|
||||||
|
default:
|
||||||
|
vs[key] = sl[1:]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
Loading…
x
Reference in New Issue
Block a user