CertC++-ERR62

Detect errors when converting a string to a number

Required inputs: IR

The process of parsing an integer or floating-point number from a string can produce many errors. The string might not contain a number. It might contain a number of the correct type that is out of range (such as an integer that is larger than  INT_MAX). The string may also contain extra information after the number, which may or may not be useful after the conversion. These error conditions must be detected and addressed when a string-to-number conversion is performed using a formatted input stream such as std::istream or the locale facet  num_get<>.

When calling a formatted input stream function like  istream::operator>>(), information about conversion errors is queried through the  basic_ios::good()basic_ios::bad(), and  basic_ios::fail() inherited member functions or through exception handling if it is enabled on the stream object.

When calling  num_get<>::get(), information about conversion errors is returned to the caller through the  ios_base::iostate& argument. The C++ Standard, section [facet.num.get.virtuals], paragraph 3 [ ISO/IEC 14882-2014], in part, states the following:

If the conversion function fails to convert the entire field, or if the field represents a value outside the range of representable values, ios_base::failbit is assigned to err.

Always explicitly check the error state of a conversion from string to a numeric value (or handle the related exception, if applicable) instead of assuming the conversion results in a valid value. This rule is in addition to  ERR34-C. Detect errors when converting a string to a number, which bans the use of conversion functions that do not perform conversion validation such as  std::atoi() and  std::scanf() from the C Standard Library.

Noncompliant Code Example

In this noncompliant code example, multiple numeric values are converted from the standard input stream. However, if the text received from the standard input stream cannot be converted into a numeric value that can be represented by an int, the resulting value stored into the variables  i and  j may be unexpected.

#include <iostream>

void f() {
  int i, j;
  std::cin >> i >> j;
  // ...
}

For instance, if the text 12345678901234567890 is the first converted value read from the standard input stream, then  i will have the value  std::numeric_limits<int>::max() (per [facet.num.get.virtuals] paragraph 3), and  j will be uninitialized (per [istream.formatted.arithmetic] paragraph 3). If the text  abcdefg is the first converted value read from the standard input stream, then  i will have the value  0 and  j will remain uninitialized.

Compliant Solution

In this compliant solution, exceptions are enabled so that any conversion failure results in an exception being thrown. However, this approach cannot distinguish between which values are valid and which values are invalid and must assume that all values are invalid. Both the  badbit and  failbit flags are set to ensure that conversion errors as well as loss of integrity with the stream are treated as exceptions.

#include <iostream>

void f() {
  int i, j;

  std::cin.exceptions(std::istream::failbit | std::istream::badbit);
  try {
    std::cin >> i >> j;
    // ...
  } catch (std::istream::failure &E) {
    // Handle error
  }
}
Compliant Solution

In this compliant solution, each converted value read from the standard input stream is tested for validity before reading the next value in the sequence, allowing error recovery on a per-value basis. It checks  std::istream::fail() to see if the failure bit was set due to a conversion failure or whether the bad bit was set due to a loss of integrity with the stream object. If a failure condition is encountered, it is cleared on the input stream and then characters are read and discarded until a  ' ' (space) character occurs. The error handling in this case only works if a space character is what delimits the two numeric values to be converted.

#include <iostream>
#include <limits>

void f() {
  int i;
  std::cin >> i;
  if (std::cin.fail()) {
    // Handle failure to convert the value.
    std::cin.clear();
    std::cin.ignore(std::numeric_limits<std::streamsize>::max(), ' ');
  }

  int j;
  std::cin >> j;
  if (std::cin.fail()) {
    std::cin.clear();
    std::cin.ignore(std::numeric_limits<std::streamsize>::max(), ' ');
  }
 
  // ...
}
Risk Assessment

It is rare for a violation of this rule to result in a security  vulnerability unless it occurs in security-sensitive code. However, violations of this rule can easily result in lost or misinterpreted data. 

Rule Severity Likelihood Remediation Cost Priority Level
ERR62-CPP Medium Unlikely Medium P4 L3
Related Guidelines
SEI CERT C Coding Standard ERR34-C. Detect errors when converting a string to a number
MITRE CWE CWE-676, Use of potentially dangerous function
CWE-20, Insufficient input validation
Bibliography
[ ISO/IEC 9899:1999] Subclause 7.22.1, "Numeric conversion functions"
Subclause 7.21.6, "Formatted input/output functions"
[ ISO/IEC 14882-2014] Subclause 22.4.2.1.1, "num_get members"
Subclause 27.7.2.2, "Formatted input functions"
Excerpt from SEI CERT C++ Coding Standard [https://cmu-sei.github.io/secure-coding-standards/sei-cert-cpp-coding-standard/rules/exceptions-and-error-handling-err/err62-cpp], Copyright (C) 1995-2026 Carnegie Mellon University. See section 9.4. "3rd-Party Licenses" in the documentation for full details.

Possible Messages

Key

Text

Severity

Disabled

missing_basic_ios_fail

Use basic_ios::fail() after reading from input streams.

None

False

Options

check_functions

check_functions : set[bauhaus.analysis.config.QualifiedName] = {'std::basic_ios::fail', 'std::ios_base::fail'}

Fully qualified names of functions to check the state of the input stream.
 

functions_to_check

functions_to_check : set[bauhaus.analysis.config.QualifiedName] = {'std::basic_istream::operator>>'}

Fully qualified names of functions after which a call to one of the check functions is required.