feat: Inital commit
This commit is contained in:
+201
@@ -0,0 +1,201 @@
|
||||
/*
|
||||
* Copyright 2020 National Library of Norway.
|
||||
*
|
||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||
* you may not use this file except in compliance with the License.
|
||||
* You may obtain a copy of the License at
|
||||
*
|
||||
* http://www.apache.org/licenses/LICENSE-2.0
|
||||
*
|
||||
* Unless required by applicable law or agreed to in writing, software
|
||||
* distributed under the License is distributed on an "AS IS" BASIS,
|
||||
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||
* See the License for the specific language governing permissions and
|
||||
* limitations under the License.
|
||||
*/
|
||||
|
||||
package url
|
||||
|
||||
import (
|
||||
"sort"
|
||||
"strings"
|
||||
)
|
||||
|
||||
type NameValuePair struct {
|
||||
Name, Value string
|
||||
}
|
||||
|
||||
// SearchParams represents a set of query parameters.
|
||||
type SearchParams struct {
|
||||
url *Url
|
||||
params []*NameValuePair
|
||||
}
|
||||
|
||||
func (s *SearchParams) init(query string) {
|
||||
s.params = s.params[:0]
|
||||
p := strings.Split(query, "&")
|
||||
for _, q := range p {
|
||||
if q == "" {
|
||||
continue
|
||||
}
|
||||
kv := strings.SplitN(q, "=", 2)
|
||||
name := s.url.parser.DecodePercentEncoded(kv[0])
|
||||
name = strings.ReplaceAll(name, "+", " ")
|
||||
nvp := &NameValuePair{Name: name}
|
||||
if len(kv) == 2 {
|
||||
value := s.url.parser.DecodePercentEncoded(kv[1])
|
||||
value = strings.ReplaceAll(value, "+", " ")
|
||||
nvp.Value = value
|
||||
}
|
||||
s.params = append(s.params, nvp)
|
||||
}
|
||||
}
|
||||
|
||||
func (s *SearchParams) update() {
|
||||
if s.url == nil {
|
||||
return
|
||||
}
|
||||
query := s.String()
|
||||
if (query == "" && s.url.query != nil) || query != "" {
|
||||
s.url.query = &query
|
||||
}
|
||||
}
|
||||
|
||||
// Append appends a new name/value pair to the search parameters.
|
||||
func (s *SearchParams) Append(name, value string) {
|
||||
s.params = append(s.params, &NameValuePair{Name: name, Value: value})
|
||||
s.update()
|
||||
}
|
||||
|
||||
// Delete deletes the given search parameter, and its associated value(s), from the search parameters.
|
||||
func (s *SearchParams) Delete(name string) {
|
||||
var result []*NameValuePair
|
||||
for _, nvp := range s.params {
|
||||
if nvp.Name != name {
|
||||
result = append(result, nvp)
|
||||
}
|
||||
}
|
||||
s.params = result
|
||||
s.update()
|
||||
}
|
||||
|
||||
// Get returns the first value associated with the given search parameter name.
|
||||
func (s *SearchParams) Get(name string) string {
|
||||
for _, nvp := range s.params {
|
||||
if nvp.Name == name {
|
||||
return nvp.Value
|
||||
}
|
||||
}
|
||||
return ""
|
||||
}
|
||||
|
||||
// GetAll returns all the values associated with the given search parameter name.
|
||||
func (s *SearchParams) GetAll(name string) []string {
|
||||
var result []string
|
||||
for _, nvp := range s.params {
|
||||
if nvp.Name == name {
|
||||
result = append(result, nvp.Value)
|
||||
}
|
||||
}
|
||||
return result
|
||||
}
|
||||
|
||||
// Has returns true if the search parameters contains a parameter with the given name.
|
||||
func (s *SearchParams) Has(name string) bool {
|
||||
for _, nvp := range s.params {
|
||||
if nvp.Name == name {
|
||||
return true
|
||||
}
|
||||
}
|
||||
return false
|
||||
}
|
||||
|
||||
// Set sets the value associated with name to value. It replaces any existing values associated with name.
|
||||
func (s *SearchParams) Set(name, value string) {
|
||||
isSet := false
|
||||
params := s.params[:0]
|
||||
for i, nvp := range s.params {
|
||||
if nvp.Name == name {
|
||||
if isSet {
|
||||
s.params[i] = nil
|
||||
continue
|
||||
}
|
||||
nvp.Value = value
|
||||
isSet = true
|
||||
}
|
||||
params = append(params, nvp)
|
||||
}
|
||||
if !isSet {
|
||||
s.params = append(params, &NameValuePair{Name: name, Value: value})
|
||||
} else {
|
||||
s.params = params
|
||||
}
|
||||
s.update()
|
||||
}
|
||||
|
||||
// Sort sorts the search parameters by name.
|
||||
func (s *SearchParams) Sort() {
|
||||
sort.SliceStable(s.params, func(i, j int) bool {
|
||||
return s.params[i].Name < s.params[j].Name
|
||||
})
|
||||
s.update()
|
||||
}
|
||||
|
||||
// SortAbsolute sorts the search parameters by name and value.
|
||||
func (s *SearchParams) SortAbsolute() {
|
||||
sort.SliceStable(s.params, func(i, j int) bool {
|
||||
return s.params[i].Name+s.params[i].Value < s.params[j].Name+s.params[j].Value
|
||||
})
|
||||
s.update()
|
||||
}
|
||||
|
||||
// Iterate iterates over the search parameters.
|
||||
func (s *SearchParams) Iterate(f func(pair *NameValuePair)) {
|
||||
for _, nvp := range s.params {
|
||||
f(nvp)
|
||||
}
|
||||
s.update()
|
||||
}
|
||||
|
||||
func (s *SearchParams) String() string {
|
||||
output := strings.Builder{}
|
||||
for idx, nvp := range s.params {
|
||||
if idx > 0 {
|
||||
output.WriteRune('&')
|
||||
}
|
||||
|
||||
s.QueryEscape(nvp.Name, &output)
|
||||
if !s.url.parser.opts.skipEqualsForEmptySearchParamsValue || nvp.Value != "" {
|
||||
output.WriteRune('=')
|
||||
}
|
||||
if nvp.Value != "" {
|
||||
s.QueryEscape(nvp.Value, &output)
|
||||
}
|
||||
}
|
||||
return output.String()
|
||||
}
|
||||
|
||||
func (s *SearchParams) QueryEscape(st string, output *strings.Builder) {
|
||||
for _, b := range st {
|
||||
if b == 0x0020 {
|
||||
output.WriteRune(0x002B)
|
||||
} else {
|
||||
output.WriteString(s.url.parser.percentEncodeRune(b, s.url.parser.opts.queryPercentEncodeSet))
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
// Clone returns a deep copy of the search parameters.
|
||||
func (s *SearchParams) Clone() *SearchParams {
|
||||
sp := &SearchParams{
|
||||
url: s.url,
|
||||
params: make([]*NameValuePair, len(s.params)),
|
||||
}
|
||||
for i, nvp := range s.params {
|
||||
sp.params[i] = &NameValuePair{
|
||||
Name: nvp.Name,
|
||||
Value: nvp.Value,
|
||||
}
|
||||
}
|
||||
return sp
|
||||
}
|
||||
Reference in New Issue
Block a user