#include "mysettings.h" #include "modellist.h" #include #include #include #include #include #include static int default_threadCount = std::min(4, (int32_t) std::thread::hardware_concurrency()); static bool default_saveChats = false; static bool default_saveChatGPTChats = true; static bool default_serverChat = false; static QString default_userDefaultModel = "Application default"; static bool default_forceMetal = false; static QString default_lastVersionStarted = ""; static int default_localDocsChunkSize = 256; static QString default_chatTheme = "Dark"; static int default_localDocsRetrievalSize = 3; static bool default_localDocsShowReferences = true; static QString default_networkAttribution = ""; static bool default_networkIsActive = false; static bool default_networkUsageStatsActive = false; static QString defaultLocalModelsPath() { QString localPath = QStandardPaths::writableLocation(QStandardPaths::AppLocalDataLocation) + "/"; QString testWritePath = localPath + QString("test_write.txt"); QString canonicalLocalPath = QFileInfo(localPath).canonicalFilePath() + "/"; QDir localDir(localPath); if (!localDir.exists()) { if (!localDir.mkpath(localPath)) { qWarning() << "ERROR: Local download directory can't be created:" << canonicalLocalPath; return canonicalLocalPath; } } if (QFileInfo::exists(testWritePath)) return canonicalLocalPath; QFile testWriteFile(testWritePath); if (testWriteFile.open(QIODeviceBase::ReadWrite)) { testWriteFile.close(); return canonicalLocalPath; } qWarning() << "ERROR: Local download path appears not writeable:" << canonicalLocalPath; return canonicalLocalPath; } class MyPrivateSettings: public MySettings { }; Q_GLOBAL_STATIC(MyPrivateSettings, settingsInstance) MySettings *MySettings::globalInstance() { return settingsInstance(); } MySettings::MySettings() : QObject{nullptr} { QSettings::setDefaultFormat(QSettings::IniFormat); } void MySettings::restoreModelDefaults(const ModelInfo &model) { setModelTemperature(model, model.m_temperature); setModelTopP(model, model.m_topP); setModelTopK(model, model.m_topK);; setModelMaxLength(model, model.m_maxLength); setModelPromptBatchSize(model, model.m_promptBatchSize); setModelRepeatPenalty(model, model.m_repeatPenalty); setModelRepeatPenaltyTokens(model, model.m_repeatPenaltyTokens); setModelPromptTemplate(model, model.m_promptTemplate); setModelSystemPrompt(model, model.m_systemPrompt); } void MySettings::restoreApplicationDefaults() { setThreadCount(default_threadCount); setSaveChats(default_saveChats); setSaveChatGPTChats(default_saveChatGPTChats); setServerChat(default_serverChat); setModelPath(defaultLocalModelsPath()); setUserDefaultModel(default_userDefaultModel); setForceMetal(default_forceMetal); } void MySettings::restoreLocalDocsDefaults() { setLocalDocsChunkSize(default_localDocsChunkSize); setLocalDocsRetrievalSize(default_localDocsRetrievalSize); setLocalDocsShowReferences(default_localDocsShowReferences); } void MySettings::eraseModel(const ModelInfo &m) { QSettings settings; settings.remove(QString("model-%1").arg(m.id())); settings.sync(); } QString MySettings::modelName(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/name", !m.m_name.isEmpty() ? m.m_name : m.m_filename).toString(); } void MySettings::setModelName(const ModelInfo &m, const QString &name, bool force) { if ((modelName(m) == name || m.id().isEmpty()) && !force) return; QSettings setting; if ((m.m_name == name || m.m_filename == name) && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/name"); else setting.setValue(QString("model-%1").arg(m.id()) + "/name", name); if (m.isClone) setting.setValue(QString("model-%1").arg(m.id()) + "/isClone", "true"); setting.sync(); if (!force) emit nameChanged(m); } QString MySettings::modelFilename(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/filename", m.m_filename).toString(); } void MySettings::setModelFilename(const ModelInfo &m, const QString &filename, bool force) { if ((modelFilename(m) == filename || m.id().isEmpty()) && !force) return; QSettings setting; if (m.m_filename == filename && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/filename"); else setting.setValue(QString("model-%1").arg(m.id()) + "/filename", filename); setting.sync(); if (!force) emit filenameChanged(m); } double MySettings::modelTemperature(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/temperature", m.m_temperature).toDouble(); } void MySettings::setModelTemperature(const ModelInfo &m, double t, bool force) { if (modelTemperature(m) == t && !force) return; QSettings setting; if (m.m_temperature == t && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/temperature"); else setting.setValue(QString("model-%1").arg(m.id()) + "/temperature", t); setting.sync(); if (!force) emit temperatureChanged(m); } double MySettings::modelTopP(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/topP", m.m_topP).toDouble(); } void MySettings::setModelTopP(const ModelInfo &m, double p, bool force) { if (modelTopP(m) == p && !force) return; QSettings setting; if (m.m_topP == p && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/topP"); else setting.setValue(QString("model-%1").arg(m.id()) + "/topP", p); setting.sync(); if (!force) emit topPChanged(m); } int MySettings::modelTopK(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/topK", m.m_topK).toInt(); } void MySettings::setModelTopK(const ModelInfo &m, int k, bool force) { if (modelTopK(m) == k && !force) return; QSettings setting; if (m.m_topK == k && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/topK"); else setting.setValue(QString("model-%1").arg(m.id()) + "/topK", k); setting.sync(); if (!force) emit topKChanged(m); } int MySettings::modelMaxLength(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/maxLength", m.m_maxLength).toInt(); } void MySettings::setModelMaxLength(const ModelInfo &m, int l, bool force) { if (modelMaxLength(m) == l && !force) return; QSettings setting; if (m.m_maxLength == l && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/maxLength"); else setting.setValue(QString("model-%1").arg(m.id()) + "/maxLength", l); setting.sync(); if (!force) emit maxLengthChanged(m); } int MySettings::modelPromptBatchSize(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/promptBatchSize", m.m_promptBatchSize).toInt(); } void MySettings::setModelPromptBatchSize(const ModelInfo &m, int s, bool force) { if (modelPromptBatchSize(m) == s && !force) return; QSettings setting; if (m.m_promptBatchSize == s && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/promptBatchSize"); else setting.setValue(QString("model-%1").arg(m.id()) + "/promptBatchSize", s); setting.sync(); if (!force) emit promptBatchSizeChanged(m); } double MySettings::modelRepeatPenalty(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/repeatPenalty", m.m_repeatPenalty).toDouble(); } void MySettings::setModelRepeatPenalty(const ModelInfo &m, double p, bool force) { if (modelRepeatPenalty(m) == p && !force) return; QSettings setting; if (m.m_repeatPenalty == p && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/repeatPenalty"); else setting.setValue(QString("model-%1").arg(m.id()) + "/repeatPenalty", p); setting.sync(); if (!force) emit repeatPenaltyChanged(m); } int MySettings::modelRepeatPenaltyTokens(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/repeatPenaltyTokens", m.m_repeatPenaltyTokens).toInt(); } void MySettings::setModelRepeatPenaltyTokens(const ModelInfo &m, int t, bool force) { if (modelRepeatPenaltyTokens(m) == t && !force) return; QSettings setting; if (m.m_repeatPenaltyTokens == t && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/repeatPenaltyTokens"); else setting.setValue(QString("model-%1").arg(m.id()) + "/repeatPenaltyTokens", t); setting.sync(); if (!force) emit repeatPenaltyTokensChanged(m); } QString MySettings::modelPromptTemplate(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/promptTemplate", m.m_promptTemplate).toString(); } void MySettings::setModelPromptTemplate(const ModelInfo &m, const QString &t, bool force) { if (modelPromptTemplate(m) == t && !force) return; QSettings setting; if (m.m_promptTemplate == t && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/promptTemplate"); else setting.setValue(QString("model-%1").arg(m.id()) + "/promptTemplate", t); setting.sync(); if (!force) emit promptTemplateChanged(m); } QString MySettings::modelSystemPrompt(const ModelInfo &m) const { QSettings setting; setting.sync(); return setting.value(QString("model-%1").arg(m.id()) + "/systemPrompt", m.m_systemPrompt).toString(); } void MySettings::setModelSystemPrompt(const ModelInfo &m, const QString &p, bool force) { if (modelSystemPrompt(m) == p && !force) return; QSettings setting; if (m.m_systemPrompt == p && !m.isClone) setting.remove(QString("model-%1").arg(m.id()) + "/systemPrompt"); else setting.setValue(QString("model-%1").arg(m.id()) + "/systemPrompt", p); setting.sync(); if (!force) emit systemPromptChanged(m); } int MySettings::threadCount() const { QSettings setting; setting.sync(); int c = setting.value("threadCount", default_threadCount).toInt(); // The old thread setting likely left many people with 0 in settings config file, which means // we should reset it to the default going forward if (c <= 0) c = default_threadCount; c = std::max(c, 1); c = std::min(c, QThread::idealThreadCount()); return c; } void MySettings::setThreadCount(int c) { if (threadCount() == c) return; c = std::max(c, 1); c = std::min(c, QThread::idealThreadCount()); QSettings setting; setting.setValue("threadCount", c); setting.sync(); emit threadCountChanged(); } bool MySettings::saveChats() const { QSettings setting; setting.sync(); return setting.value("saveChats", default_saveChats).toBool(); } void MySettings::setSaveChats(bool b) { if (saveChats() == b) return; QSettings setting; setting.setValue("saveChats", b); setting.sync(); emit saveChatsChanged(); } bool MySettings::saveChatGPTChats() const { QSettings setting; setting.sync(); return setting.value("saveChatGPTChats", default_saveChatGPTChats).toBool(); } void MySettings::setSaveChatGPTChats(bool b) { if (saveChatGPTChats() == b) return; QSettings setting; setting.setValue("saveChatGPTChats", b); setting.sync(); emit saveChatGPTChatsChanged(); } bool MySettings::serverChat() const { QSettings setting; setting.sync(); return setting.value("serverChat", default_serverChat).toBool(); } void MySettings::setServerChat(bool b) { if (serverChat() == b) return; QSettings setting; setting.setValue("serverChat", b); setting.sync(); emit serverChatChanged(); } QString MySettings::modelPath() const { QSettings setting; setting.sync(); // We have to migrate the old setting because I changed the setting key recklessly in v2.4.11 // which broke a lot of existing installs const bool containsOldSetting = setting.contains("modelPaths"); if (containsOldSetting) { const bool containsNewSetting = setting.contains("modelPath"); if (!containsNewSetting) setting.setValue("modelPath", setting.value("modelPaths")); setting.remove("modelPaths"); setting.sync(); } return setting.value("modelPath", defaultLocalModelsPath()).toString(); } void MySettings::setModelPath(const QString &p) { QString filePath = (p.startsWith("file://") ? QUrl(p).toLocalFile() : p); QString canonical = QFileInfo(filePath).canonicalFilePath() + "/"; if (modelPath() == canonical) return; QSettings setting; setting.setValue("modelPath", canonical); setting.sync(); emit modelPathChanged(); } QString MySettings::userDefaultModel() const { QSettings setting; setting.sync(); return setting.value("userDefaultModel", default_userDefaultModel).toString(); } void MySettings::setUserDefaultModel(const QString &u) { if (userDefaultModel() == u) return; QSettings setting; setting.setValue("userDefaultModel", u); setting.sync(); emit userDefaultModelChanged(); } QString MySettings::chatTheme() const { QSettings setting; setting.sync(); return setting.value("chatTheme", default_chatTheme).toString(); } void MySettings::setChatTheme(const QString &u) { if(chatTheme() == u) return; QSettings setting; setting.setValue("chatTheme", u); setting.sync(); emit chatThemeChanged(); } bool MySettings::forceMetal() const { return m_forceMetal; } void MySettings::setForceMetal(bool b) { if (m_forceMetal == b) return; m_forceMetal = b; emit forceMetalChanged(b); } QString MySettings::lastVersionStarted() const { QSettings setting; setting.sync(); return setting.value("lastVersionStarted", default_lastVersionStarted).toString(); } void MySettings::setLastVersionStarted(const QString &v) { if (lastVersionStarted() == v) return; QSettings setting; setting.setValue("lastVersionStarted", v); setting.sync(); emit lastVersionStartedChanged(); } int MySettings::localDocsChunkSize() const { QSettings setting; setting.sync(); return setting.value("localdocs/chunkSize", default_localDocsChunkSize).toInt(); } void MySettings::setLocalDocsChunkSize(int s) { if (localDocsChunkSize() == s) return; QSettings setting; setting.setValue("localdocs/chunkSize", s); setting.sync(); emit localDocsChunkSizeChanged(); } int MySettings::localDocsRetrievalSize() const { QSettings setting; setting.sync(); return setting.value("localdocs/retrievalSize", default_localDocsRetrievalSize).toInt(); } void MySettings::setLocalDocsRetrievalSize(int s) { if (localDocsRetrievalSize() == s) return; QSettings setting; setting.setValue("localdocs/retrievalSize", s); setting.sync(); emit localDocsRetrievalSizeChanged(); } bool MySettings::localDocsShowReferences() const { QSettings setting; setting.sync(); return setting.value("localdocs/showReferences", default_localDocsShowReferences).toBool(); } void MySettings::setLocalDocsShowReferences(bool b) { if (localDocsShowReferences() == b) return; QSettings setting; setting.setValue("localdocs/showReferences", b); setting.sync(); emit localDocsShowReferencesChanged(); } QString MySettings::networkAttribution() const { QSettings setting; setting.sync(); return setting.value("network/attribution", default_networkAttribution).toString(); } void MySettings::setNetworkAttribution(const QString &a) { if (networkAttribution() == a) return; QSettings setting; setting.setValue("network/attribution", a); setting.sync(); emit networkAttributionChanged(); } bool MySettings::networkIsActive() const { QSettings setting; setting.sync(); return setting.value("network/isActive", default_networkIsActive).toBool(); } void MySettings::setNetworkIsActive(bool b) { if (networkIsActive() == b) return; QSettings setting; setting.setValue("network/isActive", b); setting.sync(); emit networkIsActiveChanged(); } bool MySettings::networkUsageStatsActive() const { QSettings setting; setting.sync(); return setting.value("network/usageStatsActive", default_networkUsageStatsActive).toBool(); } void MySettings::setNetworkUsageStatsActive(bool b) { if (networkUsageStatsActive() == b) return; QSettings setting; setting.setValue("network/usageStatsActive", b); setting.sync(); emit networkUsageStatsActiveChanged(); } QString MySettings::attemptModelLoad() const { QSettings setting; setting.sync(); return setting.value("attemptModelLoad", QString()).toString(); } void MySettings::setAttemptModelLoad(const QString &modelFile) { if (attemptModelLoad() == modelFile) return; QSettings setting; if (modelFile.isEmpty()) setting.remove("attemptModelLoad"); else setting.setValue("attemptModelLoad", modelFile); setting.sync(); emit attemptModelLoadChanged(); }