diff options
Diffstat (limited to 'src/libserver/css')
-rw-r--r-- | src/libserver/css/css_tokeniser.cxx | 66 |
1 files changed, 34 insertions, 32 deletions
diff --git a/src/libserver/css/css_tokeniser.cxx b/src/libserver/css/css_tokeniser.cxx index dc570d64b..e44c14011 100644 --- a/src/libserver/css/css_tokeniser.cxx +++ b/src/libserver/css/css_tokeniser.cxx @@ -20,6 +20,7 @@ #include "frozen/unordered_map.h" #include "frozen/string.h" #include <string> +#include <cmath> namespace rspamd::css { @@ -29,8 +30,8 @@ namespace rspamd::css { * This helper is intended to create tokens either with a tag and value * or with just a tag. */ -template<css_parser_token::token_type T, typename ...Args> -auto make_token(const Args&... args) -> css_parser_token; +template<css_parser_token::token_type T, class Arg> +auto make_token(const Arg &arg) -> css_parser_token; template<> auto make_token<css_parser_token::token_type::string_token, std::string_view>(const std::string_view &s) @@ -76,7 +77,7 @@ auto make_token<css_parser_token::token_type::delim_token, char>(const char &c) template<> auto make_token<css_parser_token::token_type::number_token, float>(const float &d) --> css_parser_token + -> css_parser_token { return css_parser_token{css_parser_token::token_type::number_token, d}; } @@ -360,51 +361,52 @@ auto css_tokeniser::consume_number() -> struct css_parser_token } if (i > offset) { - float num; - /* I wish it was supported properly */ //auto conv_res = std::from_chars(&input[offset], &input[i], num); char numbuf[128], *endptr = NULL; rspamd_strlcpy(numbuf, &input[offset], MIN(i - offset + 1, sizeof(numbuf))); - num = g_ascii_strtod(numbuf, &endptr); + auto num = g_ascii_strtod(numbuf, &endptr); + offset = i; - if (endptr && *endptr != '\0') { + if ((endptr && *endptr != '\0') || num >= G_MAXFLOAT || num <= G_MINFLOAT || isnan(num)) { msg_debug_css("invalid number: %s", numbuf); + return make_token<css_parser_token::token_type::delim_token>(input[i - 1]); } - offset = i; + else { - auto ret = make_token<css_parser_token::token_type::number_token>(num); + auto ret = make_token<css_parser_token::token_type::number_token>(static_cast<float>(num)); - if (i < input.size()) { - if (input[i] == '%') { - ret.flags |= css_parser_token::number_percent; - i ++; + if (i < input.size()) { + if (input[i] == '%') { + ret.flags |= css_parser_token::number_percent; + i++; - offset = i; - } - else if (is_plain_ident_start(input[i])) { - auto dim_token = consume_ident(); - - if (dim_token.type == css_parser_token::token_type::ident_token) { - if (!ret.adjust_dim(dim_token)) { - auto sv = std::get<std::string_view>(dim_token.value); - msg_debug_css("cannot apply dimension from the token %*s; number value = %.1f", - (int)sv.size(), sv.begin(), num); - /* Unconsume ident */ - offset = i; + offset = i; + } + else if (is_plain_ident_start(input[i])) { + auto dim_token = consume_ident(); + + if (dim_token.type == css_parser_token::token_type::ident_token) { + if (!ret.adjust_dim(dim_token)) { + auto sv = std::get<std::string_view>(dim_token.value); + msg_debug_css("cannot apply dimension from the token %*s; number value = %.1f", + (int) sv.size(), sv.begin(), num); + /* Unconsume ident */ + offset = i; + } + } + else { + /* We have no option but to uncosume ident token in this case */ + msg_debug_css("got invalid ident like token after number, unconsume it"); } } else { - /* We have no option but to uncosume ident token in this case */ - msg_debug_css("got invalid ident like token after number, unconsume it"); + /* Plain number, nothing to do */ } } - else { - /* Plain number, nothing to do */ - } - } - return ret; + return ret; + } } else { msg_err_css("internal error: invalid number, empty token"); |