switch (args->event) {
case NB_EV_VALIDATE:
ifname = yang_dnode_get_string(args->dnode, "./interface");
- if (ifname != NULL) {
+ nh_type = yang_dnode_get_enum(args->dnode, "./nh-type");
+ if (ifname != NULL && nh_type != STATIC_BLACKHOLE) {
if (strcasecmp(ifname, "Null0") == 0
|| strcasecmp(ifname, "reject") == 0
|| strcasecmp(ifname, "blackhole") == 0) {
{
struct static_nexthop *nh;
enum static_nh_type nh_type;
+ const char *nh_ifname;
+ const char *nh_vrf;
switch (args->event) {
case NB_EV_VALIDATE:
nh_type = yang_dnode_get_enum(args->dnode, "../nh-type");
+ nh_ifname = yang_dnode_get_string(args->dnode, "../interface");
+ nh_vrf = yang_dnode_get_string(args->dnode, "../vrf");
+ if (nh_ifname && nh_vrf) {
+ struct vrf *vrf = vrf_lookup_by_name(nh_vrf);
+ struct interface *ifp = if_lookup_by_name(nh_ifname,
+ vrf->vrf_id);
+
+ if (ifp && (!strmatch(nh_ifname, "blackhole") ||
+ !strmatch(nh_ifname, "reject"))) {
+ snprintf(args->errmsg, args->errmsg_len,
+ "nexthop interface name must be (reject, blackhole)");
+ return NB_ERR_VALIDATION;
+ }
+ }
if (nh_type != STATIC_BLACKHOLE) {
snprintf(args->errmsg, args->errmsg_len,
"nexthop type is not the blackhole type");
bool bfd_multi_hop;
const char *bfd_source;
const char *bfd_profile;
+
+ const char *input;
};
static int static_route_nb_run(struct vty *vty, struct static_route_args *args)
else
buf_gate_str = "";
- if (args->gateway == NULL && args->interface_name == NULL)
+ if (args->gateway == NULL && args->interface_name == NULL) {
type = STATIC_BLACKHOLE;
- else if (args->gateway && args->interface_name) {
+ /* If this is blackhole/reject flagged route, then
+ * specify interface_name with the value of what was really
+ * entered.
+ * interface_name will be validated later in NB functions
+ * to check if we don't create blackhole/reject routes that
+ * match the real interface names.
+ * E.g.: `ip route 10.0.0.1/32 bla` will create a blackhole
+ * route despite the real interface named `bla` exists.
+ */
+ if (args->input)
+ args->interface_name = args->input;
+ } else if (args->gateway && args->interface_name) {
if (args->afi == AFI_IP)
type = STATIC_IPV4_GATEWAY_IFNAME;
else
"Table to configure\n"
"The table number to configure\n")
{
+ int idx_flag = 0;
+
struct static_route_args args = {
.delete = !!no,
.afi = AFI_IP,
.vrf = vrf,
};
+ if (flag && argv_find(argv, argc, flag, &idx_flag))
+ args.input = argv[idx_flag]->arg;
+
return static_route_nb_run(vty, &args);
}
"Table to configure\n"
"The table number to configure\n")
{
+ int idx_flag = 0;
+
struct static_route_args args = {
.delete = !!no,
.afi = AFI_IP,
*/
assert(args.prefix);
+ if (flag && argv_find(argv, argc, flag, &idx_flag))
+ args.input = argv[idx_flag]->arg;
+
return static_route_nb_run(vty, &args);
}
"Table to configure\n"
"The table number to configure\n")
{
+ int idx_flag = 0;
+
struct static_route_args args = {
.delete = !!no,
.afi = AFI_IP6,
.vrf = vrf,
};
+ if (flag && argv_find(argv, argc, flag, &idx_flag))
+ args.input = argv[idx_flag]->arg;
+
return static_route_nb_run(vty, &args);
}
"Table to configure\n"
"The table number to configure\n")
{
+ int idx_flag = 0;
+
struct static_route_args args = {
.delete = !!no,
.afi = AFI_IP6,
*/
assert(args.prefix);
+ if (flag && argv_find(argv, argc, flag, &idx_flag))
+ args.input = argv[idx_flag]->arg;
+
return static_route_nb_run(vty, &args);
}