diff --git a/docs/topics/Reference.adoc b/docs/topics/Reference.adoc index f81f09d70..b01676083 100644 --- a/docs/topics/Reference.adoc +++ b/docs/topics/Reference.adoc @@ -49,6 +49,8 @@ This section contains full details on advanced features available in KeePassXC. |{DB_DIR} |Absolute directory path of database file |=== +NOTE: You can insert literal placeholder strings by escaping the beginning and ending curly braces. For example, to insert the string `{USERNAME}`, you would type `++\{USERNAME\}++`. + === Entry Cross-Reference A reference to another entry's field is possible using the shorthand syntax: `{REF:<FIELD>@<SEARCH_IN>:<SEARCH_TEXT>}` diff --git a/src/core/Entry.cpp b/src/core/Entry.cpp index 0147ec0f9..70bb57ed8 100644 --- a/src/core/Entry.cpp +++ b/src/core/Entry.cpp @@ -1059,6 +1059,15 @@ QString Entry::resolveMultiplePlaceholdersRecursive(const QString& str, int maxD return str; } + // Short circuit if we have escaped the placeholder brackets + if (str.startsWith("\\{") && str.endsWith("\\}")) { + // Replace the escaped brackets with actuals and move on + auto ret = str; + ret.replace(0, 2, "{"); + ret.replace(ret.size() - 2, 2, "}"); + return ret; + } + QString result; auto matches = placeholderRegEx.globalMatch(str); int capEnd = 0; diff --git a/tests/TestEntry.cpp b/tests/TestEntry.cpp index 51cb4799c..4ad23f017 100644 --- a/tests/TestEntry.cpp +++ b/tests/TestEntry.cpp @@ -583,6 +583,17 @@ void TestEntry::testResolveReplacePlaceholders() // Test complicated and nested replacements QCOMPARE(entry2->resolveMultiplePlaceholders(entry2->url()), QString("cmd://sap.exe -system=server1 -client=12345 -user=Username2 -pw=Password1")); + + auto* entry3 = new Entry(); + entry3->setGroup(root); + entry3->setUuid(QUuid::createUuid()); + entry3->setTitle("Entry 3"); + entry3->setUsername("HMAC-SHA-256"); + entry3->setUrl("{T-REPLACE-RX:!{USERNAME}!\\{USERNAME\\}!!}"); + + // Test escaped enclosures + QCOMPARE(entry3->resolveMultiplePlaceholders(entry3->url()), entry3->username()); + // Test invalid syntax QString error; entry1->resolveRegexPlaceholder("{T-REPLACE-RX:/{USERNAME}/.*+?/test/}", &error); // invalid regex