2
0
mirror of https://github.com/openvswitch/ovs synced 2025-10-25 15:07:05 +00:00

ovsdb-monitor: Support monitor_cond_since.

Support the new monitor method monitor_cond_since so that a client
can request monitoring start from a specific point instead of always
from beginning. This will reduce the cost at scenarios when server
is restarted/failed-over but client still has all existing data. In
these scenarios only new changes (and in most cases no change) needed
to be transfered to client. When ovsdb-server restarted, history
transactions are read from disk file; when ovsdb-server failed over,
history transactions exists already in the memory of the new server.

There are situations that the requested transaction may not be found.
For example, a transaction that is too old and has been discarded
from the maintained history list in memory, or the transactions on
disk has been compacted during ovsdb compaction. In those situations
the server fall backs to transfer all data start from begining.

For more details of the protocol change, see
Documentation/ref/ovsdb-server.7.rst.

This change includes both server side and ovsdb-client side changes
with the new protocol. IDLs using this capability will be added in
future patches.

Now the feature takes effect only for cluster mode of ovsdb-server,
because cluster mode is the only mode that supports unique transcation
uuid today. For other modes, the monitor_cond_since always fall back
to transfer all data with found = false. Support for those modes can
be added in the future.

Signed-off-by: Han Zhou <hzhou8@ebay.com>
Signed-off-by: Ben Pfaff <blp@ovn.org>
This commit is contained in:
Han Zhou
2019-02-28 09:15:18 -08:00
committed by Ben Pfaff
parent 695e815027
commit 9167cb52fa
9 changed files with 689 additions and 39 deletions

View File

@@ -364,7 +364,79 @@ Initial views of rows are not presented in update2 notifications, but in the
response object to the ``monitor_cond`` request. The formatting of the response object to the ``monitor_cond`` request. The formatting of the
<table-updates2> object, however, is the same in either case. <table-updates2> object, however, is the same in either case.
4.1.15 Get Server ID 4.1.15 Monitor_cond_since
-------------------------
A new monitor method added in Open vSwitch version 2.12. The
``monitor_cond_since`` request enables a client to request changes that
happened after a specific transaction id. A client can use this feature to
request only latest changes after a server connection reset instead of
re-transfer all data from the server again.
The ``monitor_cond`` method described in Section 4.1.12 also applies to
``monitor_cond_since``, with the following exceptions:
* RPC request method becomes ``monitor_cond_since``.
* Reply result includes extra parameters.
* Subsequent changes are sent to the client using the ``update3`` monitor
notification, described in Section 4.1.16
The request object has the following members::
"method": "monitor_cond_since"
"params": [<db-name>, <json-value>, <monitor-cond-requests>, <last-txn-id>]
"id": <nonnull-json-value>
The <last-txn-id> parameter is the transaction id that identifies the latest
data the client already has, and it requests server to send changes AFTER this
transaction (exclusive).
All other parameters are the same as ``monitor_cond`` method.
The response object has the following members::
"result": [<found>, <last-txn-id>, <table-updates2>]
"error": null
"id": same "id" as request
The <found> is a boolean value that tells if the <last-txn-id> requested by
client is found in server's history or not. If true, the changes after that
version up to current is sent. Otherwise, all data is sent.
The <last-txn-id> is the transaction id that identifies the latest transaction
included in the changes in <table-updates2> of this response, so that client
can keep tracking. If there is no change involved in this response, it is the
same as the <last-txn-id> in the request if <found> is true, or zero uuid if
<found> is false. If the server does not support transaction uuid, it will
be zero uuid as well.
All other parameters are the same as in response object of ``monitor_cond``
method.
Like in ``monitor_cond``, subsequent changes that match conditions in
<monitor-cond-request> are automatically sent to the client, but using
``update3`` monitor notification (see Section 4.1.16), instead of ``update2``.
4.1.16 Update3 notification
---------------------------
The ``update3`` notification is sent by the server to the client to report
changes in tables that are being monitored following a ``monitor_cond_since``
request as described above. The notification has the following members::
"method": "update3"
"params": [<json-value>, <last-txn-id>, <table-updates2>]
"id": null
The <last-txn-id> is the same as described in the response object of
``monitor_cond_since``.
All other parameters are the same as in ``update2`` monitor notification (see
Section 4.1.14).
4.1.17 Get Server ID
-------------------- --------------------
A new RPC method added in Open vSwitch version 2.7. The request contains the A new RPC method added in Open vSwitch version 2.7. The request contains the
@@ -384,7 +456,7 @@ The response object contains the following members::
running OVSDB server process. A fresh UUID is generated when the process running OVSDB server process. A fresh UUID is generated when the process
restarts. restarts.
4.1.16 Database Change Awareness 4.1.18 Database Change Awareness
-------------------------------- --------------------------------
RFC 7047 does not provide a way for a client to find out about some kinds of RFC 7047 does not provide a way for a client to find out about some kinds of
@@ -414,7 +486,7 @@ The reply is always the same::
"error": null "error": null
"id": same "id" as request "id": same "id" as request
4.1.17 Schema Conversion 4.1.19 Schema Conversion
------------------------ ------------------------
Open vSwitch 2.9 adds a new JSON-RPC request to convert an online database from Open vSwitch 2.9 adds a new JSON-RPC request to convert an online database from

View File

