Skip to content

Commit 79cc598

Browse files
committed
Fix #13144 (regression: misra.json has stopped working since 2.10)
1 parent 4903a41 commit 79cc598

9 files changed

+68
-47
lines changed

addons/misra.py

+26-34
Original file line numberDiff line numberDiff line change
@@ -4353,9 +4353,8 @@ def loadRuleTexts(self, filename):
43534353
num1 = 0
43544354
num2 = 0
43554355
appendixA = False
4356-
expect_more = False
43574356

4358-
Rule_pattern = re.compile(r'^Rule ([0-9]+).([0-9]+)')
4357+
Rule_pattern = re.compile(r'^Rule ([0-9]+)\.([0-9]+)')
43594358
severity_pattern = re.compile(r'.*[ ]*(Advisory|Required|Mandatory)$')
43604359
xA_Z_pattern = re.compile(r'^[#A-Z].*')
43614360
a_z_pattern = re.compile(r'^[a-z].*')
@@ -4383,70 +4382,63 @@ def loadRuleTexts(self, filename):
43834382
file_stream = open(filename, 'rt')
43844383

43854384
rule = None
4386-
have_severity = False
4387-
severity_loc = 0
4385+
rule_line_number = 0
43884386

43894387
for line in file_stream:
43904388

4391-
line = line.replace('\r', '').replace('\n', '')
4389+
line = line.strip()
4390+
if len(line) == 0:
4391+
expect_more = False
4392+
continue
43924393

43934394
if not appendixA:
43944395
if line.find('Appendix A') >= 0 and line.find('Summary of guidelines') >= 10:
43954396
appendixA = True
43964397
continue
43974398
if line.find('Appendix B') >= 0:
43984399
break
4399-
if len(line) == 0:
4400-
continue
44014400

44024401
# Parse rule declaration.
44034402
res = Rule_pattern.match(line)
44044403

44054404
if res:
4406-
have_severity = False
4407-
expect_more = False
4408-
severity_loc = 0
4405+
rule_line_number = 0
44094406
num1 = int(res.group(1))
44104407
num2 = int(res.group(2))
44114408
rule = Rule(num1, num2)
44124409

4413-
if not have_severity and rule is not None:
44144410
res = severity_pattern.match(line)
4415-
44164411
if res:
44174412
rule.misra_severity = res.group(1)
4418-
have_severity = True
4419-
else:
4420-
severity_loc += 1
4413+
rule_line_number = 1
4414+
continue
4415+
4416+
if rule is None:
4417+
continue
44214418

4422-
# Only look for severity on the Rule line
4423-
# or the next non-blank line after
4424-
# If it's not in either of those locations then
4425-
# assume a severity was not provided.
4419+
rule_line_number += 1
44264420

4427-
if severity_loc < 2:
4421+
if rule_line_number == 1:
4422+
res = severity_pattern.match(line)
4423+
4424+
if res:
4425+
rule.misra_severity = res.group(1)
44284426
continue
44294427

4430-
rule.misra_severity = ''
4431-
have_severity = True
4428+
rule_line_number = 2
44324429

4433-
if rule is None:
4430+
# Parse beginning of rule text.
4431+
if not rule.text and xA_Z_pattern.match(line):
4432+
rule.text = line.strip()
4433+
self.ruleTexts[rule.num] = rule
44344434
continue
44354435

44364436
# Parse continuing of rule text.
4437-
if expect_more:
4438-
if a_z_pattern.match(line):
4439-
self.ruleTexts[rule.num].text += ' ' + line
4440-
continue
4441-
4442-
expect_more = False
4437+
if a_z_pattern.match(line):
4438+
self.ruleTexts[rule.num].text += ' ' + line.strip()
44434439
continue
44444440

4445-
# Parse beginning of rule text.
4446-
if xA_Z_pattern.match(line):
4447-
rule.text = line
4448-
self.ruleTexts[rule.num] = rule
4449-
expect_more = True
4441+
rule = None
44504442

44514443
file_stream.close()
44524444

addons/test/misra/misra_rules_structure.txt

+4-1
Original file line numberDiff line numberDiff line change
@@ -11,7 +11,10 @@ Here we go:
1111
Appendix A Summary of guidelines
1212

1313
Rule 1.2
14-
Rule text.
14+
Rule text.
15+
16+
Rule 2.1
17+
Rule text for 2.1.
1518

