]> git.ipfire.org Git - thirdparty/nftables.git/commitdiff
json: Make inet_service_type_json() respect literal level
authorPhil Sutter <phil@nwl.cc>
Tue, 11 Sep 2018 20:14:25 +0000 (22:14 +0200)
committerFlorian Westphal <fw@strlen.de>
Thu, 13 Sep 2018 08:42:11 +0000 (10:42 +0200)
This brings inet_service_type_json() on par with
inet_service_type_print(). Despite datatype_print()'s ability to use the
'print' callback, a dedicated 'json' callback is required to make port
numbers appear as numbers in JSON output instead of strings. Therefore
go with a bit of code duplication here.

Signed-off-by: Phil Sutter <phil@nwl.cc>
Signed-off-by: Florian Westphal <fw@strlen.de>
src/json.c

index bb1d0f2d1bfcdadffc923b1a74ffd2196a57a024..c0fc42263c1a0ba3c615460c1f7231753672de08 100644 (file)
@@ -857,7 +857,23 @@ json_t *inet_protocol_type_json(const struct expr *expr,
 
 json_t *inet_service_type_json(const struct expr *expr, struct output_ctx *octx)
 {
-       return json_integer(ntohs(mpz_get_be16(expr->value)));
+       struct sockaddr_in sin = {
+               .sin_family = AF_INET,
+               .sin_port = mpz_get_be16(expr->value),
+       };
+       char buf[NI_MAXSERV];
+
+       if (octx->literal < NFT_LITERAL_PORT ||
+           getnameinfo((struct sockaddr *)&sin, sizeof(sin),
+                       NULL, 0, buf, sizeof(buf), 0))
+               return json_integer(ntohs(sin.sin_port));
+
+       if (htons(atoi(buf)) == sin.sin_port ||
+           getnameinfo((struct sockaddr *)&sin, sizeof(sin),
+                       NULL, 0, buf, sizeof(buf), NI_DGRAM))
+               return json_integer(ntohs(sin.sin_port));
+
+       return json_string(buf);
 }
 
 json_t *mark_type_json(const struct expr *expr, struct output_ctx *octx)