@@ -986,13 +986,19 @@ ovsdb_jsonrpc_session_got_request(struct ovsdb_jsonrpc_session *s,
ovsdb_jsonrpc_trigger_create(s, db, request); ovsdb_jsonrpc_trigger_create(s, db, request);
} }
} else if (!strcmp(request->method, "monitor") || } else if (!strcmp(request->method, "monitor") ||
(monitor_cond_enable__ && !strcmp(request->method, (monitor_cond_enable__ &&
"monitor_cond"))) { (!strcmp(request->method, "monitor_cond") ||
!strcmp(request->method, "monitor_cond_since")))) {
struct ovsdb *db = ovsdb_jsonrpc_lookup_db(s, request, &reply); struct ovsdb *db = ovsdb_jsonrpc_lookup_db(s, request, &reply);
if (!reply) { if (!reply) {
int l = strlen(request->method) - strlen("monitor"); enum ovsdb_monitor_version version;
enum ovsdb_monitor_version version = l ? OVSDB_MONITOR_V2 if (!strcmp(request->method, "monitor")) {
: OVSDB_MONITOR_V1; version = OVSDB_MONITOR_V1;
} else if (!strcmp(request->method, "monitor_cond")) {
version = OVSDB_MONITOR_V2;
} else {
version = OVSDB_MONITOR_V3;
}
reply = ovsdb_jsonrpc_monitor_create(s, db, request->params, reply = ovsdb_jsonrpc_monitor_create(s, db, request->params,
version, request->id); version, request->id);
} }
@@ -1364,7 +1370,8 @@ ovsdb_jsonrpc_monitor_create(struct ovsdb_jsonrpc_session *s, struct ovsdb *db,
struct shash_node *node; struct shash_node *node;
struct json *json; struct json *json;
if (json_array(params)->n != 3) { if ((version == OVSDB_MONITOR_V2 && json_array(params)->n != 3) ||
(version == OVSDB_MONITOR_V3 && json_array(params)->n != 4)) {
error = ovsdb_syntax_error(params, NULL, "invalid parameters"); error = ovsdb_syntax_error(params, NULL, "invalid parameters");
goto error; goto error;
} }
@@ -1385,7 +1392,7 @@ ovsdb_jsonrpc_monitor_create(struct ovsdb_jsonrpc_session *s, struct ovsdb *db,
m->session = s; m->session = s;
m->db = db; m->db = db;
m->dbmon = ovsdb_monitor_create(db, m); m->dbmon = ovsdb_monitor_create(db, m);
if (version == OVSDB_MONITOR_V2) { if (version == OVSDB_MONITOR_V2 || version == OVSDB_MONITOR_V3) {
m->condition = ovsdb_monitor_session_condition_create(); m->condition = ovsdb_monitor_session_condition_create();
} }
m->version = version; m->version = version;
@@ -1444,9 +1451,42 @@ ovsdb_jsonrpc_monitor_create(struct ovsdb_jsonrpc_session *s, struct ovsdb *db,
ovsdb_monitor_condition_bind(m->dbmon, m->condition); ovsdb_monitor_condition_bind(m->dbmon, m->condition);
} }
bool initial = false;
if (version == OVSDB_MONITOR_V3) {
struct json *last_id = params->array.elems[3];
if (last_id->type != JSON_STRING) {
error = ovsdb_syntax_error(last_id, NULL,
"last-txn-id must be string");
goto error;
}
struct uuid txn_uuid;
if (!uuid_from_string(&txn_uuid, last_id->string)) {
error = ovsdb_syntax_error(last_id, NULL,
"last-txn-id must be UUID format.");
goto error;
}
if (!uuid_is_zero(&txn_uuid)) {
ovsdb_monitor_get_changes_after(&txn_uuid, m->dbmon,
&m->change_set);
}
}
if (!m->change_set) {
ovsdb_monitor_get_initial(m->dbmon, &m->change_set); ovsdb_monitor_get_initial(m->dbmon, &m->change_set);
json = ovsdb_jsonrpc_monitor_compose_update(m, true); initial = true;
}
json = ovsdb_jsonrpc_monitor_compose_update(m, initial);
json = json ? json : json_object_create(); json = json ? json : json_object_create();
if (m->version == OVSDB_MONITOR_V3) {
struct json *json_last_id = json_string_create_nocopy(
xasprintf(UUID_FMT,
UUID_ARGS(ovsdb_monitor_get_last_txnid(
m->dbmon))));
struct json *json_found = json_boolean_create(!initial);
json = json_array_create_3(json_found, json_last_id, json);
}
return jsonrpc_create_reply(json, request_id); return jsonrpc_create_reply(json, request_id);
error: error:
@@ -1580,8 +1620,17 @@ ovsdb_jsonrpc_monitor_cond_change(struct ovsdb_jsonrpc_session *s,
if (update_json) { if (update_json) {
struct jsonrpc_msg *msg; struct jsonrpc_msg *msg;
struct json *p; struct json *p;
if (m->version == OVSDB_MONITOR_V3) {
struct json *json_last_id = json_string_create_nocopy(
xasprintf(UUID_FMT,
UUID_ARGS(ovsdb_monitor_get_last_txnid(
m->dbmon))));
p = json_array_create_3(json_clone(m->monitor_id), json_last_id,
update_json);
} else {
p = json_array_create_2(json_clone(m->monitor_id), update_json); p = json_array_create_2(json_clone(m->monitor_id), update_json);
}
msg = ovsdb_jsonrpc_create_notify(m, p); msg = ovsdb_jsonrpc_create_notify(m, p);
jsonrpc_session_send(s->js, msg); jsonrpc_session_send(s->js, msg);
} }
@@ -1702,6 +1751,9 @@ ovsdb_jsonrpc_create_notify(const struct ovsdb_jsonrpc_monitor *m,
case OVSDB_MONITOR_V2: case OVSDB_MONITOR_V2:
method = "update2"; method = "update2";
break; break;
case OVSDB_MONITOR_V3:
method = "update3";
break;
case OVSDB_MONITOR_VERSION_MAX: case OVSDB_MONITOR_VERSION_MAX:
default: default:
OVS_NOT_REACHED(); OVS_NOT_REACHED();
@@ -1728,8 +1780,17 @@ ovsdb_jsonrpc_monitor_flush_all(struct ovsdb_jsonrpc_session *s)
if (json) { if (json) {
struct jsonrpc_msg *msg; struct jsonrpc_msg *msg;
struct json *params; struct json *params;
if (m->version == OVSDB_MONITOR_V3) {
struct json *json_last_id = json_string_create_nocopy(
xasprintf(UUID_FMT,
UUID_ARGS(ovsdb_monitor_get_last_txnid(
m->dbmon))));
params = json_array_create_3(json_clone(m->monitor_id),
json_last_id, json);
} else {
params = json_array_create_2(json_clone(m->monitor_id), json); params = json_array_create_2(json_clone(m->monitor_id), json);
}
msg = ovsdb_jsonrpc_create_notify(m, params); msg = ovsdb_jsonrpc_create_notify(m, params);
jsonrpc_session_send(s->js, msg); jsonrpc_session_send(s->js, msg);
} }

View File

@@ -136,6 +136,9 @@ struct ovsdb_monitor_change_set {
struct ovs_list change_set_for_tables; struct ovs_list change_set_for_tables;
int n_refs; int n_refs;
/* The previous txn id before this change set's start point. */
struct uuid prev_txn;
}; };
/* Contains 'struct ovsdb_monitor_row's for rows in a specific table /* Contains 'struct ovsdb_monitor_row's for rows in a specific table
@@ -200,7 +203,9 @@ typedef struct json *
static void ovsdb_monitor_destroy(struct ovsdb_monitor *); static void ovsdb_monitor_destroy(struct ovsdb_monitor *);
static struct ovsdb_monitor_change_set * ovsdb_monitor_add_change_set( static struct ovsdb_monitor_change_set * ovsdb_monitor_add_change_set(
struct ovsdb_monitor *, bool init_only); struct ovsdb_monitor *, bool init_only, const struct uuid *prev_txn);
static struct ovsdb_monitor_change_set * ovsdb_monitor_find_change_set(
const struct ovsdb_monitor *, const struct uuid *prev_txn);
static void ovsdb_monitor_change_set_destroy( static void ovsdb_monitor_change_set_destroy(
struct ovsdb_monitor_change_set *); struct ovsdb_monitor_change_set *);
static void ovsdb_monitor_track_new_change_set(struct ovsdb_monitor *); static void ovsdb_monitor_track_new_change_set(struct ovsdb_monitor *);
@@ -540,13 +545,14 @@ ovsdb_monitor_table_exists(struct ovsdb_monitor *m,
static struct ovsdb_monitor_change_set * static struct ovsdb_monitor_change_set *
ovsdb_monitor_add_change_set(struct ovsdb_monitor *dbmon, ovsdb_monitor_add_change_set(struct ovsdb_monitor *dbmon,
bool init_only) bool init_only, const struct uuid *prev_txn)
{ {
struct ovsdb_monitor_change_set *change_set = xzalloc(sizeof *change_set); struct ovsdb_monitor_change_set *change_set = xzalloc(sizeof *change_set);
change_set->uuid = uuid_random(); change_set->uuid = uuid_random();
ovs_list_push_back(&(dbmon->change_sets), &change_set->list_node); ovs_list_push_back(&(dbmon->change_sets), &change_set->list_node);
ovs_list_init(&change_set->change_set_for_tables); ovs_list_init(&change_set->change_set_for_tables);
change_set->n_refs = 1; change_set->n_refs = 1;
change_set->prev_txn = prev_txn ? *prev_txn : UUID_ZERO;
struct shash_node *node; struct shash_node *node;
SHASH_FOR_EACH (node, &dbmon->tables) { SHASH_FOR_EACH (node, &dbmon->tables) {
@@ -567,6 +573,33 @@ ovsdb_monitor_add_change_set(struct ovsdb_monitor *dbmon,
return change_set; return change_set;
}; };
static struct ovsdb_monitor_change_set *
ovsdb_monitor_find_change_set(const struct ovsdb_monitor *dbmon,
const struct uuid *prev_txn)
{
struct ovsdb_monitor_change_set *cs;
LIST_FOR_EACH (cs, list_node, &dbmon->change_sets) {
if (uuid_equals(&cs->prev_txn, prev_txn)) {
/* Check n_columns for each table in dbmon, in case it is changed
* after the change set is populated. */
bool n_col_is_equal = true;
struct ovsdb_monitor_change_set_for_table *mcst;
LIST_FOR_EACH (mcst, list_in_change_set,
&cs->change_set_for_tables) {
struct ovsdb_monitor_table *mt = mcst->mt;
if (mt->n_columns != mcst->n_columns) {
n_col_is_equal = false;
break;
}
}
if (n_col_is_equal) {
return cs;
}
}
}
return NULL;
}
static void static void
ovsdb_monitor_untrack_change_set(struct ovsdb_monitor *dbmon, ovsdb_monitor_untrack_change_set(struct ovsdb_monitor *dbmon,
struct ovsdb_monitor_change_set *mcs) struct ovsdb_monitor_change_set *mcs)
@@ -591,7 +624,8 @@ ovsdb_monitor_track_new_change_set(struct ovsdb_monitor *dbmon)
if (change_set) { if (change_set) {
change_set->n_refs++; change_set->n_refs++;
} else { } else {
change_set = ovsdb_monitor_add_change_set(dbmon, false); change_set = ovsdb_monitor_add_change_set(dbmon, false,
ovsdb_monitor_get_last_txnid(dbmon));
dbmon->new_change_set = change_set; dbmon->new_change_set = change_set;
} }
} }
@@ -1190,12 +1224,13 @@ ovsdb_monitor_get_update(
condition, condition,
ovsdb_monitor_compose_row_update); ovsdb_monitor_compose_row_update);
} else { } else {
ovs_assert(version == OVSDB_MONITOR_V2); ovs_assert(version == OVSDB_MONITOR_V2 ||
version == OVSDB_MONITOR_V3);
if (!cond_updated) { if (!cond_updated) {
json = ovsdb_monitor_compose_update(dbmon, initial, mcs, json = ovsdb_monitor_compose_update(dbmon, initial, mcs,
condition, condition,
ovsdb_monitor_compose_row_update2); ovsdb_monitor_compose_row_update2);
if (!condition || !condition->conditional) { if (!condition || !condition->conditional) {
ovsdb_monitor_json_cache_insert(dbmon, version, mcs, ovsdb_monitor_json_cache_insert(dbmon, version, mcs,
json); json);
@@ -1434,7 +1469,7 @@ ovsdb_monitor_get_initial(struct ovsdb_monitor *dbmon,
{ {
if (!dbmon->init_change_set) { if (!dbmon->init_change_set) {
struct ovsdb_monitor_change_set *change_set = struct ovsdb_monitor_change_set *change_set =
ovsdb_monitor_add_change_set(dbmon, true); ovsdb_monitor_add_change_set(dbmon, true, NULL);
dbmon->init_change_set = change_set; dbmon->init_change_set = change_set;
struct ovsdb_monitor_change_set_for_table *mcst; struct ovsdb_monitor_change_set_for_table *mcst;
@@ -1454,6 +1489,68 @@ ovsdb_monitor_get_initial(struct ovsdb_monitor *dbmon,
*p_mcs = dbmon->init_change_set; *p_mcs = dbmon->init_change_set;
} }
static bool
ovsdb_monitor_history_change_cb(const struct ovsdb_row *old,
const struct ovsdb_row *new,
const unsigned long int *changed,
void *aux)
{
struct ovsdb_monitor_change_set *change_set = aux;
struct ovsdb_table *table = new ? new->table : old->table;
struct ovsdb_monitor_change_set_for_table *mcst;
enum ovsdb_monitor_selection type =
ovsdb_monitor_row_update_type(false, old, new);
LIST_FOR_EACH (mcst, list_in_change_set,
&change_set->change_set_for_tables) {
if (mcst->mt->table == table) {
enum ovsdb_monitor_changes_efficacy efficacy =
ovsdb_monitor_changes_classify(type, mcst->mt, changed);
if (efficacy > OVSDB_CHANGES_NO_EFFECT) {
ovsdb_monitor_changes_update(old, new, mcst->mt, mcst);
}
return true;
}
}
return false;
}
void
ovsdb_monitor_get_changes_after(const struct uuid *txn_uuid,
struct ovsdb_monitor *dbmon,
struct ovsdb_monitor_change_set **p_mcs)
{
ovs_assert(*p_mcs == NULL);
ovs_assert(!uuid_is_zero(txn_uuid));
struct ovsdb_monitor_change_set *change_set =
ovsdb_monitor_find_change_set(dbmon, txn_uuid);
if (change_set) {
change_set->n_refs++;
*p_mcs = change_set;
return;
}
struct ovsdb_txn_history_node *h_node;
bool found = false;
LIST_FOR_EACH (h_node, node, &dbmon->db->txn_history) {
struct ovsdb_txn *txn = h_node->txn;
if (!found) {
/* find the txn with last_id in history */
if (uuid_equals(ovsdb_txn_get_txnid(txn), txn_uuid)) {
found = true;
change_set = ovsdb_monitor_add_change_set(dbmon, false,
txn_uuid);
}
} else {
/* Already found. Add changes in each follow up transaction to
* the new change_set. */
ovsdb_txn_for_each_change(txn, ovsdb_monitor_history_change_cb,
change_set);
}
}
*p_mcs = change_set;
}
void void
ovsdb_monitor_remove_jsonrpc_monitor(struct ovsdb_monitor *dbmon, ovsdb_monitor_remove_jsonrpc_monitor(struct ovsdb_monitor *dbmon,
struct ovsdb_jsonrpc_monitor *jsonrpc_monitor, struct ovsdb_jsonrpc_monitor *jsonrpc_monitor,
@@ -1692,3 +1789,15 @@ ovsdb_monitor_prereplace_db(struct ovsdb *db)
} }
} }
} }
const struct uuid *
ovsdb_monitor_get_last_txnid(struct ovsdb_monitor *dbmon) {
static struct uuid dummy = { .parts = { 0, 0, 0, 0 } };
if (dbmon->db->n_txn_history) {
struct ovsdb_txn_history_node *thn = CONTAINER_OF(
ovs_list_back(&dbmon->db->txn_history),
struct ovsdb_txn_history_node, node);
return ovsdb_txn_get_txnid(thn->txn);
}
return &dummy;
}

View File

@@ -39,6 +39,8 @@ enum ovsdb_monitor_version {
OVSDB_MONITOR_V1, /* RFC 7047 "monitor" method. */ OVSDB_MONITOR_V1, /* RFC 7047 "monitor" method. */
OVSDB_MONITOR_V2, /* Extension to RFC 7047, see ovsdb-server OVSDB_MONITOR_V2, /* Extension to RFC 7047, see ovsdb-server
man page for details. */ man page for details. */
OVSDB_MONITOR_V3, /* Extension to V2, see ovsdb-server man
page for details. */
/* Last entry. */ /* Last entry. */
OVSDB_MONITOR_VERSION_MAX OVSDB_MONITOR_VERSION_MAX
@@ -80,6 +82,8 @@ struct json *ovsdb_monitor_get_update(struct ovsdb_monitor *,
enum ovsdb_monitor_version, enum ovsdb_monitor_version,
struct ovsdb_monitor_change_set **p_mcs); struct ovsdb_monitor_change_set **p_mcs);
const struct uuid *ovsdb_monitor_get_last_txnid(struct ovsdb_monitor *);
void ovsdb_monitor_table_add_select(struct ovsdb_monitor *, void ovsdb_monitor_table_add_select(struct ovsdb_monitor *,
const struct ovsdb_table *, const struct ovsdb_table *,
enum ovsdb_monitor_selection); enum ovsdb_monitor_selection);
@@ -89,6 +93,8 @@ bool ovsdb_monitor_needs_flush(struct ovsdb_monitor *,
void ovsdb_monitor_get_initial(struct ovsdb_monitor *, void ovsdb_monitor_get_initial(struct ovsdb_monitor *,
struct ovsdb_monitor_change_set **); struct ovsdb_monitor_change_set **);
void ovsdb_monitor_get_changes_after(const struct uuid *txn_uuid,
struct ovsdb_monitor *, struct ovsdb_monitor_change_set **);
void ovsdb_monitor_get_memory_usage(struct simap *); void ovsdb_monitor_get_memory_usage(struct simap *);

View File

@@ -44,6 +44,9 @@ ovsdb\-client \- command-line interface to \fBovsdb-server\fR(1)
\fBovsdb\-client\fR [\fIoptions\fR] \fBmonitor\-cond\fR [\fIserver\fR] [\fIdatabase\fR] \fIconditions \fBovsdb\-client\fR [\fIoptions\fR] \fBmonitor\-cond\fR [\fIserver\fR] [\fIdatabase\fR] \fIconditions
\fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]... \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]...
.br .br
\fBovsdb\-client\fR [\fIoptions\fR] \fBmonitor\-cond\-since\fR [\fIserver\fR] [\fIdatabase\fR]
[\fIlast-id\fR] \fIconditions \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]...
.br
\fBovsdb\-client \fR[\fIoptions\fR] \fBwait\fR \fR[\fIserver\fR] \fIdatabase\fR \fIstate\fR \fBovsdb\-client \fR[\fIoptions\fR] \fBwait\fR \fR[\fIserver\fR] \fIdatabase\fR \fIstate\fR
.IP "Testing Commands:" .IP "Testing Commands:"
\fBovsdb\-client\fR [\fIoptions\fR] \fBlock\fR [\fIserver\fR] \fIlock\fR \fBovsdb\-client\fR [\fIoptions\fR] \fBlock\fR [\fIserver\fR] \fIlock\fR
@@ -243,6 +246,7 @@ with an error or succeed with surprising results.
. .
.IP "\fBmonitor\fR [\fIserver\fR] [\fIdatabase\fR] \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]..." .IP "\fBmonitor\fR [\fIserver\fR] [\fIdatabase\fR] \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]..."
.IQ "\fBmonitor\-cond\fR [\fIserver\fR] [\fIdatabase\fR] \fIconditions\fR \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]..." .IQ "\fBmonitor\-cond\fR [\fIserver\fR] [\fIdatabase\fR] \fIconditions\fR \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]..."
.IQ "\fBmonitor\-cond\-since\fR [\fIserver\fR] [\fIdatabase\fR] [\fIlast-id\fR] \fIconditions\fR \fItable\fR [\fIcolumn\fR[\fB,\fIcolumn\fR]...]..."
Connects to \fIserver\fR and monitors the contents of rows that match conditions in Connects to \fIserver\fR and monitors the contents of rows that match conditions in
\fItable\fR in \fIdatabase\fR. By default, the initial contents of \fItable\fR are \fItable\fR in \fIdatabase\fR. By default, the initial contents of \fItable\fR are
printed, followed by each change as it occurs. If conditions empty, printed, followed by each change as it occurs. If conditions empty,
@@ -269,13 +273,14 @@ line.
with the following change: A condition can be either a 3-element JSON array with the following change: A condition can be either a 3-element JSON array
as deescribed in the RFC or a boolean value.. as deescribed in the RFC or a boolean value..
.IP .IP
If \fB\-\-detach\fR is used with \fBmonitor\fR or \fBmonitor\-cond\fR, then If \fB\-\-detach\fR is used with \fBmonitor\fR, \fBmonitor\-cond\fR or
\fBovsdb\-client\fR detaches after it has successfully received and \fBmonitor\-cond\-since\fR, then \fBovsdb\-client\fR detaches after it has
printed the initial contents of \fItable\fR. successfully received and printed the initial contents of \fItable\fR.
.IP .IP
The \fBmonitor\fR command uses RFC 7047 "monitor" method to open a monitor The \fBmonitor\fR command uses RFC 7047 "monitor" method to open a monitor
session with the server. The \fBmonitor\-cond\fR command uses RFC 7047 session with the server. The \fBmonitor\-cond\fR and \fBmonitor\-cond\-since\fR
extension "monitor_cond" method. See \fBovsdb\-server\fR(1) for details. commandls uses RFC 7047 extension "monitor_cond" and "monitor_cond_since"
methods. See \fBovsdb\-server\fR(1) for details.
.IP "\fBmonitor\fI \fR[\fIserver\fR] \fR[\fIdatabase\fR] \fBALL\fR" .IP "\fBmonitor\fI \fR[\fIserver\fR] \fR[\fIdatabase\fR] \fBALL\fR"
Connects to \fIserver\fR and monitors the contents of all tables in Connects to \fIserver\fR and monitors the contents of all tables in
\fIdatabase\fR. Prints initial values and all kinds of changes to all \fIdatabase\fR. Prints initial values and all kinds of changes to all
@@ -340,10 +345,10 @@ The following options controlling output formatting:
.so lib/table.man .so lib/table.man
. .
.IP "\fB\-\-timestamp\fR" .IP "\fB\-\-timestamp\fR"
For the \fBmonitor\fR and \fBmonitor\-cond\fR commands, add a timestamp to each For the \fBmonitor\fR, \fBmonitor\-cond\fR and \fBmonitor\-cond\-since\fR
table update. Most output formats add the timestamp on a line of its own commands, add a timestamp to each table update. Most output formats add the
just above the table. The JSON output format puts the timestamp in a timestamp on a line of its own just above the table. The JSON output format
member of the top-level JSON object named \fBtime\fR. puts the timestamp in a member of the top-level JSON object named \fBtime\fR.
. .
.IP "\fB\-t\fR" .IP "\fB\-t\fR"
.IQ "\fB\-\-timeout=\fIsecs\fR" .IQ "\fB\-\-timeout=\fIsecs\fR"
@@ -352,8 +357,9 @@ seconds. If the timeout expires, \fBovsdb\-client\fR will exit with a
\fBSIGALRM\fR signal. \fBSIGALRM\fR signal.
. .
.SS "Daemon Options" .SS "Daemon Options"
The daemon options apply only to the \fBmonitor\fR and \fBmonitor\-cond\fR commands. The daemon options apply only to the \fBmonitor\fR, \fBmonitor\-cond\fR and
With any other command, they have no effect. \fBmonitor\-cond\-since\fR commands. With any other command, they have no
effect.
.ds DD .ds DD
.so lib/daemon.man .so lib/daemon.man
.SS "Logging Options" .SS "Logging Options"

View File

@@ -430,6 +430,13 @@ usage(void)
" DATABASE on SERVER.\n" " DATABASE on SERVER.\n"
" COLUMNs may include !initial, !insert, !delete, !modify\n" " COLUMNs may include !initial, !insert, !delete, !modify\n"
" to avoid seeing the specified kinds of changes.\n" " to avoid seeing the specified kinds of changes.\n"
"\n monitor-cond-since [SERVER] [DATABASE] [LASTID] CONDITION TABLE [COLUMN,...]...\n"
" monitor contents that match CONDITION of COLUMNs in TABLE in\n"
" DATABASE on SERVER, since change after LASTID.\n"
" LASTID specifies transaction ID after which the monitoring\n"
" starts, which works only for cluster mode. If ignored, it\n"
" defaults to an all-zero uuid.\n"
" Other arguments are the same as in monitor-cond.\n"
"\n convert [SERVER] SCHEMA\n" "\n convert [SERVER] SCHEMA\n"
" convert database on SERVER named in SCHEMA to SCHEMA.\n" " convert database on SERVER named in SCHEMA to SCHEMA.\n"
"\n needs-conversion [SERVER] SCHEMA\n" "\n needs-conversion [SERVER] SCHEMA\n"
@@ -1126,6 +1133,35 @@ monitor2_print(struct json *table_updates2,
} }
} }
static void
monitor3_print(struct json *result,
const struct monitored_table *mts, size_t n_mts)
{
if (result->type != JSON_ARRAY) {
ovs_error(0, "<result> is not array");
}
if (result->array.n != 3) {
ovs_error(0, "<result> should have 3 elements, but has %"PRIuSIZE".",
result->array.n);
}
bool found = json_boolean(result->array.elems[0]);
const char *last_id = json_string(result->array.elems[1]);
printf("found: %s, last_id: %s\n", found ? "true" : "false", last_id);
struct json *table_updates2 = result->array.elems[2];
monitor2_print(table_updates2, mts, n_mts);
}
static void
monitor3_notify_print(const char *last_id, struct json *table_updates2,
const struct monitored_table *mts, size_t n_mts)
{
printf("\nlast_id: %s", last_id);
monitor2_print(table_updates2, mts, n_mts);
}
static void static void
add_column(const char *server, const struct ovsdb_column *column, add_column(const char *server, const struct ovsdb_column *column,
struct ovsdb_column_set *columns, struct json *columns_json) struct ovsdb_column_set *columns, struct json *columns_json)
@@ -1333,7 +1369,8 @@ destroy_monitored_table(struct monitored_table *mts, size_t n)
static void static void
do_monitor__(struct jsonrpc *rpc, const char *database, do_monitor__(struct jsonrpc *rpc, const char *database,
enum ovsdb_monitor_version version, enum ovsdb_monitor_version version,
int argc, char *argv[], struct json *condition) int argc, char *argv[], struct json *condition,
const struct uuid *last_id)
{ {
const char *server = jsonrpc_get_name(rpc); const char *server = jsonrpc_get_name(rpc);
const char *table_name = argv[0]; const char *table_name = argv[0];
@@ -1411,8 +1448,24 @@ do_monitor__(struct jsonrpc *rpc, const char *database,
monitor = json_array_create_3(json_string_create(database), monitor = json_array_create_3(json_string_create(database),
json_null_create(), monitor_requests); json_null_create(), monitor_requests);
const char *method = version == OVSDB_MONITOR_V2 ? "monitor_cond" const char *method;
: "monitor"; switch (version) {
case OVSDB_MONITOR_V1:
method = "monitor";
break;
case OVSDB_MONITOR_V2:
method = "monitor_cond";
break;
case OVSDB_MONITOR_V3:
method = "monitor_cond_since";
struct json *json_last_id = json_string_create_nocopy(
xasprintf(UUID_FMT, UUID_ARGS(last_id)));
json_array_add(monitor, json_last_id);
break;
case OVSDB_MONITOR_VERSION_MAX:
default:
OVS_NOT_REACHED();
}
struct jsonrpc_msg *request; struct jsonrpc_msg *request;
request = jsonrpc_create_request(method, monitor, NULL); request = jsonrpc_create_request(method, monitor, NULL);
@@ -1444,6 +1497,9 @@ do_monitor__(struct jsonrpc *rpc, const char *database,
case OVSDB_MONITOR_V2: case OVSDB_MONITOR_V2:
monitor2_print(msg->result, mts, n_mts); monitor2_print(msg->result, mts, n_mts);
break; break;
case OVSDB_MONITOR_V3:
monitor3_print(msg->result, mts, n_mts);
break;
case OVSDB_MONITOR_VERSION_MAX: case OVSDB_MONITOR_VERSION_MAX:
default: default:
OVS_NOT_REACHED(); OVS_NOT_REACHED();
@@ -1469,6 +1525,17 @@ do_monitor__(struct jsonrpc *rpc, const char *database,
monitor2_print(params->array.elems[1], mts, n_mts); monitor2_print(params->array.elems[1], mts, n_mts);
fflush(stdout); fflush(stdout);
} }
} else if (msg->type == JSONRPC_NOTIFY
&& version == OVSDB_MONITOR_V3
&& !strcmp(msg->method, "update3")) {
struct json *params = msg->params;
if (params->type == JSON_ARRAY
&& params->array.n == 3
&& params->array.elems[0]->type == JSON_NULL) {
monitor3_notify_print(json_string(params->array.elems[1]),
params->array.elems[2], mts, n_mts);
fflush(stdout);
}
} else if (msg->type == JSONRPC_NOTIFY } else if (msg->type == JSONRPC_NOTIFY
&& !strcmp(msg->method, "monitor_canceled")) { && !strcmp(msg->method, "monitor_canceled")) {
ovs_fatal(0, "%s: %s database was removed", ovs_fatal(0, "%s: %s database was removed",
@@ -1500,12 +1567,13 @@ static void
do_monitor(struct jsonrpc *rpc, const char *database, do_monitor(struct jsonrpc *rpc, const char *database,
int argc, char *argv[]) int argc, char *argv[])
{ {
do_monitor__(rpc, database, OVSDB_MONITOR_V1, argc, argv, NULL); do_monitor__(rpc, database, OVSDB_MONITOR_V1, argc, argv, NULL, NULL);
} }
static void static void
do_monitor_cond(struct jsonrpc *rpc, const char *database, do_monitor_cond__(struct jsonrpc *rpc, const char *database,
int argc, char *argv[]) enum ovsdb_monitor_version version,
struct uuid *last_id, int argc, char *argv[])
{ {
struct ovsdb_condition cnd; struct ovsdb_condition cnd;
struct json *condition = NULL; struct json *condition = NULL;
@@ -1524,10 +1592,31 @@ do_monitor_cond(struct jsonrpc *rpc, const char *database,
check_ovsdb_error(ovsdb_condition_from_json(table, condition, check_ovsdb_error(ovsdb_condition_from_json(table, condition,
NULL, &cnd)); NULL, &cnd));
ovsdb_condition_destroy(&cnd); ovsdb_condition_destroy(&cnd);
do_monitor__(rpc, database, OVSDB_MONITOR_V2, --argc, ++argv, condition); do_monitor__(rpc, database, version, --argc, ++argv, condition,
last_id);
ovsdb_schema_destroy(schema); ovsdb_schema_destroy(schema);
} }
static void
do_monitor_cond(struct jsonrpc *rpc, const char *database,
int argc, char *argv[])
{
do_monitor_cond__(rpc, database, OVSDB_MONITOR_V2, NULL, argc, argv);
}
static void
do_monitor_cond_since(struct jsonrpc *rpc, const char *database,
int argc, char *argv[])
{
ovs_assert(argc > 1);
struct uuid last_id;
if (uuid_from_string(&last_id, argv[0])) {
argc--;
argv++;
}
do_monitor_cond__(rpc, database, OVSDB_MONITOR_V3, &last_id, argc, argv);
}
static bool static bool
is_database_clustered(struct jsonrpc *rpc, const char *database) is_database_clustered(struct jsonrpc *rpc, const char *database)
{ {
@@ -2409,6 +2498,7 @@ static const struct ovsdb_client_command all_commands[] = {
{ "query", NEED_NONE, 1, 2, do_query }, { "query", NEED_NONE, 1, 2, do_query },
{ "monitor", NEED_DATABASE, 1, INT_MAX, do_monitor }, { "monitor", NEED_DATABASE, 1, INT_MAX, do_monitor },
{ "monitor-cond", NEED_DATABASE, 2, 3, do_monitor_cond }, { "monitor-cond", NEED_DATABASE, 2, 3, do_monitor_cond },
{ "monitor-cond-since", NEED_DATABASE, 2, 4, do_monitor_cond_since },
{ "wait", NEED_NONE, 2, 3, do_wait }, { "wait", NEED_NONE, 2, 3, do_wait },
{ "convert", NEED_NONE, 1, 2, do_convert }, { "convert", NEED_NONE, 1, 2, do_convert },
{ "needs-conversion", NEED_NONE, 1, 2, do_needs_conversion }, { "needs-conversion", NEED_NONE, 1, 2, do_needs_conversion },

View File

@@ -120,6 +120,12 @@ ovsdb_txn_set_txnid(const struct uuid *txnid, struct ovsdb_txn *txn)
txn->txnid = *txnid; txn->txnid = *txnid;
} }
const struct uuid *
ovsdb_txn_get_txnid(const struct ovsdb_txn *txn)
{
return &txn->txnid;
}
static void static void
ovsdb_txn_free(struct ovsdb_txn *txn) ovsdb_txn_free(struct ovsdb_txn *txn)
{ {

View File

@@ -26,6 +26,7 @@ struct uuid;
struct ovsdb_txn *ovsdb_txn_create(struct ovsdb *); struct ovsdb_txn *ovsdb_txn_create(struct ovsdb *);
void ovsdb_txn_set_txnid(const struct uuid *, struct ovsdb_txn *); void ovsdb_txn_set_txnid(const struct uuid *, struct ovsdb_txn *);
const struct uuid *ovsdb_txn_get_txnid(const struct ovsdb_txn *);
void ovsdb_txn_abort(struct ovsdb_txn *); void ovsdb_txn_abort(struct ovsdb_txn *);
struct ovsdb_error *ovsdb_txn_replay_commit(struct ovsdb_txn *) struct ovsdb_error *ovsdb_txn_replay_commit(struct ovsdb_txn *)

View File

@@ -595,8 +595,8 @@ AT_SETUP(monitor-cond-change with many sessions pending)
AT_KEYWORDS([ovsdb server monitor monitor-cond negative]) AT_KEYWORDS([ovsdb server monitor monitor-cond negative])
ordinal_schema > schema ordinal_schema > schema
AT_CHECK([ovsdb-tool create db schema], [0], [stdout], [ignore]) AT_CHECK([ovsdb-tool create db schema], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log]) AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1]) AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`' on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals", for txn in m4_foreach([txn], [[[["ordinals",
@@ -657,3 +657,302 @@ row,action,name
<1>,insert,"""one""" <1>,insert,"""one"""
]], [ignore]) ]], [ignore])
AT_CLEANUP AT_CLEANUP
# Test monitor-cond-since with zero uuid, which shouldn't
# be found in server and server should send all rows
# as initial.
AT_SETUP([monitor-cond-since not found])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create-cluster db schema unix:db.raft], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 0, "name": "zero"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 1, "name": "one"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 2, "name": "two"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0], [ignore], [ignore])
done
# Omitting the last_id parameter in ovsdb-client monitor-cond-since command
# will by default using all zero uuid, which doesn't exist in any history txn.
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 10, "name": "ten"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 11, "name": "eleven"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0],
[ignore], [ignore], [kill `cat server-pid client-pid`])
done
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: false, last_id: <0>
row,action,name,number,_version
<1>,initial,"""one""",1,"[""uuid"",""<2>""]"
last_id: <3>
row,action,name,number,_version
<4>,insert,"""ten""",10,"[""uuid"",""<5>""]"
]], [ignore])
AT_CLEANUP
# Test monitor-cond-since in ovsdb server restart scenario.
# ovsdb-client should receive only new changes after the
# specific transaction id.
AT_SETUP([monitor-cond-since db restart])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create-cluster db schema unix:db.raft], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 0, "name": "zero"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 1, "name": "one"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 2, "name": "two"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0], [ignore], [ignore])
done
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
OVS_WAIT_UNTIL([grep last_id output])
kill `cat ovsdb-client.pid`
kill `cat ovsdb-server.pid`
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
# Remember the last_id, which will be used for monitor-cond-since later.
last_id=`grep last_id output | awk '{print $4}'`
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
# Some new changes made to db after restarting the server.
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 10, "name": "ten"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 11, "name": "eleven"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0],
[ignore], [ignore], [kill `cat server-pid client-pid`])
done
# Use last_id to monitor and get only the new changes.
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals $last_id '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: true, last_id: <0>
row,action,name,number,_version
<1>,insert,"""ten""",10,"[""uuid"",""<2>""]"
]], [ignore])
AT_CLEANUP
# Test monitor-cond-since with last_id found in server
# but there is no new change after that transaction.
AT_SETUP([monitor-cond-since found but no new rows])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create-cluster db schema unix:db.raft], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 0, "name": "zero"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 1, "name": "one"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 2, "name": "two"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0], [ignore], [ignore])
done
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
OVS_WAIT_UNTIL([grep last_id output])
kill `cat ovsdb-client.pid`
OVS_WAIT_UNTIL([test ! -e ovsdb-client.pid])
last_id=`grep last_id output | awk '{print $4}'`
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals $last_id '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: true, last_id: <0>
]], [ignore])
AT_CLEANUP
# Test monitor-cond-since against empty DB
AT_SETUP([monitor-cond-since empty db])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create-cluster db schema unix:db.raft], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
OVS_WAIT_UNTIL([grep last_id output])
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: false, last_id: <0>
]], [ignore])
AT_CLEANUP
# Test monitor-cond-since with cond-change followed.
AT_SETUP([monitor-cond-since condition change])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create-cluster db schema unix:db.raft], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 0, "name": "zero"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 1, "name": "one"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 2, "name": "two"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0], [ignore], [ignore])
done
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[]]' ordinals > output], [0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
for cond in m4_foreach([cond],
[[[[["name","==","one"],["name","==","two"]]]],
[[[["name","==","one"]]]],
[[[false]]],
[[[true]]]], ['cond' ]); do
AT_CHECK([ovs-appctl -t ovsdb-client ovsdb-client/cond_change ordinals "$cond"], [0], [ignore], [ignore])
done
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: false, last_id: <0>
row,action,name,number,_version
<1>,initial,"""one""",1,"[""uuid"",""<2>""]"
<3>,initial,"""two""",2,"[""uuid"",""<4>""]"
<5>,initial,"""zero""",,"[""uuid"",""<6>""]"
last_id: <0>
row,action,name,number,_version
<5>,delete,,,
last_id: <0>
row,action,name,number,_version
<3>,delete,,,
last_id: <0>
row,action,name,number,_version
<1>,delete,,,
last_id: <0>
row,action,name,number,_version
<1>,insert,"""one""",1,"[""uuid"",""<2>""]"
<3>,insert,"""two""",2,"[""uuid"",""<4>""]"
<5>,insert,"""zero""",,"[""uuid"",""<6>""]"
]], [ignore])
AT_CLEANUP
# Test monitor-cond-since with non-cluster mode server
AT_SETUP([monitor-cond-since non-cluster])
AT_KEYWORDS([ovsdb server monitor monitor-cond-since positive])
ordinal_schema > schema
AT_CHECK([ovsdb-tool create db schema], [0], [stdout], [ignore])
AT_CAPTURE_FILE([ovsdb-server-log])
AT_CHECK([ovsdb-server --detach --no-chdir --pidfile --remote=punix:socket --log-file="`pwd`"/ovsdb-server-log db >/dev/null 2>&1])
on_exit 'kill `cat ovsdb-server.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 0, "name": "zero"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 1, "name": "one"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 2, "name": "two"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0], [ignore], [ignore])
done
AT_CHECK([ovsdb-client -vjsonrpc --pidfile --detach --no-chdir -d json monitor-cond-since --format=csv unix:socket ordinals '[[["name","==","one"],["name","==","ten"]]]' ordinals > output],
[0], [ignore], [ignore])
on_exit 'kill `cat ovsdb-client.pid`'
for txn in m4_foreach([txn], [[[["ordinals",
{"op": "insert",
"table": "ordinals",
"row": {"number": 10, "name": "ten"}},
{"op": "insert",
"table": "ordinals",
"row": {"number": 11, "name": "eleven"}}]]]], ['txn' ]); do
AT_CHECK([ovsdb-client transact unix:socket "$txn"], [0],
[ignore], [ignore], [kill `cat server-pid client-pid`])
done
AT_CHECK([ovsdb-client transact unix:socket '[["ordinals"]]'], [0],
[ignore], [ignore])
AT_CHECK([ovs-appctl -t ovsdb-server -e exit], [0], [ignore], [ignore])
OVS_WAIT_UNTIL([test ! -e ovsdb-server.pid && test ! -e ovsdb-client.pid])
# Transaction shouldn't be found, and last_id returned should always
# be the same (all zero uuid)
AT_CHECK([$PYTHON $srcdir/ovsdb-monitor-sort.py < output | uuidfilt], [0],
[[found: false, last_id: <0>
row,action,name,number,_version
<1>,initial,"""one""",1,"[""uuid"",""<2>""]"
last_id: <0>
row,action,name,number,_version
<3>,insert,"""ten""",10,"[""uuid"",""<4>""]"
]], [ignore])
AT_CLEANUP