diff options
| author | David Faure <faure@kde.org> | 2021-06-23 12:05:33 +0200 | 
|---|---|---|
| committer | David Faure <faure@kde.org> | 2021-08-03 10:01:15 +0000 | 
| commit | ed28682265bd95416a98d5ccc6a72e96563f84f3 (patch) | |
| tree | f02637dae257a7330b47731257c6811945b448e8 | |
| parent | dcc5f6529d3008f2cf3a678a4c42d5092b406690 (diff) | |
| download | kconfig-ed28682265bd95416a98d5ccc6a72e96563f84f3.tar.gz kconfig-ed28682265bd95416a98d5ccc6a72e96563f84f3.tar.bz2 | |
KConfig: fix deletion of an entry that is also in kdeglobals
This is the case where we expected to see Key[$d] in the config file,
and it was somehow broken. When saving, the key was omitted, so when
reloading the kdeglobals key was present again.
Detected when trying to write a unittest for a different patch...
I had to reshuffle the unittest a bit because testThreads calls
testSimple which didn't expect that the "[AAA]" group would actually
be deleted now.
| -rw-r--r-- | autotests/kconfigtest.cpp | 80 | ||||
| -rw-r--r-- | autotests/kconfigtest.h | 1 | ||||
| -rw-r--r-- | src/core/kconfigdata.cpp | 7 | ||||
| -rw-r--r-- | src/core/kconfigini.cpp | 2 | 
4 files changed, 75 insertions, 15 deletions
| diff --git a/autotests/kconfigtest.cpp b/autotests/kconfigtest.cpp index 49f22d1b..af62b14d 100644 --- a/autotests/kconfigtest.cpp +++ b/autotests/kconfigtest.cpp @@ -116,9 +116,15 @@ void KConfigTest::initTestCase()      KConfig sc(s_kconfig_test_subdir); -    KConfigGroup cg(&sc, "AAA"); +    KConfigGroup cg(&sc, "AAA"); // deleted later by testDelete      cg.writeEntry("stringEntry1", s_string_entry1, KConfig::Persistent | KConfig::Global); -    cg.deleteEntry("stringEntry2", KConfig::Global); + +    cg = KConfigGroup(&sc, "GlobalGroup"); +    cg.writeEntry("globalEntry", s_string_entry1, KConfig::Persistent | KConfig::Global); +    cg.deleteEntry("globalEntry2", KConfig::Global); + +    cg = KConfigGroup(&sc, "LocalGroupToBeDeleted"); // deleted later by testDelete +    cg.writeEntry("stringEntry1", s_string_entry1);      cg = KConfigGroup(&sc, "Hello");      cg.writeEntry("boolEntry1", s_bool_entry1); @@ -339,16 +345,16 @@ void KConfigTest::testSimple()          QVERIFY(!group.contains(QChar(0x1d)));      } -    KConfigGroup sc3(&sc2, "AAA"); +    KConfigGroup sc3(&sc2, "GlobalGroup"); -    QVERIFY(sc3.hasKey("stringEntry1")); // from kdeglobals -    QVERIFY(!sc3.isEntryImmutable("stringEntry1")); -    QCOMPARE(sc3.readEntry("stringEntry1"), s_string_entry1); +    QVERIFY(sc3.hasKey("globalEntry")); // from kdeglobals +    QVERIFY(!sc3.isEntryImmutable("globalEntry")); +    QCOMPARE(sc3.readEntry("globalEntry"), s_string_entry1); -    QVERIFY(!sc3.hasKey("stringEntry2")); -    QCOMPARE(sc3.readEntry("stringEntry2", QStringLiteral("bla")), QStringLiteral("bla")); +    QVERIFY(!sc3.hasKey("globalEntry2")); +    QCOMPARE(sc3.readEntry("globalEntry2", QStringLiteral("bla")), QStringLiteral("bla")); -    QVERIFY(!sc3.hasDefault("stringEntry1")); +    QVERIFY(!sc3.hasDefault("globalEntry"));      sc3 = KConfigGroup(&sc2, "Hello");      QCOMPARE(sc3.readEntry("Test", QByteArray()), QByteArray(s_utf8bit_entry)); @@ -824,6 +830,8 @@ void KConfigTest::testDelete()      QVERIFY(!sc.entryMap(QStringLiteral("Hello")).isEmpty()); // not deleted group      QVERIFY(sc.entryMap(QStringLiteral("FooBar")).isEmpty()); // inexistent group +    KConfigGroup(&sc, "LocalGroupToBeDeleted").deleteGroup(); +      QVERIFY(cg.sync());      // Check what happens on disk      const QList<QByteArray> lines = readLines(); @@ -832,7 +840,8 @@ void KConfigTest::testDelete()      QVERIFY(!lines.contains("[Complex Types][Nested Group 1]\n"));      QVERIFY(!lines.contains("[Complex Types][Nested Group 2]\n"));      QVERIFY(!lines.contains("[Complex Types][Nested Group 2.1]\n")); -    QVERIFY(!lines.contains("[AAA]\n")); +    QVERIFY(!lines.contains("[LocalGroupToBeDeleted]\n")); +    QVERIFY(lines.contains("[AAA]\n")); // deleted from kconfigtest, but present in kdeglobals, so [$d]      QVERIFY(lines.contains("[Hello]\n")); // a group that was not deleted      // test for entries that are marked as deleted when there is no default @@ -1712,8 +1721,57 @@ void KConfigTest::testKdeGlobals()          const KConfigGroup generalNoGlob(&globReadNoGlob, "General");          QCOMPARE(generalNoGlob.readEntry("testKG"), QStringLiteral("2"));      } +} + +void KConfigTest::testLocalDeletion() +{ +    // Prepare kdeglobals +    { +        KConfig glob(QStringLiteral("kdeglobals")); +        KConfigGroup general(&glob, "OwnTestGroup"); +        general.writeEntry("GlobalKey", "DontTouchMe"); +        QVERIFY(glob.sync()); +    } -    // TODO now use kconfigtest and writeEntry(,Global) -> should go into kdeglobals +    QStringList expectedKeys{QStringLiteral("LocalKey")}; +    expectedKeys.prepend(QStringLiteral("GlobalWrite")); + +    // Write into kconfigtest, including deleting GlobalKey +    { +        KConfig mainConfig(s_kconfig_test_subdir); +        KConfigGroup mainGroup(&mainConfig, "OwnTestGroup"); +        mainGroup.writeEntry("LocalKey", QStringLiteral("LocalValue")); +        mainGroup.writeEntry("GlobalWrite", QStringLiteral("GlobalValue"), KConfig::Persistent | KConfig::Global); // goes to kdeglobals +        QCOMPARE(mainGroup.readEntry("GlobalKey"), QStringLiteral("DontTouchMe")); +        mainGroup.deleteEntry("GlobalKey"); // local deletion ([$d]), kdeglobals is unchanged +        QCOMPARE(mainGroup.readEntry("GlobalKey", "Default"), QStringLiteral("Default")); // key is gone +        QCOMPARE(mainGroup.keyList(), expectedKeys); +    } + +    // Check what ended up in kconfigtest +    const QList<QByteArray> lines = readLines(); +    QVERIFY(lines.contains("[OwnTestGroup]\n")); +    QVERIFY(lines.contains("GlobalKey[$d]\n")); + +    // Check what ended up in kdeglobals +    { +        KConfig globReadNoGlob(QStringLiteral("kdeglobals"), KConfig::NoGlobals); +        const KConfigGroup generalNoGlob(&globReadNoGlob, "OwnTestGroup"); +        QCOMPARE(generalNoGlob.readEntry("GlobalKey"), QStringLiteral("DontTouchMe")); +        QCOMPARE(generalNoGlob.readEntry("GlobalWrite"), QStringLiteral("GlobalValue")); +        QVERIFY(!generalNoGlob.hasKey("LocalValue")); +        QStringList expectedGlobalKeys{QStringLiteral("GlobalKey")}; +        expectedGlobalKeys.append(QStringLiteral("GlobalWrite")); +        QCOMPARE(generalNoGlob.keyList(), expectedGlobalKeys); +    } + +    // Check what we see when re-reading the config file +    { +        KConfig mainConfig(s_kconfig_test_subdir); +        KConfigGroup mainGroup(&mainConfig, "OwnTestGroup"); +        QCOMPARE(mainGroup.readEntry("GlobalKey", "Default"), QStringLiteral("Default")); // key is gone +        QCOMPARE(mainGroup.keyList(), expectedKeys); +    }  }  void KConfigTest::testAnonymousConfig() diff --git a/autotests/kconfigtest.h b/autotests/kconfigtest.h index d6502331..9d519eda 100644 --- a/autotests/kconfigtest.h +++ b/autotests/kconfigtest.h @@ -76,6 +76,7 @@ private Q_SLOTS:      void testLocaleConfig();      void testDirtyAfterRevert();      void testKdeGlobals(); +    void testLocalDeletion();      void testNewlines();      void testXdgListEntry();      void testNotify(); diff --git a/src/core/kconfigdata.cpp b/src/core/kconfigdata.cpp index b479a598..0d6a7bb9 100644 --- a/src/core/kconfigdata.cpp +++ b/src/core/kconfigdata.cpp @@ -18,8 +18,9 @@ QDebug operator<<(QDebug dbg, const KEntryKey &key)  QDebug operator<<(QDebug dbg, const KEntry &entry)  { -    dbg.nospace() << "[" << entry.mValue << (entry.bDirty ? " dirty" : "") << (entry.bGlobal ? " global" : "") << (entry.bImmutable ? " immutable" : "") -                  << (entry.bDeleted ? " deleted" : "") << (entry.bReverted ? " reverted" : "") << (entry.bExpand ? " expand" : "") << "]"; +    dbg.nospace() << "[" << entry.mValue << (entry.bDirty ? " dirty" : "") << (entry.bGlobal ? " global" : "") +                  << (entry.bOverridesGlobal ? " overrides global" : "") << (entry.bImmutable ? " immutable" : "") << (entry.bDeleted ? " deleted" : "") +                  << (entry.bReverted ? " reverted" : "") << (entry.bExpand ? " expand" : "") << "]";      return dbg.space();  } @@ -164,7 +165,7 @@ bool KEntryMap::setEntry(const QByteArray &group, const QByteArray &key, const Q              }          }          if (it.value() != e) { -            // qDebug() << "changing" << k << "from" << e.mValue << "to" << value; +            // qDebug() << "changing" << k << "from" << it.value().mValue << "to" << value << e;              it.value() = e;              if (k.bDefault) {                  KEntryKey nonDefaultKey(k); diff --git a/src/core/kconfigini.cpp b/src/core/kconfigini.cpp index a67e490f..2792cb4f 100644 --- a/src/core/kconfigini.cpp +++ b/src/core/kconfigini.cpp @@ -437,7 +437,7 @@ bool KConfigIniBackend::writeConfig(const QByteArray &locale, KEntryMap &entryMa              } else {                  KEntryKey defaultKey = key;                  defaultKey.bDefault = true; -                if (!entryMap.contains(defaultKey)) { +                if (!entryMap.contains(defaultKey) && !it->bOverridesGlobal) {                      writeMap.remove(key); // remove the deleted entry if there is no default                      // qDebug() << "Detected as deleted=>removed:" << key.mGroup << key.mKey << "global=" << bGlobal;                  } else { | 
