2019-11-07 19:05:39 +00:00
|
|
|
// Copyright 2011 Google LLC. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style
|
|
|
|
// license that can be found in the LICENSE file.
|
|
|
|
|
|
|
|
// Package googleapi contains the common code shared by all Google API
|
|
|
|
// libraries.
|
|
|
|
package googleapi // import "google.golang.org/api/googleapi"
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"strings"
|
2022-02-07 13:07:36 +00:00
|
|
|
"time"
|
2019-11-07 19:05:39 +00:00
|
|
|
|
2019-11-19 19:29:35 +00:00
|
|
|
"google.golang.org/api/internal/third_party/uritemplates"
|
2019-11-07 19:05:39 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
// ContentTyper is an interface for Readers which know (or would like
|
|
|
|
// to override) their Content-Type. If a media body doesn't implement
|
|
|
|
// ContentTyper, the type is sniffed from the content using
|
|
|
|
// http.DetectContentType.
|
|
|
|
type ContentTyper interface {
|
|
|
|
ContentType() string
|
|
|
|
}
|
|
|
|
|
|
|
|
// A SizeReaderAt is a ReaderAt with a Size method.
|
|
|
|
// An io.SectionReader implements SizeReaderAt.
|
|
|
|
type SizeReaderAt interface {
|
|
|
|
io.ReaderAt
|
|
|
|
Size() int64
|
|
|
|
}
|
|
|
|
|
|
|
|
// ServerResponse is embedded in each Do response and
|
|
|
|
// provides the HTTP status code and header sent by the server.
|
|
|
|
type ServerResponse struct {
|
|
|
|
// HTTPStatusCode is the server's response status code. When using a
|
|
|
|
// resource method's Do call, this will always be in the 2xx range.
|
|
|
|
HTTPStatusCode int
|
|
|
|
// Header contains the response header fields from the server.
|
|
|
|
Header http.Header
|
|
|
|
}
|
|
|
|
|
|
|
|
const (
|
|
|
|
// Version defines the gax version being used. This is typically sent
|
|
|
|
// in an HTTP header to services.
|
|
|
|
Version = "0.5"
|
|
|
|
|
|
|
|
// UserAgent is the header string used to identify this package.
|
|
|
|
UserAgent = "google-api-go-client/" + Version
|
|
|
|
|
|
|
|
// DefaultUploadChunkSize is the default chunk size to use for resumable
|
|
|
|
// uploads if not specified by the user.
|
2020-02-10 21:35:08 +00:00
|
|
|
DefaultUploadChunkSize = 16 * 1024 * 1024
|
2019-11-07 19:05:39 +00:00
|
|
|
|
|
|
|
// MinUploadChunkSize is the minimum chunk size that can be used for
|
|
|
|
// resumable uploads. All user-specified chunk sizes must be multiple of
|
|
|
|
// this value.
|
|
|
|
MinUploadChunkSize = 256 * 1024
|
|
|
|
)
|
|
|
|
|
|
|
|
// Error contains an error response from the server.
|
|
|
|
type Error struct {
|
|
|
|
// Code is the HTTP response status code and will always be populated.
|
|
|
|
Code int `json:"code"`
|
|
|
|
// Message is the server response message and is only populated when
|
|
|
|
// explicitly referenced by the JSON server response.
|
|
|
|
Message string `json:"message"`
|
2020-05-12 11:26:19 +00:00
|
|
|
// Details provide more context to an error.
|
|
|
|
Details []interface{} `json:"details"`
|
2019-11-07 19:05:39 +00:00
|
|
|
// Body is the raw response returned by the server.
|
|
|
|
// It is often but not always JSON, depending on how the request fails.
|
|
|
|
Body string
|
|
|
|
// Header contains the response header fields from the server.
|
|
|
|
Header http.Header
|
|
|
|
|
|
|
|
Errors []ErrorItem
|
2022-11-10 11:46:33 +00:00
|
|
|
// err is typically a wrapped apierror.APIError, see
|
|
|
|
// google-api-go-client/internal/gensupport/error.go.
|
|
|
|
err error
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ErrorItem is a detailed error code & message from the Google API frontend.
|
|
|
|
type ErrorItem struct {
|
|
|
|
// Reason is the typed error code. For example: "some_example".
|
|
|
|
Reason string `json:"reason"`
|
|
|
|
// Message is the human-readable description of the error.
|
|
|
|
Message string `json:"message"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Error) Error() string {
|
|
|
|
if len(e.Errors) == 0 && e.Message == "" {
|
|
|
|
return fmt.Sprintf("googleapi: got HTTP response code %d with body: %v", e.Code, e.Body)
|
|
|
|
}
|
|
|
|
var buf bytes.Buffer
|
|
|
|
fmt.Fprintf(&buf, "googleapi: Error %d: ", e.Code)
|
|
|
|
if e.Message != "" {
|
|
|
|
fmt.Fprintf(&buf, "%s", e.Message)
|
|
|
|
}
|
2020-05-12 11:26:19 +00:00
|
|
|
if len(e.Details) > 0 {
|
|
|
|
var detailBuf bytes.Buffer
|
|
|
|
enc := json.NewEncoder(&detailBuf)
|
|
|
|
enc.SetIndent("", " ")
|
|
|
|
if err := enc.Encode(e.Details); err == nil {
|
|
|
|
fmt.Fprint(&buf, "\nDetails:")
|
|
|
|
fmt.Fprintf(&buf, "\n%s", detailBuf.String())
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
2019-11-07 19:05:39 +00:00
|
|
|
if len(e.Errors) == 0 {
|
|
|
|
return strings.TrimSpace(buf.String())
|
|
|
|
}
|
|
|
|
if len(e.Errors) == 1 && e.Errors[0].Message == e.Message {
|
|
|
|
fmt.Fprintf(&buf, ", %s", e.Errors[0].Reason)
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
fmt.Fprintln(&buf, "\nMore details:")
|
|
|
|
for _, v := range e.Errors {
|
|
|
|
fmt.Fprintf(&buf, "Reason: %s, Message: %s\n", v.Reason, v.Message)
|
|
|
|
}
|
|
|
|
return buf.String()
|
|
|
|
}
|
|
|
|
|
2022-11-10 11:46:33 +00:00
|
|
|
// Wrap allows an existing Error to wrap another error. See also [Error.Unwrap].
|
|
|
|
func (e *Error) Wrap(err error) {
|
|
|
|
e.err = err
|
|
|
|
}
|
|
|
|
|
|
|
|
func (e *Error) Unwrap() error {
|
|
|
|
return e.err
|
|
|
|
}
|
|
|
|
|
2019-11-07 19:05:39 +00:00
|
|
|
type errorReply struct {
|
|
|
|
Error *Error `json:"error"`
|
|
|
|
}
|
|
|
|
|
|
|
|
// CheckResponse returns an error (of type *Error) if the response
|
|
|
|
// status code is not 2xx.
|
|
|
|
func CheckResponse(res *http.Response) error {
|
|
|
|
if res.StatusCode >= 200 && res.StatusCode <= 299 {
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-12 19:43:23 +00:00
|
|
|
slurp, err := io.ReadAll(res.Body)
|
2019-11-07 19:05:39 +00:00
|
|
|
if err == nil {
|
|
|
|
jerr := new(errorReply)
|
|
|
|
err = json.Unmarshal(slurp, jerr)
|
|
|
|
if err == nil && jerr.Error != nil {
|
|
|
|
if jerr.Error.Code == 0 {
|
|
|
|
jerr.Error.Code = res.StatusCode
|
|
|
|
}
|
|
|
|
jerr.Error.Body = string(slurp)
|
2022-02-11 23:12:19 +00:00
|
|
|
jerr.Error.Header = res.Header
|
2019-11-07 19:05:39 +00:00
|
|
|
return jerr.Error
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return &Error{
|
|
|
|
Code: res.StatusCode,
|
|
|
|
Body: string(slurp),
|
|
|
|
Header: res.Header,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// IsNotModified reports whether err is the result of the
|
|
|
|
// server replying with http.StatusNotModified.
|
|
|
|
// Such error values are sometimes returned by "Do" methods
|
|
|
|
// on calls when If-None-Match is used.
|
|
|
|
func IsNotModified(err error) bool {
|
|
|
|
if err == nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
ae, ok := err.(*Error)
|
|
|
|
return ok && ae.Code == http.StatusNotModified
|
|
|
|
}
|
|
|
|
|
|
|
|
// CheckMediaResponse returns an error (of type *Error) if the response
|
|
|
|
// status code is not 2xx. Unlike CheckResponse it does not assume the
|
|
|
|
// body is a JSON error document.
|
|
|
|
// It is the caller's responsibility to close res.Body.
|
|
|
|
func CheckMediaResponse(res *http.Response) error {
|
|
|
|
if res.StatusCode >= 200 && res.StatusCode <= 299 {
|
|
|
|
return nil
|
|
|
|
}
|
2023-08-12 19:43:23 +00:00
|
|
|
slurp, _ := io.ReadAll(io.LimitReader(res.Body, 1<<20))
|
2019-11-07 19:05:39 +00:00
|
|
|
return &Error{
|
2022-12-19 21:02:39 +00:00
|
|
|
Code: res.StatusCode,
|
|
|
|
Body: string(slurp),
|
|
|
|
Header: res.Header,
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// MarshalStyle defines whether to marshal JSON with a {"data": ...} wrapper.
|
|
|
|
type MarshalStyle bool
|
|
|
|
|
|
|
|
// WithDataWrapper marshals JSON with a {"data": ...} wrapper.
|
|
|
|
var WithDataWrapper = MarshalStyle(true)
|
|
|
|
|
|
|
|
// WithoutDataWrapper marshals JSON without a {"data": ...} wrapper.
|
|
|
|
var WithoutDataWrapper = MarshalStyle(false)
|
|
|
|
|
|
|
|
func (wrap MarshalStyle) JSONReader(v interface{}) (io.Reader, error) {
|
|
|
|
buf := new(bytes.Buffer)
|
|
|
|
if wrap {
|
|
|
|
buf.Write([]byte(`{"data": `))
|
|
|
|
}
|
|
|
|
err := json.NewEncoder(buf).Encode(v)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
if wrap {
|
|
|
|
buf.Write([]byte(`}`))
|
|
|
|
}
|
|
|
|
return buf, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ProgressUpdater is a function that is called upon every progress update of a resumable upload.
|
|
|
|
// This is the only part of a resumable upload (from googleapi) that is usable by the developer.
|
|
|
|
// The remaining usable pieces of resumable uploads is exposed in each auto-generated API.
|
|
|
|
type ProgressUpdater func(current, total int64)
|
|
|
|
|
|
|
|
// MediaOption defines the interface for setting media options.
|
|
|
|
type MediaOption interface {
|
|
|
|
setOptions(o *MediaOptions)
|
|
|
|
}
|
|
|
|
|
|
|
|
type contentTypeOption string
|
|
|
|
|
|
|
|
func (ct contentTypeOption) setOptions(o *MediaOptions) {
|
|
|
|
o.ContentType = string(ct)
|
|
|
|
if o.ContentType == "" {
|
|
|
|
o.ForceEmptyContentType = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ContentType returns a MediaOption which sets the Content-Type header for media uploads.
|
|
|
|
// If ctype is empty, the Content-Type header will be omitted.
|
|
|
|
func ContentType(ctype string) MediaOption {
|
|
|
|
return contentTypeOption(ctype)
|
|
|
|
}
|
|
|
|
|
|
|
|
type chunkSizeOption int
|
|
|
|
|
|
|
|
func (cs chunkSizeOption) setOptions(o *MediaOptions) {
|
|
|
|
size := int(cs)
|
|
|
|
if size%MinUploadChunkSize != 0 {
|
|
|
|
size += MinUploadChunkSize - (size % MinUploadChunkSize)
|
|
|
|
}
|
|
|
|
o.ChunkSize = size
|
|
|
|
}
|
|
|
|
|
|
|
|
// ChunkSize returns a MediaOption which sets the chunk size for media uploads.
|
|
|
|
// size will be rounded up to the nearest multiple of 256K.
|
|
|
|
// Media which contains fewer than size bytes will be uploaded in a single request.
|
|
|
|
// Media which contains size bytes or more will be uploaded in separate chunks.
|
|
|
|
// If size is zero, media will be uploaded in a single request.
|
|
|
|
func ChunkSize(size int) MediaOption {
|
|
|
|
return chunkSizeOption(size)
|
|
|
|
}
|
|
|
|
|
2022-02-07 13:07:36 +00:00
|
|
|
type chunkRetryDeadlineOption time.Duration
|
|
|
|
|
|
|
|
func (cd chunkRetryDeadlineOption) setOptions(o *MediaOptions) {
|
|
|
|
o.ChunkRetryDeadline = time.Duration(cd)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ChunkRetryDeadline returns a MediaOption which sets a per-chunk retry
|
|
|
|
// deadline. If a single chunk has been attempting to upload for longer than
|
|
|
|
// this time and the request fails, it will no longer be retried, and the error
|
|
|
|
// will be returned to the caller.
|
|
|
|
// This is only applicable for files which are large enough to require
|
|
|
|
// a multi-chunk resumable upload.
|
|
|
|
// The default value is 32s.
|
|
|
|
// To set a deadline on the entire upload, use context timeout or cancellation.
|
|
|
|
func ChunkRetryDeadline(deadline time.Duration) MediaOption {
|
|
|
|
return chunkRetryDeadlineOption(deadline)
|
|
|
|
}
|
|
|
|
|
2019-11-07 19:05:39 +00:00
|
|
|
// MediaOptions stores options for customizing media upload. It is not used by developers directly.
|
|
|
|
type MediaOptions struct {
|
|
|
|
ContentType string
|
|
|
|
ForceEmptyContentType bool
|
2022-02-07 13:07:36 +00:00
|
|
|
ChunkSize int
|
|
|
|
ChunkRetryDeadline time.Duration
|
2019-11-07 19:05:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// ProcessMediaOptions stores options from opts in a MediaOptions.
|
|
|
|
// It is not used by developers directly.
|
|
|
|
func ProcessMediaOptions(opts []MediaOption) *MediaOptions {
|
|
|
|
mo := &MediaOptions{ChunkSize: DefaultUploadChunkSize}
|
|
|
|
for _, o := range opts {
|
|
|
|
o.setOptions(mo)
|
|
|
|
}
|
|
|
|
return mo
|
|
|
|
}
|
|
|
|
|
|
|
|
// ResolveRelative resolves relatives such as "http://www.golang.org/" and
|
|
|
|
// "topics/myproject/mytopic" into a single string, such as
|
|
|
|
// "http://www.golang.org/topics/myproject/mytopic". It strips all parent
|
|
|
|
// references (e.g. ../..) as well as anything after the host
|
|
|
|
// (e.g. /bar/gaz gets stripped out of foo.com/bar/gaz).
|
|
|
|
//
|
|
|
|
// ResolveRelative panics if either basestr or relstr is not able to be parsed.
|
|
|
|
func ResolveRelative(basestr, relstr string) string {
|
|
|
|
u, err := url.Parse(basestr)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("failed to parse %q", basestr))
|
|
|
|
}
|
|
|
|
afterColonPath := ""
|
|
|
|
if i := strings.IndexRune(relstr, ':'); i > 0 {
|
|
|
|
afterColonPath = relstr[i+1:]
|
|
|
|
relstr = relstr[:i]
|
|
|
|
}
|
|
|
|
rel, err := url.Parse(relstr)
|
|
|
|
if err != nil {
|
|
|
|
panic(fmt.Sprintf("failed to parse %q", relstr))
|
|
|
|
}
|
|
|
|
u = u.ResolveReference(rel)
|
|
|
|
us := u.String()
|
|
|
|
if afterColonPath != "" {
|
|
|
|
us = fmt.Sprintf("%s:%s", us, afterColonPath)
|
|
|
|
}
|
|
|
|
us = strings.Replace(us, "%7B", "{", -1)
|
|
|
|
us = strings.Replace(us, "%7D", "}", -1)
|
|
|
|
us = strings.Replace(us, "%2A", "*", -1)
|
|
|
|
return us
|
|
|
|
}
|
|
|
|
|
|
|
|
// Expand subsitutes any {encoded} strings in the URL passed in using
|
|
|
|
// the map supplied.
|
|
|
|
//
|
|
|
|
// This calls SetOpaque to avoid encoding of the parameters in the URL path.
|
|
|
|
func Expand(u *url.URL, expansions map[string]string) {
|
|
|
|
escaped, unescaped, err := uritemplates.Expand(u.Path, expansions)
|
|
|
|
if err == nil {
|
|
|
|
u.Path = unescaped
|
|
|
|
u.RawPath = escaped
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// CloseBody is used to close res.Body.
|
|
|
|
// Prior to calling Close, it also tries to Read a small amount to see an EOF.
|
|
|
|
// Not seeing an EOF can prevent HTTP Transports from reusing connections.
|
|
|
|
func CloseBody(res *http.Response) {
|
|
|
|
if res == nil || res.Body == nil {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
// Justification for 3 byte reads: two for up to "\r\n" after
|
|
|
|
// a JSON/XML document, and then 1 to see EOF if we haven't yet.
|
|
|
|
// TODO(bradfitz): detect Go 1.3+ and skip these reads.
|
|
|
|
// See https://codereview.appspot.com/58240043
|
|
|
|
// and https://codereview.appspot.com/49570044
|
|
|
|
buf := make([]byte, 1)
|
|
|
|
for i := 0; i < 3; i++ {
|
|
|
|
_, err := res.Body.Read(buf)
|
|
|
|
if err != nil {
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
res.Body.Close()
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
// VariantType returns the type name of the given variant.
|
|
|
|
// If the map doesn't contain the named key or the value is not a []interface{}, "" is returned.
|
|
|
|
// This is used to support "variant" APIs that can return one of a number of different types.
|
|
|
|
func VariantType(t map[string]interface{}) string {
|
|
|
|
s, _ := t["type"].(string)
|
|
|
|
return s
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConvertVariant uses the JSON encoder/decoder to fill in the struct 'dst' with the fields found in variant 'v'.
|
|
|
|
// This is used to support "variant" APIs that can return one of a number of different types.
|
|
|
|
// It reports whether the conversion was successful.
|
|
|
|
func ConvertVariant(v map[string]interface{}, dst interface{}) bool {
|
|
|
|
var buf bytes.Buffer
|
|
|
|
err := json.NewEncoder(&buf).Encode(v)
|
|
|
|
if err != nil {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
return json.Unmarshal(buf.Bytes(), dst) == nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// A Field names a field to be retrieved with a partial response.
|
2019-11-19 19:29:35 +00:00
|
|
|
// https://cloud.google.com/storage/docs/json_api/v1/how-tos/performance
|
2019-11-07 19:05:39 +00:00
|
|
|
//
|
|
|
|
// Partial responses can dramatically reduce the amount of data that must be sent to your application.
|
|
|
|
// In order to request partial responses, you can specify the full list of fields
|
|
|
|
// that your application needs by adding the Fields option to your request.
|
|
|
|
//
|
|
|
|
// Field strings use camelCase with leading lower-case characters to identify fields within the response.
|
|
|
|
//
|
|
|
|
// For example, if your response has a "NextPageToken" and a slice of "Items" with "Id" fields,
|
|
|
|
// you could request just those fields like this:
|
|
|
|
//
|
2022-08-14 21:53:41 +00:00
|
|
|
// svc.Events.List().Fields("nextPageToken", "items/id").Do()
|
2019-11-07 19:05:39 +00:00
|
|
|
//
|
|
|
|
// or if you were also interested in each Item's "Updated" field, you can combine them like this:
|
|
|
|
//
|
2022-08-14 21:53:41 +00:00
|
|
|
// svc.Events.List().Fields("nextPageToken", "items(id,updated)").Do()
|
2019-11-07 19:05:39 +00:00
|
|
|
//
|
|
|
|
// Another way to find field names is through the Google API explorer:
|
|
|
|
// https://developers.google.com/apis-explorer/#p/
|
|
|
|
type Field string
|
|
|
|
|
|
|
|
// CombineFields combines fields into a single string.
|
|
|
|
func CombineFields(s []Field) string {
|
|
|
|
r := make([]string, len(s))
|
|
|
|
for i, v := range s {
|
|
|
|
r[i] = string(v)
|
|
|
|
}
|
|
|
|
return strings.Join(r, ",")
|
|
|
|
}
|
|
|
|
|
|
|
|
// A CallOption is an optional argument to an API call.
|
|
|
|
// It should be treated as an opaque value by users of Google APIs.
|
|
|
|
//
|
|
|
|
// A CallOption is something that configures an API call in a way that is
|
|
|
|
// not specific to that API; for instance, controlling the quota user for
|
|
|
|
// an API call is common across many APIs, and is thus a CallOption.
|
|
|
|
type CallOption interface {
|
|
|
|
Get() (key, value string)
|
|
|
|
}
|
|
|
|
|
2021-08-13 10:03:03 +00:00
|
|
|
// A MultiCallOption is an option argument to an API call and can be passed
|
|
|
|
// anywhere a CallOption is accepted. It additionally supports returning a slice
|
|
|
|
// of values for a given key.
|
|
|
|
type MultiCallOption interface {
|
|
|
|
CallOption
|
|
|
|
GetMulti() (key string, value []string)
|
|
|
|
}
|
|
|
|
|
2019-11-07 19:05:39 +00:00
|
|
|
// QuotaUser returns a CallOption that will set the quota user for a call.
|
|
|
|
// The quota user can be used by server-side applications to control accounting.
|
|
|
|
// It can be an arbitrary string up to 40 characters, and will override UserIP
|
|
|
|
// if both are provided.
|
|
|
|
func QuotaUser(u string) CallOption { return quotaUser(u) }
|
|
|
|
|
|
|
|
type quotaUser string
|
|
|
|
|
|
|
|
func (q quotaUser) Get() (string, string) { return "quotaUser", string(q) }
|
|
|
|
|
|
|
|
// UserIP returns a CallOption that will set the "userIp" parameter of a call.
|
|
|
|
// This should be the IP address of the originating request.
|
|
|
|
func UserIP(ip string) CallOption { return userIP(ip) }
|
|
|
|
|
|
|
|
type userIP string
|
|
|
|
|
|
|
|
func (i userIP) Get() (string, string) { return "userIp", string(i) }
|
|
|
|
|
|
|
|
// Trace returns a CallOption that enables diagnostic tracing for a call.
|
|
|
|
// traceToken is an ID supplied by Google support.
|
|
|
|
func Trace(traceToken string) CallOption { return traceTok(traceToken) }
|
|
|
|
|
|
|
|
type traceTok string
|
|
|
|
|
|
|
|
func (t traceTok) Get() (string, string) { return "trace", "token:" + string(t) }
|
|
|
|
|
2021-08-13 10:03:03 +00:00
|
|
|
type queryParameter struct {
|
|
|
|
key string
|
|
|
|
values []string
|
|
|
|
}
|
|
|
|
|
|
|
|
// QueryParameter allows setting the value(s) of an arbitrary key.
|
|
|
|
func QueryParameter(key string, values ...string) CallOption {
|
|
|
|
return queryParameter{key: key, values: append([]string{}, values...)}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Get will never actually be called -- GetMulti will.
|
|
|
|
func (q queryParameter) Get() (string, string) {
|
|
|
|
return "", ""
|
|
|
|
}
|
|
|
|
|
|
|
|
// GetMulti returns the key and values values associated to that key.
|
|
|
|
func (q queryParameter) GetMulti() (string, []string) {
|
|
|
|
return q.key, q.values
|
|
|
|
}
|
|
|
|
|
2019-11-07 19:05:39 +00:00
|
|
|
// TODO: Fields too
|