1619
Stop parsing after this line:
1720
Appendix B

addons/test/misra_test.py

+4-7
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,6 @@
1-
# Running the test with Python 2:
2-
# Be sure to install pytest version 4.6.4 (newer should also work)
3-
# Command in cppcheck directory:
4-
# python -m pytest addons/test/test-misra.py
5-
#
61
# Running the test with Python 3:
72
# Command in cppcheck directory:
8-
# PYTHONPATH=./addons python3 -m pytest addons/test/test-misra.py
3+
# PYTHONPATH=./addons python3 -m pytest addons/test/misra_test.py
94

105
import pytest
116
import re
@@ -48,6 +43,8 @@ def test_loadRuleTexts_structure(checker):
4843
assert(checker.ruleTexts.get(101, None) is None)
4944
assert(checker.ruleTexts[102].text == "Rule text.")
5045
assert(checker.ruleTexts.get(103, None) is None)
46+
assert(checker.ruleTexts[201].text == "Rule text for 2.1.")
47+
assert(checker.ruleTexts.get(202, None) is None)
5148

5249

5350
def test_loadRuleTexts_empty_lines(checker):
@@ -62,7 +59,7 @@ def test_loadRuleTexts_mutiple_lines(checker):
6259
assert(checker.ruleTexts[102].text == "Multiple lines text.")
6360
assert(checker.ruleTexts[103].text == "Multiple lines text.")
6461
assert(checker.ruleTexts[104].text == "Should")
65-
assert(checker.ruleTexts[105].text == "Should")
62+
assert(checker.ruleTexts[105].text == "Should starts from lowercase letter.")
6663
assert(checker.ruleTexts[106].text == "Can contain empty lines.")
6764

6865

cli/cmdlineparser.cpp

+2-2
Original file line numberDiff line numberDiff line change
@@ -340,9 +340,9 @@ CmdLineParser::Result CmdLineParser::parseFromArgs(int argc, const char* const a
340340
return Result::Fail;
341341
{
342342
XMLErrorMessagesLogger xmlLogger;
343-
std::cout << ErrorMessage::getXMLHeader(mSettings.cppcheckCfgProductName, mSettings.xml_version);
343+
std::cout << ErrorMessage::getXMLHeader(mSettings.cppcheckCfgProductName, 2);
344344
CppCheck::getErrorMessages(xmlLogger);
345-
std::cout << ErrorMessage::getXMLFooter() << std::endl;
345+
std::cout << ErrorMessage::getXMLFooter(2) << std::endl;
346346
}
347347
return Result::Exit;
348348
}

cli/cppcheckexecutor.cpp

