KConfig

kconfigini.cpp
1/*
2 This file is part of the KDE libraries
3 SPDX-FileCopyrightText: 2006, 2007 Thomas Braxton <kde.braxton@gmail.com>
4 SPDX-FileCopyrightText: 1999 Preston Brown <pbrown@kde.org>
5 SPDX-FileCopyrightText: 1997-1999 Matthias Kalle Dalheimer <kalle@kde.org>
6
7 SPDX-License-Identifier: LGPL-2.0-or-later
8*/
9
10#include "kconfigini_p.h"
11
12#include "kconfig_core_log_settings.h"
13#include "kconfigdata_p.h"
14
15#include <QDateTime>
16#include <QDebug>
17#include <QDir>
18#include <QFile>
19#include <QFileInfo>
20#include <QLockFile>
21#include <QSaveFile>
22#include <QStandardPaths>
23#include <qplatformdefs.h>
24
25#ifndef Q_OS_WIN
26#include <unistd.h> // getuid, close
27#endif
28#include <fcntl.h> // open
29#include <sys/types.h> // uid_t
30
31using namespace Qt::StringLiterals;
32
33KCONFIGCORE_EXPORT bool kde_kiosk_exception = false; // flag to disable kiosk restrictions
34
36{
37 auto it = cache->constFind(fragment);
38 if (it != cache->constEnd()) {
39 return it.value();
40 }
41
42 return cache->insert(fragment, fragment.toByteArray()).value();
43}
44
45QString KConfigIniBackend::warningProlog(const QFile &file, int line)
46{
47 // %2 then %1 i.e. int before QString, so that the QString is last
48 // This avoids a wrong substitution if the fileName itself contains %1
49 return QStringLiteral("KConfigIni: In file %2, line %1:").arg(line).arg(file.fileName());
50}
51
52KConfigIniBackend::KConfigIniBackend()
53{
54}
55
56KConfigIniBackend::ParseInfo KConfigIniBackend::parseConfig(const QByteArray &currentLocale, KEntryMap &entryMap, ParseOptions options)
57{
58 return parseConfig(currentLocale, entryMap, options, false);
59}
60
61// merging==true is the merging that happens at the beginning of writeConfig:
62// merge changes in the on-disk file with the changes in the KConfig object.
63KConfigIniBackend::ParseInfo KConfigIniBackend::parseConfig(const QByteArray &currentLocale, KEntryMap &entryMap, ParseOptions options, bool merging)
64{
65 if (filePath().isEmpty()) {
66 return ParseOk;
67 }
68
69 QFile file(filePath());
71 return file.exists() ? ParseOpenError : ParseOk;
72 }
73
74 QList<QString> immutableGroups;
75
76 bool fileOptionImmutable = false;
77 bool groupOptionImmutable = false;
78 bool groupSkip = false;
79
80 int lineNo = 0;
81 // on systems using \r\n as end of line, \r will be taken care of by
82 // trim() below
83 QByteArray buffer = file.readAll();
84 QByteArrayView contents(buffer.data(), buffer.size());
85
86 const int langIdx = currentLocale.indexOf('_');
87 const QByteArray currentLanguage = langIdx >= 0 ? currentLocale.left(langIdx) : currentLocale;
88
89 QString currentGroup = QStringLiteral("<default>");
90 bool bDefault = options & ParseDefaults;
91 bool allowExecutableValues = options & ParseExpansions;
92
93 // Reduce memory overhead by making use of implicit sharing
94 // This assumes that config files contain only a small amount of
95 // different fragments which are repeated often.
96 // This is often the case, especially sub groups will all have
97 // the same list of keys and similar values as well.
99 cache.reserve(4096);
100
101 while (!contents.isEmpty()) {
102 QByteArrayView line;
103 if (const auto idx = contents.indexOf('\n'); idx < 0) {
104 line = contents;
105 contents = {};
106 } else {
107 line = contents.left(idx);
108 contents = contents.mid(idx + 1);
109 }
110 line = line.trimmed();
111 ++lineNo;
112
113 // skip empty lines and lines beginning with '#'
114 if (line.isEmpty() || line.at(0) == '#') {
115 continue;
116 }
117
118 if (line.at(0) == '[') { // found a group
119 groupOptionImmutable = fileOptionImmutable;
120
121 QByteArray newGroup;
122 int start = 1;
123 int end = 0;
124 do {
125 end = start;
126 for (;;) {
127 if (end == line.length()) {
128 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, lineNo) << "Invalid group header.";
129 // XXX maybe reset the current group here?
130 goto next_line;
131 }
132 if (line.at(end) == ']') {
133 break;
134 }
135 ++end;
136 }
137 /* clang-format off */
138 if (end + 1 == line.length()
139 && start + 2 == end
140 && line.at(start) == '$'
141 && line.at(start + 1) == 'i') { /* clang-format on */
142 if (newGroup.isEmpty()) {
143 fileOptionImmutable = !kde_kiosk_exception;
144 } else {
145 groupOptionImmutable = !kde_kiosk_exception;
146 }
147 } else {
148 if (!newGroup.isEmpty()) {
149 newGroup += '\x1d';
150 }
151 QByteArrayView namePart = line.mid(start, end - start);
152 printableToString(namePart, file, lineNo);
153 newGroup += namePart.toByteArray();
154 }
155 } while ((start = end + 2) <= line.length() && line.at(end + 1) == '[');
156 currentGroup = QString::fromUtf8(newGroup);
157
158 groupSkip = entryMap.getEntryOption(currentGroup, {}, {}, KEntryMap::EntryImmutable);
159
160 if (groupSkip && !bDefault) {
161 continue;
162 }
163
164 if (groupOptionImmutable)
165 // Do not make the groups immutable until the entries from
166 // this file have been added.
167 {
168 immutableGroups.append(currentGroup);
169 }
170 } else {
171 if (groupSkip && !bDefault) {
172 continue; // skip entry
173 }
174
175 QByteArrayView aKey;
176 int eqpos = line.indexOf('=');
177 if (eqpos < 0) {
178 aKey = line;
179 line = {};
180 } else {
181 QByteArrayView temp = line.left(eqpos);
182 aKey = temp.trimmed();
183 line = line.mid(eqpos + 1);
184 line = line.trimmed();
185 }
186 if (aKey.isEmpty()) {
187 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, lineNo) << "Invalid entry (empty key)";
188 continue;
189 }
190
191 KEntryMap::EntryOptions entryOptions = {};
192 if (groupOptionImmutable) {
193 entryOptions |= KEntryMap::EntryImmutable;
194 }
195
196 QByteArrayView locale;
197 int start;
198 while ((start = aKey.lastIndexOf('[')) >= 0) {
199 int end = aKey.indexOf(']', start);
200 if (end < 0) {
201 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, lineNo) << "Invalid entry (missing ']')";
202 goto next_line;
203 } else if (end > start + 1 && aKey.at(start + 1) == '$') { // found option(s)
204 int i = start + 2;
205 while (i < end) {
206 switch (aKey.at(i)) {
207 case 'i':
208 if (!kde_kiosk_exception) {
209 entryOptions |= KEntryMap::EntryImmutable;
210 }
211 break;
212 case 'e':
213 if (allowExecutableValues) {
214 entryOptions |= KEntryMap::EntryExpansion;
215 }
216 break;
217 case 'd':
218 entryOptions |= KEntryMap::EntryDeleted;
219 aKey.truncate(start);
220 printableToString(aKey, file, lineNo);
221 entryMap.setEntry(currentGroup, aKey.toByteArray(), QByteArray(), entryOptions);
222 goto next_line;
223 default:
224 break;
225 }
226 ++i;
227 }
228 } else { // found a locale
229 if (!locale.isNull()) {
230 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, lineNo) << "Invalid entry (second locale!?)";
231 goto next_line;
232 }
233
234 locale = aKey.mid(start + 1, end - start - 1);
235 }
236 aKey.truncate(start);
237 }
238 if (eqpos < 0) { // Do this here after [$d] was checked
239 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, lineNo) << "Invalid entry (missing '=')";
240 continue;
241 }
242 printableToString(aKey, file, lineNo);
243 if (!locale.isEmpty()) {
244 if (locale != currentLocale && locale != currentLanguage) {
245 // backward compatibility. C == en_US
246 if (locale.at(0) != 'C' || currentLocale != "en_US") {
247 if (merging) {
248 entryOptions |= KEntryMap::EntryRawKey;
249 } else {
250 goto next_line; // skip this entry if we're not merging
251 }
252 }
253 }
254 }
255
256 if (options & ParseGlobal) {
257 entryOptions |= KEntryMap::EntryGlobal;
258 }
259 if (bDefault) {
260 entryOptions |= KEntryMap::EntryDefault;
261 }
262 if (!locale.isNull()) {
263 entryOptions |= KEntryMap::EntryLocalized;
264 if (locale.indexOf('_') != -1) {
265 entryOptions |= KEntryMap::EntryLocalizedCountry;
266 }
267 }
268 printableToString(line, file, lineNo);
269 if (entryOptions & KEntryMap::EntryRawKey) {
270 QByteArray rawKey;
271 rawKey.reserve(aKey.length() + locale.length() + 2);
272 rawKey.append(aKey);
273 rawKey.append('[').append(locale).append(']');
274 entryMap.setEntry(currentGroup, rawKey, lookup(line, &cache), entryOptions);
275 } else {
276 entryMap.setEntry(currentGroup, lookup(aKey, &cache), lookup(line, &cache), entryOptions);
277 }
278 }
279 next_line:
280 continue;
281 }
282
283 // now make sure immutable groups are marked immutable
284 for (const QString &group : std::as_const(immutableGroups)) {
285 entryMap.setEntry(group, QByteArray(), QByteArray(), KEntryMap::EntryImmutable);
286 }
287
288 return fileOptionImmutable ? ParseImmutable : ParseOk;
289}
290
291void KConfigIniBackend::writeEntries(const QByteArray &locale, QIODevice &file, const KEntryMap &map, bool defaultGroup, bool primaryGroup, bool &firstEntry)
292{
293 QString currentGroup;
294 bool groupIsImmutable = false;
295 for (const auto &[key, entry] : map) {
296 // Either process the default group or all others
297 if ((key.mGroup != QStringLiteral("<default>")) == defaultGroup) {
298 continue; // skip
299 }
300 // Either process the primary group or all others
301 if ((mPrimaryGroup.isNull() || key.mGroup != mPrimaryGroup) == primaryGroup) {
302 continue; // skip
303 }
304
305 // the only thing we care about groups is, is it immutable?
306 if (key.mKey.isNull()) {
307 groupIsImmutable = entry.bImmutable;
308 continue; // skip
309 }
310
311 const KEntry &currentEntry = entry;
312 if (!defaultGroup && currentGroup != key.mGroup) {
313 if (!firstEntry) {
314 file.putChar('\n');
315 }
316 currentGroup = key.mGroup;
317 for (int start = 0, end;; start = end + 1) {
318 file.putChar('[');
319 end = currentGroup.indexOf(QLatin1Char('\x1d'), start);
320 if (end < 0) {
321 int cgl = currentGroup.length();
322 if (currentGroup.at(start) == QLatin1Char('$') && cgl - start <= 10) {
323 for (int i = start + 1; i < cgl; i++) {
324 const QChar c = currentGroup.at(i);
325 if (c < QLatin1Char('a') || c > QLatin1Char('z')) {
326 goto nope;
327 }
328 }
329 file.write("\\x24");
330 ++start;
331 }
332 nope:
333 // TODO: make stringToPrintable also process QString, to save the conversion here and below
334 file.write(stringToPrintable(QStringView(currentGroup).mid(start).toUtf8(), GroupString));
335 file.putChar(']');
336 if (groupIsImmutable) {
337 file.write("[$i]", 4);
338 }
339 file.putChar('\n');
340 break;
341 } else {
342 file.write(stringToPrintable(QStringView(currentGroup).mid(start, end - start).toUtf8(), GroupString));
343 file.putChar(']');
344 }
345 }
346 }
347
348 firstEntry = false;
349 // it is data for a group
350
351 if (key.bRaw) { // unprocessed key with attached locale from merge
352 file.write(key.mKey);
353 } else {
354 file.write(stringToPrintable(key.mKey, KeyString)); // Key
355 if (key.bLocal && locale != "C") { // 'C' locale == untranslated
356 file.putChar('[');
357 file.write(locale); // locale tag
358 file.putChar(']');
359 }
360 }
361 if (currentEntry.bDeleted) {
362 if (currentEntry.bImmutable) {
363 file.write("[$di]", 5); // Deleted + immutable
364 } else {
365 file.write("[$d]", 4); // Deleted
366 }
367 } else {
368 if (currentEntry.bImmutable || currentEntry.bExpand) {
369 file.write("[$", 2);
370 if (currentEntry.bImmutable) {
371 file.putChar('i');
372 }
373 if (currentEntry.bExpand) {
374 file.putChar('e');
375 }
376 file.putChar(']');
377 }
378 file.putChar('=');
379 file.write(stringToPrintable(currentEntry.mValue, ValueString));
380 }
381 file.putChar('\n');
382 }
383}
384
385void KConfigIniBackend::writeEntries(const QByteArray &locale, QIODevice &file, const KEntryMap &map)
386{
387 bool firstEntry = true;
388
389 // write default group
390 writeEntries(locale, file, map, true, false, firstEntry);
391
392 if (!mPrimaryGroup.isNull()) {
393 // write the primary group - it needs to be written before all other groups
394 writeEntries(locale, file, map, false, true, firstEntry);
395 }
396
397 // write all other groups
398 writeEntries(locale, file, map, false, false, firstEntry);
399}
400
401bool KConfigIniBackend::writeConfig(const QByteArray &locale, KEntryMap &entryMap, WriteOptions options)
402{
403 Q_ASSERT(!filePath().isEmpty());
404
405 KEntryMap writeMap;
406 const bool bGlobal = options & WriteGlobal;
407
408 // First, reparse the file on disk, to merge our changes with the ones done by other apps
409 // Store the result into writeMap.
410 {
411 ParseOptions opts = ParseExpansions;
412 if (bGlobal) {
413 opts |= ParseGlobal;
414 }
415 ParseInfo info = parseConfig(locale, writeMap, opts, true);
416 if (info != ParseOk) { // either there was an error or the file became immutable
417 return false;
418 }
419 }
420
421 for (auto &[key, entry] : entryMap) {
422 if (!key.mKey.isEmpty() && !entry.bDirty) { // not dirty, doesn't overwrite entry in writeMap. skips default entries, too.
423 continue;
424 }
425
426 // only write entries that have the same "globality" as the file
427 if (entry.bGlobal == bGlobal) {
428 if (entry.bReverted && entry.bOverridesGlobal) {
429 entry.bDeleted = true;
430 writeMap[key] = entry;
431 } else if (entry.bReverted) {
432 writeMap.erase(key);
433 } else if (!entry.bDeleted) {
434 writeMap[key] = entry;
435 } else {
436 KEntryKey defaultKey = key;
437 defaultKey.bDefault = true;
438 if (entryMap.find(defaultKey) == entryMap.end() && !entry.bOverridesGlobal) {
439 writeMap.erase(key); // remove the deleted entry if there is no default
440 // qDebug() << "Detected as deleted=>removed:" << key.mGroup << key.mKey << "global=" << bGlobal;
441 } else {
442 writeMap[key] = entry; // otherwise write an explicitly deleted entry
443 // qDebug() << "Detected as deleted=>[$d]:" << key.mGroup << key.mKey << "global=" << bGlobal;
444 }
445 }
446 entry.bDirty = false;
447 }
448 }
449
450 // now writeMap should contain only entries to be written
451 // so write it out to disk
452
453 // check if file exists
454 QFile::Permissions fileMode = filePath().startsWith(u"/etc/xdg/"_s) ? QFile::ReadUser | QFile::WriteUser | QFile::ReadGroup | QFile::ReadOther //
456
457 bool createNew = true;
458
459 QFileInfo fi(filePath());
460 if (fi.exists()) {
461#ifdef Q_OS_WIN
462 // TODO: getuid does not exist on windows, use GetSecurityInfo and GetTokenInformation instead
463 createNew = false;
464#else
465 if (fi.ownerId() == ::getuid()) {
466 // Preserve file mode if file exists and is owned by user.
467 fileMode = fi.permissions();
468 } else {
469 // File is not owned by user:
470 // Don't create new file but write to existing file instead.
471 createNew = false;
472 }
473#endif
474 }
475
476 if (createNew) {
477 QSaveFile file(filePath());
478 if (!file.open(QIODevice::WriteOnly)) {
479#ifdef Q_OS_ANDROID
480 // HACK: when we are dealing with content:// URIs, QSaveFile has to rely on DirectWrite.
481 // Otherwise this method returns a false and we're done.
482 file.setDirectWriteFallback(true);
483 if (!file.open(QIODevice::WriteOnly)) {
484 qWarning(KCONFIG_CORE_LOG) << "Couldn't create a new file:" << filePath() << ". Error:" << file.errorString();
485 return false;
486 }
487#else
488 qWarning(KCONFIG_CORE_LOG) << "Couldn't create a new file:" << filePath() << ". Error:" << file.errorString();
489 return false;
490#endif
491 }
492
493 file.setTextModeEnabled(true); // to get eol translation
494 writeEntries(locale, file, writeMap);
495
496 if (!file.size() && (fileMode == (QFile::ReadUser | QFile::WriteUser))) {
497 // File is empty and doesn't have special permissions: delete it.
498 file.cancelWriting();
499
500 if (fi.exists()) {
501 // also remove the old file in case it existed. this can happen
502 // when we delete all the entries in an existing config file.
503 // if we don't do this, then deletions and revertToDefault's
504 // will mysteriously fail
505 QFile::remove(filePath());
506 }
507 } else {
508 // Normal case: Close the file
509 if (file.commit()) {
510 QFile::setPermissions(filePath(), fileMode);
511 return true;
512 }
513 // Couldn't write. Disk full?
514 qCWarning(KCONFIG_CORE_LOG) << "Couldn't write" << filePath() << ". Disk full?";
515 return false;
516 }
517 } else {
518 // Open existing file. *DON'T* create it if it suddenly does not exist!
519#if defined(Q_OS_UNIX) && !defined(Q_OS_ANDROID)
520 int fd = QT_OPEN(QFile::encodeName(filePath()).constData(), O_WRONLY | O_TRUNC);
521 if (fd < 0) {
522 return false;
523 }
524 QFile f;
525 if (!f.open(fd, QIODevice::WriteOnly)) {
526 QT_CLOSE(fd);
527 return false;
528 }
529 writeEntries(locale, f, writeMap);
530 f.close();
531 QT_CLOSE(fd);
532#else
533 QFile f(filePath());
534 // XXX This is broken - it DOES create the file if it is suddenly gone.
536 return false;
537 }
538 f.setTextModeEnabled(true);
539 writeEntries(locale, f, writeMap);
540#endif
541 }
542 return true;
543}
544
545bool KConfigIniBackend::isWritable() const
546{
547 const QString filePath = this->filePath();
548 if (filePath.isEmpty()) {
549 return false;
550 }
551
552 QFileInfo file(filePath);
553 if (file.exists()) {
554 return file.isWritable();
555 }
556
557 // If the file does not exist, check if the deepest existing dir is writable
558 QFileInfo dir(file.absolutePath());
559 while (!dir.exists()) {
560 QString parent = dir.absolutePath(); // Go up. Can't use cdUp() on non-existing dirs.
561 if (parent == dir.filePath()) {
562 // no parent
563 return false;
564 }
565 dir.setFile(parent);
566 }
567 return dir.isDir() && dir.isWritable();
568}
569
570QString KConfigIniBackend::nonWritableErrorMessage() const
571{
572 return tr("Configuration file \"%1\" not writable.\n").arg(filePath());
573}
574
575void KConfigIniBackend::createEnclosing()
576{
577 const QString file = filePath();
578 if (file.isEmpty()) {
579 return; // nothing to do
580 }
581
582 // Create the containing dir, maybe it wasn't there
583 QDir().mkpath(QFileInfo(file).absolutePath());
584}
585
586void KConfigIniBackend::setFilePath(const QString &path)
587{
588 if (path.isEmpty()) {
589 return;
590 }
591
592 Q_ASSERT(QDir::isAbsolutePath(path));
593
594 const QFileInfo info(path);
595 if (info.exists()) {
596 setLocalFilePath(info.canonicalFilePath());
597 return;
598 }
599
600 if (QString filePath = info.dir().canonicalPath(); !filePath.isEmpty()) {
601 filePath += QLatin1Char('/') + info.fileName();
602 setLocalFilePath(filePath);
603 } else {
604 setLocalFilePath(path);
605 }
606}
607
608KConfigBase::AccessMode KConfigIniBackend::accessMode() const
609{
610 if (filePath().isEmpty()) {
611 return KConfigBase::NoAccess;
612 }
613
614 if (isWritable()) {
615 return KConfigBase::ReadWrite;
616 }
617
618 return KConfigBase::ReadOnly;
619}
620
621bool KConfigIniBackend::lock()
622{
623 Q_ASSERT(!filePath().isEmpty());
624
625 m_mutex.lock();
626#ifdef Q_OS_ANDROID
627 if (!lockFile) {
628 // handle content Uris properly
629 if (filePath().startsWith(QLatin1String("content://"))) {
630 // we can't create file at an arbitrary location, so use internal storage to create one
631
632 // NOTE: filename can be the same, but because this lock is short lived we may never have a collision
633 lockFile = std::make_unique<QLockFile>(QStandardPaths::writableLocation(QStandardPaths::GenericCacheLocation) + QLatin1String("/")
634 + QFileInfo(filePath()).fileName() + QLatin1String(".lock"));
635 } else {
636 lockFile = std::make_unique<QLockFile>(filePath() + QLatin1String(".lock"));
637 }
638 }
639#else
640 if (!lockFile) {
641 lockFile = std::make_unique<QLockFile>(filePath() + QLatin1String(".lock"));
642 }
643#endif
644
645 if (!lockFile->lock()) {
646 m_mutex.unlock();
647 }
648
649 return lockFile->isLocked();
650}
651
652void KConfigIniBackend::unlock()
653{
654 lockFile->unlock();
655 lockFile = nullptr;
656 m_mutex.unlock();
657}
658
659bool KConfigIniBackend::isLocked() const
660{
661 return lockFile && lockFile->isLocked();
662}
663
664namespace
665{
666// serialize an escaped byte at the end of @param data
667// @param data should have room for 4 bytes
668char *escapeByte(char *data, unsigned char s)
669{
670 static const char nibbleLookup[] = {'0', '1', '2', '3', '4', '5', '6', '7', '8', '9', 'a', 'b', 'c', 'd', 'e', 'f'};
671 *data++ = '\\';
672 *data++ = 'x';
673 *data++ = nibbleLookup[s >> 4];
674 *data++ = nibbleLookup[s & 0x0f];
675 return data;
676}
677
678// Struct that represents a multi-byte UTF-8 character.
679// This struct is used to keep track of bytes that seem to be valid
680// UTF-8.
681struct Utf8Char {
682public:
683 unsigned char bytes[4];
684 unsigned char count;
685 unsigned char charLength;
686
687 Utf8Char()
688 {
689 clear();
690 charLength = 0;
691 }
692 void clear()
693 {
694 count = 0;
695 }
696 // Add a byte to the UTF8 character.
697 // When an additional byte leads to an invalid character, return false.
698 bool addByte(unsigned char b)
699 {
700 if (count == 0) {
701 if (b > 0xc1 && (b & 0xe0) == 0xc0) {
702 charLength = 2;
703 } else if ((b & 0xf0) == 0xe0) {
704 charLength = 3;
705 } else if (b < 0xf5 && (b & 0xf8) == 0xf0) {
706 charLength = 4;
707 } else {
708 return false;
709 }
710 bytes[0] = b;
711 count = 1;
712 } else if (count < 4 && (b & 0xc0) == 0x80) {
713 if (count == 1) {
714 if (charLength == 3 && bytes[0] == 0xe0 && b < 0xa0) {
715 return false; // overlong 3 byte sequence
716 }
717 if (charLength == 4) {
718 if (bytes[0] == 0xf0 && b < 0x90) {
719 return false; // overlong 4 byte sequence
720 }
721 if (bytes[0] == 0xf4 && b > 0x8f) {
722 return false; // Unicode value larger than U+10FFFF
723 }
724 }
725 }
726 bytes[count++] = b;
727 } else {
728 return false;
729 }
730 return true;
731 }
732 // Return true if Utf8Char contains one valid character.
733 bool isComplete() const
734 {
735 return count > 0 && count == charLength;
736 }
737 // Add the bytes in this UTF8 character in escaped form to data.
738 char *escapeBytes(char *data)
739 {
740 for (unsigned char i = 0; i < count; ++i) {
741 data = escapeByte(data, bytes[i]);
742 }
743 clear();
744 return data;
745 }
746 // Add the bytes of the UTF8 character to a buffer.
747 // Only call this if isComplete() returns true.
748 char *writeUtf8(char *data)
749 {
750 for (unsigned char i = 0; i < count; ++i) {
751 *data++ = bytes[i];
752 }
753 clear();
754 return data;
755 }
756 // Write the bytes in the UTF8 character literally, or, if the
757 // character is not complete, write the escaped bytes.
758 // This is useful to handle the state that remains after handling
759 // all bytes in a buffer.
760 char *write(char *data)
761 {
762 if (isComplete()) {
763 data = writeUtf8(data);
764 } else {
765 data = escapeBytes(data);
766 }
767 return data;
768 }
769};
770}
771
772QByteArray KConfigIniBackend::stringToPrintable(const QByteArray &aString, StringType type)
773{
774 const int len = aString.size();
775 if (len == 0) {
776 return aString;
777 }
778
779 QByteArray result; // Guesstimated that it's good to avoid data() initialization for a length of len*4
780 result.resize(len * 4); // Maximum 4x as long as source string due to \x<ab> escape sequences
781 const char *s = aString.constData();
782 int i = 0;
783 char *data = result.data();
784 char *start = data;
785
786 // Protect leading space
787 if (s[0] == ' ' && type != GroupString) {
788 *data++ = '\\';
789 *data++ = 's';
790 ++i;
791 }
792 Utf8Char utf8;
793
794 for (; i < len; ++i) {
795 switch (s[i]) {
796 default:
797 if (utf8.addByte(s[i])) {
798 break;
799 } else {
800 data = utf8.escapeBytes(data);
801 }
802 // The \n, \t, \r cases (all < 32) are handled below; we can ignore them here
803 if (((unsigned char)s[i]) < 32) {
804 goto doEscape;
805 }
806 // GroupString and KeyString should be valid UTF-8, but ValueString
807 // can be a bytearray with non-UTF-8 bytes that should be escaped.
808 if (type == ValueString && ((unsigned char)s[i]) >= 127) {
809 goto doEscape;
810 }
811 *data++ = s[i];
812 break;
813 case '\n':
814 *data++ = '\\';
815 *data++ = 'n';
816 break;
817 case '\t':
818 *data++ = '\\';
819 *data++ = 't';
820 break;
821 case '\r':
822 *data++ = '\\';
823 *data++ = 'r';
824 break;
825 case '\\':
826 *data++ = '\\';
827 *data++ = '\\';
828 break;
829 case '=':
830 if (type != KeyString) {
831 *data++ = s[i];
832 break;
833 }
834 goto doEscape;
835 case '[':
836 case ']':
837 // Above chars are OK to put in *value* strings as plaintext
838 if (type == ValueString) {
839 *data++ = s[i];
840 break;
841 }
842 doEscape:
843 data = escapeByte(data, s[i]);
844 break;
845 }
846 if (utf8.isComplete()) {
847 data = utf8.writeUtf8(data);
848 }
849 }
850 data = utf8.write(data);
851 *data = 0;
852 result.resize(data - start);
853
854 // Protect trailing space
855 if (result.endsWith(' ') && type != GroupString) {
856 result.replace(result.length() - 1, 1, "\\s");
857 }
858
859 return result;
860}
861
862char KConfigIniBackend::charFromHex(const char *str, const QFile &file, int line)
863{
864 unsigned char ret = 0;
865 for (int i = 0; i < 2; i++) {
866 ret <<= 4;
867 quint8 c = quint8(str[i]);
868
869 if (c >= '0' && c <= '9') {
870 ret |= c - '0';
871 } else if (c >= 'a' && c <= 'f') {
872 ret |= c - 'a' + 0x0a;
873 } else if (c >= 'A' && c <= 'F') {
874 ret |= c - 'A' + 0x0a;
875 } else {
876 QByteArray e(str, 2);
877 e.prepend("\\x");
878 qCWarning(KCONFIG_CORE_LOG) << warningProlog(file, line) << "Invalid hex character " << c << " in \\x<nn>-type escape sequence \"" << e.constData()
879 << "\".";
880 return 'x';
881 }
882 }
883 return char(ret);
884}
885
886void KConfigIniBackend::printableToString(QByteArrayView &aString, const QFile &file, int line)
887{
888 if (aString.isEmpty() || aString.indexOf('\\') == -1) {
889 return;
890 }
891 aString = aString.trimmed();
892 int l = aString.size();
893 char *r = const_cast<char *>(aString.data());
894 char *str = r;
895
896 for (int i = 0; i < l; i++, r++) {
897 if (str[i] != '\\') {
898 *r = str[i];
899 } else {
900 // Probable escape sequence
901 ++i;
902 if (i >= l) { // Line ends after backslash - stop.
903 *r = '\\';
904 break;
905 }
906
907 switch (str[i]) {
908 case 's':
909 *r = ' ';
910 break;
911 case 't':
912 *r = '\t';
913 break;
914 case 'n':
915 *r = '\n';
916 break;
917 case 'r':
918 *r = '\r';
919 break;
920 case '\\':
921 *r = '\\';
922 break;
923 case ';':
924 // not really an escape sequence, but allowed in .desktop files, don't strip '\;' from the string
925 *r = '\\';
926 ++r;
927 *r = ';';
928 break;
929 case ',':
930 // not really an escape sequence, but allowed in .desktop files, don't strip '\,' from the string
931 *r = '\\';
932 ++r;
933 *r = ',';
934 break;
935 case 'x':
936 if (i + 2 < l) {
937 *r = charFromHex(str + i + 1, file, line);
938 i += 2;
939 } else {
940 *r = 'x';
941 i = l - 1;
942 }
943 break;
944 default:
945 *r = '\\';
946 qCWarning(KCONFIG_CORE_LOG).noquote() << warningProlog(file, line) << QStringLiteral("Invalid escape sequence: «\\%1»").arg(str[i]);
947 }
948 }
949 }
950 aString.truncate(r - aString.constData());
951}
952
953QString KConfigIniBackend::filePath() const
954{
955 return mLocalFilePath;
956}
957
958void KConfigIniBackend::setLocalFilePath(const QString &file)
959{
960 mLocalFilePath = file;
961}
962
963void KConfigIniBackend::setPrimaryGroup(const QString &group)
964{
965 mPrimaryGroup = group;
966}
967
968#include "moc_kconfigini_p.cpp"
AccessMode
Possible return values for accessMode().
Q_SCRIPTABLE QString start(QString train="")
QAction * end(const QObject *recvr, const char *slot, QObject *parent)
QString path(const QString &relativePath)
KIOCORE_EXPORT QString dir(const QString &fileClass)
KGuiItem clear()
QByteArray & append(QByteArrayView data)
const char * constData() const const
char * data()
bool endsWith(QByteArrayView bv) const const
qsizetype indexOf(QByteArrayView bv, qsizetype from) const const
bool isEmpty() const const
QByteArray left(qsizetype len) const const
qsizetype length() const const
QByteArray & replace(QByteArrayView before, QByteArrayView after)
void reserve(qsizetype size)
void resize(qsizetype newSize, char c)
qsizetype size() const const
QByteArrayView left(qsizetype length) const const
QByteArrayView mid(qsizetype start, qsizetype length) const const
char at(qsizetype n) const const
const_pointer constData() const const
const_pointer data() const const
qsizetype indexOf(QByteArrayView bv, qsizetype from) const const
bool isEmpty() const const
bool isNull() const const
qsizetype lastIndexOf(QByteArrayView bv) const const
qsizetype length() const const
qsizetype size() const const
QByteArray toByteArray() const const
QByteArrayView trimmed() const const
void truncate(qsizetype length)
bool isAbsolutePath(const QString &path)
bool mkpath(const QString &dirPath) const const
QByteArray encodeName(const QString &fileName)
bool exists(const QString &fileName)
virtual QString fileName() const const override
bool open(FILE *fh, OpenMode mode, FileHandleFlags handleFlags)
bool remove()
virtual bool setPermissions(Permissions permissions) override
typedef Permissions
virtual void close() override
const_iterator constEnd() const const
const_iterator constFind(const Key &key) const const
iterator insert(const Key &key, const T &value)
void reserve(qsizetype size)
QString errorString() const const
bool isWritable() const const
virtual bool open(QIODeviceBase::OpenMode mode)
bool putChar(char c)
QByteArray readAll()
void setTextModeEnabled(bool enabled)
virtual qint64 size() const const
qint64 write(const QByteArray &data)
void append(QList< T > &&value)
QString writableLocation(StandardLocation type)
QString arg(Args &&... args) const const
const QChar at(qsizetype position) const const
QString fromUtf8(QByteArrayView str)
qsizetype indexOf(QChar ch, qsizetype from, Qt::CaseSensitivity cs) const const
bool isEmpty() const const
qsizetype length() const const
This file is part of the KDE documentation.
Documentation copyright © 1996-2025 The KDE developers.
Generated on Fri Mar 28 2025 11:52:18 by doxygen 1.13.2 written by Dimitri van Heesch, © 1997-2006

KDE's Doxygen guidelines are available online.