mirror of
https://gitlab.bertha.cloud/partitio/Nextcloud-Partitio/gonextcloud
synced 2024-11-23 00:26:24 +00:00
586 lines
15 KiB
Go
586 lines
15 KiB
Go
|
package grequests
|
|||
|
|
|||
|
import (
|
|||
|
"bytes"
|
|||
|
"crypto/tls"
|
|||
|
"encoding/json"
|
|||
|
"encoding/xml"
|
|||
|
"errors"
|
|||
|
"fmt"
|
|||
|
"io"
|
|||
|
"mime"
|
|||
|
"mime/multipart"
|
|||
|
"net"
|
|||
|
"net/http"
|
|||
|
"net/http/cookiejar"
|
|||
|
"net/textproto"
|
|||
|
"net/url"
|
|||
|
"strconv"
|
|||
|
"strings"
|
|||
|
"time"
|
|||
|
|
|||
|
"github.com/google/go-querystring/query"
|
|||
|
|
|||
|
"context"
|
|||
|
|
|||
|
"golang.org/x/net/publicsuffix"
|
|||
|
)
|
|||
|
|
|||
|
// RequestOptions is the location that of where the data
|
|||
|
type RequestOptions struct {
|
|||
|
|
|||
|
// Data is a map of key values that will eventually convert into the
|
|||
|
// query string of a GET request or the body of a POST request.
|
|||
|
Data map[string]string
|
|||
|
|
|||
|
// Params is a map of query strings that may be used within a GET request
|
|||
|
Params map[string]string
|
|||
|
|
|||
|
// QueryStruct is a struct that encapsulates a set of URL query params
|
|||
|
// this paramter is mutually exclusive with `Params map[string]string` (they cannot be combined)
|
|||
|
// for more information please see https://godoc.org/github.com/google/go-querystring/query
|
|||
|
QueryStruct interface{}
|
|||
|
|
|||
|
// Files is where you can include files to upload. The use of this data
|
|||
|
// structure is limited to POST requests
|
|||
|
Files []FileUpload
|
|||
|
|
|||
|
// JSON can be used when you wish to send JSON within the request body
|
|||
|
JSON interface{}
|
|||
|
|
|||
|
// XML can be used if you wish to send XML within the request body
|
|||
|
XML interface{}
|
|||
|
|
|||
|
// Headers if you want to add custom HTTP headers to the request,
|
|||
|
// this is your friend
|
|||
|
Headers map[string]string
|
|||
|
|
|||
|
// InsecureSkipVerify is a flag that specifies if we should validate the
|
|||
|
// server's TLS certificate. It should be noted that Go's TLS verify mechanism
|
|||
|
// doesn't validate if a certificate has been revoked
|
|||
|
InsecureSkipVerify bool
|
|||
|
|
|||
|
// DisableCompression will disable gzip compression on requests
|
|||
|
DisableCompression bool
|
|||
|
|
|||
|
// UserAgent allows you to set an arbitrary custom user agent
|
|||
|
UserAgent string
|
|||
|
|
|||
|
// Host allows you to set an arbitrary custom host
|
|||
|
Host string
|
|||
|
|
|||
|
// Auth allows you to specify a user name and password that you wish to
|
|||
|
// use when requesting the URL. It will use basic HTTP authentication
|
|||
|
// formatting the username and password in base64 the format is:
|
|||
|
// []string{username, password}
|
|||
|
Auth []string
|
|||
|
|
|||
|
// IsAjax is a flag that can be set to make the request appear
|
|||
|
// to be generated by browser Javascript
|
|||
|
IsAjax bool
|
|||
|
|
|||
|
// Cookies is an array of `http.Cookie` that allows you to attach
|
|||
|
// cookies to your request
|
|||
|
Cookies []*http.Cookie
|
|||
|
|
|||
|
// UseCookieJar will create a custom HTTP client that will
|
|||
|
// process and store HTTP cookies when they are sent down
|
|||
|
UseCookieJar bool
|
|||
|
|
|||
|
// Proxies is a map in the following format
|
|||
|
// *protocol* => proxy address e.g http => http://127.0.0.1:8080
|
|||
|
Proxies map[string]*url.URL
|
|||
|
|
|||
|
// TLSHandshakeTimeout specifies the maximum amount of time waiting to
|
|||
|
// wait for a TLS handshake. Zero means no timeout.
|
|||
|
TLSHandshakeTimeout time.Duration
|
|||
|
|
|||
|
// DialTimeout is the maximum amount of time a dial will wait for
|
|||
|
// a connect to complete.
|
|||
|
DialTimeout time.Duration
|
|||
|
|
|||
|
// KeepAlive specifies the keep-alive period for an active
|
|||
|
// network connection. If zero, keep-alive are not enabled.
|
|||
|
DialKeepAlive time.Duration
|
|||
|
|
|||
|
// RequestTimeout is the maximum amount of time a whole request(include dial / request / redirect)
|
|||
|
// will wait.
|
|||
|
RequestTimeout time.Duration
|
|||
|
|
|||
|
// HTTPClient can be provided if you wish to supply a custom HTTP client
|
|||
|
// this is useful if you want to use an OAUTH client with your request.
|
|||
|
HTTPClient *http.Client
|
|||
|
|
|||
|
// SensitiveHTTPHeaders is a map of sensitive HTTP headers that a user
|
|||
|
// doesn't want passed on a redirect.
|
|||
|
SensitiveHTTPHeaders map[string]struct{}
|
|||
|
|
|||
|
// RedirectLimit is the acceptable amount of redirects that we should expect
|
|||
|
// before returning an error be default this is set to 30. You can change this
|
|||
|
// globally by modifying the `RedirectLimit` variable.
|
|||
|
RedirectLimit int
|
|||
|
|
|||
|
// RequestBody allows you to put anything matching an `io.Reader` into the request
|
|||
|
// this option will take precedence over any other request option specified
|
|||
|
RequestBody io.Reader
|
|||
|
|
|||
|
// CookieJar allows you to specify a special cookiejar to use with your request.
|
|||
|
// this option will take precedence over the `UseCookieJar` option above.
|
|||
|
CookieJar http.CookieJar
|
|||
|
|
|||
|
// Context can be used to maintain state between requests https://golang.org/pkg/context/#Context
|
|||
|
Context context.Context
|
|||
|
}
|
|||
|
|
|||
|
func doRegularRequest(requestVerb, url string, ro *RequestOptions) (*Response, error) {
|
|||
|
return buildResponse(buildRequest(requestVerb, url, ro, nil))
|
|||
|
}
|
|||
|
|
|||
|
func doSessionRequest(requestVerb, url string, ro *RequestOptions, httpClient *http.Client) (*Response, error) {
|
|||
|
return buildResponse(buildRequest(requestVerb, url, ro, httpClient))
|
|||
|
}
|
|||
|
|
|||
|
var quoteEscaper = strings.NewReplacer("\\", "\\\\", `"`, "\\\"")
|
|||
|
|
|||
|
func escapeQuotes(s string) string {
|
|||
|
return quoteEscaper.Replace(s)
|
|||
|
}
|
|||
|
|
|||
|
// buildRequest is where most of the magic happens for request processing
|
|||
|
func buildRequest(httpMethod, url string, ro *RequestOptions, httpClient *http.Client) (*http.Response, error) {
|
|||
|
if ro == nil {
|
|||
|
ro = &RequestOptions{}
|
|||
|
}
|
|||
|
|
|||
|
if ro.CookieJar != nil {
|
|||
|
ro.UseCookieJar = true
|
|||
|
}
|
|||
|
|
|||
|
// Create our own HTTP client
|
|||
|
|
|||
|
if httpClient == nil {
|
|||
|
httpClient = BuildHTTPClient(*ro)
|
|||
|
}
|
|||
|
|
|||
|
var err error // we don't want to shadow url so we won't use :=
|
|||
|
switch {
|
|||
|
case len(ro.Params) != 0:
|
|||
|
if url, err = buildURLParams(url, ro.Params); err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
case ro.QueryStruct != nil:
|
|||
|
if url, err = buildURLStruct(url, ro.QueryStruct); err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// Build the request
|
|||
|
req, err := buildHTTPRequest(httpMethod, url, ro)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
// Do we need to add any HTTP headers or Basic Auth?
|
|||
|
addHTTPHeaders(ro, req)
|
|||
|
addCookies(ro, req)
|
|||
|
|
|||
|
addRedirectFunctionality(httpClient, ro)
|
|||
|
|
|||
|
if ro.Context != nil {
|
|||
|
req = req.WithContext(ro.Context)
|
|||
|
}
|
|||
|
|
|||
|
return httpClient.Do(req)
|
|||
|
}
|
|||
|
|
|||
|
func buildHTTPRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
if ro.RequestBody != nil {
|
|||
|
return http.NewRequest(httpMethod, userURL, ro.RequestBody)
|
|||
|
}
|
|||
|
|
|||
|
if ro.JSON != nil {
|
|||
|
return createBasicJSONRequest(httpMethod, userURL, ro)
|
|||
|
}
|
|||
|
|
|||
|
if ro.XML != nil {
|
|||
|
return createBasicXMLRequest(httpMethod, userURL, ro)
|
|||
|
}
|
|||
|
|
|||
|
if ro.Files != nil {
|
|||
|
return createFileUploadRequest(httpMethod, userURL, ro)
|
|||
|
}
|
|||
|
|
|||
|
if ro.Data != nil {
|
|||
|
return createBasicRequest(httpMethod, userURL, ro)
|
|||
|
}
|
|||
|
|
|||
|
return http.NewRequest(httpMethod, userURL, nil)
|
|||
|
}
|
|||
|
|
|||
|
func createFileUploadRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
if httpMethod == "POST" {
|
|||
|
return createMultiPartPostRequest(httpMethod, userURL, ro)
|
|||
|
}
|
|||
|
|
|||
|
// This may be a PUT or PATCH request so we will just put the raw
|
|||
|
// io.ReadCloser in the request body
|
|||
|
// and guess the MIME type from the file name
|
|||
|
|
|||
|
// At the moment, we will only support 1 file upload as a time
|
|||
|
// when uploading using PUT or PATCH
|
|||
|
|
|||
|
req, err := http.NewRequest(httpMethod, userURL, ro.Files[0].FileContents)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
req.Header.Set("Content-Type", mime.TypeByExtension(ro.Files[0].FileName))
|
|||
|
|
|||
|
return req, nil
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
func createBasicXMLRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
var reader io.Reader
|
|||
|
|
|||
|
switch ro.XML.(type) {
|
|||
|
case string:
|
|||
|
reader = strings.NewReader(ro.XML.(string))
|
|||
|
case []byte:
|
|||
|
reader = bytes.NewReader(ro.XML.([]byte))
|
|||
|
default:
|
|||
|
byteSlice, err := xml.Marshal(ro.XML)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
reader = bytes.NewReader(byteSlice)
|
|||
|
}
|
|||
|
|
|||
|
req, err := http.NewRequest(httpMethod, userURL, reader)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
req.Header.Set("Content-Type", "application/xml")
|
|||
|
|
|||
|
return req, nil
|
|||
|
|
|||
|
}
|
|||
|
func createMultiPartPostRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
requestBody := &bytes.Buffer{}
|
|||
|
|
|||
|
multipartWriter := multipart.NewWriter(requestBody)
|
|||
|
|
|||
|
for i, f := range ro.Files {
|
|||
|
|
|||
|
if f.FileContents == nil {
|
|||
|
return nil, errors.New("grequests: Pointer FileContents cannot be nil")
|
|||
|
}
|
|||
|
|
|||
|
fieldName := f.FieldName
|
|||
|
|
|||
|
if fieldName == "" {
|
|||
|
if len(ro.Files) > 1 {
|
|||
|
fieldName = strings.Join([]string{"file", strconv.Itoa(i + 1)}, "")
|
|||
|
} else {
|
|||
|
fieldName = "file"
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
var writer io.Writer
|
|||
|
var err error
|
|||
|
|
|||
|
if f.FileMime != "" {
|
|||
|
h := make(textproto.MIMEHeader)
|
|||
|
h.Set("Content-Disposition", fmt.Sprintf(`form-data; name="%s"; filename="%s"`, escapeQuotes(fieldName), escapeQuotes(f.FileName)))
|
|||
|
h.Set("Content-Type", f.FileMime)
|
|||
|
writer, err = multipartWriter.CreatePart(h)
|
|||
|
} else {
|
|||
|
writer, err = multipartWriter.CreateFormFile(fieldName, f.FileName)
|
|||
|
}
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
if _, err = io.Copy(writer, f.FileContents); err != nil && err != io.EOF {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
f.FileContents.Close()
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
// Populate the other parts of the form (if there are any)
|
|||
|
for key, value := range ro.Data {
|
|||
|
multipartWriter.WriteField(key, value)
|
|||
|
}
|
|||
|
|
|||
|
if err := multipartWriter.Close(); err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
req, err := http.NewRequest(httpMethod, userURL, requestBody)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
req.Header.Add("Content-Type", multipartWriter.FormDataContentType())
|
|||
|
|
|||
|
return req, err
|
|||
|
}
|
|||
|
|
|||
|
func createBasicJSONRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
|
|||
|
var reader io.Reader
|
|||
|
switch ro.JSON.(type) {
|
|||
|
case string:
|
|||
|
reader = strings.NewReader(ro.JSON.(string))
|
|||
|
case []byte:
|
|||
|
reader = bytes.NewReader(ro.JSON.([]byte))
|
|||
|
default:
|
|||
|
byteSlice, err := json.Marshal(ro.JSON)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
reader = bytes.NewReader(byteSlice)
|
|||
|
}
|
|||
|
|
|||
|
req, err := http.NewRequest(httpMethod, userURL, reader)
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
req.Header.Set("Content-Type", "application/json")
|
|||
|
|
|||
|
return req, nil
|
|||
|
|
|||
|
}
|
|||
|
func createBasicRequest(httpMethod, userURL string, ro *RequestOptions) (*http.Request, error) {
|
|||
|
|
|||
|
req, err := http.NewRequest(httpMethod, userURL, strings.NewReader(encodePostValues(ro.Data)))
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return nil, err
|
|||
|
}
|
|||
|
|
|||
|
// The content type must be set to a regular form
|
|||
|
req.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
|||
|
|
|||
|
return req, nil
|
|||
|
}
|
|||
|
|
|||
|
func encodePostValues(postValues map[string]string) string {
|
|||
|
urlValues := &url.Values{}
|
|||
|
|
|||
|
for key, value := range postValues {
|
|||
|
urlValues.Set(key, value)
|
|||
|
}
|
|||
|
|
|||
|
return urlValues.Encode() // This will sort all of the string values
|
|||
|
}
|
|||
|
|
|||
|
// proxySettings will default to the default proxy settings if none are provided
|
|||
|
// if settings are provided – they will override the environment variables
|
|||
|
func (ro RequestOptions) proxySettings(req *http.Request) (*url.URL, error) {
|
|||
|
// No proxies – lets use the default
|
|||
|
if len(ro.Proxies) == 0 {
|
|||
|
return http.ProxyFromEnvironment(req)
|
|||
|
}
|
|||
|
|
|||
|
// There was a proxy specified – do we support the protocol?
|
|||
|
if _, ok := ro.Proxies[req.URL.Scheme]; ok {
|
|||
|
return ro.Proxies[req.URL.Scheme], nil
|
|||
|
}
|
|||
|
|
|||
|
// Proxies were specified but not for any protocol that we use
|
|||
|
return http.ProxyFromEnvironment(req)
|
|||
|
|
|||
|
}
|
|||
|
|
|||
|
// dontUseDefaultClient will tell the "client creator" if a custom client is needed
|
|||
|
// it checks the following items (and will create a custom client of these are)
|
|||
|
// true
|
|||
|
// 1. Do we want to accept invalid SSL certificates?
|
|||
|
// 2. Do we want to disable compression?
|
|||
|
// 3. Do we want a custom proxy?
|
|||
|
// 4. Do we want to change the default timeout for TLS Handshake?
|
|||
|
// 5. Do we want to change the default request timeout?
|
|||
|
// 6. Do we want to change the default connection timeout?
|
|||
|
// 7. Do you want to use the http.Client's cookieJar?
|
|||
|
func (ro RequestOptions) dontUseDefaultClient() bool {
|
|||
|
switch {
|
|||
|
case ro.InsecureSkipVerify == true:
|
|||
|
case ro.DisableCompression == true:
|
|||
|
case len(ro.Proxies) != 0:
|
|||
|
case ro.TLSHandshakeTimeout != 0:
|
|||
|
case ro.DialTimeout != 0:
|
|||
|
case ro.DialKeepAlive != 0:
|
|||
|
case len(ro.Cookies) != 0:
|
|||
|
case ro.UseCookieJar != false:
|
|||
|
case ro.RequestTimeout != 0:
|
|||
|
default:
|
|||
|
return false
|
|||
|
}
|
|||
|
return true
|
|||
|
}
|
|||
|
|
|||
|
// BuildHTTPClient is a function that will return a custom HTTP client based on the request options provided
|
|||
|
// the check is in UseDefaultClient
|
|||
|
func BuildHTTPClient(ro RequestOptions) *http.Client {
|
|||
|
|
|||
|
if ro.HTTPClient != nil {
|
|||
|
return ro.HTTPClient
|
|||
|
}
|
|||
|
|
|||
|
// Does the user want to change the defaults?
|
|||
|
if !ro.dontUseDefaultClient() {
|
|||
|
return http.DefaultClient
|
|||
|
}
|
|||
|
|
|||
|
// Using the user config for tls timeout or default
|
|||
|
if ro.TLSHandshakeTimeout == 0 {
|
|||
|
ro.TLSHandshakeTimeout = tlsHandshakeTimeout
|
|||
|
}
|
|||
|
|
|||
|
// Using the user config for dial timeout or default
|
|||
|
if ro.DialTimeout == 0 {
|
|||
|
ro.DialTimeout = dialTimeout
|
|||
|
}
|
|||
|
|
|||
|
// Using the user config for dial keep alive or default
|
|||
|
if ro.DialKeepAlive == 0 {
|
|||
|
ro.DialKeepAlive = dialKeepAlive
|
|||
|
}
|
|||
|
|
|||
|
if ro.RequestTimeout == 0 {
|
|||
|
ro.RequestTimeout = requestTimeout
|
|||
|
}
|
|||
|
|
|||
|
var cookieJar http.CookieJar
|
|||
|
|
|||
|
if ro.UseCookieJar {
|
|||
|
if ro.CookieJar != nil {
|
|||
|
cookieJar = ro.CookieJar
|
|||
|
} else {
|
|||
|
// The function does not return an error ever... so we are just ignoring it
|
|||
|
cookieJar, _ = cookiejar.New(&cookiejar.Options{PublicSuffixList: publicsuffix.List})
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return &http.Client{
|
|||
|
Jar: cookieJar,
|
|||
|
Transport: createHTTPTransport(ro),
|
|||
|
Timeout: ro.RequestTimeout,
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func createHTTPTransport(ro RequestOptions) *http.Transport {
|
|||
|
ourHTTPTransport := &http.Transport{
|
|||
|
// These are borrowed from the default transporter
|
|||
|
Proxy: ro.proxySettings,
|
|||
|
Dial: (&net.Dialer{
|
|||
|
Timeout: ro.DialTimeout,
|
|||
|
KeepAlive: ro.DialKeepAlive,
|
|||
|
}).Dial,
|
|||
|
TLSHandshakeTimeout: ro.TLSHandshakeTimeout,
|
|||
|
|
|||
|
// Here comes the user settings
|
|||
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: ro.InsecureSkipVerify},
|
|||
|
DisableCompression: ro.DisableCompression,
|
|||
|
}
|
|||
|
EnsureTransporterFinalized(ourHTTPTransport)
|
|||
|
return ourHTTPTransport
|
|||
|
}
|
|||
|
|
|||
|
// buildURLParams returns a URL with all of the params
|
|||
|
// Note: This function will override current URL params if they contradict what is provided in the map
|
|||
|
// That is what the "magic" is on the last line
|
|||
|
func buildURLParams(userURL string, params map[string]string) (string, error) {
|
|||
|
parsedURL, err := url.Parse(userURL)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return "", err
|
|||
|
}
|
|||
|
|
|||
|
parsedQuery, err := url.ParseQuery(parsedURL.RawQuery)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return "", nil
|
|||
|
}
|
|||
|
|
|||
|
for key, value := range params {
|
|||
|
parsedQuery.Set(key, value)
|
|||
|
}
|
|||
|
|
|||
|
return addQueryParams(parsedURL, parsedQuery), nil
|
|||
|
}
|
|||
|
|
|||
|
// addHTTPHeaders adds any additional HTTP headers that need to be added are added here including:
|
|||
|
// 1. Custom User agent
|
|||
|
// 2. Authorization Headers
|
|||
|
// 3. Any other header requested
|
|||
|
func addHTTPHeaders(ro *RequestOptions, req *http.Request) {
|
|||
|
for key, value := range ro.Headers {
|
|||
|
req.Header.Set(key, value)
|
|||
|
}
|
|||
|
|
|||
|
if ro.UserAgent != "" {
|
|||
|
req.Header.Set("User-Agent", ro.UserAgent)
|
|||
|
} else {
|
|||
|
req.Header.Set("User-Agent", localUserAgent)
|
|||
|
}
|
|||
|
|
|||
|
if ro.Host != "" {
|
|||
|
req.Host = ro.Host
|
|||
|
}
|
|||
|
|
|||
|
if ro.Auth != nil {
|
|||
|
req.SetBasicAuth(ro.Auth[0], ro.Auth[1])
|
|||
|
}
|
|||
|
|
|||
|
if ro.IsAjax == true {
|
|||
|
req.Header.Set("X-Requested-With", "XMLHttpRequest")
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func addCookies(ro *RequestOptions, req *http.Request) {
|
|||
|
for _, c := range ro.Cookies {
|
|||
|
req.AddCookie(c)
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
func addQueryParams(parsedURL *url.URL, parsedQuery url.Values) string {
|
|||
|
return strings.Join([]string{strings.Replace(parsedURL.String(), "?"+parsedURL.RawQuery, "", -1), parsedQuery.Encode()}, "?")
|
|||
|
}
|
|||
|
|
|||
|
func buildURLStruct(userURL string, URLStruct interface{}) (string, error) {
|
|||
|
parsedURL, err := url.Parse(userURL)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return "", err
|
|||
|
}
|
|||
|
|
|||
|
parsedQuery, err := url.ParseQuery(parsedURL.RawQuery)
|
|||
|
|
|||
|
if err != nil {
|
|||
|
return "", err
|
|||
|
}
|
|||
|
|
|||
|
queryStruct, err := query.Values(URLStruct)
|
|||
|
if err != nil {
|
|||
|
return "", err
|
|||
|
}
|
|||
|
|
|||
|
for key, value := range queryStruct {
|
|||
|
for _, v := range value {
|
|||
|
parsedQuery.Add(key, v)
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
return addQueryParams(parsedURL, parsedQuery), nil
|
|||
|
}
|