+7
Original file line numberDiff line numberDiff line change
@@ -274,6 +274,7 @@ namespace {
274274
* @brief Write the checkers report
275275
*/
276276
void writeCheckersReport();
277+
void writeXmlCheckersReport();
277278

278279
bool hasCriticalErrors() const {
279280
return !mCriticalErrors.empty();
@@ -464,6 +465,8 @@ int CppCheckExecutor::check_internal(const Settings& settings) const
464465

465466
if (settings.safety || settings.severity.isEnabled(Severity::information) || !settings.checkersReportFilename.empty())
466467
stdLogger.writeCheckersReport();
468+
if (settings.xml_version == 3)
469+
stdLogger.writeXmlCheckersReport();
467470

468471
if (settings.xml) {
469472
stdLogger.reportErr(ErrorMessage::getXMLFooter(settings.xml_version));
@@ -512,9 +515,13 @@ void StdLogger::writeCheckersReport()
512515
if (fout.is_open())
513516
fout << checkersReport.getReport(mCriticalErrors);
514517
}
518+
}
515519

520+
void StdLogger::writeXmlCheckersReport()
521+
{
516522
if (mSettings.xml && mSettings.xml_version == 3) {
517523
reportErr(" </errors>\n");
524+
const CheckersReport checkersReport(mSettings, mActiveCheckers);
518525
reportErr(checkersReport.getXmlReport(mCriticalErrors));
519526
}
520527
}

lib/checkersreport.cpp

+4
Original file line numberDiff line numberDiff line change
@@ -40,6 +40,10 @@ static int getMisraCVersion(const Settings& settings) {
4040
return 2023;
4141
if (settings.addons.count("misra"))
4242
return 2012;
43+
for (const AddonInfo& addonInfo: settings.addonInfos) {
44+
if (addonInfo.name == "misra")
45+
return 2012;
46+
}
4347
return 0;
4448
}
4549

lib/errorlogger.h

+1-1
Original file line numberDiff line numberDiff line change
@@ -143,7 +143,7 @@ class CPPCHECKLIB ErrorMessage {
143143
std::string toXML() const;
144144

145145
static std::string getXMLHeader(std::string productName, int xmlVersion = 2);
146-
static std::string getXMLFooter(int xmlVersion = 2);
146+
static std::string getXMLFooter(int xmlVersion);
147147

148148
/**
149149
* Format the error message into a string.

test/cli/other_test.py

+13
Original file line numberDiff line numberDiff line change
@@ -1970,6 +1970,19 @@ def test_checkers_report(tmpdir):
19701970
assert '--checkers-report' not in stderr
19711971

19721972

1973+
def test_checkers_report_misra_json(tmpdir):
1974+
"""check that misra checkers are reported properly when --addon=misra.json is used"""
1975+
test_file = os.path.join(tmpdir, 'test.c')
1976+
with open(test_file, 'wt') as f:
1977+
f.write('x=1;')
1978+
misra_json = os.path.join(tmpdir, 'misra.json')
1979+
with open(misra_json, 'wt') as f:
1980+
f.write('{"script":"misra.py"}')
1981+
exitcode, stdout, stderr = cppcheck('--enable=style --addon=misra.json --xml-version=3 test.c'.split(), cwd=tmpdir)
1982+
assert exitcode == 0, stdout
1983+
assert '<checker id="Misra C 2012: 8.1"/>' in stderr
1984+
1985+
19731986
def test_ignore(tmpdir):
19741987
os.mkdir(os.path.join(tmpdir, 'src'))
19751988
test_file = os.path.join(tmpdir, 'src', 'test.cpp')

test/testerrorlogger.cpp

+7-2
Original file line numberDiff line numberDiff line change
@@ -54,6 +54,7 @@ class TestErrorLogger : public TestFixture {
5454
TEST_CASE(ToXmlV2Locations);
5555
TEST_CASE(ToXmlV2Encoding);
5656
TEST_CASE(FromXmlV2);
57+
TEST_CASE(ToXmlV3);
5758

5859
// Inconclusive results in xml reports..
5960
TEST_CASE(InconclusiveXml);
@@ -232,7 +233,7 @@ class TestErrorLogger : public TestFixture {
232233
header += CppCheck::version();
233234
header += "\"/>\n <errors>";
234235
ASSERT_EQUALS(header, ErrorMessage::getXMLHeader(""));
235-
ASSERT_EQUALS(" </errors>\n</results>", ErrorMessage::getXMLFooter());
236+
ASSERT_EQUALS(" </errors>\n</results>", ErrorMessage::getXMLFooter(2));
236237
std::string message(" <error id=\"errorId\" severity=\"error\"");
237238
message += " msg=\"Programming error.\" verbose=\"Verbose error\">\n";
238239
message += " <location file=\"foo.cpp\" line=\"5\" column=\"1\"/>\n </error>";
@@ -253,7 +254,7 @@ class TestErrorLogger : public TestFixture {
253254
header += CppCheck::version();
254255
header += "\"/>\n <errors>";
255256
ASSERT_EQUALS(header, ErrorMessage::getXMLHeader(""));
256-
ASSERT_EQUALS(" </errors>\n</results>", ErrorMessage::getXMLFooter());
257+
ASSERT_EQUALS(" </errors>\n</results>", ErrorMessage::getXMLFooter(2));
257258
std::string message(" <error id=\"errorId\" severity=\"error\"");
258259
message += " msg=\"Programming error.\" verbose=\"Verbose error\">\n";
259260
message += " <location file=\"bar.cpp\" line=\"8\" column=\"1\" info=\"\\303\\244\"/>\n";
@@ -312,6 +313,10 @@ class TestErrorLogger : public TestFixture {
312313
ASSERT_EQUALS(1u, msg.callStack.back().column);
313314
}
314315

316+
void ToXmlV3() const {
317+
ASSERT_EQUALS("</results>", ErrorMessage::getXMLFooter(3));
318+
}
319+
315320
void InconclusiveXml() const {
316321
// Location
317322
std::list<ErrorMessage::FileLocation> locs(1, fooCpp5);

0 commit comments

Comments
 (0)