]> git.ipfire.org Git - thirdparty/nftables.git/commitdiff
src: introduce simple hints on incorrect object
authorPablo Neira Ayuso <pablo@netfilter.org>
Sat, 1 Dec 2018 10:33:31 +0000 (11:33 +0100)
committerPablo Neira Ayuso <pablo@netfilter.org>
Sat, 1 Dec 2018 10:35:34 +0000 (11:35 +0100)
 # nft add counter x test
 # nft list counter x test
 Error: No such file or directory; did you mean obj ‘test’ in table ip ‘x’?
 list counter x text
                ^^^^

Signed-off-by: Pablo Neira Ayuso <pablo@netfilter.org>
include/rule.h
src/evaluate.c
src/rule.c

index 85b9f80d04feb770a3f3f65ef0fbc1451ac7674a..88fed62e7cba329d0b040b24383b72f50e8319f0 100644 (file)
@@ -392,6 +392,9 @@ void obj_free(struct obj *obj);
 void obj_add_hash(struct obj *obj, struct table *table);
 struct obj *obj_lookup(const struct table *table, const char *name,
                       uint32_t type);
+struct obj *obj_lookup_fuzzy(const char *obj_name,
+                            const struct nft_cache *cache,
+                            const struct table **t);
 void obj_print(const struct obj *n, struct output_ctx *octx);
 void obj_print_plain(const struct obj *obj, struct output_ctx *octx);
 const char *obj_type_name(uint32_t type);
index 14c76420d524f8006538aacb5438b56d0e3d19a2..329fd42dd0a896c8f0364ddb69ce1ae51780fd67 100644 (file)
@@ -3450,6 +3450,23 @@ static int cmd_evaluate_get(struct eval_ctx *ctx, struct cmd *cmd)
        }
 }
 
+static int obj_not_found(struct eval_ctx *ctx, const struct location *loc,
+                        const char *obj_name)
+{
+       const struct table *table;
+       struct obj *obj;
+
+       obj = obj_lookup_fuzzy(obj_name, &ctx->nft->cache, &table);
+       if (obj == NULL)
+               return cmd_error(ctx, loc, "%s", strerror(ENOENT));
+
+       return cmd_error(ctx, loc,
+                        "%s; did you mean obj ‘%s’ in table %s ‘%s’?",
+                        strerror(ENOENT), obj->handle.obj.name,
+                                family2str(obj->handle.family),
+                                table->handle.table.name);
+}
+
 static int cmd_evaluate_list_obj(struct eval_ctx *ctx, const struct cmd *cmd,
                                 uint32_t obj_type)
 {
@@ -3463,9 +3480,9 @@ static int cmd_evaluate_list_obj(struct eval_ctx *ctx, const struct cmd *cmd,
                return table_not_found(ctx);
 
        if (obj_lookup(table, cmd->handle.obj.name, obj_type) == NULL)
-               return cmd_error(ctx, &cmd->handle.obj.location,
-                                "Could not process rule: %s",
-                                strerror(ENOENT));
+               return obj_not_found(ctx, &cmd->handle.obj.location,
+                                    cmd->handle.obj.name);
+
        return 0;
 }
 
index c244d0ba6b027149b2b9389e5c099f3a8a6b86fc..0a3c1970c83a459ee3cddd4ff83378d74c893ecb 100644 (file)
@@ -1710,6 +1710,30 @@ struct obj *obj_lookup(const struct table *table, const char *name,
        return NULL;
 }
 
+struct obj *obj_lookup_fuzzy(const char *obj_name,
+                            const struct nft_cache *cache,
+                            const struct table **t)
+{
+       struct string_misspell_state st;
+       struct table *table;
+       struct obj *obj;
+
+       string_misspell_init(&st);
+
+       list_for_each_entry(table, &cache->list, list) {
+               list_for_each_entry(obj, &table->objs, list) {
+                       if (!strcmp(obj->handle.obj.name, obj_name)) {
+                               *t = table;
+                               return obj;
+                       }
+                       if (string_misspell_update(obj->handle.obj.name,
+                                                  obj_name, obj, &st))
+                               *t = table;
+               }
+       }
+       return st.obj;
+}
+
 static void print_proto_name_proto(uint8_t l4, struct output_ctx *octx)
 {
        const struct protoent *p = getprotobynumber(l4);