| 1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117 |
- package etcd
- // Ensures that a value can be retrieve for a given key.
- import (
- "fmt"
- "net/http"
- "net/url"
- "testing"
- "time"
- "github.com/coreos/etcd/third_party/github.com/stretchr/testify/assert"
- )
- // Ensures that a directory is created
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar?dir=true
- //
- func TestV2SetDirectory(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"set","node":{"key":"/foo","dir":true,"modifiedIndex":2,"createdIndex":2}}`, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a time-to-live is added to a key.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d ttl=20
- //
- func TestV2SetKeyWithTTL(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- t0 := time.Now()
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("ttl", "20")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBodyJSON(resp)
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["ttl"], 20, "")
- // Make sure the expiration date is correct.
- expiration, _ := time.Parse(time.RFC3339Nano, node["expiration"].(string))
- assert.Equal(t, expiration.Sub(t0)/time.Second, 20, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an invalid time-to-live is returned as an error.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d ttl=bad_ttl
- //
- func TestV2SetKeyWithBadTTL(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("ttl", "bad_ttl")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusBadRequest)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 202, "")
- assert.Equal(t, body["message"], "The given TTL in POST form is not a number", "")
- assert.Equal(t, body["cause"], "Update", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is conditionally set if it previously did not exist.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d prevExist=false
- //
- func TestV2CreateKeySuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("prevExist", "false")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBodyJSON(resp)
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["value"], "XXX", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not conditionally set because it previously existed.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d prevExist=false
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d prevExist=false -> fail
- //
- func TestV2CreateKeyFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("prevExist", "false")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 105, "")
- assert.Equal(t, body["message"], "Key already exists", "")
- assert.Equal(t, body["cause"], "/foo/bar", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is conditionally set only if it previously did exist.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevExist=true
- //
- func TestV2UpdateKeySuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevExist", "true")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "update", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not conditionally set if it previously did not exist.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo?dir=true
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d prevExist=true
- //
- func TestV2UpdateKeyFailOnValue(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), v)
- resp.Body.Close()
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- v.Set("value", "YYY")
- v.Set("prevExist", "true")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusNotFound)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 100, "")
- assert.Equal(t, body["message"], "Key not found", "")
- assert.Equal(t, body["cause"], "/foo/bar", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not conditionally set if it previously did not exist.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=YYY -d prevExist=true -> fail
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevExist=true -> fail
- //
- func TestV2UpdateKeyFailOnMissingDirectory(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "YYY")
- v.Set("prevExist", "true")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- assert.Equal(t, resp.StatusCode, http.StatusNotFound)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 100, "")
- assert.Equal(t, body["message"], "Key not found", "")
- assert.Equal(t, body["cause"], "/foo", "")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusNotFound)
- body = ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 100, "")
- assert.Equal(t, body["message"], "Key not found", "")
- assert.Equal(t, body["cause"], "/foo", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key could update TTL.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=XXX -d ttl=1000 -d prevExist=true
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=XXX -d ttl= -d prevExist=true
- //
- func TestV2UpdateKeySuccessWithTTL(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- node := (ReadBodyJSON(resp)["node"]).(map[string]interface{})
- createdIndex := node["createdIndex"]
- v.Set("ttl", "1000")
- v.Set("prevExist", "true")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- node = (ReadBodyJSON(resp)["node"]).(map[string]interface{})
- assert.Equal(t, node["value"], "XXX", "")
- assert.Equal(t, node["ttl"], 1000, "")
- assert.NotEqual(t, node["expiration"], "", "")
- assert.Equal(t, node["createdIndex"], createdIndex, "")
- v.Del("ttl")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- node = (ReadBodyJSON(resp)["node"]).(map[string]interface{})
- assert.Equal(t, node["value"], "XXX", "")
- assert.Equal(t, node["ttl"], nil, "")
- assert.Equal(t, node["expiration"], nil, "")
- assert.Equal(t, node["createdIndex"], createdIndex, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is set only if the previous index matches.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevIndex=1
- //
- func TestV2SetKeyCASOnIndexSuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevIndex", "2")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "compareAndSwap", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["value"], "YYY", "")
- assert.Equal(t, node["modifiedIndex"], 3, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not set if the previous index does not match.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevIndex=10
- //
- func TestV2SetKeyCASOnIndexFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevIndex", "10")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101, "")
- assert.Equal(t, body["message"], "Compare failed", "")
- assert.Equal(t, body["cause"], "[10 != 2]", "")
- assert.Equal(t, body["index"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an error is thrown if an invalid previous index is provided.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevIndex=bad_index
- //
- func TestV2SetKeyCASWithInvalidIndex(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "YYY")
- v.Set("prevIndex", "bad_index")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusBadRequest)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 203, "")
- assert.Equal(t, body["message"], "The given index in POST form is not a number", "")
- assert.Equal(t, body["cause"], "CompareAndSwap", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is set only if the previous value matches.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevValue=XXX
- //
- func TestV2SetKeyCASOnValueSuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevValue", "XXX")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "compareAndSwap", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["value"], "YYY", "")
- assert.Equal(t, node["modifiedIndex"], 3, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not set if the previous value does not match.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevValue=AAA
- //
- func TestV2SetKeyCASOnValueFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevValue", "AAA")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101, "")
- assert.Equal(t, body["message"], "Compare failed", "")
- assert.Equal(t, body["cause"], "[AAA != XXX]", "")
- assert.Equal(t, body["index"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an error is returned if a blank prevValue is set.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX -d prevValue=
- //
- func TestV2SetKeyCASWithMissingValueFails(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("prevValue", "")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusBadRequest)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 201, "")
- assert.Equal(t, body["message"], "PrevValue is Required in POST form", "")
- assert.Equal(t, body["cause"], "CompareAndSwap", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not set if both previous value and index do not match.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevValue=AAA -d prevIndex=4
- //
- func TestV2SetKeyCASOnValueAndIndexFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevValue", "AAA")
- v.Set("prevIndex", "4")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101, "")
- assert.Equal(t, body["message"], "Compare failed", "")
- assert.Equal(t, body["cause"], "[AAA != XXX] [4 != 2]", "")
- assert.Equal(t, body["index"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not set if previous value match but index does not.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevValue=XXX -d prevIndex=4
- //
- func TestV2SetKeyCASOnValueMatchAndIndexFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevValue", "XXX")
- v.Set("prevIndex", "4")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101, "")
- assert.Equal(t, body["message"], "Compare failed", "")
- assert.Equal(t, body["cause"], "[4 != 2]", "")
- assert.Equal(t, body["index"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not set if previous index matches but value does not.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY -d prevValue=AAA -d prevIndex=3
- //
- func TestV2SetKeyCASOnIndexMatchAndValueFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- ReadBody(resp)
- v.Set("value", "YYY")
- v.Set("prevValue", "AAA")
- v.Set("prevIndex", "2")
- resp, _ = PutForm(fullURL, v)
- assert.Equal(t, resp.StatusCode, http.StatusPreconditionFailed)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101, "")
- assert.Equal(t, body["message"], "Compare failed", "")
- assert.Equal(t, body["cause"], "[AAA != XXX]", "")
- assert.Equal(t, body["index"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensure that we can set an empty value
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=
- //
- func TestV2SetKeyCASWithEmptyValueSuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBody(resp)
- assert.Equal(t, string(body), `{"action":"set","node":{"key":"/foo/bar","value":"","modifiedIndex":2,"createdIndex":2}}`)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- func TestV2SetKey(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"set","node":{"key":"/foo/bar","value":"XXX","modifiedIndex":2,"createdIndex":2}}`, "")
- resp.Body.Close()
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- func TestV2SetKeyRedirect(t *testing.T) {
- es, hs := buildCluster(3)
- waitCluster(t, es)
- u := hs[1].URL
- ru := fmt.Sprintf("%s%s", hs[0].URL, "/v2/keys/foo/bar")
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- assert.Equal(t, resp.StatusCode, http.StatusTemporaryRedirect)
- location, err := resp.Location()
- if err != nil {
- t.Errorf("want err = %, want nil", err)
- }
- if location.String() != ru {
- t.Errorf("location = %v, want %v", location.String(), ru)
- }
- resp.Body.Close()
- for i := range es {
- es[len(es)-i-1].Stop()
- }
- for i := range hs {
- hs[len(hs)-i-1].Close()
- }
- afterTest(t)
- }
- // Ensures that a key is deleted.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo/bar
- //
- func TestV2DeleteKey(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- resp.Body.Close()
- ReadBody(resp)
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"delete","node":{"key":"/foo/bar","modifiedIndex":3,"createdIndex":2},"prevNode":{"key":"/foo/bar","value":"XXX","modifiedIndex":2,"createdIndex":2}}`, "")
- resp.Body.Close()
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an empty directory is deleted when dir is set.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo?dir=true
- // $ curl -X DELETE localhost:4001/v2/keys/foo ->fail
- // $ curl -X DELETE localhost:4001/v2/keys/foo?dir=true
- //
- func TestV2DeleteEmptyDirectory(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), url.Values{})
- resp.Body.Close()
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusForbidden)
- bodyJson := ReadBodyJSON(resp)
- assert.Equal(t, bodyJson["errorCode"], 102, "")
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"delete","node":{"key":"/foo","dir":true,"modifiedIndex":3,"createdIndex":2},"prevNode":{"key":"/foo","dir":true,"modifiedIndex":2,"createdIndex":2}}`, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a not-empty directory is deleted when dir is set.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar?dir=true
- // $ curl -X DELETE localhost:4001/v2/keys/foo?dir=true ->fail
- // $ curl -X DELETE localhost:4001/v2/keys/foo?dir=true&recursive=true
- //
- func TestV2DeleteNonEmptyDirectory(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?dir=true"), url.Values{})
- ReadBody(resp)
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusForbidden)
- bodyJson := ReadBodyJSON(resp)
- assert.Equal(t, bodyJson["errorCode"], 108, "")
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true&recursive=true"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"delete","node":{"key":"/foo","dir":true,"modifiedIndex":3,"createdIndex":2},"prevNode":{"key":"/foo","dir":true,"modifiedIndex":2,"createdIndex":2}}`, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a directory is deleted when recursive is set.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo?dir=true
- // $ curl -X DELETE localhost:4001/v2/keys/foo?recursive=true
- //
- func TestV2DeleteDirectoryRecursiveImpliesDir(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?dir=true"), url.Values{})
- ReadBody(resp)
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?recursive=true"), url.Values{})
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBody(resp)
- assert.Nil(t, err, "")
- assert.Equal(t, string(body), `{"action":"delete","node":{"key":"/foo","dir":true,"modifiedIndex":3,"createdIndex":2},"prevNode":{"key":"/foo","dir":true,"modifiedIndex":2,"createdIndex":2}}`, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is deleted if the previous index matches
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo?prevIndex=3
- //
- func TestV2DeleteKeyCADOnIndexSuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- ReadBody(resp)
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?prevIndex=2"), url.Values{})
- assert.Nil(t, err, "")
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "compareAndDelete", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo", "")
- assert.Equal(t, node["modifiedIndex"], 3, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not deleted if the previous index does not match
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo?prevIndex=100
- //
- func TestV2DeleteKeyCADOnIndexFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, err := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo"), v)
- ReadBody(resp)
- resp, err = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo?prevIndex=100"), url.Values{})
- assert.Nil(t, err, "")
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an error is thrown if an invalid previous index is provided.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo/bar?prevIndex=bad_index
- //
- func TestV2DeleteKeyCADWithInvalidIndex(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- resp, _ = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?prevIndex=bad_index"), v)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 203)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is deleted only if the previous value matches.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo/bar?prevValue=XXX
- //
- func TestV2DeleteKeyCADOnValueSuccess(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- resp, _ = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?prevValue=XXX"), v)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "compareAndDelete", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["modifiedIndex"], 3, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a key is not deleted if the previous value does not match.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo/bar?prevValue=YYY
- //
- func TestV2DeleteKeyCADOnValueFail(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- resp, _ = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?prevValue=YYY"), v)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 101)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that an error is thrown if an invalid previous value is provided.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X DELETE localhost:4001/v2/keys/foo/bar?prevIndex=
- //
- func TestV2DeleteKeyCADWithInvalidValue(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- resp, _ = DeleteForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?prevValue="), v)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["errorCode"], 201)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures a unique value is added to the key's children.
- //
- // $ curl -X POST localhost:4001/v2/keys/foo/bar
- // $ curl -X POST localhost:4001/v2/keys/foo/bar
- // $ curl -X POST localhost:4001/v2/keys/foo/baz
- //
- func TestV2CreateUnique(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- // POST should add index to list.
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := PostForm(fullURL, nil)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "create", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/bar/2", "")
- assert.Nil(t, node["dir"], "")
- assert.Equal(t, node["modifiedIndex"], 2, "")
- // Second POST should add next index to list.
- resp, _ = PostForm(fullURL, nil)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body = ReadBodyJSON(resp)
- node = body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/bar/3", "")
- // POST to a different key should add index to that list.
- resp, _ = PostForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/baz"), nil)
- assert.Equal(t, resp.StatusCode, http.StatusCreated)
- body = ReadBodyJSON(resp)
- node = body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/baz/4", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- //
- // $ curl localhost:4001/v2/keys/foo/bar -> fail
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl localhost:4001/v2/keys/foo/bar
- //
- func TestV2GetKey(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := Get(fullURL)
- resp.Body.Close()
- resp, _ = PutForm(fullURL, v)
- resp.Body.Close()
- resp, _ = Get(fullURL)
- assert.Equal(t, resp.Header.Get("Content-Type"), "application/json")
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBodyJSON(resp)
- resp.Body.Close()
- assert.Equal(t, body["action"], "get", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/bar", "")
- assert.Equal(t, node["value"], "XXX", "")
- assert.Equal(t, node["modifiedIndex"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a directory of values can be recursively retrieved for a given key.
- //
- // $ curl -X PUT localhost:4001/v2/keys/foo/x -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/y/z -d value=YYY
- // $ curl localhost:4001/v2/keys/foo -d recursive=true
- //
- func TestV2GetKeyRecursively(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- v.Set("ttl", "10")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/x"), v)
- ReadBody(resp)
- v.Set("value", "YYY")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/y/z"), v)
- ReadBody(resp)
- resp, _ = Get(fmt.Sprintf("%s%s", u, "/v2/keys/foo?recursive=true"))
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- body := ReadBodyJSON(resp)
- assert.Equal(t, body["action"], "get", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo", "")
- assert.Equal(t, node["dir"], true, "")
- assert.Equal(t, node["modifiedIndex"], 2, "")
- assert.Equal(t, len(node["nodes"].([]interface{})), 2, "")
- node0 := node["nodes"].([]interface{})[0].(map[string]interface{})
- assert.Equal(t, node0["key"], "/foo/x", "")
- assert.Equal(t, node0["value"], "XXX", "")
- assert.Equal(t, node0["ttl"], 10, "")
- node1 := node["nodes"].([]interface{})[1].(map[string]interface{})
- assert.Equal(t, node1["key"], "/foo/y", "")
- assert.Equal(t, node1["dir"], true, "")
- node2 := node1["nodes"].([]interface{})[0].(map[string]interface{})
- assert.Equal(t, node2["key"], "/foo/y/z", "")
- assert.Equal(t, node2["value"], "YYY", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a watcher can wait for a value to be set and return it to the client.
- //
- // $ curl localhost:4001/v2/keys/foo/bar?wait=true
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- //
- func TestV2WatchKey(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- // There exists a little gap between etcd ready to serve and
- // it actually serves the first request, which means the response
- // delay could be a little bigger.
- // This test is time sensitive, so it does one request to ensure
- // that the server is working.
- resp, _ := Get(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"))
- resp.Body.Close()
- var watchResp *http.Response
- c := make(chan bool)
- go func() {
- watchResp, _ = Get(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?wait=true"))
- c <- true
- }()
- // Make sure response didn't fire early.
- time.Sleep(1 * time.Millisecond)
- // Set a value.
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- // A response should follow from the GET above.
- time.Sleep(1 * time.Millisecond)
- select {
- case <-c:
- default:
- t.Fatal("cannot get watch result")
- }
- body := ReadBodyJSON(watchResp)
- watchResp.Body.Close()
- assert.NotNil(t, body, "")
- assert.Equal(t, body["action"], "set", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/bar", "")
- assert.Equal(t, node["value"], "XXX", "")
- assert.Equal(t, node["modifiedIndex"], 2, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a watcher can wait for a value to be set after a given index.
- //
- // $ curl localhost:4001/v2/keys/foo/bar?wait=true&waitIndex=3
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=YYY
- //
- func TestV2WatchKeyWithIndex(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- var body map[string]interface{}
- c := make(chan bool)
- go func() {
- resp, _ := Get(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar?wait=true&waitIndex=3"))
- body = ReadBodyJSON(resp)
- c <- true
- }()
- // Make sure response didn't fire early.
- time.Sleep(1 * time.Millisecond)
- assert.Nil(t, body, "")
- // Set a value (before given index).
- v := url.Values{}
- v.Set("value", "XXX")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- // Make sure response didn't fire early.
- time.Sleep(1 * time.Millisecond)
- assert.Nil(t, body, "")
- // Set a value (before given index).
- v.Set("value", "YYY")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar"), v)
- ReadBody(resp)
- // A response should follow from the GET above.
- time.Sleep(1 * time.Millisecond)
- select {
- case <-c:
- default:
- t.Fatal("cannot get watch result")
- }
- assert.NotNil(t, body, "")
- assert.Equal(t, body["action"], "set", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/foo/bar", "")
- assert.Equal(t, node["value"], "YYY", "")
- assert.Equal(t, node["modifiedIndex"], 3, "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that a watcher can wait for a value to be set after a given index.
- //
- // $ curl localhost:4001/v2/keys/keyindir/bar?wait=true
- // $ curl -X PUT localhost:4001/v2/keys/keyindir -d dir=true -d ttl=1
- // $ curl -X PUT localhost:4001/v2/keys/keyindir/bar -d value=YYY
- //
- func TestV2WatchKeyInDir(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- var body map[string]interface{}
- c := make(chan bool)
- // Set a value (before given index).
- v := url.Values{}
- v.Set("dir", "true")
- v.Set("ttl", "1")
- resp, _ := PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/keyindir"), v)
- ReadBody(resp)
- // Set a value (before given index).
- v = url.Values{}
- v.Set("value", "XXX")
- resp, _ = PutForm(fmt.Sprintf("%s%s", u, "/v2/keys/keyindir/bar"), v)
- ReadBody(resp)
- go func() {
- resp, _ := Get(fmt.Sprintf("%s%s", u, "/v2/keys/keyindir/bar?wait=true"))
- body = ReadBodyJSON(resp)
- c <- true
- }()
- // wait for expiration, we do have a up to 500 millisecond delay
- time.Sleep(time.Second + time.Millisecond*500)
- select {
- case <-c:
- default:
- t.Fatal("cannot get watch result")
- }
- assert.NotNil(t, body, "")
- assert.Equal(t, body["action"], "expire", "")
- node := body["node"].(map[string]interface{})
- assert.Equal(t, node["key"], "/keyindir", "")
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
- // Ensures that HEAD could work.
- //
- // $ curl -I localhost:4001/v2/keys/foo/bar -> fail
- // $ curl -X PUT localhost:4001/v2/keys/foo/bar -d value=XXX
- // $ curl -I localhost:4001/v2/keys/foo/bar
- //
- func TestV2HeadKey(t *testing.T) {
- es, hs := buildCluster(1)
- u := hs[0].URL
- v := url.Values{}
- v.Set("value", "XXX")
- fullURL := fmt.Sprintf("%s%s", u, "/v2/keys/foo/bar")
- resp, _ := Head(fullURL)
- assert.Equal(t, resp.StatusCode, http.StatusNotFound)
- assert.Equal(t, resp.ContentLength, -1)
- resp, _ = PutForm(fullURL, v)
- ReadBody(resp)
- resp, _ = Head(fullURL)
- assert.Equal(t, resp.StatusCode, http.StatusOK)
- assert.Equal(t, resp.ContentLength, -1)
- es[0].Stop()
- hs[0].Close()
- afterTest(t)
- }
|