* Restconf with startup feature will now copy all edit changes to startup db (as it should according to RFC 8040)
* See [Restconf does not handle startup datastore according to the RFC](https://github.com/clicon/clixon/issues/74) * Netconf Startup feature is no longer hardcoded, you need to explicitly enable it (See RFC 6241, Section 8.7) * Enable in config file with: `<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>`, or use `*:*`
This commit is contained in:
parent
161ef9c7b0
commit
6bf2a74e24
26 changed files with 270 additions and 128 deletions
|
|
@ -44,6 +44,9 @@
|
||||||
|
|
||||||
### API changes on existing features (you may need to change your code)
|
### API changes on existing features (you may need to change your code)
|
||||||
|
|
||||||
|
* Restconf with startup feature will now copy all edit changes to startup db (as it should according to RFC 8040)
|
||||||
|
* Netconf Startup feature is no longer hardcoded, you need to explicitly enable it (See RFC 6241, Section 8.7)
|
||||||
|
* Enable in config file with: `<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>`, or use `*:*`
|
||||||
* The directory `docker/system` has been moved to `docker/main`, to reflect that it runs the main example.
|
* The directory `docker/system` has been moved to `docker/main`, to reflect that it runs the main example.
|
||||||
* xmldb_get() removed "config" parameter:
|
* xmldb_get() removed "config" parameter:
|
||||||
* Change all calls to dbget from: `xmldb_get(h, db, xpath, 0|1, &xret, msd)` to `xmldb_get(h, db, xpath, &xret, msd)`
|
* Change all calls to dbget from: `xmldb_get(h, db, xpath, 0|1, &xret, msd)` to `xmldb_get(h, db, xpath, &xret, msd)`
|
||||||
|
|
@ -111,7 +114,7 @@
|
||||||
|
|
||||||
### Minor changes
|
### Minor changes
|
||||||
|
|
||||||
* A new "hello world" example is added
|
* A new minimal "hello world" example has been added
|
||||||
* Experimental customized error output strings, see [lib/clixon/clixon_err_string.h]
|
* Experimental customized error output strings, see [lib/clixon/clixon_err_string.h]
|
||||||
* Empty leaf values, eg <a></a> are now checked at validation.
|
* Empty leaf values, eg <a></a> are now checked at validation.
|
||||||
* Empty values were skipped in validation.
|
* Empty values were skipped in validation.
|
||||||
|
|
@ -138,6 +141,7 @@
|
||||||
* Added libgen.h for baseline()
|
* Added libgen.h for baseline()
|
||||||
|
|
||||||
### Corrected Bugs
|
### Corrected Bugs
|
||||||
|
* [Restconf does not handle startup datastore according to the RFC](https://github.com/clicon/clixon/issues/74)
|
||||||
* Failure in startup with -m startup or running left running_db cleared.
|
* Failure in startup with -m startup or running left running_db cleared.
|
||||||
* Running-db should not be changed on failure. Unless failure-db defined. Or if SEGV, etc. In those cases, tmp_db should include the original running-db.
|
* Running-db should not be changed on failure. Unless failure-db defined. Or if SEGV, etc. In those cases, tmp_db should include the original running-db.
|
||||||
* Backend plugin returning NULL was still installed - is now logged and skipped.
|
* Backend plugin returning NULL was still installed - is now logged and skipped.
|
||||||
|
|
|
||||||
|
|
@ -162,10 +162,12 @@ Clixon implements the following NETCONF proposals or standards:
|
||||||
The following RFC6241 capabilities/features are hardcoded in Clixon:
|
The following RFC6241 capabilities/features are hardcoded in Clixon:
|
||||||
- :candidate (RFC6241 8.3)
|
- :candidate (RFC6241 8.3)
|
||||||
- :validate (RFC6241 8.6)
|
- :validate (RFC6241 8.6)
|
||||||
- :startup (RFC6241 8.7)
|
|
||||||
- :xpath (RFC6241 8.9)
|
- :xpath (RFC6241 8.9)
|
||||||
- :notification: (RFC5277)
|
- :notification: (RFC5277)
|
||||||
|
|
||||||
|
The following features are optional and can be enabled by setting CLICON_FEATURE:
|
||||||
|
- :startup (RFC6241 8.7)
|
||||||
|
|
||||||
Clixon does not support the following netconf features:
|
Clixon does not support the following netconf features:
|
||||||
|
|
||||||
- :url capability
|
- :url capability
|
||||||
|
|
|
||||||
|
|
@ -616,6 +616,13 @@ main(int argc,
|
||||||
clicon_log(LOG_ERR, "Startup mode undefined. Specify option CLICON_STARTUP_MODE or specify -s option to clicon_backend.");
|
clicon_log(LOG_ERR, "Startup mode undefined. Specify option CLICON_STARTUP_MODE or specify -s option to clicon_backend.");
|
||||||
goto done;
|
goto done;
|
||||||
}
|
}
|
||||||
|
/* Check that netconf :startup is enabled */
|
||||||
|
if (startup_mode == SM_STARTUP &&
|
||||||
|
!if_feature(yspec, "ietf-netconf", "startup")){
|
||||||
|
clicon_log(LOG_ERR, "Startup mode selected but Netconf :startup feature is not enabled. Enable with option: <CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>");
|
||||||
|
goto done;
|
||||||
|
}
|
||||||
|
|
||||||
/* Init running db if it is not there
|
/* Init running db if it is not there
|
||||||
*/
|
*/
|
||||||
if (xmldb_exists(h, "running") != 1)
|
if (xmldb_exists(h, "running") != 1)
|
||||||
|
|
|
||||||
|
|
@ -485,6 +485,10 @@ main(int argc, char **argv)
|
||||||
if (yang_modules_init(h) < 0)
|
if (yang_modules_init(h) < 0)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
|
/* Add netconf yang spec, used as internal protocol */
|
||||||
|
if (netconf_module_load(h) < 0)
|
||||||
|
goto done;
|
||||||
|
|
||||||
/* Create tree generated from dataspec. If no other trees exists, this is
|
/* Create tree generated from dataspec. If no other trees exists, this is
|
||||||
* the only one.
|
* the only one.
|
||||||
* The following code creates the tree @datamodel
|
* The following code creates the tree @datamodel
|
||||||
|
|
|
||||||
|
|
@ -668,6 +668,11 @@ main(int argc,
|
||||||
/* Load yang module library, RFC7895 */
|
/* Load yang module library, RFC7895 */
|
||||||
if (yang_modules_init(h) < 0)
|
if (yang_modules_init(h) < 0)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
|
/* Add netconf yang spec, used as internal protocol */
|
||||||
|
if (netconf_module_load(h) < 0)
|
||||||
|
goto done;
|
||||||
|
|
||||||
/* Add system modules */
|
/* Add system modules */
|
||||||
if (clicon_option_bool(h, "CLICON_STREAM_DISCOVERY_RFC8040") &&
|
if (clicon_option_bool(h, "CLICON_STREAM_DISCOVERY_RFC8040") &&
|
||||||
yang_spec_parse_module(h, "ietf-restconf-monitoring", NULL, yspec)< 0)
|
yang_spec_parse_module(h, "ietf-restconf-monitoring", NULL, yspec)< 0)
|
||||||
|
|
@ -675,6 +680,11 @@ main(int argc,
|
||||||
if (clicon_option_bool(h, "CLICON_STREAM_DISCOVERY_RFC5277") &&
|
if (clicon_option_bool(h, "CLICON_STREAM_DISCOVERY_RFC5277") &&
|
||||||
yang_spec_parse_module(h, "clixon-rfc5277", NULL, yspec)< 0)
|
yang_spec_parse_module(h, "clixon-rfc5277", NULL, yspec)< 0)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
||||||
|
/* Dump configuration options on debug */
|
||||||
|
if (debug)
|
||||||
|
clicon_option_dump(h, debug);
|
||||||
|
|
||||||
/* Call start function in all plugins before we go interactive
|
/* Call start function in all plugins before we go interactive
|
||||||
*/
|
*/
|
||||||
if (clixon_plugin_start(h) < 0)
|
if (clixon_plugin_start(h) < 0)
|
||||||
|
|
|
||||||
|
|
@ -588,6 +588,25 @@ api_data_post(clicon_handle h,
|
||||||
goto done;
|
goto done;
|
||||||
goto ok;
|
goto ok;
|
||||||
}
|
}
|
||||||
|
if (if_feature(yspec, "ietf-netconf", "startup")){
|
||||||
|
/* RFC8040 Sec 1.4:
|
||||||
|
* If the NETCONF server supports :startup, the RESTCONF server MUST
|
||||||
|
* automatically update the non-volatile startup configuration
|
||||||
|
* datastore, after the "running" datastore has been altered as a
|
||||||
|
* consequence of a RESTCONF edit operation.
|
||||||
|
*/
|
||||||
|
cbuf_reset(cbx);
|
||||||
|
cprintf(cbx, "<rpc username=\"%s\">", NACM_RECOVERY_USER);
|
||||||
|
cprintf(cbx, "<copy-config><source><running/></source><target><startup/></target></copy-config></rpc>");
|
||||||
|
if (clicon_rpc_netconf(h, cbuf_get(cbx), &xretcom, NULL) < 0)
|
||||||
|
goto done;
|
||||||
|
/* If copy-config failed, log and ignore (already committed) */
|
||||||
|
if ((xe = xpath_first(xretcom, "//rpc-error")) != NULL){
|
||||||
|
|
||||||
|
clicon_log(LOG_WARNING, "%s: copy-config running->startup failed", __FUNCTION__);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
FCGX_SetExitStatus(201, r->out); /* Created */
|
FCGX_SetExitStatus(201, r->out); /* Created */
|
||||||
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
||||||
FCGX_FPrintF(r->out, "\r\n");
|
FCGX_FPrintF(r->out, "\r\n");
|
||||||
|
|
@ -914,6 +933,24 @@ api_data_put(clicon_handle h,
|
||||||
goto done;
|
goto done;
|
||||||
goto ok;
|
goto ok;
|
||||||
}
|
}
|
||||||
|
if (if_feature(yspec, "ietf-netconf", "startup")){
|
||||||
|
/* RFC8040 Sec 1.4:
|
||||||
|
* If the NETCONF server supports :startup, the RESTCONF server MUST
|
||||||
|
* automatically update the non-volatile startup configuration
|
||||||
|
* datastore, after the "running" datastore has been altered as a
|
||||||
|
* consequence of a RESTCONF edit operation.
|
||||||
|
*/
|
||||||
|
cbuf_reset(cbx);
|
||||||
|
cprintf(cbx, "<rpc username=\"%s\">", NACM_RECOVERY_USER);
|
||||||
|
cprintf(cbx, "<copy-config><source><running/></source><target><startup/></target></copy-config></rpc>");
|
||||||
|
if (clicon_rpc_netconf(h, cbuf_get(cbx), &xretcom, NULL) < 0)
|
||||||
|
goto done;
|
||||||
|
/* If copy-config failed, log and ignore (already committed) */
|
||||||
|
if ((xe = xpath_first(xretcom, "//rpc-error")) != NULL){
|
||||||
|
|
||||||
|
clicon_log(LOG_WARNING, "%s: copy-config running->startup failed", __FUNCTION__);
|
||||||
|
}
|
||||||
|
}
|
||||||
FCGX_SetExitStatus(201, r->out); /* Created */
|
FCGX_SetExitStatus(201, r->out); /* Created */
|
||||||
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
||||||
FCGX_FPrintF(r->out, "\r\n");
|
FCGX_FPrintF(r->out, "\r\n");
|
||||||
|
|
@ -1071,6 +1108,24 @@ api_data_delete(clicon_handle h,
|
||||||
goto done;
|
goto done;
|
||||||
goto ok;
|
goto ok;
|
||||||
}
|
}
|
||||||
|
if (if_feature(yspec, "ietf-netconf", "startup")){
|
||||||
|
/* RFC8040 Sec 1.4:
|
||||||
|
* If the NETCONF server supports :startup, the RESTCONF server MUST
|
||||||
|
* automatically update the non-volatile startup configuration
|
||||||
|
* datastore, after the "running" datastore has been altered as a
|
||||||
|
* consequence of a RESTCONF edit operation.
|
||||||
|
*/
|
||||||
|
cbuf_reset(cbx);
|
||||||
|
cprintf(cbx, "<rpc username=\"%s\">", NACM_RECOVERY_USER);
|
||||||
|
cprintf(cbx, "<copy-config><source><running/></source><target><startup/></target></copy-config></rpc>");
|
||||||
|
if (clicon_rpc_netconf(h, cbuf_get(cbx), &xretcom, NULL) < 0)
|
||||||
|
goto done;
|
||||||
|
/* If copy-config failed, log and ignore (already committed) */
|
||||||
|
if ((xe = xpath_first(xretcom, "//rpc-error")) != NULL){
|
||||||
|
|
||||||
|
clicon_log(LOG_WARNING, "%s: copy-config running->startup failed", __FUNCTION__);
|
||||||
|
}
|
||||||
|
}
|
||||||
FCGX_SetExitStatus(201, r->out);
|
FCGX_SetExitStatus(201, r->out);
|
||||||
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
FCGX_FPrintF(r->out, "Content-Type: text/plain\r\n");
|
||||||
FCGX_FPrintF(r->out, "\r\n");
|
FCGX_FPrintF(r->out, "\r\n");
|
||||||
|
|
|
||||||
11
doc/FAQ.md
11
doc/FAQ.md
|
|
@ -298,6 +298,17 @@ The example below shows enabling a specific feature; enabling all features in mo
|
||||||
Features can be probed by using RFC 7895 Yang module library which provides
|
Features can be probed by using RFC 7895 Yang module library which provides
|
||||||
information on all modules and which features are enabled.
|
information on all modules and which features are enabled.
|
||||||
|
|
||||||
|
Clixon have three hardcoded features:
|
||||||
|
- :candidate (RFC6241 8.3)
|
||||||
|
- :validate (RFC6241 8.6)
|
||||||
|
- :xpath (RFC6241 8.9)
|
||||||
|
|
||||||
|
You can select the startup feature by including it in the config file:
|
||||||
|
```
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
|
```
|
||||||
|
(or just `ietf-netconf:*`).
|
||||||
|
|
||||||
## Can I run Clixon as a container?
|
## Can I run Clixon as a container?
|
||||||
|
|
||||||
Yes, Clixon has two examples on how to build docker containers. A [base](../docker/base) image and a complete [example system](../docker/system).
|
Yes, Clixon has two examples on how to build docker containers. A [base](../docker/base) image and a complete [example system](../docker/system).
|
||||||
|
|
|
||||||
|
|
@ -1,6 +1,6 @@
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>/usr/local/etc/example.xml</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>/usr/local/etc/example.xml</CLICON_CONFIGFILE>
|
||||||
<CLICON_FEATURE>*:*</CLICON_FEATURE>
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MODULE_MAIN>clixon-example</CLICON_YANG_MODULE_MAIN>
|
<CLICON_YANG_MODULE_MAIN>clixon-example</CLICON_YANG_MODULE_MAIN>
|
||||||
<CLICON_CLI_MODE>example</CLICON_CLI_MODE>
|
<CLICON_CLI_MODE>example</CLICON_CLI_MODE>
|
||||||
|
|
|
||||||
|
|
@ -70,18 +70,6 @@ int clicon_config_yang_set(clicon_handle h, yang_stmt *ys);
|
||||||
cxobj *clicon_conf_xml(clicon_handle h);
|
cxobj *clicon_conf_xml(clicon_handle h);
|
||||||
int clicon_conf_xml_set(clicon_handle h, cxobj *x);
|
int clicon_conf_xml_set(clicon_handle h, cxobj *x);
|
||||||
|
|
||||||
#ifdef XXX
|
|
||||||
plghndl_t clicon_xmldb_plugin_get(clicon_handle h);
|
|
||||||
int clicon_xmldb_plugin_set(clicon_handle h, plghndl_t handle);
|
|
||||||
|
|
||||||
void *clicon_xmldb_api_get(clicon_handle h);
|
|
||||||
int clicon_xmldb_api_set(clicon_handle h, void *xa_api);
|
|
||||||
|
|
||||||
|
|
||||||
void *clicon_xmldb_handle_get(clicon_handle h);
|
|
||||||
int clicon_xmldb_handle_set(clicon_handle h, void *xh);
|
|
||||||
#endif
|
|
||||||
|
|
||||||
db_elmnt *clicon_db_elmnt_get(clicon_handle h, const char *db);
|
db_elmnt *clicon_db_elmnt_get(clicon_handle h, const char *db);
|
||||||
int clicon_db_elmnt_set(clicon_handle h, const char *db, db_elmnt *xc);
|
int clicon_db_elmnt_set(clicon_handle h, const char *db, db_elmnt *xc);
|
||||||
|
|
||||||
|
|
|
||||||
|
|
@ -184,6 +184,7 @@ yang_stmt *yang_choice(yang_stmt *y);
|
||||||
int yang_order(yang_stmt *y);
|
int yang_order(yang_stmt *y);
|
||||||
int yang_print(FILE *f, yang_stmt *yn);
|
int yang_print(FILE *f, yang_stmt *yn);
|
||||||
int yang_print_cbuf(cbuf *cb, yang_stmt *yn, int marginal);
|
int yang_print_cbuf(cbuf *cb, yang_stmt *yn, int marginal);
|
||||||
|
int if_feature(yang_stmt *yspec, char *module, char *feature);
|
||||||
int ys_populate(yang_stmt *ys, void *arg);
|
int ys_populate(yang_stmt *ys, void *arg);
|
||||||
yang_stmt *yang_parse_file(int fd, const char *name, yang_stmt *ysp);
|
yang_stmt *yang_parse_file(int fd, const char *name, yang_stmt *ysp);
|
||||||
int yang_apply(yang_stmt *yn, enum rfc_6020 key, yang_applyfn_t fn,
|
int yang_apply(yang_stmt *yn, enum rfc_6020 key, yang_applyfn_t fn,
|
||||||
|
|
|
||||||
|
|
@ -214,104 +214,6 @@ clicon_conf_xml_set(clicon_handle h,
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef XXX
|
|
||||||
/*! Get xmldb datastore plugin handle, as used by dlopen/dlsym/dlclose */
|
|
||||||
plghndl_t
|
|
||||||
clicon_xmldb_plugin_get(clicon_handle h)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
size_t len;
|
|
||||||
void *p;
|
|
||||||
|
|
||||||
if ((p = hash_value(cdat, "xmldb_plugin", &len)) != NULL)
|
|
||||||
return *(plghndl_t*)p;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! Set xmldb datastore plugin handle, as used by dlopen/dlsym/dlclose */
|
|
||||||
int
|
|
||||||
clicon_xmldb_plugin_set(clicon_handle h,
|
|
||||||
plghndl_t handle)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
|
|
||||||
if (hash_add(cdat, "xmldb_plugin", &handle, sizeof(void*)) == NULL)
|
|
||||||
return -1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*! Get XMLDB API struct pointer
|
|
||||||
* @param[in] h Clicon handle
|
|
||||||
* @retval xa XMLDB API struct
|
|
||||||
* @note xa is really of type struct xmldb_api*
|
|
||||||
*/
|
|
||||||
void *
|
|
||||||
clicon_xmldb_api_get(clicon_handle h)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
size_t len;
|
|
||||||
void *xa;
|
|
||||||
|
|
||||||
if ((xa = hash_value(cdat, "xmldb_api", &len)) != NULL)
|
|
||||||
return *(void**)xa;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! Set or reset XMLDB API struct pointer
|
|
||||||
* @param[in] h Clicon handle
|
|
||||||
* @param[in] xa XMLDB API struct
|
|
||||||
* @note xa is really of type struct xmldb_api*
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
clicon_xmldb_api_set(clicon_handle h,
|
|
||||||
void *xa)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
|
|
||||||
/* It is the pointer to xa_api that should be copied by hash,
|
|
||||||
so we send a ptr to the ptr to indicate what to copy.
|
|
||||||
*/
|
|
||||||
if (hash_add(cdat, "xmldb_api", &xa, sizeof(void*)) == NULL)
|
|
||||||
return -1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
/*! Get XMLDB storage handle
|
|
||||||
* @param[in] h Clicon handle
|
|
||||||
* @retval xh XMLDB storage handle. If not connected return NULL
|
|
||||||
*/
|
|
||||||
void *
|
|
||||||
clicon_xmldb_handle_get(clicon_handle h)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
size_t len;
|
|
||||||
void *xh;
|
|
||||||
|
|
||||||
if ((xh = hash_value(cdat, "xmldb_handle", &len)) != NULL)
|
|
||||||
return *(void**)xh;
|
|
||||||
return NULL;
|
|
||||||
}
|
|
||||||
|
|
||||||
/*! Set or reset XMLDB storage handle
|
|
||||||
* @param[in] h Clicon handle
|
|
||||||
* @param[in] xh XMLDB storage handle. If NULL reset it
|
|
||||||
* @note Just keep note of it, dont allocate it or so.
|
|
||||||
*/
|
|
||||||
int
|
|
||||||
clicon_xmldb_handle_set(clicon_handle h,
|
|
||||||
void *xh)
|
|
||||||
{
|
|
||||||
clicon_hash_t *cdat = clicon_data(h);
|
|
||||||
|
|
||||||
if (hash_add(cdat, "xmldb_handle", &xh, sizeof(void*)) == NULL)
|
|
||||||
return -1;
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
#endif /* XXX */
|
|
||||||
|
|
||||||
|
|
||||||
/*! Get authorized user name
|
/*! Get authorized user name
|
||||||
* @param[in] h Clicon handle
|
* @param[in] h Clicon handle
|
||||||
* @retval xh XMLDB storage handle. If not connected return NULL
|
* @retval xh XMLDB storage handle. If not connected return NULL
|
||||||
|
|
|
||||||
|
|
@ -1030,14 +1030,8 @@ netconf_module_load(clicon_handle h)
|
||||||
goto done;
|
goto done;
|
||||||
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:validate</CLICON_FEATURE>", yspec, &xc) < 0)
|
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:validate</CLICON_FEATURE>", yspec, &xc) < 0)
|
||||||
goto done;
|
goto done;
|
||||||
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>", yspec, &xc) < 0)
|
|
||||||
goto done;
|
|
||||||
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:xpath</CLICON_FEATURE>", yspec, &xc) < 0)
|
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:xpath</CLICON_FEATURE>", yspec, &xc) < 0)
|
||||||
goto done;
|
goto done;
|
||||||
#ifdef NYI
|
|
||||||
if (xml_parse_string("<CLICON_FEATURE>ietf-netconf:confirmed-commit</CLICON_FEATURE>", yspec, &xc) < 0)
|
|
||||||
goto done;
|
|
||||||
#endif
|
|
||||||
/* Load yang spec */
|
/* Load yang spec */
|
||||||
if (yang_spec_parse_module(h, "ietf-netconf", NULL, yspec)< 0)
|
if (yang_spec_parse_module(h, "ietf-netconf", NULL, yspec)< 0)
|
||||||
goto done;
|
goto done;
|
||||||
|
|
|
||||||
|
|
@ -1591,6 +1591,33 @@ ys_populate_identity(yang_stmt *ys,
|
||||||
return retval;
|
return retval;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*! Return 1 if feature is enabled, 0 if not using the populated yang tree
|
||||||
|
*
|
||||||
|
* @param[in] yspec yang specification
|
||||||
|
* @param[in] module Name of module
|
||||||
|
* @param[in] feature Name of feature
|
||||||
|
* @retval 0 Not found or not set
|
||||||
|
* @retval 1 Found and set
|
||||||
|
* XXX: should the in-param be h, ymod, or yspec?
|
||||||
|
*/
|
||||||
|
int
|
||||||
|
if_feature(yang_stmt *yspec,
|
||||||
|
char *module,
|
||||||
|
char *feature)
|
||||||
|
{
|
||||||
|
yang_stmt *ym; /* module */
|
||||||
|
yang_stmt *yf; /* feature */
|
||||||
|
cg_var *cv;
|
||||||
|
|
||||||
|
if ((ym = yang_find_module_by_name(yspec, module)) == NULL)
|
||||||
|
return 0;
|
||||||
|
if ((yf = yang_find(ym, Y_FEATURE, feature)) == NULL)
|
||||||
|
return 0;
|
||||||
|
if ((cv = yang_cv_get(yf)) == NULL)
|
||||||
|
return 0;
|
||||||
|
return cv_bool_get(cv);
|
||||||
|
}
|
||||||
|
|
||||||
/*! Populate yang feature statement - set cv to 1 if enabled
|
/*! Populate yang feature statement - set cv to 1 if enabled
|
||||||
*
|
*
|
||||||
* @param[in] ys Feature yang statement to populate.
|
* @param[in] ys Feature yang statement to populate.
|
||||||
|
|
@ -1608,6 +1635,8 @@ ys_populate_feature(clicon_handle h,
|
||||||
char *module;
|
char *module;
|
||||||
char *feature;
|
char *feature;
|
||||||
cxobj *xc;
|
cxobj *xc;
|
||||||
|
char *m;
|
||||||
|
char *f;
|
||||||
|
|
||||||
/* get clicon config file in xml form */
|
/* get clicon config file in xml form */
|
||||||
if ((x = clicon_conf_xml(h)) == NULL)
|
if ((x = clicon_conf_xml(h)) == NULL)
|
||||||
|
|
@ -1620,11 +1649,12 @@ ys_populate_feature(clicon_handle h,
|
||||||
feature = ys->ys_argument;
|
feature = ys->ys_argument;
|
||||||
xc = NULL;
|
xc = NULL;
|
||||||
while ((xc = xml_child_each(x, xc, CX_ELMNT)) != NULL && found == 0) {
|
while ((xc = xml_child_each(x, xc, CX_ELMNT)) != NULL && found == 0) {
|
||||||
char *m = NULL;
|
m = NULL;
|
||||||
char *f = NULL;
|
f = NULL;
|
||||||
if (strcmp(xml_name(xc), "CLICON_FEATURE") != 0)
|
if (strcmp(xml_name(xc), "CLICON_FEATURE") != 0)
|
||||||
continue;
|
continue;
|
||||||
/* get m and f from configuration feature rules */
|
/* CLICON_FEATURE is on the form <module>:<feature>.
|
||||||
|
* Split on colon to get module(m) and feature(f) respectively */
|
||||||
if (nodeid_split(xml_body(xc), &m, &f) < 0)
|
if (nodeid_split(xml_body(xc), &m, &f) < 0)
|
||||||
goto done;
|
goto done;
|
||||||
if (m && f &&
|
if (m && f &&
|
||||||
|
|
|
||||||
|
|
@ -32,6 +32,7 @@ cfg=$dir/conf_yang.xml
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_MODULE_SET_ID>42</CLICON_MODULE_SET_ID>
|
<CLICON_MODULE_SET_ID>42</CLICON_MODULE_SET_ID>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
||||||
|
|
|
||||||
|
|
@ -158,7 +158,7 @@ fi
|
||||||
|
|
||||||
# Note order of features in ietf-netconf yang is alphabetically: candidate, startup, validate, xpath
|
# Note order of features in ietf-netconf yang is alphabetically: candidate, startup, validate, xpath
|
||||||
new "netconf module ietf-netconf"
|
new "netconf module ietf-netconf"
|
||||||
expect="<module><name>ietf-netconf</name><revision>2011-06-01</revision><namespace>urn:ietf:params:xml:ns:netconf:base:1.0</namespace><feature>candidate</feature><feature>validate</feature><feature>startup</feature><feature>xpath</feature><conformance-type>implement</conformance-type></module>"
|
expect="<module><name>ietf-netconf</name><revision>2011-06-01</revision><namespace>urn:ietf:params:xml:ns:netconf:base:1.0</namespace><feature>candidate</feature><feature>validate</feature><feature>xpath</feature><conformance-type>implement</conformance-type></module>"
|
||||||
match=`echo "$ret" | grep -GZo "$expect"`
|
match=`echo "$ret" | grep -GZo "$expect"`
|
||||||
if [ -z "$match" ]; then
|
if [ -z "$match" ]; then
|
||||||
err "$expect" "$ret"
|
err "$expect" "$ret"
|
||||||
|
|
|
||||||
|
|
@ -13,6 +13,7 @@ fyang=$dir/nacm-example.yang
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MAIN_FILE>$fyang</CLICON_YANG_MAIN_FILE>
|
<CLICON_YANG_MAIN_FILE>$fyang</CLICON_YANG_MAIN_FILE>
|
||||||
|
|
|
||||||
|
|
@ -37,6 +37,7 @@ fyang=$dir/nacm-example.yang
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MAIN_FILE>$fyang</CLICON_YANG_MAIN_FILE>
|
<CLICON_YANG_MAIN_FILE>$fyang</CLICON_YANG_MAIN_FILE>
|
||||||
|
|
|
||||||
|
|
@ -14,6 +14,7 @@ tmp=$dir/tmp.x
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_MODULE_SET_ID>42</CLICON_MODULE_SET_ID>
|
<CLICON_MODULE_SET_ID>42</CLICON_MODULE_SET_ID>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
||||||
|
|
|
||||||
|
|
@ -73,12 +73,12 @@ expecteq "$(curl -s -H 'Accept: application/yang-data+xml' -G http://localhost/r
|
||||||
|
|
||||||
# Should be alphabetically ordered
|
# Should be alphabetically ordered
|
||||||
new "restconf get restconf/operations. RFC8040 3.3.2 (json)"
|
new "restconf get restconf/operations. RFC8040 3.3.2 (json)"
|
||||||
expecteq "$(curl -sG http://localhost/restconf/operations)" 0 '{"operations": {"clixon-example:client-rpc": null,"clixon-example:empty": null,"clixon-example:optional": null,"clixon-example:example": null,"clixon-lib:debug": null}}
|
expecteq "$(curl -sG http://localhost/restconf/operations)" 0 '{"operations": {"clixon-example:client-rpc": null,"clixon-example:empty": null,"clixon-example:optional": null,"clixon-example:example": null,"clixon-lib:debug": null,"ietf-netconf:get-config": null,"ietf-netconf:edit-config": null,"ietf-netconf:copy-config": null,"ietf-netconf:delete-config": null,"ietf-netconf:lock": null,"ietf-netconf:unlock": null,"ietf-netconf:get": null,"ietf-netconf:close-session": null,"ietf-netconf:kill-session": null,"ietf-netconf:commit": null,"ietf-netconf:discard-changes": null,"ietf-netconf:validate": null,"clixon-rfc5277:create-subscription": null}}
|
||||||
'
|
'
|
||||||
|
|
||||||
new "restconf get restconf/operations. RFC8040 3.3.2 (xml)"
|
new "restconf get restconf/operations. RFC8040 3.3.2 (xml)"
|
||||||
ret=$(curl -s -H "Accept: application/yang-data+xml" -G http://localhost/restconf/operations)
|
ret=$(curl -s -H "Accept: application/yang-data+xml" -G http://localhost/restconf/operations)
|
||||||
expect='<operations><client-rpc xmlns="urn:example:clixon"/><empty xmlns="urn:example:clixon"/><optional xmlns="urn:example:clixon"/><example xmlns="urn:example:clixon"/><debug xmlns="http://clicon.org/lib"/></operations>'
|
expect='<operations><client-rpc xmlns="urn:example:clixon"/><empty xmlns="urn:example:clixon"/><optional xmlns="urn:example:clixon"/><example xmlns="urn:example:clixon"/><debug xmlns="http://clicon.org/lib"/><get-config xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><edit-config xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><copy-config xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><delete-config xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><lock xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><unlock xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><get xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><close-session xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><kill-session xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><commit xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><discard-changes xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><validate xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"/><create-subscription xmlns="urn:ietf:params:xml:ns:netmod:notification"/></operations>'
|
||||||
match=`echo $ret | grep -EZo "$expect"`
|
match=`echo $ret | grep -EZo "$expect"`
|
||||||
if [ -z "$match" ]; then
|
if [ -z "$match" ]; then
|
||||||
err "$expect" "$ret"
|
err "$expect" "$ret"
|
||||||
|
|
|
||||||
126
test/test_restconf_startup.sh
Executable file
126
test/test_restconf_startup.sh
Executable file
|
|
@ -0,0 +1,126 @@
|
||||||
|
#!/bin/bash
|
||||||
|
# Test restconf :startup
|
||||||
|
# RFC 8040 Sec 1.4 says:
|
||||||
|
# the NETCONF server supports :startup, the RESTCONF server MUST
|
||||||
|
# automatically update the non-volatile startup configuration
|
||||||
|
# datastore, after the "running" datastore has been altered as a
|
||||||
|
# consequence of a RESTCONF edit operation.
|
||||||
|
# Magic line must be first in script (see README.md)
|
||||||
|
s="$_" ; . ./lib.sh || if [ "$s" = $0 ]; then exit 0; else return 0; fi
|
||||||
|
|
||||||
|
APPNAME=example
|
||||||
|
|
||||||
|
cfg=$dir/conf.xml
|
||||||
|
fyang=$dir/example.yang
|
||||||
|
|
||||||
|
cat <<EOF > $fyang
|
||||||
|
module example{
|
||||||
|
yang-version 1.1;
|
||||||
|
namespace "urn:example:clixon";
|
||||||
|
prefix ip;
|
||||||
|
container x {
|
||||||
|
list y {
|
||||||
|
key "a";
|
||||||
|
leaf a {
|
||||||
|
type string;
|
||||||
|
}
|
||||||
|
leaf b {
|
||||||
|
type string;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
EOF
|
||||||
|
|
||||||
|
# Use yang in example
|
||||||
|
|
||||||
|
cat <<EOF > $cfg
|
||||||
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
|
<CLICON_BACKEND_DIR>/usr/local/lib/$APPNAME/backend</CLICON_BACKEND_DIR>
|
||||||
|
<CLICON_BACKEND_REGEXP>example_backend.so$</CLICON_BACKEND_REGEXP>
|
||||||
|
<CLICON_RESTCONF_DIR>/usr/local/lib/$APPNAME/restconf</CLICON_RESTCONF_DIR>
|
||||||
|
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
||||||
|
<CLICON_BACKEND_PIDFILE>/usr/local/var/$APPNAME/$APPNAME.pidfile</CLICON_BACKEND_PIDFILE>
|
||||||
|
<CLICON_XMLDB_DIR>$dir</CLICON_XMLDB_DIR>
|
||||||
|
</clixon-config>
|
||||||
|
EOF
|
||||||
|
|
||||||
|
testrun(){
|
||||||
|
option=$1
|
||||||
|
|
||||||
|
new "test params: -f $cfg -y $fyang $option"
|
||||||
|
if [ $BE -ne 0 ]; then
|
||||||
|
new "kill old backend"
|
||||||
|
sudo clixon_backend -zf $cfg
|
||||||
|
if [ $? -ne 0 ]; then
|
||||||
|
err
|
||||||
|
fi
|
||||||
|
new "start backend -s init -f $cfg -y $fyang $option"
|
||||||
|
start_backend -s init -f $cfg -y $fyang $option
|
||||||
|
fi
|
||||||
|
|
||||||
|
new "kill old restconf daemon"
|
||||||
|
sudo pkill -u www-data clixon_restconf
|
||||||
|
|
||||||
|
new "start restconf daemon"
|
||||||
|
start_restconf -f $cfg -y $fyang $option
|
||||||
|
|
||||||
|
new "waiting"
|
||||||
|
sleep $RCWAIT
|
||||||
|
|
||||||
|
new "restconf put 42"
|
||||||
|
expecteq "$(curl -s -X PUT http://localhost/restconf/data/example:x/y=42 -d '{"example:y":{"a":"42","b":"42"}}')" 0 ""
|
||||||
|
|
||||||
|
new "restconf put 99"
|
||||||
|
expecteq "$(curl -s -X PUT http://localhost/restconf/data/example:x/y=99 -d '{"example:y":{"a":"99","b":"99"}}')" 0 ""
|
||||||
|
|
||||||
|
new "restconf post 123"
|
||||||
|
expecteq "$(curl -s -X POST http://localhost/restconf/data/example:x -d '{"example:y":{"a":"123","b":"123"}}')" 0 ""
|
||||||
|
|
||||||
|
new "restconf delete 42"
|
||||||
|
expecteq "$(curl -s -X DELETE http://localhost/restconf/data/example:x/y=42)" 0 ""
|
||||||
|
|
||||||
|
new "Kill restconf daemon"
|
||||||
|
stop_restconf
|
||||||
|
|
||||||
|
if [ $BE -ne 0 ]; then
|
||||||
|
new "Kill backend"
|
||||||
|
# Check if premature kill
|
||||||
|
pid=`pgrep -u root -f clixon_backend`
|
||||||
|
if [ -z "$pid" ]; then
|
||||||
|
err "backend already dead"
|
||||||
|
fi
|
||||||
|
# kill backend
|
||||||
|
stop_backend -f $cfg
|
||||||
|
fi
|
||||||
|
}
|
||||||
|
|
||||||
|
# clear startup
|
||||||
|
sudo rm -f $dir/startup_db;
|
||||||
|
|
||||||
|
new "Run with startup option, check running is copied"
|
||||||
|
testrun "-o CLICON_FEATURE=ietf-netconf:startup"
|
||||||
|
|
||||||
|
new "Check running and startup exists and are same"
|
||||||
|
if [ ! -f $dir/startup_db ]; then
|
||||||
|
err "startup should exist but does not"
|
||||||
|
fi
|
||||||
|
echo "diff $dir/startup_db $dir/running_db"
|
||||||
|
d=$(sudo diff $dir/startup_db $dir/running_db)
|
||||||
|
if [ -n "$d" ]; then
|
||||||
|
err "running and startup should be equal" "$d"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# clear startup
|
||||||
|
sudo rm -f $dir/startup_db;
|
||||||
|
|
||||||
|
new "Run without startup option, check running is copied"
|
||||||
|
testrun ""
|
||||||
|
|
||||||
|
new "Check startup should not exist"
|
||||||
|
if [ -f $dir/startup_db ]; then
|
||||||
|
err "startup should not exist"
|
||||||
|
fi
|
||||||
|
rm -rf $dir
|
||||||
|
|
@ -127,7 +127,7 @@ new "restconf wrong method"
|
||||||
expecteq "$(curl -s -X POST -d '{"clixon-example:input":{"x":"0"}}' http://localhost/restconf/operations/clixon-example:wrong)" 0 '{"ietf-restconf:errors" : {"error": {"error-type": "application","error-tag": "missing-element","error-info": {"bad-element": "wrong"},"error-severity": "error","error-message": "RPC not defined"}}}
'
|
expecteq "$(curl -s -X POST -d '{"clixon-example:input":{"x":"0"}}' http://localhost/restconf/operations/clixon-example:wrong)" 0 '{"ietf-restconf:errors" : {"error": {"error-type": "application","error-tag": "missing-element","error-info": {"bad-element": "wrong"},"error-severity": "error","error-message": "RPC not defined"}}}
'
|
||||||
|
|
||||||
new "restconf example missing input"
|
new "restconf example missing input"
|
||||||
expecteq "$(curl -s -X POST -d '{"clixon-example:input":null}' http://localhost/restconf/operations/ietf-netconf:edit-config)" 0 '{"ietf-restconf:errors" : {"error": {"error-type": "protocol","error-tag": "operation-failed","error-severity": "error","error-message": "yang module not found"}}}
'
|
expecteq "$(curl -s -X POST -d '{"clixon-example:input":null}' http://localhost/restconf/operations/ietf-netconf:edit-config)" 0 '{"ietf-restconf:errors" : {"error": {"error-type": "application","error-tag": "missing-element","error-info": {"bad-element": "target"},"error-severity": "error","error-message": "Mandatory variable"}}}
'
|
||||||
|
|
||||||
new "netconf kill-session missing session-id mandatory"
|
new "netconf kill-session missing session-id mandatory"
|
||||||
expecteof "$clixon_netconf -qf $cfg" 0 '<rpc xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"><kill-session/></rpc>]]>]]>' '^<rpc-reply><rpc-error><error-type>application</error-type><error-tag>missing-element</error-tag><error-info><bad-element>session-id</bad-element></error-info><error-severity>error</error-severity><error-message>Mandatory variable</error-message></rpc-error></rpc-reply>]]>]]>$'
|
expecteof "$clixon_netconf -qf $cfg" 0 '<rpc xmlns="urn:ietf:params:xml:ns:netconf:base:1.0"><kill-session/></rpc>]]>]]>' '^<rpc-reply><rpc-error><error-type>application</error-type><error-tag>missing-element</error-tag><error-info><bad-element>session-id</bad-element></error-info><error-severity>error</error-severity><error-message>Mandatory variable</error-message></rpc-error></rpc-reply>]]>]]>$'
|
||||||
|
|
|
||||||
|
|
@ -8,7 +8,7 @@
|
||||||
# - startup db starts with a "start" interface
|
# - startup db starts with a "start" interface
|
||||||
# There is also an "invalid" XML and a "broken" XML
|
# There is also an "invalid" XML and a "broken" XML
|
||||||
# There are two steps, first run through everything OK
|
# There are two steps, first run through everything OK
|
||||||
# Then try with invalid and borken XML and ensure the backend quits and all is untouched
|
# Then try with invalid and broken XML and ensure the backend quits and all is untouched
|
||||||
|
|
||||||
# Magic line must be first in script (see README.md)
|
# Magic line must be first in script (see README.md)
|
||||||
s="$_" ; . ./lib.sh || if [ "$s" = $0 ]; then exit 0; else return 0; fi
|
s="$_" ; . ./lib.sh || if [ "$s" = $0 ]; then exit 0; else return 0; fi
|
||||||
|
|
@ -20,6 +20,7 @@ cfg=$dir/conf_startup.xml
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>$IETFRFC</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MODULE_MAIN>clixon-example</CLICON_YANG_MODULE_MAIN>
|
<CLICON_YANG_MODULE_MAIN>clixon-example</CLICON_YANG_MODULE_MAIN>
|
||||||
|
|
@ -65,7 +66,6 @@ testrun(){
|
||||||
edb=$4 # extradb at start
|
edb=$4 # extradb at start
|
||||||
exprun=$5 # expected running_db after startup
|
exprun=$5 # expected running_db after startup
|
||||||
|
|
||||||
|
|
||||||
sudo rm -f $dir/*_db
|
sudo rm -f $dir/*_db
|
||||||
echo "<config>$rdb</config>" > $dir/running_db
|
echo "<config>$rdb</config>" > $dir/running_db
|
||||||
echo "<config>$sdb</config>" > $dir/startup_db
|
echo "<config>$sdb</config>" > $dir/startup_db
|
||||||
|
|
|
||||||
|
|
@ -94,6 +94,7 @@ EOF
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
||||||
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
||||||
|
|
|
||||||
|
|
@ -173,6 +173,7 @@ XML='<system xmlns="urn:example:a"><a>dont change me</a><c>rename me</c><host-na
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
||||||
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
||||||
|
|
|
||||||
|
|
@ -232,6 +232,7 @@ EOF
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_FEATURE>interfaces:if-mib</CLICON_FEATURE>
|
<CLICON_FEATURE>interfaces:if-mib</CLICON_FEATURE>
|
||||||
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
||||||
|
|
|
||||||
|
|
@ -57,6 +57,7 @@ EOF
|
||||||
cat <<EOF > $cfg
|
cat <<EOF > $cfg
|
||||||
<clixon-config xmlns="http://clicon.org/config">
|
<clixon-config xmlns="http://clicon.org/config">
|
||||||
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
<CLICON_CONFIGFILE>$cfg</CLICON_CONFIGFILE>
|
||||||
|
<CLICON_FEATURE>ietf-netconf:startup</CLICON_FEATURE>
|
||||||
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
<CLICON_YANG_DIR>/usr/local/share/clixon</CLICON_YANG_DIR>
|
||||||
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
<CLICON_YANG_MAIN_DIR>$dir</CLICON_YANG_MAIN_DIR>
|
||||||
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
<CLICON_SOCK>/usr/local/var/$APPNAME/$APPNAME.sock</CLICON_SOCK>
|
||||||
|
|
|
||||||
Loading…
Add table
Add a link
Reference in a new issue