2014-04-01 00:12:10 +00:00
|
|
|
package agent
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
2014-05-06 21:18:32 +00:00
|
|
|
"fmt"
|
2014-04-01 00:12:10 +00:00
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
2014-11-25 18:54:30 +00:00
|
|
|
|
pkg refactor
command/agent/* -> agent/*
command/consul/* -> agent/consul/*
command/agent/command{,_test}.go -> command/agent{,_test}.go
command/base/command.go -> command/base.go
command/base/* -> command/*
commands.go -> command/commands.go
The script which did the refactor is:
(
cd $GOPATH/src/github.com/hashicorp/consul
git mv command/agent/command.go command/agent.go
git mv command/agent/command_test.go command/agent_test.go
git mv command/agent/flag_slice_value{,_test}.go command/
git mv command/agent .
git mv command/base/command.go command/base.go
git mv command/base/config_util{,_test}.go command/
git mv commands.go command/
git mv consul agent
rmdir command/base/
gsed -i -e 's|package agent|package command|' command/agent{,_test}.go
gsed -i -e 's|package agent|package command|' command/flag_slice_value{,_test}.go
gsed -i -e 's|package base|package command|' command/base.go command/config_util{,_test}.go
gsed -i -e 's|package main|package command|' command/commands.go
gsed -i -e 's|base.Command|BaseCommand|' command/commands.go
gsed -i -e 's|agent.Command|AgentCommand|' command/commands.go
gsed -i -e 's|\tCommand:|\tBaseCommand:|' command/commands.go
gsed -i -e 's|base\.||' command/commands.go
gsed -i -e 's|command\.||' command/commands.go
gsed -i -e 's|command|c|' main.go
gsed -i -e 's|range Commands|range command.Commands|' main.go
gsed -i -e 's|Commands: Commands|Commands: command.Commands|' main.go
gsed -i -e 's|base\.BoolValue|BoolValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.DurationValue|DurationValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.StringValue|StringValue|' command/operator_autopilot_set.go
gsed -i -e 's|base\.UintValue|UintValue|' command/operator_autopilot_set.go
gsed -i -e 's|\bCommand\b|BaseCommand|' command/base.go
gsed -i -e 's|BaseCommand Options|Command Options|' command/base.go
gsed -i -e 's|base.Command|BaseCommand|' command/*.go
gsed -i -e 's|c\.Command|c.BaseCommand|g' command/*.go
gsed -i -e 's|\tCommand:|\tBaseCommand:|' command/*_test.go
gsed -i -e 's|base\.||' command/*_test.go
gsed -i -e 's|\bCommand\b|AgentCommand|' command/agent{,_test}.go
gsed -i -e 's|cmd.AgentCommand|cmd.BaseCommand|' command/agent.go
gsed -i -e 's|cli.AgentCommand = new(Command)|cli.Command = new(AgentCommand)|' command/agent_test.go
gsed -i -e 's|exec.AgentCommand|exec.Command|' command/agent_test.go
gsed -i -e 's|exec.BaseCommand|exec.Command|' command/agent_test.go
gsed -i -e 's|NewTestAgent|agent.NewTestAgent|' command/agent_test.go
gsed -i -e 's|= TestConfig|= agent.TestConfig|' command/agent_test.go
gsed -i -e 's|: RetryJoin|: agent.RetryJoin|' command/agent_test.go
gsed -i -e 's|\.\./\.\./|../|' command/config_util_test.go
gsed -i -e 's|\bverifyUniqueListeners|VerifyUniqueListeners|' agent/config{,_test}.go command/agent.go
gsed -i -e 's|\bserfLANKeyring\b|SerfLANKeyring|g' agent/{agent,keyring,testagent}.go command/agent.go
gsed -i -e 's|\bserfWANKeyring\b|SerfWANKeyring|g' agent/{agent,keyring,testagent}.go command/agent.go
gsed -i -e 's|\bNewAgent\b|agent.New|g' command/agent{,_test}.go
gsed -i -e 's|\bNewAgent|New|' agent/{acl_test,agent,testagent}.go
gsed -i -e 's|\bAgent\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bBool\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bDefaultConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bDevConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bMergeConfig\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bReadConfigPaths\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bParseMetaPair\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bSerfLANKeyring\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|\bSerfWANKeyring\b|agent.&|g' command/agent{,_test}.go
gsed -i -e 's|circonus\.agent|circonus|g' command/agent{,_test}.go
gsed -i -e 's|logger\.agent|logger|g' command/agent{,_test}.go
gsed -i -e 's|metrics\.agent|metrics|g' command/agent{,_test}.go
gsed -i -e 's|// agent.Agent|// agent|' command/agent{,_test}.go
gsed -i -e 's|a\.agent\.Config|a.Config|' command/agent{,_test}.go
gsed -i -e 's|agent\.AppendSliceValue|AppendSliceValue|' command/{configtest,validate}.go
gsed -i -e 's|consul/consul|agent/consul|' GNUmakefile
gsed -i -e 's|\.\./test|../../test|' agent/consul/server_test.go
# fix imports
f=$(grep -rl 'github.com/hashicorp/consul/command/agent' * | grep '\.go')
gsed -i -e 's|github.com/hashicorp/consul/command/agent|github.com/hashicorp/consul/agent|' $f
goimports -w $f
f=$(grep -rl 'github.com/hashicorp/consul/consul' * | grep '\.go')
gsed -i -e 's|github.com/hashicorp/consul/consul|github.com/hashicorp/consul/agent/consul|' $f
goimports -w $f
goimports -w command/*.go main.go
)
2017-06-09 22:28:28 +00:00
|
|
|
"github.com/hashicorp/consul/agent/consul/structs"
|
2017-04-19 23:00:11 +00:00
|
|
|
"github.com/hashicorp/consul/api"
|
2014-04-01 00:12:10 +00:00
|
|
|
)
|
|
|
|
|
2014-05-06 21:18:32 +00:00
|
|
|
const (
|
|
|
|
// maxKVSize is used to limit the maximum payload length
|
|
|
|
// of a KV entry. If it exceeds this amount, the client is
|
|
|
|
// likely abusing the KV store.
|
|
|
|
maxKVSize = 512 * 1024
|
|
|
|
)
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
func (s *HTTPServer) KVSEndpoint(resp http.ResponseWriter, req *http.Request) (interface{}, error) {
|
|
|
|
// Set default DC
|
|
|
|
args := structs.KeyRequest{}
|
2014-04-21 19:25:36 +00:00
|
|
|
if done := s.parse(resp, req, &args.Datacenter, &args.QueryOptions); done {
|
2014-04-01 00:12:10 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Pull out the key name, validation left to each sub-handler
|
|
|
|
args.Key = strings.TrimPrefix(req.URL.Path, "/v1/kv/")
|
|
|
|
|
2014-04-28 23:55:11 +00:00
|
|
|
// Check for a key list
|
|
|
|
keyList := false
|
|
|
|
params := req.URL.Query()
|
|
|
|
if _, ok := params["keys"]; ok {
|
|
|
|
keyList = true
|
|
|
|
}
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
// Switch on the method
|
|
|
|
switch req.Method {
|
|
|
|
case "GET":
|
2014-04-28 23:55:11 +00:00
|
|
|
if keyList {
|
|
|
|
return s.KVSGetKeys(resp, req, &args)
|
|
|
|
}
|
2017-04-21 01:59:42 +00:00
|
|
|
return s.KVSGet(resp, req, &args)
|
2014-04-01 00:12:10 +00:00
|
|
|
case "PUT":
|
|
|
|
return s.KVSPut(resp, req, &args)
|
|
|
|
case "DELETE":
|
|
|
|
return s.KVSDelete(resp, req, &args)
|
|
|
|
default:
|
|
|
|
resp.WriteHeader(405)
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// KVSGet handles a GET request
|
|
|
|
func (s *HTTPServer) KVSGet(resp http.ResponseWriter, req *http.Request, args *structs.KeyRequest) (interface{}, error) {
|
|
|
|
// Check for recurse
|
|
|
|
method := "KVS.Get"
|
|
|
|
params := req.URL.Query()
|
|
|
|
if _, ok := params["recurse"]; ok {
|
|
|
|
method = "KVS.List"
|
|
|
|
} else if missingKey(resp, args) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make the RPC
|
|
|
|
var out structs.IndexedDirEntries
|
|
|
|
if err := s.agent.RPC(method, &args, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2014-04-26 18:20:49 +00:00
|
|
|
setMeta(resp, &out.QueryMeta)
|
2014-04-01 03:00:17 +00:00
|
|
|
|
|
|
|
// Check if we get a not found
|
|
|
|
if len(out.Entries) == 0 {
|
|
|
|
resp.WriteHeader(404)
|
|
|
|
return nil, nil
|
|
|
|
}
|
2014-05-20 23:53:43 +00:00
|
|
|
|
|
|
|
// Check if we are in raw mode with a normal get, write out
|
|
|
|
// the raw body
|
|
|
|
if _, ok := params["raw"]; ok && method == "KVS.Get" {
|
|
|
|
body := out.Entries[0].Value
|
|
|
|
resp.Header().Set("Content-Length", strconv.FormatInt(int64(len(body)), 10))
|
|
|
|
resp.Write(body)
|
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
return out.Entries, nil
|
|
|
|
}
|
|
|
|
|
2014-04-28 23:55:11 +00:00
|
|
|
// KVSGetKeys handles a GET request for keys
|
|
|
|
func (s *HTTPServer) KVSGetKeys(resp http.ResponseWriter, req *http.Request, args *structs.KeyRequest) (interface{}, error) {
|
2015-09-11 19:24:54 +00:00
|
|
|
// Check for a separator, due to historic spelling error,
|
2014-05-07 22:25:17 +00:00
|
|
|
// we now are forced to check for both spellings
|
2014-04-28 23:55:11 +00:00
|
|
|
var sep string
|
|
|
|
params := req.URL.Query()
|
|
|
|
if _, ok := params["seperator"]; ok {
|
|
|
|
sep = params.Get("seperator")
|
|
|
|
}
|
2014-05-07 22:25:17 +00:00
|
|
|
if _, ok := params["separator"]; ok {
|
|
|
|
sep = params.Get("separator")
|
|
|
|
}
|
2014-04-28 23:55:11 +00:00
|
|
|
|
|
|
|
// Construct the args
|
|
|
|
listArgs := structs.KeyListRequest{
|
|
|
|
Datacenter: args.Datacenter,
|
|
|
|
Prefix: args.Key,
|
|
|
|
Seperator: sep,
|
|
|
|
QueryOptions: args.QueryOptions,
|
|
|
|
}
|
|
|
|
|
|
|
|
// Make the RPC
|
|
|
|
var out structs.IndexedKeyList
|
|
|
|
if err := s.agent.RPC("KVS.ListKeys", &listArgs, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
setMeta(resp, &out.QueryMeta)
|
|
|
|
|
2014-04-30 21:21:02 +00:00
|
|
|
// Check if we get a not found. We do not generate
|
|
|
|
// not found for the root, but just provide the empty list
|
|
|
|
if len(out.Keys) == 0 && listArgs.Prefix != "" {
|
2014-04-28 23:55:11 +00:00
|
|
|
resp.WriteHeader(404)
|
|
|
|
return nil, nil
|
|
|
|
}
|
2014-04-30 21:21:02 +00:00
|
|
|
|
|
|
|
// Use empty list instead of null
|
|
|
|
if out.Keys == nil {
|
|
|
|
out.Keys = []string{}
|
|
|
|
}
|
2014-04-28 23:55:11 +00:00
|
|
|
return out.Keys, nil
|
|
|
|
}
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
// KVSPut handles a PUT request
|
|
|
|
func (s *HTTPServer) KVSPut(resp http.ResponseWriter, req *http.Request, args *structs.KeyRequest) (interface{}, error) {
|
|
|
|
if missingKey(resp, args) {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2015-01-23 20:48:39 +00:00
|
|
|
if conflictingFlags(resp, req, "cas", "acquire", "release") {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2014-04-01 00:12:10 +00:00
|
|
|
applyReq := structs.KVSRequest{
|
|
|
|
Datacenter: args.Datacenter,
|
2017-04-19 23:00:11 +00:00
|
|
|
Op: api.KVSet,
|
2014-04-01 00:12:10 +00:00
|
|
|
DirEnt: structs.DirEntry{
|
|
|
|
Key: args.Key,
|
|
|
|
Flags: 0,
|
|
|
|
Value: nil,
|
|
|
|
},
|
|
|
|
}
|
2014-08-15 02:34:50 +00:00
|
|
|
applyReq.Token = args.Token
|
2014-04-01 00:12:10 +00:00
|
|
|
|
|
|
|
// Check for flags
|
|
|
|
params := req.URL.Query()
|
|
|
|
if _, ok := params["flags"]; ok {
|
|
|
|
flagVal, err := strconv.ParseUint(params.Get("flags"), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
applyReq.DirEnt.Flags = flagVal
|
|
|
|
}
|
|
|
|
|
|
|
|
// Check for cas value
|
|
|
|
if _, ok := params["cas"]; ok {
|
2014-04-01 03:00:30 +00:00
|
|
|
casVal, err := strconv.ParseUint(params.Get("cas"), 10, 64)
|
2014-04-01 00:12:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
applyReq.DirEnt.ModifyIndex = casVal
|
2017-04-19 23:00:11 +00:00
|
|
|
applyReq.Op = api.KVCAS
|
2014-04-01 00:12:10 +00:00
|
|
|
}
|
|
|
|
|
2014-05-19 23:14:03 +00:00
|
|
|
// Check for lock acquisition
|
|
|
|
if _, ok := params["acquire"]; ok {
|
|
|
|
applyReq.DirEnt.Session = params.Get("acquire")
|
2017-04-19 23:00:11 +00:00
|
|
|
applyReq.Op = api.KVLock
|
2014-05-19 23:14:03 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// Check for lock release
|
|
|
|
if _, ok := params["release"]; ok {
|
|
|
|
applyReq.DirEnt.Session = params.Get("release")
|
2017-04-19 23:00:11 +00:00
|
|
|
applyReq.Op = api.KVUnlock
|
2014-05-19 23:14:03 +00:00
|
|
|
}
|
|
|
|
|
2014-05-06 21:18:32 +00:00
|
|
|
// Check the content-length
|
|
|
|
if req.ContentLength > maxKVSize {
|
|
|
|
resp.WriteHeader(413)
|
Use fmt.Fprint/Fprintf/Fprintln
Used the following rewrite rules:
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c, d))) -> fmt.Fprintf(resp, a, b, c, d)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c))) -> fmt.Fprintf(resp, a, b, c)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b))) -> fmt.Fprintf(resp, a, b)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a))) -> fmt.Fprint(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a + "\n")) -> fmt.Fprintln(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a)) -> fmt.Fprint(resp, a)' *.go
2017-04-20 14:07:42 +00:00
|
|
|
fmt.Fprintf(resp, "Value exceeds %d byte limit", maxKVSize)
|
2014-05-06 21:18:32 +00:00
|
|
|
return nil, nil
|
|
|
|
}
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
// Copy the value
|
|
|
|
buf := bytes.NewBuffer(nil)
|
|
|
|
if _, err := io.Copy(buf, req.Body); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
applyReq.DirEnt.Value = buf.Bytes()
|
|
|
|
|
|
|
|
// Make the RPC
|
|
|
|
var out bool
|
|
|
|
if err := s.agent.RPC("KVS.Apply", &applyReq, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// Only use the out value if this was a CAS
|
2017-04-19 23:00:11 +00:00
|
|
|
if applyReq.Op == api.KVSet {
|
2014-04-01 00:12:10 +00:00
|
|
|
return true, nil
|
|
|
|
}
|
2017-04-21 01:59:42 +00:00
|
|
|
return out, nil
|
2014-04-01 00:12:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// KVSPut handles a DELETE request
|
|
|
|
func (s *HTTPServer) KVSDelete(resp http.ResponseWriter, req *http.Request, args *structs.KeyRequest) (interface{}, error) {
|
2015-01-23 20:48:39 +00:00
|
|
|
if conflictingFlags(resp, req, "recurse", "cas") {
|
|
|
|
return nil, nil
|
|
|
|
}
|
2014-04-01 00:12:10 +00:00
|
|
|
applyReq := structs.KVSRequest{
|
|
|
|
Datacenter: args.Datacenter,
|
2017-04-19 23:00:11 +00:00
|
|
|
Op: api.KVDelete,
|
2014-04-01 00:12:10 +00:00
|
|
|
DirEnt: structs.DirEntry{
|
|
|
|
Key: args.Key,
|
|
|
|
},
|
|
|
|
}
|
2014-08-15 02:34:50 +00:00
|
|
|
applyReq.Token = args.Token
|
2014-04-01 00:12:10 +00:00
|
|
|
|
|
|
|
// Check for recurse
|
|
|
|
params := req.URL.Query()
|
|
|
|
if _, ok := params["recurse"]; ok {
|
2017-04-19 23:00:11 +00:00
|
|
|
applyReq.Op = api.KVDeleteTree
|
2014-04-01 03:00:01 +00:00
|
|
|
} else if missingKey(resp, args) {
|
|
|
|
return nil, nil
|
2014-04-01 00:12:10 +00:00
|
|
|
}
|
|
|
|
|
2015-01-09 01:08:58 +00:00
|
|
|
// Check for cas value
|
|
|
|
if _, ok := params["cas"]; ok {
|
|
|
|
casVal, err := strconv.ParseUint(params.Get("cas"), 10, 64)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
applyReq.DirEnt.ModifyIndex = casVal
|
2017-04-19 23:00:11 +00:00
|
|
|
applyReq.Op = api.KVDeleteCAS
|
2015-01-09 01:08:58 +00:00
|
|
|
}
|
|
|
|
|
2014-04-01 00:12:10 +00:00
|
|
|
// Make the RPC
|
|
|
|
var out bool
|
|
|
|
if err := s.agent.RPC("KVS.Apply", &applyReq, &out); err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
2015-01-09 01:08:58 +00:00
|
|
|
|
|
|
|
// Only use the out value if this was a CAS
|
2017-04-19 23:00:11 +00:00
|
|
|
if applyReq.Op == api.KVDeleteCAS {
|
2015-01-09 01:08:58 +00:00
|
|
|
return out, nil
|
|
|
|
}
|
2017-04-21 01:59:42 +00:00
|
|
|
return true, nil
|
2014-04-01 00:12:10 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
// missingKey checks if the key is missing
|
|
|
|
func missingKey(resp http.ResponseWriter, args *structs.KeyRequest) bool {
|
|
|
|
if args.Key == "" {
|
|
|
|
resp.WriteHeader(400)
|
Use fmt.Fprint/Fprintf/Fprintln
Used the following rewrite rules:
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c, d))) -> fmt.Fprintf(resp, a, b, c, d)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c))) -> fmt.Fprintf(resp, a, b, c)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b))) -> fmt.Fprintf(resp, a, b)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a))) -> fmt.Fprint(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a + "\n")) -> fmt.Fprintln(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a)) -> fmt.Fprint(resp, a)' *.go
2017-04-20 14:07:42 +00:00
|
|
|
fmt.Fprint(resp, "Missing key name")
|
2014-04-01 00:12:10 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2015-01-23 20:48:39 +00:00
|
|
|
|
|
|
|
// conflictingFlags determines if non-composable flags were passed in a request.
|
|
|
|
func conflictingFlags(resp http.ResponseWriter, req *http.Request, flags ...string) bool {
|
|
|
|
params := req.URL.Query()
|
|
|
|
|
|
|
|
found := false
|
|
|
|
for _, conflict := range flags {
|
|
|
|
if _, ok := params[conflict]; ok {
|
|
|
|
if found {
|
|
|
|
resp.WriteHeader(400)
|
Use fmt.Fprint/Fprintf/Fprintln
Used the following rewrite rules:
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c, d))) -> fmt.Fprintf(resp, a, b, c, d)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b, c))) -> fmt.Fprintf(resp, a, b, c)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a, b))) -> fmt.Fprintf(resp, a, b)' *.go
gofmt -w -r 'resp.Write([]byte(fmt.Sprintf(a))) -> fmt.Fprint(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a + "\n")) -> fmt.Fprintln(resp, a)' *.go
gofmt -w -r 'resp.Write([]byte(a)) -> fmt.Fprint(resp, a)' *.go
2017-04-20 14:07:42 +00:00
|
|
|
fmt.Fprint(resp, "Conflicting flags: "+params.Encode())
|
2015-01-23 20:48:39 +00:00
|
|
|
return true
|
|
|
|
}
|
|
|
|
found = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return false
|
|
|
|
}
|