2015-01-06 10:40:00 -08:00
|
|
|
package api
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"net/http"
|
|
|
|
"strconv"
|
|
|
|
"strings"
|
|
|
|
)
|
|
|
|
|
|
|
|
// KVPair is used to represent a single K/V entry
|
|
|
|
type KVPair struct {
|
2016-09-26 10:07:00 -05:00
|
|
|
// Key is the name of the key. It is also part of the URL path when accessed
|
|
|
|
// via the API.
|
|
|
|
Key string
|
|
|
|
|
|
|
|
// CreateIndex holds the index corresponding the creation of this KVPair. This
|
|
|
|
// is a read-only field.
|
2015-01-06 10:40:00 -08:00
|
|
|
CreateIndex uint64
|
2016-09-26 10:07:00 -05:00
|
|
|
|
2016-09-26 13:31:19 -07:00
|
|
|
// ModifyIndex is used for the Check-And-Set operations and can also be fed
|
|
|
|
// back into the WaitIndex of the QueryOptions in order to perform blocking
|
|
|
|
// queries.
|
2015-01-06 10:40:00 -08:00
|
|
|
ModifyIndex uint64
|
2016-09-26 10:07:00 -05:00
|
|
|
|
|
|
|
// LockIndex holds the index corresponding to a lock on this key, if any. This
|
|
|
|
// is a read-only field.
|
|
|
|
LockIndex uint64
|
|
|
|
|
|
|
|
// Flags are any user-defined flags on the key. It is up to the implementer
|
|
|
|
// to check these values, since Consul does not treat them specially.
|
|
|
|
Flags uint64
|
|
|
|
|
|
|
|
// Value is the value for the key. This can be any value, but it will be
|
|
|
|
// base64 encoded upon transport.
|
|
|
|
Value []byte
|
|
|
|
|
2016-09-26 16:06:44 -07:00
|
|
|
// Session is a string representing the ID of the session. Any other
|
2016-09-26 10:07:00 -05:00
|
|
|
// interactions with this key over the same session must specify the same
|
2016-09-26 13:31:26 -07:00
|
|
|
// session ID.
|
2016-09-26 10:07:00 -05:00
|
|
|
Session string
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// KVPairs is a list of KVPair objects
|
|
|
|
type KVPairs []*KVPair
|
|
|
|
|
2016-05-10 13:36:48 -07:00
|
|
|
// KVOp constants give possible operations available in a KVTxn.
|
|
|
|
type KVOp string
|
|
|
|
|
2016-05-06 17:50:58 -07:00
|
|
|
const (
|
2017-04-20 17:50:52 -07:00
|
|
|
KVSet KVOp = "set"
|
|
|
|
KVDelete KVOp = "delete"
|
|
|
|
KVDeleteCAS KVOp = "delete-cas"
|
|
|
|
KVDeleteTree KVOp = "delete-tree"
|
|
|
|
KVCAS KVOp = "cas"
|
|
|
|
KVLock KVOp = "lock"
|
|
|
|
KVUnlock KVOp = "unlock"
|
|
|
|
KVGet KVOp = "get"
|
|
|
|
KVGetTree KVOp = "get-tree"
|
|
|
|
KVCheckSession KVOp = "check-session"
|
|
|
|
KVCheckIndex KVOp = "check-index"
|
|
|
|
KVCheckNotExists KVOp = "check-not-exists"
|
2016-05-06 17:50:58 -07:00
|
|
|
)
|
|
|
|
|
|
|
|
// KVTxnOp defines a single operation inside a transaction.
|
|
|
|
type KVTxnOp struct {
|
2016-11-24 20:43:41 +01:00
|
|
|
Verb KVOp
|
2016-05-06 17:50:58 -07:00
|
|
|
Key string
|
|
|
|
Value []byte
|
|
|
|
Flags uint64
|
|
|
|
Index uint64
|
|
|
|
Session string
|
|
|
|
}
|
|
|
|
|
2016-05-11 01:35:27 -07:00
|
|
|
// KVTxnOps defines a set of operations to be performed inside a single
|
2016-05-06 17:50:58 -07:00
|
|
|
// transaction.
|
2016-05-11 01:35:27 -07:00
|
|
|
type KVTxnOps []*KVTxnOp
|
2016-05-06 17:50:58 -07:00
|
|
|
|
2016-05-11 01:35:27 -07:00
|
|
|
// KVTxnResponse has the outcome of a transaction.
|
|
|
|
type KVTxnResponse struct {
|
|
|
|
Results []*KVPair
|
|
|
|
Errors TxnErrors
|
2016-05-06 17:50:58 -07:00
|
|
|
}
|
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
// KV is used to manipulate the K/V API
|
|
|
|
type KV struct {
|
|
|
|
c *Client
|
|
|
|
}
|
|
|
|
|
|
|
|
// KV is used to return a handle to the K/V apis
|
|
|
|
func (c *Client) KV() *KV {
|
|
|
|
return &KV{c}
|
|
|
|
}
|
|
|
|
|
2016-10-03 08:24:04 +00:00
|
|
|
// Get is used to lookup a single key. The returned pointer
|
|
|
|
// to the KVPair will be nil if the key does not exist.
|
2015-01-06 10:40:00 -08:00
|
|
|
func (k *KV) Get(key string, q *QueryOptions) (*KVPair, *QueryMeta, error) {
|
|
|
|
resp, qm, err := k.getInternal(key, nil, q)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if resp == nil {
|
|
|
|
return nil, qm, nil
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var entries []*KVPair
|
|
|
|
if err := decodeBody(resp, &entries); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if len(entries) > 0 {
|
|
|
|
return entries[0], qm, nil
|
|
|
|
}
|
|
|
|
return nil, qm, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// List is used to lookup all keys under a prefix
|
|
|
|
func (k *KV) List(prefix string, q *QueryOptions) (KVPairs, *QueryMeta, error) {
|
|
|
|
resp, qm, err := k.getInternal(prefix, map[string]string{"recurse": ""}, q)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if resp == nil {
|
|
|
|
return nil, qm, nil
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var entries []*KVPair
|
|
|
|
if err := decodeBody(resp, &entries); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
return entries, qm, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Keys is used to list all the keys under a prefix. Optionally,
|
|
|
|
// a separator can be used to limit the responses.
|
|
|
|
func (k *KV) Keys(prefix, separator string, q *QueryOptions) ([]string, *QueryMeta, error) {
|
|
|
|
params := map[string]string{"keys": ""}
|
|
|
|
if separator != "" {
|
|
|
|
params["separator"] = separator
|
|
|
|
}
|
|
|
|
resp, qm, err := k.getInternal(prefix, params, q)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
if resp == nil {
|
|
|
|
return nil, qm, nil
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
var entries []string
|
|
|
|
if err := decodeBody(resp, &entries); err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
return entries, qm, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KV) getInternal(key string, params map[string]string, q *QueryOptions) (*http.Response, *QueryMeta, error) {
|
2016-11-05 00:55:10 -04:00
|
|
|
r := k.c.newRequest("GET", "/v1/kv/"+strings.TrimPrefix(key, "/"))
|
2015-01-06 10:40:00 -08:00
|
|
|
r.setQueryOptions(q)
|
|
|
|
for param, val := range params {
|
|
|
|
r.params.Set(param, val)
|
|
|
|
}
|
|
|
|
rtt, resp, err := k.c.doRequest(r)
|
|
|
|
if err != nil {
|
|
|
|
return nil, nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
qm := &QueryMeta{}
|
|
|
|
parseQueryMeta(resp, qm)
|
|
|
|
qm.RequestTime = rtt
|
|
|
|
|
|
|
|
if resp.StatusCode == 404 {
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil, qm, nil
|
|
|
|
} else if resp.StatusCode != 200 {
|
|
|
|
resp.Body.Close()
|
|
|
|
return nil, nil, fmt.Errorf("Unexpected response code: %d", resp.StatusCode)
|
|
|
|
}
|
|
|
|
return resp, qm, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Put is used to write a new value. Only the
|
|
|
|
// Key, Flags and Value is respected.
|
|
|
|
func (k *KV) Put(p *KVPair, q *WriteOptions) (*WriteMeta, error) {
|
|
|
|
params := make(map[string]string, 1)
|
|
|
|
if p.Flags != 0 {
|
|
|
|
params["flags"] = strconv.FormatUint(p.Flags, 10)
|
|
|
|
}
|
|
|
|
_, wm, err := k.put(p.Key, params, p.Value, q)
|
|
|
|
return wm, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// CAS is used for a Check-And-Set operation. The Key,
|
|
|
|
// ModifyIndex, Flags and Value are respected. Returns true
|
|
|
|
// on success or false on failures.
|
|
|
|
func (k *KV) CAS(p *KVPair, q *WriteOptions) (bool, *WriteMeta, error) {
|
|
|
|
params := make(map[string]string, 2)
|
|
|
|
if p.Flags != 0 {
|
|
|
|
params["flags"] = strconv.FormatUint(p.Flags, 10)
|
|
|
|
}
|
|
|
|
params["cas"] = strconv.FormatUint(p.ModifyIndex, 10)
|
|
|
|
return k.put(p.Key, params, p.Value, q)
|
|
|
|
}
|
|
|
|
|
2015-09-15 13:22:08 +01:00
|
|
|
// Acquire is used for a lock acquisition operation. The Key,
|
2015-01-06 10:40:00 -08:00
|
|
|
// Flags, Value and Session are respected. Returns true
|
|
|
|
// on success or false on failures.
|
|
|
|
func (k *KV) Acquire(p *KVPair, q *WriteOptions) (bool, *WriteMeta, error) {
|
|
|
|
params := make(map[string]string, 2)
|
|
|
|
if p.Flags != 0 {
|
|
|
|
params["flags"] = strconv.FormatUint(p.Flags, 10)
|
|
|
|
}
|
|
|
|
params["acquire"] = p.Session
|
|
|
|
return k.put(p.Key, params, p.Value, q)
|
|
|
|
}
|
|
|
|
|
|
|
|
// Release is used for a lock release operation. The Key,
|
|
|
|
// Flags, Value and Session are respected. Returns true
|
|
|
|
// on success or false on failures.
|
|
|
|
func (k *KV) Release(p *KVPair, q *WriteOptions) (bool, *WriteMeta, error) {
|
|
|
|
params := make(map[string]string, 2)
|
|
|
|
if p.Flags != 0 {
|
|
|
|
params["flags"] = strconv.FormatUint(p.Flags, 10)
|
|
|
|
}
|
|
|
|
params["release"] = p.Session
|
|
|
|
return k.put(p.Key, params, p.Value, q)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KV) put(key string, params map[string]string, body []byte, q *WriteOptions) (bool, *WriteMeta, error) {
|
2015-03-31 21:08:06 -06:00
|
|
|
if len(key) > 0 && key[0] == '/' {
|
|
|
|
return false, nil, fmt.Errorf("Invalid key. Key must not begin with a '/': %s", key)
|
|
|
|
}
|
|
|
|
|
2015-01-06 10:40:00 -08:00
|
|
|
r := k.c.newRequest("PUT", "/v1/kv/"+key)
|
|
|
|
r.setWriteOptions(q)
|
|
|
|
for param, val := range params {
|
|
|
|
r.params.Set(param, val)
|
|
|
|
}
|
|
|
|
r.body = bytes.NewReader(body)
|
|
|
|
rtt, resp, err := requireOK(k.c.doRequest(r))
|
|
|
|
if err != nil {
|
|
|
|
return false, nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
|
|
|
qm := &WriteMeta{}
|
|
|
|
qm.RequestTime = rtt
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if _, err := io.Copy(&buf, resp.Body); err != nil {
|
|
|
|
return false, nil, fmt.Errorf("Failed to read response: %v", err)
|
|
|
|
}
|
|
|
|
res := strings.Contains(string(buf.Bytes()), "true")
|
|
|
|
return res, qm, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Delete is used to delete a single key
|
|
|
|
func (k *KV) Delete(key string, w *WriteOptions) (*WriteMeta, error) {
|
2015-01-13 13:57:48 -08:00
|
|
|
_, qm, err := k.deleteInternal(key, nil, w)
|
|
|
|
return qm, err
|
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteCAS is used for a Delete Check-And-Set operation. The Key
|
|
|
|
// and ModifyIndex are respected. Returns true on success or false on failures.
|
|
|
|
func (k *KV) DeleteCAS(p *KVPair, q *WriteOptions) (bool, *WriteMeta, error) {
|
|
|
|
params := map[string]string{
|
|
|
|
"cas": strconv.FormatUint(p.ModifyIndex, 10),
|
|
|
|
}
|
|
|
|
return k.deleteInternal(p.Key, params, q)
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// DeleteTree is used to delete all keys under a prefix
|
|
|
|
func (k *KV) DeleteTree(prefix string, w *WriteOptions) (*WriteMeta, error) {
|
2015-01-13 13:57:48 -08:00
|
|
|
_, qm, err := k.deleteInternal(prefix, map[string]string{"recurse": ""}, w)
|
|
|
|
return qm, err
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
|
|
|
|
2015-01-13 13:57:48 -08:00
|
|
|
func (k *KV) deleteInternal(key string, params map[string]string, q *WriteOptions) (bool, *WriteMeta, error) {
|
2016-11-05 00:55:10 -04:00
|
|
|
r := k.c.newRequest("DELETE", "/v1/kv/"+strings.TrimPrefix(key, "/"))
|
2015-01-06 10:40:00 -08:00
|
|
|
r.setWriteOptions(q)
|
2015-01-13 13:57:48 -08:00
|
|
|
for param, val := range params {
|
|
|
|
r.params.Set(param, val)
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
|
|
|
rtt, resp, err := requireOK(k.c.doRequest(r))
|
|
|
|
if err != nil {
|
2015-01-13 13:57:48 -08:00
|
|
|
return false, nil, err
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
2015-01-13 13:57:48 -08:00
|
|
|
defer resp.Body.Close()
|
2015-01-06 10:40:00 -08:00
|
|
|
|
|
|
|
qm := &WriteMeta{}
|
|
|
|
qm.RequestTime = rtt
|
2015-01-13 13:57:48 -08:00
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if _, err := io.Copy(&buf, resp.Body); err != nil {
|
|
|
|
return false, nil, fmt.Errorf("Failed to read response: %v", err)
|
|
|
|
}
|
|
|
|
res := strings.Contains(string(buf.Bytes()), "true")
|
|
|
|
return res, qm, nil
|
2015-01-06 10:40:00 -08:00
|
|
|
}
|
2016-05-06 17:50:58 -07:00
|
|
|
|
2016-05-11 01:35:27 -07:00
|
|
|
// TxnOp is the internal format we send to Consul. It's not specific to KV,
|
|
|
|
// though currently only KV operations are supported.
|
|
|
|
type TxnOp struct {
|
2016-05-11 10:58:27 -07:00
|
|
|
KV *KVTxnOp
|
2016-05-11 01:35:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// TxnOps is a list of transaction operations.
|
|
|
|
type TxnOps []*TxnOp
|
|
|
|
|
|
|
|
// TxnResult is the internal format we receive from Consul.
|
|
|
|
type TxnResult struct {
|
2016-05-11 13:48:03 -07:00
|
|
|
KV *KVPair
|
2016-05-11 01:35:27 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
// TxnResults is a list of TxnResult objects.
|
|
|
|
type TxnResults []*TxnResult
|
|
|
|
|
|
|
|
// TxnError is used to return information about an operation in a transaction.
|
|
|
|
type TxnError struct {
|
|
|
|
OpIndex int
|
|
|
|
What string
|
|
|
|
}
|
|
|
|
|
|
|
|
// TxnErrors is a list of TxnError objects.
|
|
|
|
type TxnErrors []*TxnError
|
|
|
|
|
|
|
|
// TxnResponse is the internal format we receive from Consul.
|
|
|
|
type TxnResponse struct {
|
|
|
|
Results TxnResults
|
|
|
|
Errors TxnErrors
|
|
|
|
}
|
|
|
|
|
2016-05-06 17:50:58 -07:00
|
|
|
// Txn is used to apply multiple KV operations in a single, atomic transaction.
|
2016-05-12 17:38:25 -07:00
|
|
|
//
|
2016-05-06 17:50:58 -07:00
|
|
|
// Note that Go will perform the required base64 encoding on the values
|
2016-05-10 13:36:48 -07:00
|
|
|
// automatically because the type is a byte slice. Transactions are defined as a
|
|
|
|
// list of operations to perform, using the KVOp constants and KVTxnOp structure
|
|
|
|
// to define operations. If any operation fails, none of the changes are applied
|
2016-05-11 01:35:27 -07:00
|
|
|
// to the state store. Note that this hides the internal raw transaction interface
|
|
|
|
// and munges the input and output types into KV-specific ones for ease of use.
|
|
|
|
// If there are more non-KV operations in the future we may break out a new
|
|
|
|
// transaction API client, but it will be easy to keep this KV-specific variant
|
|
|
|
// supported.
|
2016-05-10 13:36:48 -07:00
|
|
|
//
|
2016-05-12 17:38:25 -07:00
|
|
|
// Even though this is generally a write operation, we take a QueryOptions input
|
|
|
|
// and return a QueryMeta output. If the transaction contains only read ops, then
|
|
|
|
// Consul will fast-path it to a different endpoint internally which supports
|
|
|
|
// consistency controls, but not blocking. If there are write operations then
|
|
|
|
// the request will always be routed through raft and any consistency settings
|
|
|
|
// will be ignored.
|
|
|
|
//
|
2016-05-10 13:36:48 -07:00
|
|
|
// Here's an example:
|
|
|
|
//
|
2016-05-11 01:35:27 -07:00
|
|
|
// ops := KVTxnOps{
|
|
|
|
// &KVTxnOp{
|
|
|
|
// Verb: KVLock,
|
|
|
|
// Key: "test/lock",
|
2016-05-10 13:36:48 -07:00
|
|
|
// Session: "adf4238a-882b-9ddc-4a9d-5b6758e4159e",
|
2016-05-11 01:35:27 -07:00
|
|
|
// Value: []byte("hello"),
|
2016-05-10 13:36:48 -07:00
|
|
|
// },
|
2016-05-11 01:35:27 -07:00
|
|
|
// &KVTxnOp{
|
|
|
|
// Verb: KVGet,
|
|
|
|
// Key: "another/key",
|
2016-05-10 13:36:48 -07:00
|
|
|
// },
|
|
|
|
// }
|
2016-05-11 01:35:27 -07:00
|
|
|
// ok, response, _, err := kv.Txn(&ops, nil)
|
2016-05-10 13:36:48 -07:00
|
|
|
//
|
|
|
|
// If there is a problem making the transaction request then an error will be
|
|
|
|
// returned. Otherwise, the ok value will be true if the transaction succeeded
|
2016-05-11 01:35:27 -07:00
|
|
|
// or false if it was rolled back. The response is a structured return value which
|
2016-05-10 13:36:48 -07:00
|
|
|
// will have the outcome of the transaction. Its Results member will have entries
|
|
|
|
// for each operation. Deleted keys will have a nil entry in the, and to save
|
|
|
|
// space, the Value of each key in the Results will be nil unless the operation
|
|
|
|
// is a KVGet. If the transaction was rolled back, the Errors member will have
|
|
|
|
// entries referencing the index of the operation that failed along with an error
|
|
|
|
// message.
|
2016-05-12 17:38:25 -07:00
|
|
|
func (k *KV) Txn(txn KVTxnOps, q *QueryOptions) (bool, *KVTxnResponse, *QueryMeta, error) {
|
2016-05-10 21:41:47 -07:00
|
|
|
r := k.c.newRequest("PUT", "/v1/txn")
|
2016-05-12 17:38:25 -07:00
|
|
|
r.setQueryOptions(q)
|
2016-05-06 17:50:58 -07:00
|
|
|
|
2016-05-11 01:35:27 -07:00
|
|
|
// Convert into the internal format since this is an all-KV txn.
|
|
|
|
ops := make(TxnOps, 0, len(txn))
|
2016-05-11 10:58:27 -07:00
|
|
|
for _, kvOp := range txn {
|
|
|
|
ops = append(ops, &TxnOp{KV: kvOp})
|
2016-05-11 01:35:27 -07:00
|
|
|
}
|
|
|
|
r.obj = ops
|
2016-05-06 17:50:58 -07:00
|
|
|
rtt, resp, err := k.c.doRequest(r)
|
|
|
|
if err != nil {
|
|
|
|
return false, nil, nil, err
|
|
|
|
}
|
|
|
|
defer resp.Body.Close()
|
|
|
|
|
2016-05-12 17:38:25 -07:00
|
|
|
qm := &QueryMeta{}
|
|
|
|
parseQueryMeta(resp, qm)
|
|
|
|
qm.RequestTime = rtt
|
2016-05-06 17:50:58 -07:00
|
|
|
|
|
|
|
if resp.StatusCode == http.StatusOK || resp.StatusCode == http.StatusConflict {
|
2016-05-11 01:35:27 -07:00
|
|
|
var txnResp TxnResponse
|
|
|
|
if err := decodeBody(resp, &txnResp); err != nil {
|
2016-05-06 17:50:58 -07:00
|
|
|
return false, nil, nil, err
|
|
|
|
}
|
2016-05-11 01:35:27 -07:00
|
|
|
|
|
|
|
// Convert from the internal format.
|
|
|
|
kvResp := KVTxnResponse{
|
|
|
|
Errors: txnResp.Errors,
|
|
|
|
}
|
|
|
|
for _, result := range txnResp.Results {
|
2016-05-11 13:48:03 -07:00
|
|
|
kvResp.Results = append(kvResp.Results, result.KV)
|
2016-05-11 01:35:27 -07:00
|
|
|
}
|
2016-05-12 17:38:25 -07:00
|
|
|
return resp.StatusCode == http.StatusOK, &kvResp, qm, nil
|
2016-05-06 17:50:58 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
var buf bytes.Buffer
|
|
|
|
if _, err := io.Copy(&buf, resp.Body); err != nil {
|
|
|
|
return false, nil, nil, fmt.Errorf("Failed to read response: %v", err)
|
|
|
|
}
|
|
|
|
return false, nil, nil, fmt.Errorf("Failed request: %s", buf.String())
|
|
|
|
}
|