/**************************************************************************** ** ** Copyright (C) 2016 The Qt Company Ltd. ** Contact: https://www.qt.io/licensing/ ** ** This file is part of Qt Creator. ** ** Commercial License Usage ** Licensees holding valid commercial Qt licenses may use this file in ** accordance with the commercial license agreement provided with the ** Software or, alternatively, in accordance with the terms contained in ** a written agreement between you and The Qt Company. For licensing terms ** and conditions see https://www.qt.io/terms-conditions. For further ** information use the contact form at https://www.qt.io/contact-us. ** ** GNU General Public License Usage ** Alternatively, this file may be used under the terms of the GNU ** General Public License version 3 as published by the Free Software ** Foundation with exceptions as appearing in the file LICENSE.GPL3-EXCEPT ** included in the packaging of this file. Please review the following ** information to ensure the GNU General Public License requirements will ** be met: https://www.gnu.org/licenses/gpl-3.0.html. ** ****************************************************************************/ #include #include //TESTED_COMPONENT=src/libs/utils class TestMacroExpander : public Utils::AbstractMacroExpander { public: virtual bool resolveMacro(const QString &name, QString *ret) { if (name == QLatin1String("foo")) { *ret = QLatin1String("a"); return true; } if (name == QLatin1String("a")) { *ret = QLatin1String("hi"); return true; } if (name == QLatin1String("hi")) { *ret = QLatin1String("ho"); return true; } if (name == QLatin1String("hihi")) { *ret = QLatin1String("bar"); return true; } if (name == "slash") { *ret = "foo/bar"; return true; } if (name == "sl/sh") { *ret = "slash"; return true; } if (name == "JS:foo") { *ret = "bar"; return true; } return false; } }; class tst_StringUtils : public QObject { Q_OBJECT private slots: void testWithTildeHomePath(); void testMacroExpander_data(); void testMacroExpander(); private: TestMacroExpander mx; }; void tst_StringUtils::testWithTildeHomePath() { #ifndef Q_OS_WIN // home path itself QCOMPARE(Utils::withTildeHomePath(QDir::homePath()), QString::fromLatin1("~")); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QLatin1Char('/')), QString::fromLatin1("~")); QCOMPARE(Utils::withTildeHomePath(QString::fromLatin1("/unclean/..") + QDir::homePath()), QString::fromLatin1("~")); // sub of home path QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/foo")), QString::fromLatin1("~/foo")); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/foo/")), QString::fromLatin1("~/foo")); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/some/path/file.txt")), QString::fromLatin1("~/some/path/file.txt")); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/some/unclean/../path/file.txt")), QString::fromLatin1("~/some/path/file.txt")); // not sub of home path QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/../foo")), QString(QDir::homePath() + QString::fromLatin1("/../foo"))); #else // windows: should return same as input QCOMPARE(Utils::withTildeHomePath(QDir::homePath()), QDir::homePath()); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/foo")), QDir::homePath() + QString::fromLatin1("/foo")); QCOMPARE(Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/../foo")), Utils::withTildeHomePath(QDir::homePath() + QString::fromLatin1("/../foo"))); #endif } void tst_StringUtils::testMacroExpander_data() { QTest::addColumn("in"); QTest::addColumn("out"); static const struct { const char * const in; const char * const out; } vals[] = { { "text", "text" }, { "%{a}", "hi" }, { "%%{a}", "%hi" }, { "%%%{a}", "%%hi" }, { "%{b}", "%{b}" }, { "pre%{a}", "prehi" }, { "%{a}post", "hipost" }, { "pre%{a}post", "prehipost" }, { "%{a}%{a}", "hihi" }, { "%{a}text%{a}", "hitexthi" }, { "%{foo}%{a}text%{a}", "ahitexthi" }, { "%{}{a}", "%{a}" }, { "%{}", "%" }, { "test%{}", "test%" }, { "%{}test", "%test" }, { "%{abc", "%{abc" }, { "%{%{a}", "%{hi" }, { "%{%{a}}", "ho" }, { "%{%{a}}}post", "ho}post" }, { "%{hi%{a}}", "bar" }, { "%{hi%{%{foo}}}", "bar" }, { "%{hihi/b/c}", "car" }, { "%{hihi/a/}", "br" }, // empty replacement { "%{hihi/b}", "bar" }, // incomplete substitution { "%{hihi/./c}", "car" }, { "%{hihi//./c}", "ccc" }, { "%{hihi/(.)(.)r/\\2\\1c}", "abc" }, // no escape for capture groups { "%{hihi/b/c/d}", "c/dar" }, { "%{hihi/a/e{\\}e}", "be{}er" }, // escape closing brace { "%{slash/o\\/b/ol's c}", "fool's car" }, { "%{sl\\/sh/(.)(a)(.)/\\2\\1\\3as}", "salsash" }, // escape in variable name { "%{JS:foo/b/c}", "%{JS:foo/b/c}" }, // No replacement for JS (all considered varName) { "%{%{a}%{a}/b/c}", "car" }, { "%{nonsense:-sense}", "sense" }, }; for (unsigned i = 0; i < sizeof(vals)/sizeof(vals[0]); i++) QTest::newRow(vals[i].in) << QString::fromLatin1(vals[i].in) << QString::fromLatin1(vals[i].out); } void tst_StringUtils::testMacroExpander() { QFETCH(QString, in); QFETCH(QString, out); Utils::expandMacros(&in, &mx); QCOMPARE(in, out); } QTEST_MAIN(tst_StringUtils) #include "tst_stringutils.moc"