// Copyright (C) 2016 The Qt Company Ltd. // SPDX-License-Identifier: LicenseRef-Qt-Commercial OR GPL-3.0-only WITH Qt-GPL-exception-1.0 #include #include #include #include #include //TESTED_COMPONENT=src/libs/utils using namespace Utils; class tst_StringUtils : public QObject { Q_OBJECT private slots: void testWithTildeHomePath(); void testStripAccelerator_data(); void testStripAccelerator(); void testParseUsedPortFromNetstatOutput_data(); void testParseUsedPortFromNetstatOutput(); void testJoinStrings_data(); void testJoinStrings(); void testTrim_data(); void testTrim(); void testWildcardToRegularExpression_data(); void testWildcardToRegularExpression(); void testSplitAtFirst_data(); void testSplitAtFirst(); void testAsciify_data(); void testAsciify(); void testNormalizeNewlinesInString(); void testNormalizeNewlinesInString_data(); void testNormalizeNewlinesInByteArray(); void testNormalizeNewlinesInByteArray_data(); void testRemoveCommentsFromJson(); void testRemoveCommentsFromJson_data(); void testRemoveExtraCommasFromJson(); void testRemoveExtraCommasFromJson_data(); void testCleanJson(); void testCleanJson_data(); void testApplyJsonPatch(); }; void tst_StringUtils::testCleanJson_data() { QTest::addColumn("input"); QTest::addColumn("expected"); QTest::newRow("simple object with trailing comma") << QByteArray(R"({"key1": "value1", "key2": "value2",})") << QByteArray(R"({"key1": "value1", "key2": "value2"})"); QTest::newRow("simple with comment") << QByteArray(R"({"key1": "value1", "key2": "value2"} // comment)") << QByteArray(R"({"key1": "value1", "key2": "value2"} )"); QTest::newRow("comma in comment") << QByteArray(R"({"key1": "value1", "key2": "value2"} // comment,)") << QByteArray(R"({"key1": "value1", "key2": "value2"} )"); QTest::newRow("comma after comment") << QByteArray(R"({"key1": "value1", "key2": "value2" /* comment*/,})") << QByteArray(R"({"key1": "value1", "key2": "value2" })"); QTest::newRow("comma before comment") << QByteArray(R"({"key1": "value1", "key2": "value2", /* comment*/})") << QByteArray(R"({"key1": "value1", "key2": "value2" })"); QTest::newRow("escaped comma in string") << QByteArray(R"({"key1": "value1", "key2": "value2\,"})") << QByteArray(R"({"key1": "value1", "key2": "value2\,"})"); QTest::newRow("escaped quote in string") << QByteArray(R"({"key1": "val\"ue1", "key2": "\,value2\"",})") << QByteArray(R"({"key1": "val\"ue1", "key2": "\,value2\""})"); } void tst_StringUtils::testCleanJson() { QFETCH(QByteArray, input); QFETCH(QByteArray, expected); QByteArray result = cleanJson(input); QCOMPARE(result, expected); } void tst_StringUtils::testApplyJsonPatch() { QString targetJson = R"json({"v1":1,"v2":{"v21":1,"v22":2},"v3":{},"v4":[1,2],"v5":{}})json"; QString patchJson = R"json({"v1":2,"v2":{"v21":2},"v4":[1,2,3],"v5":{},"v6":{"v61":[1,2],"v62":"1"}})json"; QString expectedJson = R"json({"v1":2,"v2":{"v21":2,"v22":2},"v3":{},"v4":[1,2,1,2,3],"v5":{},"v6":{"v61":[1,2],"v62":"1"}})json"; QJsonDocument targetDoc = QJsonDocument::fromJson(targetJson.toUtf8()); QJsonDocument patchDoc = QJsonDocument::fromJson(patchJson.toUtf8()); QJsonValue target = targetDoc.object(); QJsonValue patch = patchDoc.object(); applyJsonPatch(target, patch); QJsonDocument resultDoc = QJsonDocument(target.toObject()); QCOMPARE(resultDoc.toJson(QJsonDocument::Compact), expectedJson.toUtf8()); // Empty value should not change target ... target = targetDoc.object(); applyJsonPatch(target, QJsonValue()); resultDoc = QJsonDocument(target.toObject()); QCOMPARE(resultDoc.toJson(QJsonDocument::Compact), targetJson.toUtf8()); } void tst_StringUtils::testRemoveExtraCommasFromJson_data() { QTest::addColumn("input"); QTest::addColumn("expected"); QTest::newRow("object with trailing comma") << QByteArray(R"({"a": 1, "b": 2,})") << QByteArray(R"({"a": 1, "b": 2})"); QTest::newRow("array with trailing comma") << QByteArray(R"([1, 2, 3,])") << QByteArray(R"([1, 2, 3])"); QTest::newRow("array with many trailing comma") << QByteArray(R"([1, 2, 3,,,,])") << QByteArray(R"([1, 2, 3,,,])"); QTest::newRow("nested structures") << QByteArray(R"({"a": [1,2,3,], "b": {"x": 5,},})") << QByteArray(R"({"a": [1,2,3], "b": {"x": 5}})"); QTest::newRow("empty array and object") << QByteArray(R"({"emptyArray": [], "emptyObject": {},})") << QByteArray(R"({"emptyArray": [], "emptyObject": {}})"); QTest::newRow("already valid JSON") << QByteArray(R"({"valid": [1,2], "obj": {"a": 1}})") << QByteArray(R"({"valid": [1,2], "obj": {"a": 1}})"); QTest::newRow("commas in strings") << QByteArray(R"({"text": "hello, world,", "another": "brace } test",})") << QByteArray(R"({"text": "hello, world,", "another": "brace } test"})"); QTest::newRow("whitespace before closing") << QByteArray("{\n \"a\": 1, \n \"b\": 2,\n}") << QByteArray("{\n \"a\": 1, \n \"b\": 2\n}"); QTest::newRow("whitespace before comma") << QByteArray(R"({"a": 1, "b": 2 , })") << QByteArray(R"({"a": 1, "b": 2 })"); } void tst_StringUtils::testRemoveExtraCommasFromJson() { QFETCH(QByteArray, input); QFETCH(QByteArray, expected); QByteArray result = removeExtraCommasFromJson(input); QCOMPARE(result, expected); } void tst_StringUtils::testRemoveCommentsFromJson_data() { QTest::addColumn("json"); QTest::addColumn("expected"); QTest::newRow("single-line comments") << QByteArray(R"( { "key1": "value1", // comment "key2": "value2", "key3": "value3" // another comment } )") << QByteArray(R"( { "key1": "value1", "key2": "value2", "key3": "value3" } )"); QTest::newRow("multi-line comments") << QByteArray(R"( { "key1": "value1", /* comment */ "key2": "value2", "key3": "value3" /* another comment */ } )") << QByteArray(R"( { "key1": "value1", "key2": "value2", "key3": "value3" } )"); QTest::newRow("comment in string value") << QByteArray(R"( { "key1": "value1", // comment "key2": "value2", "key3": "value3 // another comment" } )") << QByteArray(R"( { "key1": "value1", "key2": "value2", "key3": "value3 // another comment" } )"); QTest::newRow("comment after string value") << QByteArray(R"( { "key1": "value1", // comment "key2": "value2", "key3": "value3 // another comment" /* comment */ } )") << QByteArray(R"( { "key1": "value1", "key2": "value2", "key3": "value3 // another comment" } )"); QTest::newRow("multi-line block comment") << QByteArray(R"( { "key1": "value1", /* comment */ "key2": "value2", "key3": "value3" /* another comment and another line */ } )") << QByteArray(R"( { "key1": "value1", "key2": "value2", "key3": "value3" } )"); } void tst_StringUtils::testRemoveCommentsFromJson() { QFETCH(QByteArray, json); QFETCH(QByteArray, expected); QCOMPARE(Utils::removeCommentsFromJson(json), expected); } void tst_StringUtils::testWithTildeHomePath() { const QString home = QDir::homePath(); const FilePath homePath = FilePath::fromString(home); #ifndef Q_OS_WIN // home path itself QCOMPARE(homePath.withTildeHomePath(), QString("~")); QCOMPARE(homePath.pathAppended("/").withTildeHomePath(), QString("~")); QCOMPARE(FilePath::fromString("/unclean/../" + home).withTildeHomePath(), QString("~")); // sub of home path QCOMPARE(homePath.pathAppended("/foo").withTildeHomePath(), QString("~/foo")); QCOMPARE(homePath.pathAppended("/foo/").withTildeHomePath(), QString("~/foo")); QCOMPARE(homePath.pathAppended("/some/path/file.txt").withTildeHomePath(), QString("~/some/path/file.txt")); QCOMPARE(homePath.pathAppended("/some/unclean/../path/file.txt").withTildeHomePath(), QString("~/some/path/file.txt")); // not sub of home path QCOMPARE(homePath.pathAppended("/../foo").withTildeHomePath(), QString(home + "/../foo")); #else // windows: should return same as input QCOMPARE(homePath.withTildeHomePath(), home); QCOMPARE(homePath.pathAppended("/foo").withTildeHomePath(), home + QString("/foo")); QCOMPARE(homePath.pathAppended("/../foo").withTildeHomePath(), homePath.pathAppended("/../foo").withTildeHomePath()); #endif } void tst_StringUtils::testStripAccelerator_data() { QTest::addColumn("expected"); QTest::newRow("Test") << "Test"; QTest::newRow("&Test") << "Test"; QTest::newRow("&&Test") << "&Test"; QTest::newRow("T&est") << "Test"; QTest::newRow("&Te&&st") << "Te&st"; QTest::newRow("T&e&st") << "Test"; QTest::newRow("T&&est") << "T&est"; QTest::newRow("T&&e&st") << "T&est"; QTest::newRow("T&&&est") << "T&est"; QTest::newRow("Tes&t") << "Test"; QTest::newRow("Test&") << "Test"; } void tst_StringUtils::testStripAccelerator() { QFETCH(QString, expected); QCOMPARE(Utils::stripAccelerator(QString::fromUtf8(QTest::currentDataTag())), expected); } void tst_StringUtils::testParseUsedPortFromNetstatOutput_data() { QTest::addColumn("line"); QTest::addColumn("port"); QTest::newRow("Empty") << "" << -1; // Windows netstat. QTest::newRow("Win1") << "Active Connection" << -1; QTest::newRow("Win2") << " Proto Local Address Foreign Address State" << -1; QTest::newRow("Win3") << " TCP 0.0.0.0:80 0.0.0.0:0 LISTENING" << 80; QTest::newRow("Win4") << " TCP 0.0.0.0:113 0.0.0.0:0 LISTENING" << 113; QTest::newRow("Win5") << " TCP 10.9.78.4:14714 0.0.0.0:0 LISTENING" << 14714; QTest::newRow("Win6") << " TCP 10.9.78.4:50233 12.13.135.180:993 ESTABLISHED" << 50233; QTest::newRow("Win7") << " TCP [::]:445 [::]:0 LISTENING" << 445; QTest::newRow("Win8") << " TCP 192.168.0.80:51905 169.55.74.50:443 ESTABLISHED" << 51905; QTest::newRow("Win9") << " UDP [fe80::840a:2942:8def:abcd%6]:1900 *:* " << 1900; // Linux QTest::newRow("Linux1") << "sl local_address rem_address st tx_queue rx_queue tr tm->when retrnsmt ..." << -1; QTest::newRow("Linux2") << "0: 00000000:2805 00000000:0000 0A 00000000:00000000 00:00000000 00000000 ..." << 10245; QTest::newRow("Linux3") << " 1: 0100007F:193F 00000000:0000 0A 00000000:00000000 00:00000000 00000000 ..." << 6463; // Mac QTest::newRow("Mac1") << "Active Internet connections (including servers)" << -1; QTest::newRow("Mac2") << "Proto Recv-Q Send-Q Local Address Foreign Address (state)" << -1; QTest::newRow("Mac3") << "tcp4 0 0 192.168.1.12.55687 88.198.14.66.443 ESTABLISHED" << 55687; QTest::newRow("Mac4") << "tcp6 0 0 2a01:e34:ee42:d0.55684 2a02:26f0:ff::5c.443 ESTABLISHED" << 55684; QTest::newRow("Mac5") << "tcp4 0 0 *.631 *.* LISTEN" << 631; QTest::newRow("Mac6") << "tcp6 0 0 *.631 *.* LISTEN" << 631; QTest::newRow("Mac7") << "udp4 0 0 192.168.79.1.123 *.*" << 123; QTest::newRow("Mac9") << "udp4 0 0 192.168.8.1.123 *.*" << 123; // QNX QTest::newRow("Qnx1") << "Active Internet connections (including servers)" << -1; QTest::newRow("Qnx2") << "Proto Recv-Q Send-Q Local Address Foreign Address State " << -1; QTest::newRow("Qnx3") << "tcp 0 0 10.9.7.5.22 10.9.7.4.46592 ESTABLISHED" << 22; QTest::newRow("Qnx4") << "tcp 0 0 *.8000 *.* LISTEN " << 8000; QTest::newRow("Qnx5") << "tcp 0 0 *.22 *.* LISTEN " << 22; QTest::newRow("Qnx6") << "udp 0 0 *.* *.* " << -1; QTest::newRow("Qnx7") << "udp 0 0 *.* *.* " << -1; QTest::newRow("Qnx8") << "Active Internet6 connections (including servers)" << -1; QTest::newRow("Qnx9") << "Proto Recv-Q Send-Q Local Address Foreign Address (state) " << -1; QTest::newRow("QnxA") << "tcp6 0 0 *.22 *.* LISTEN " << 22; // Android QTest::newRow("Android1") << "tcp 0 0 10.0.2.16:49088 142.250.180.74:443 ESTABLISHED" << 49088; QTest::newRow("Android2") << "tcp 0 0 10.0.2.16:48380 142.250.186.196:443 CLOSE_WAIT" << 48380; QTest::newRow("Android3") << "tcp6 0 0 [::]:5555 [::]:* LISTEN" << 5555; QTest::newRow("Android4") << "tcp6 0 0 ::ffff:127.0.0.1:39417 [::]:* LISTEN" << 39417; QTest::newRow("Android5") << "tcp6 0 0 ::ffff:10.0.2.16:35046 ::ffff:142.250.203.:443 ESTABLISHED" << 35046; QTest::newRow("Android6") << "tcp6 0 0 ::ffff:127.0.0.1:46265 ::ffff:127.0.0.1:33155 TIME_WAIT" << 46265; QTest::newRow("Android7") << "udp 0 0 10.0.2.16:50950 142.250.75.14:443 ESTABLISHED" << 50950; QTest::newRow("Android8") << "udp 2560 0 10.0.2.16:68 10.0.2.2:67 ESTABLISHED" << 68; QTest::newRow("Android9") << "udp 0 0 0.0.0.0:5353 0.0.0.0:*" << 5353; QTest::newRow("Android10") << "udp6 0 0 [::]:36662 [::]:*" << 36662; } void tst_StringUtils::testParseUsedPortFromNetstatOutput() { QFETCH(QString, line); QFETCH(int, port); QCOMPARE(Utils::parseUsedPortFromNetstatOutput(line.toUtf8()), port); } struct JoinData { QStringList input; QString output = {}; QChar separator = '\n'; }; void tst_StringUtils::testJoinStrings_data() { QTest::addColumn("data"); QTest::newRow("0") << JoinData{}; QTest::newRow("1") << JoinData{{"one"}, "one"}; QTest::newRow("1_Null") << JoinData{{{}}}; QTest::newRow("1_Empty") << JoinData{{""}}; QTest::newRow("2") << JoinData{{"first", "second"}, "first\nsecond"}; QTest::newRow("2_Null") << JoinData{{{}, {}}}; QTest::newRow("2_Empty") << JoinData{{"", ""}}; QTest::newRow("2_1stNull") << JoinData{{{}, "second"}, "second"}; QTest::newRow("2_1stEmpty") << JoinData{{"", "second"}, "second"}; QTest::newRow("2_2ndNull") << JoinData{{"first", {}}, "first"}; QTest::newRow("2_2ndEmpty") << JoinData{{"first", ""}, "first"}; QTest::newRow("3") << JoinData{{"first", "second", "third"}, "first\nsecond\nthird"}; QTest::newRow("3_Null") << JoinData{{{}, {}, {}}}; QTest::newRow("3_Empty") << JoinData{{"", "", ""}}; QTest::newRow("3_1stNull") << JoinData{{{}, "second", "third"}, "second\nthird"}; QTest::newRow("3_1stEmpty") << JoinData{{"", "second", "third"}, "second\nthird"}; QTest::newRow("3_2ndNull") << JoinData{{"first", {}, "third"}, "first\nthird"}; QTest::newRow("3_2ndEmpty") << JoinData{{"first", "", "third"}, "first\nthird"}; QTest::newRow("3_3rdNull") << JoinData{{"first", "second", {}}, "first\nsecond"}; QTest::newRow("3_3rdEmpty") << JoinData{{"first", "second", ""}, "first\nsecond"}; QTest::newRow("3_1stNonNull") << JoinData{{"first", {}, {}}, "first"}; QTest::newRow("3_1stNonEmpty") << JoinData{{"first", "", ""}, "first"}; QTest::newRow("3_2ndNonNull") << JoinData{{{}, "second", {}}, "second"}; QTest::newRow("3_2ndNonEmpty") << JoinData{{"", "second", ""}, "second"}; QTest::newRow("3_2ndNonNull") << JoinData{{{}, {}, "third"}, "third"}; QTest::newRow("3_3ndNonEmpty") << JoinData{{"", "", "third"}, "third"}; QTest::newRow("DotSeparator") << JoinData{{"first", "second"}, "first.second", '.'}; } void tst_StringUtils::testJoinStrings() { QFETCH(JoinData, data); QCOMPARE(Utils::joinStrings(data.input, data.separator), data.output); } struct TrimData { QString input; QString front = {}; QString back = {}; QString bothSides = {}; QChar ch = ' '; }; void tst_StringUtils::testTrim_data() { QTest::addColumn("data"); QTest::newRow("Empty") << TrimData{}; QTest::newRow("AllToRemove") << TrimData{" "}; QTest::newRow("BothSides") << TrimData{" foo ", "foo ", " foo", "foo"}; QTest::newRow("BothSidesLong") << TrimData{" foo ", "foo ", " foo", "foo"}; QTest::newRow("CharInside") << TrimData{" foo bar ", "foo bar ", " foo bar", "foo bar"}; } void tst_StringUtils::testTrim() { QFETCH(TrimData, data); QCOMPARE(Utils::trimFront(data.input, data.ch), data.front); QCOMPARE(Utils::trimBack(data.input, data.ch), data.back); QCOMPARE(Utils::trim(data.input, data.ch), data.bothSides); } void tst_StringUtils::testWildcardToRegularExpression_data() { QTest::addColumn("pattern"); QTest::addColumn("string"); QTest::addColumn("matches"); auto addRow = [](const char *pattern, const char *string, bool matchesNonPathGlob) { QTest::addRow("%s@%s", pattern, string) << pattern << string << matchesNonPathGlob; }; addRow("*.html", "test.html", true); addRow("*.html", "test.htm", false); addRow("*bar*", "foobarbaz", true); addRow("*", "Qt Rocks!", true); addRow("*.h", "test.cpp", false); addRow("*.???l", "test.html", true); addRow("*?", "test.html", true); addRow("*?ml", "test.html", true); addRow("*[*]", "test.html", false); addRow("*[?]", "test.html", false); addRow("*[?]ml", "test.h?ml", true); addRow("*[[]ml", "test.h[ml", true); addRow("*[]]ml", "test.h]ml", true); addRow("*.h[a-z]ml", "test.html", true); addRow("*.h[A-Z]ml", "test.html", false); addRow("*.h[A-Z]ml", "test.hTml", true); addRow("*.h[!A-Z]ml", "test.hTml", false); addRow("*.h[!A-Z]ml", "test.html", true); addRow("*.h[!T]ml", "test.hTml", false); addRow("*.h[!T]ml", "test.html", true); addRow("*.h[!T]m[!L]", "test.htmL", false); addRow("*.h[!T]m[!L]", "test.html", true); addRow("*.h[][!]ml", "test.h]ml", true); addRow("*.h[][!]ml", "test.h[ml", true); addRow("*.h[][!]ml", "test.h!ml", true); addRow("foo/*/bar", "foo/baz/bar", true); addRow("foo/*/bar", "foo/fie/baz/bar", true); addRow("foo?bar", "foo/bar", true); addRow("foo/(*)/bar", "foo/baz/bar", false); addRow("foo/(*)/bar", "foo/(baz)/bar", true); addRow("foo/?/bar", "foo/Q/bar", true); addRow("foo/?/bar", "foo/Qt/bar", false); addRow("foo/(?)/bar", "foo/Q/bar", false); addRow("foo/(?)/bar", "foo/(Q)/bar", true); addRow("foo\\*\\bar", "foo\\baz\\bar", true); addRow("foo\\*\\bar", "foo/baz/bar", false); addRow("foo\\*\\bar", "foo/baz\\bar", false); addRow("foo\\*\\bar", "foo\\fie\\baz\\bar", true); addRow("foo\\*\\bar", "foo/fie/baz/bar", false); addRow("foo/*/bar", "foo\\baz\\bar", false); addRow("foo/*/bar", "foo/baz/bar", true); addRow("foo/*/bar", "foo\\fie\\baz\\bar", false); addRow("foo/*/bar", "foo/fie/baz/bar", true); addRow("foo\\(*)\\bar", "foo\\baz\\bar", false); addRow("foo\\(*)\\bar", "foo\\(baz)\\bar", true); addRow("foo\\?\\bar", "foo\\Q\\bar", true); addRow("foo\\?\\bar", "foo\\Qt\\bar", false); addRow("foo\\(?)\\bar", "foo\\Q\\bar", false); addRow("foo\\(?)\\bar", "foo\\(Q)\\bar", true); addRow("foo*bar", "foo/fie/baz/bar", true); addRow("fie*bar", "foo/fie/baz/bar", false); } void tst_StringUtils::testWildcardToRegularExpression() { QFETCH(QString, pattern); QFETCH(QString, string); QFETCH(bool, matches); const QRegularExpression re(Utils::wildcardToRegularExpression(pattern)); QCOMPARE(string.contains(re), matches); } void tst_StringUtils::testSplitAtFirst_data() { QTest::addColumn("string"); QTest::addColumn("separator"); QTest::addColumn("left"); QTest::addColumn("right"); QTest::newRow("Empty") << QString{} << QChar{} << QString{} << QString{}; QTest::newRow("EmptyString") << QString{} << QChar{'a'} << QString{} << QString{}; QTest::newRow("EmptySeparator") << QString{"abc"} << QChar{} << QString{"abc"} << QString{}; QTest::newRow("NoSeparator") << QString{"abc"} << QChar{'d'} << QString{"abc"} << QString{}; QTest::newRow("SeparatorAtStart") << QString{"abc"} << QChar{'a'} << QString{} << QString{"bc"}; QTest::newRow("SeparatorAtEnd") << QString{"abc"} << QChar{'c'} << QString{"ab"} << QString{}; QTest::newRow("SeparatorInMiddle") << QString{"abc"} << QChar{'b'} << QString{"a"} << QString{"c"}; QTest::newRow("SeparatorAtStartAndEnd") << QString{"abca"} << QChar{'a'} << QString{} << QString{"bca"}; } void tst_StringUtils::testSplitAtFirst() { QFETCH(QString, string); QFETCH(QChar, separator); QFETCH(QString, left); QFETCH(QString, right); const auto [l, r] = Utils::splitAtFirst(string, separator); QCOMPARE(l, left); QCOMPARE(r, right); } void tst_StringUtils::testAsciify_data() { QTest::addColumn("input"); QTest::addColumn("expected"); QTest::newRow("Basic Latin") << QString("Basic text") << QString("Basic text"); QTest::newRow("Control character") << QString("\x07 text") << QString("u0007 text"); QTest::newRow("Miscellaneous Technical") << QString(u8"\u23F0 text") << QString("u23f0 text"); } void tst_StringUtils::testAsciify() { QFETCH(QString, input); QFETCH(QString, expected); const QString asciified = Utils::asciify(input); QCOMPARE(asciified, expected); } void tst_StringUtils::testNormalizeNewlinesInString_data() { QTest::addColumn("input"); QTest::addColumn("expected"); QTest::newRow("none") << QString("asd\nfoo\r\nbar\nfoo\r") << QString("asd\nfoo\nbar\nfoo\r"); QTest::newRow("one") << QString("asd\nfoo\r\r\nbar\nfoo\r") << QString("asd\nfoo\nbar\nfoo\r"); QTest::newRow("more") << QString("asd\nfoo\r\r\n\r\nbar\nfoo\r") << QString("asd\nfoo\n\nbar\nfoo\r"); } void tst_StringUtils::testNormalizeNewlinesInString() { QFETCH(QString, input); QFETCH(QString, expected); const QString normalized = Utils::normalizeNewlines(input); QCOMPARE(normalized, expected); } void tst_StringUtils::testNormalizeNewlinesInByteArray_data() { QTest::addColumn("input"); QTest::addColumn("expected"); QTest::newRow("none") << QByteArray("asd\nfoo\r\nbar\nfoo\r") << QByteArray("asd\nfoo\nbar\nfoo\r"); QTest::newRow("one") << QByteArray("asd\nfoo\r\r\nbar\nfoo\r") << QByteArray("asd\nfoo\nbar\nfoo\r"); QTest::newRow("more") << QByteArray("asd\nfoo\r\r\r\n\r\nbar\nfoo\r") << QByteArray("asd\nfoo\n\nbar\nfoo\r"); } void tst_StringUtils::testNormalizeNewlinesInByteArray() { QFETCH(QByteArray, input); QFETCH(QByteArray, expected); const QByteArray normalized = Utils::normalizeNewlines(input); QCOMPARE(normalized, expected); } QTEST_GUILESS_MAIN(tst_StringUtils) #include "tst_stringutils.moc"