2023-06-22 15:44:49 -04:00
# include "modellist.h"
2023-06-28 16:05:35 -04:00
# include "mysettings.h"
2023-07-01 11:34:21 -04:00
# include "network.h"
2023-06-22 15:44:49 -04:00
2023-07-10 16:14:57 -04:00
# include <QFile>
# include <QStandardPaths>
2023-06-22 15:44:49 -04:00
# include <algorithm>
2023-06-28 11:13:33 -04:00
//#define USE_LOCAL_MODELSJSON
2023-07-01 11:34:21 -04:00
QString ModelInfo : : id ( ) const
{
return m_id ;
}
void ModelInfo : : setId ( const QString & id )
{
m_id = id ;
}
QString ModelInfo : : name ( ) const
{
return MySettings : : globalInstance ( ) - > modelName ( * this ) ;
}
void ModelInfo : : setName ( const QString & name )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelName ( * this , name , isClone /*force*/ ) ;
m_name = name ;
}
QString ModelInfo : : filename ( ) const
{
return MySettings : : globalInstance ( ) - > modelFilename ( * this ) ;
}
void ModelInfo : : setFilename ( const QString & filename )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelFilename ( * this , filename , isClone /*force*/ ) ;
m_filename = filename ;
}
double ModelInfo : : temperature ( ) const
{
return MySettings : : globalInstance ( ) - > modelTemperature ( * this ) ;
}
void ModelInfo : : setTemperature ( double t )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelTemperature ( * this , t , isClone /*force*/ ) ;
m_temperature = t ;
}
double ModelInfo : : topP ( ) const
{
return MySettings : : globalInstance ( ) - > modelTopP ( * this ) ;
}
void ModelInfo : : setTopP ( double p )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelTopP ( * this , p , isClone /*force*/ ) ;
m_topP = p ;
}
int ModelInfo : : topK ( ) const
{
return MySettings : : globalInstance ( ) - > modelTopK ( * this ) ;
}
void ModelInfo : : setTopK ( int k )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelTopK ( * this , k , isClone /*force*/ ) ;
m_topK = k ;
}
int ModelInfo : : maxLength ( ) const
{
return MySettings : : globalInstance ( ) - > modelMaxLength ( * this ) ;
}
void ModelInfo : : setMaxLength ( int l )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelMaxLength ( * this , l , isClone /*force*/ ) ;
m_maxLength = l ;
}
int ModelInfo : : promptBatchSize ( ) const
{
return MySettings : : globalInstance ( ) - > modelPromptBatchSize ( * this ) ;
}
void ModelInfo : : setPromptBatchSize ( int s )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelPromptBatchSize ( * this , s , isClone /*force*/ ) ;
m_promptBatchSize = s ;
}
double ModelInfo : : repeatPenalty ( ) const
{
return MySettings : : globalInstance ( ) - > modelRepeatPenalty ( * this ) ;
}
void ModelInfo : : setRepeatPenalty ( double p )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelRepeatPenalty ( * this , p , isClone /*force*/ ) ;
m_repeatPenalty = p ;
}
int ModelInfo : : repeatPenaltyTokens ( ) const
{
return MySettings : : globalInstance ( ) - > modelRepeatPenaltyTokens ( * this ) ;
}
void ModelInfo : : setRepeatPenaltyTokens ( int t )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelRepeatPenaltyTokens ( * this , t , isClone /*force*/ ) ;
m_repeatPenaltyTokens = t ;
}
QString ModelInfo : : promptTemplate ( ) const
{
return MySettings : : globalInstance ( ) - > modelPromptTemplate ( * this ) ;
}
void ModelInfo : : setPromptTemplate ( const QString & t )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelPromptTemplate ( * this , t , isClone /*force*/ ) ;
m_promptTemplate = t ;
}
QString ModelInfo : : systemPrompt ( ) const
{
return MySettings : : globalInstance ( ) - > modelSystemPrompt ( * this ) ;
}
void ModelInfo : : setSystemPrompt ( const QString & p )
{
if ( isClone ) MySettings : : globalInstance ( ) - > setModelSystemPrompt ( * this , p , isClone /*force*/ ) ;
m_systemPrompt = p ;
}
2023-06-26 09:35:29 -04:00
InstalledModels : : InstalledModels ( QObject * parent )
: QSortFilterProxyModel ( parent )
{
connect ( this , & InstalledModels : : rowsInserted , this , & InstalledModels : : countChanged ) ;
connect ( this , & InstalledModels : : rowsRemoved , this , & InstalledModels : : countChanged ) ;
connect ( this , & InstalledModels : : modelReset , this , & InstalledModels : : countChanged ) ;
connect ( this , & InstalledModels : : layoutChanged , this , & InstalledModels : : countChanged ) ;
}
2023-06-22 15:44:49 -04:00
bool InstalledModels : : filterAcceptsRow ( int sourceRow ,
const QModelIndex & sourceParent ) const
{
QModelIndex index = sourceModel ( ) - > index ( sourceRow , 0 , sourceParent ) ;
bool isInstalled = sourceModel ( ) - > data ( index , ModelList : : InstalledRole ) . toBool ( ) ;
return isInstalled ;
}
2023-06-26 09:35:29 -04:00
int InstalledModels : : count ( ) const
{
return rowCount ( ) ;
}
2023-06-22 15:44:49 -04:00
DownloadableModels : : DownloadableModels ( QObject * parent )
: QSortFilterProxyModel ( parent )
, m_expanded ( false )
, m_limit ( 5 )
{
connect ( this , & DownloadableModels : : rowsInserted , this , & DownloadableModels : : countChanged ) ;
connect ( this , & DownloadableModels : : rowsRemoved , this , & DownloadableModels : : countChanged ) ;
connect ( this , & DownloadableModels : : modelReset , this , & DownloadableModels : : countChanged ) ;
connect ( this , & DownloadableModels : : layoutChanged , this , & DownloadableModels : : countChanged ) ;
}
bool DownloadableModels : : filterAcceptsRow ( int sourceRow ,
const QModelIndex & sourceParent ) const
{
bool withinLimit = sourceRow < ( m_expanded ? sourceModel ( ) - > rowCount ( ) : m_limit ) ;
QModelIndex index = sourceModel ( ) - > index ( sourceRow , 0 , sourceParent ) ;
bool isDownloadable = ! sourceModel ( ) - > data ( index , ModelList : : DescriptionRole ) . toString ( ) . isEmpty ( ) ;
bool showInGUI = ! sourceModel ( ) - > data ( index , ModelList : : DisableGUIRole ) . toBool ( ) ;
return withinLimit & & isDownloadable & & showInGUI ;
}
int DownloadableModels : : count ( ) const
{
return rowCount ( ) ;
}
bool DownloadableModels : : isExpanded ( ) const
{
return m_expanded ;
}
void DownloadableModels : : setExpanded ( bool expanded )
{
if ( m_expanded ! = expanded ) {
m_expanded = expanded ;
invalidateFilter ( ) ;
emit expandedChanged ( m_expanded ) ;
}
}
class MyModelList : public ModelList { } ;
Q_GLOBAL_STATIC ( MyModelList , modelListInstance )
ModelList * ModelList : : globalInstance ( )
{
return modelListInstance ( ) ;
}
ModelList : : ModelList ( )
: QAbstractListModel ( nullptr )
, m_installedModels ( new InstalledModels ( this ) )
, m_downloadableModels ( new DownloadableModels ( this ) )
2023-07-12 11:46:40 -04:00
, m_asyncModelRequestOngoing ( false )
2023-06-22 15:44:49 -04:00
{
m_installedModels - > setSourceModel ( this ) ;
m_downloadableModels - > setSourceModel ( this ) ;
m_watcher = new QFileSystemWatcher ( this ) ;
const QString exePath = QCoreApplication : : applicationDirPath ( ) + QDir : : separator ( ) ;
m_watcher - > addPath ( exePath ) ;
2023-06-28 16:05:35 -04:00
m_watcher - > addPath ( MySettings : : globalInstance ( ) - > modelPath ( ) ) ;
2023-06-22 15:44:49 -04:00
connect ( m_watcher , & QFileSystemWatcher : : directoryChanged , this , & ModelList : : updateModelsFromDirectory ) ;
2023-07-01 11:34:21 -04:00
connect ( MySettings : : globalInstance ( ) , & MySettings : : modelPathChanged , this , & ModelList : : updateModelsFromDirectory ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : modelPathChanged , this , & ModelList : : updateModelsFromJson ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : modelPathChanged , this , & ModelList : : updateModelsFromSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : nameChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : temperatureChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : topPChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : topKChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : maxLengthChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : promptBatchSizeChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : repeatPenaltyChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : repeatPenaltyTokensChanged , this , & ModelList : : updateDataForSettings ) ; ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : promptTemplateChanged , this , & ModelList : : updateDataForSettings ) ;
connect ( MySettings : : globalInstance ( ) , & MySettings : : systemPromptChanged , this , & ModelList : : updateDataForSettings ) ;
2023-07-10 16:14:57 -04:00
connect ( & m_networkManager , & QNetworkAccessManager : : sslErrors , this , & ModelList : : handleSslErrors ) ;
2023-07-01 11:34:21 -04:00
updateModelsFromJson ( ) ;
updateModelsFromSettings ( ) ;
2023-06-22 15:44:49 -04:00
updateModelsFromDirectory ( ) ;
}
QString ModelList : : incompleteDownloadPath ( const QString & modelFile )
{
2023-06-28 16:05:35 -04:00
return MySettings : : globalInstance ( ) - > modelPath ( ) + " incomplete- " + modelFile ;
2023-06-22 15:44:49 -04:00
}
const QList < ModelInfo > ModelList : : exportModelList ( ) const
{
QMutexLocker locker ( & m_mutex ) ;
QList < ModelInfo > infos ;
for ( ModelInfo * info : m_models )
if ( info - > installed )
infos . append ( * info ) ;
return infos ;
}
const QList < QString > ModelList : : userDefaultModelList ( ) const
{
QMutexLocker locker ( & m_mutex ) ;
2023-06-28 20:42:40 -04:00
const QString userDefaultModelName = MySettings : : globalInstance ( ) - > userDefaultModel ( ) ;
2023-06-22 15:44:49 -04:00
QList < QString > models ;
bool foundUserDefault = false ;
for ( ModelInfo * info : m_models ) {
2023-07-01 11:34:21 -04:00
if ( info - > installed & & info - > id ( ) = = userDefaultModelName ) {
2023-06-22 15:44:49 -04:00
foundUserDefault = true ;
2023-07-01 11:34:21 -04:00
models . prepend ( info - > name ( ) ) ;
2023-06-22 15:44:49 -04:00
} else if ( info - > installed ) {
2023-07-01 11:34:21 -04:00
models . append ( info - > name ( ) ) ;
2023-06-22 15:44:49 -04:00
}
}
2023-07-01 11:34:21 -04:00
const QString defaultId = " Application default " ;
2023-06-22 15:44:49 -04:00
if ( foundUserDefault )
2023-07-01 11:34:21 -04:00
models . append ( defaultId ) ;
2023-06-22 15:44:49 -04:00
else
2023-07-01 11:34:21 -04:00
models . prepend ( defaultId ) ;
2023-06-22 15:44:49 -04:00
return models ;
}
ModelInfo ModelList : : defaultModelInfo ( ) const
{
QMutexLocker locker ( & m_mutex ) ;
QSettings settings ;
settings . sync ( ) ;
// The user default model can be set by the user in the settings dialog. The "default" user
2023-07-12 12:45:08 -04:00
// default model is "Application default" which signals we should use the logic here.
2023-06-28 20:42:40 -04:00
const QString userDefaultModelName = MySettings : : globalInstance ( ) - > userDefaultModel ( ) ;
2023-06-26 17:50:11 -04:00
const bool hasUserDefaultName = ! userDefaultModelName . isEmpty ( ) & & userDefaultModelName ! = " Application default " ;
2023-06-22 15:44:49 -04:00
ModelInfo * defaultModel = nullptr ;
for ( ModelInfo * info : m_models ) {
if ( ! info - > installed )
continue ;
defaultModel = info ;
2023-06-26 17:50:11 -04:00
2023-07-12 12:45:08 -04:00
const size_t ramrequired = defaultModel - > ramrequired ;
2023-06-26 17:50:11 -04:00
2023-07-12 12:45:08 -04:00
// If we don't have either setting, then just use the first model that requires less than 16GB that is installed
if ( ! hasUserDefaultName & & ! info - > isChatGPT & & ramrequired > 0 & & ramrequired < 16 )
2023-06-26 17:50:11 -04:00
break ;
// If we have a user specified default and match, then use it
2023-07-01 11:34:21 -04:00
if ( hasUserDefaultName & & ( defaultModel - > id ( ) = = userDefaultModelName ) )
2023-06-26 17:50:11 -04:00
break ;
2023-06-22 15:44:49 -04:00
}
if ( defaultModel )
return * defaultModel ;
return ModelInfo ( ) ;
}
2023-07-01 11:34:21 -04:00
bool ModelList : : contains ( const QString & id ) const
{
QMutexLocker locker ( & m_mutex ) ;
return m_modelMap . contains ( id ) ;
}
bool ModelList : : containsByFilename ( const QString & filename ) const
2023-06-22 15:44:49 -04:00
{
QMutexLocker locker ( & m_mutex ) ;
2023-07-01 11:34:21 -04:00
for ( ModelInfo * info : m_models )
if ( info - > filename ( ) = = filename )
return true ;
return false ;
2023-06-22 15:44:49 -04:00
}
bool ModelList : : lessThan ( const ModelInfo * a , const ModelInfo * b )
{
2023-07-01 11:34:21 -04:00
// Rule 0: Non-clone before clone
if ( a - > isClone ! = b - > isClone ) {
return ! a - > isClone ;
}
2023-06-22 15:44:49 -04:00
// Rule 1: Non-empty 'order' before empty
if ( a - > order . isEmpty ( ) ! = b - > order . isEmpty ( ) ) {
return ! a - > order . isEmpty ( ) ;
}
// Rule 2: Both 'order' are non-empty, sort alphanumerically
if ( ! a - > order . isEmpty ( ) & & ! b - > order . isEmpty ( ) ) {
return a - > order < b - > order ;
}
2023-07-01 11:34:21 -04:00
// Rule 3: Both 'order' are empty, sort by id
return a - > id ( ) < b - > id ( ) ;
2023-06-22 15:44:49 -04:00
}
2023-07-01 11:34:21 -04:00
void ModelList : : addModel ( const QString & id )
2023-06-22 15:44:49 -04:00
{
2023-07-01 11:34:21 -04:00
const bool hasModel = contains ( id ) ;
2023-06-25 20:22:38 -04:00
Q_ASSERT ( ! hasModel ) ;
if ( hasModel ) {
2023-07-01 11:34:21 -04:00
qWarning ( ) < < " ERROR: model list already contains " < < id ;
2023-06-22 15:44:49 -04:00
return ;
}
2023-07-01 11:34:21 -04:00
int modelSizeBefore = 0 ;
2023-06-25 20:22:38 -04:00
int modelSizeAfter = 0 ;
2023-07-01 11:34:21 -04:00
{
QMutexLocker locker ( & m_mutex ) ;
modelSizeBefore = m_models . size ( ) ;
}
beginInsertRows ( QModelIndex ( ) , modelSizeBefore , modelSizeBefore ) ;
2023-06-25 20:22:38 -04:00
{
QMutexLocker locker ( & m_mutex ) ;
ModelInfo * info = new ModelInfo ;
2023-07-01 11:34:21 -04:00
info - > setId ( id ) ;
2023-06-25 20:22:38 -04:00
m_models . append ( info ) ;
2023-07-01 11:34:21 -04:00
m_modelMap . insert ( id , info ) ;
2023-06-25 20:22:38 -04:00
std : : stable_sort ( m_models . begin ( ) , m_models . end ( ) , ModelList : : lessThan ) ;
modelSizeAfter = m_models . size ( ) ;
}
2023-06-22 15:44:49 -04:00
endInsertRows ( ) ;
2023-06-25 20:22:38 -04:00
emit dataChanged ( index ( 0 , 0 ) , index ( modelSizeAfter - 1 , 0 ) ) ;
2023-06-22 15:44:49 -04:00
emit userDefaultModelListChanged ( ) ;
}
2023-07-10 16:14:57 -04:00
void ModelList : : changeId ( const QString & oldId , const QString & newId )
{
const bool hasModel = contains ( oldId ) ;
Q_ASSERT ( hasModel ) ;
if ( ! hasModel ) {
qWarning ( ) < < " ERROR: model list does not contain " < < oldId ;
return ;
}
QMutexLocker locker ( & m_mutex ) ;
ModelInfo * info = m_modelMap . take ( oldId ) ;
info - > setId ( newId ) ;
m_modelMap . insert ( newId , info ) ;
}
2023-06-22 15:44:49 -04:00
int ModelList : : rowCount ( const QModelIndex & parent ) const
{
Q_UNUSED ( parent )
QMutexLocker locker ( & m_mutex ) ;
return m_models . size ( ) ;
}
QVariant ModelList : : dataInternal ( const ModelInfo * info , int role ) const
{
switch ( role ) {
2023-07-01 11:34:21 -04:00
case IdRole :
return info - > id ( ) ;
2023-06-22 15:44:49 -04:00
case NameRole :
2023-07-01 11:34:21 -04:00
return info - > name ( ) ;
2023-06-22 15:44:49 -04:00
case FilenameRole :
2023-07-01 11:34:21 -04:00
return info - > filename ( ) ;
2023-06-22 15:44:49 -04:00
case DirpathRole :
return info - > dirpath ;
case FilesizeRole :
return info - > filesize ;
case Md5sumRole :
return info - > md5sum ;
case CalcHashRole :
return info - > calcHash ;
case InstalledRole :
return info - > installed ;
case DefaultRole :
return info - > isDefault ;
case ChatGPTRole :
return info - > isChatGPT ;
case DisableGUIRole :
return info - > disableGUI ;
case DescriptionRole :
return info - > description ;
case RequiresVersionRole :
return info - > requiresVersion ;
case DeprecatedVersionRole :
return info - > deprecatedVersion ;
case UrlRole :
return info - > url ;
case BytesReceivedRole :
return info - > bytesReceived ;
case BytesTotalRole :
return info - > bytesTotal ;
case TimestampRole :
return info - > timestamp ;
case SpeedRole :
return info - > speed ;
case DownloadingRole :
return info - > isDownloading ;
case IncompleteRole :
return info - > isIncomplete ;
case DownloadErrorRole :
return info - > downloadError ;
case OrderRole :
return info - > order ;
case RamrequiredRole :
return info - > ramrequired ;
case ParametersRole :
return info - > parameters ;
case QuantRole :
return info - > quant ;
case TypeRole :
return info - > type ;
2023-07-01 11:34:21 -04:00
case IsCloneRole :
return info - > isClone ;
case TemperatureRole :
return info - > temperature ( ) ;
case TopPRole :
return info - > topP ( ) ;
case TopKRole :
return info - > topK ( ) ;
case MaxLengthRole :
return info - > maxLength ( ) ;
case PromptBatchSizeRole :
return info - > promptBatchSize ( ) ;
case RepeatPenaltyRole :
return info - > repeatPenalty ( ) ;
case RepeatPenaltyTokensRole :
return info - > repeatPenaltyTokens ( ) ;
case PromptTemplateRole :
return info - > promptTemplate ( ) ;
case SystemPromptRole :
return info - > systemPrompt ( ) ;
2023-06-22 15:44:49 -04:00
}
return QVariant ( ) ;
}
2023-07-01 11:34:21 -04:00
QVariant ModelList : : data ( const QString & id , int role ) const
2023-06-22 15:44:49 -04:00
{
QMutexLocker locker ( & m_mutex ) ;
2023-07-01 11:34:21 -04:00
ModelInfo * info = m_modelMap . value ( id ) ;
2023-06-22 15:44:49 -04:00
return dataInternal ( info , role ) ;
}
2023-07-05 20:12:37 -04:00
QVariant ModelList : : dataByFilename ( const QString & filename , int role ) const
{
QMutexLocker locker ( & m_mutex ) ;
for ( ModelInfo * info : m_models )
if ( info - > filename ( ) = = filename )
return dataInternal ( info , role ) ;
return QVariant ( ) ;
}
2023-06-22 15:44:49 -04:00
QVariant ModelList : : data ( const QModelIndex & index , int role ) const
{
QMutexLocker locker ( & m_mutex ) ;
if ( ! index . isValid ( ) | | index . row ( ) < 0 | | index . row ( ) > = m_models . size ( ) )
return QVariant ( ) ;
const ModelInfo * info = m_models . at ( index . row ( ) ) ;
return dataInternal ( info , role ) ;
}
2023-07-01 11:34:21 -04:00
void ModelList : : updateData ( const QString & id , int role , const QVariant & value )
2023-06-22 15:44:49 -04:00
{
2023-06-25 20:22:38 -04:00
int modelSize ;
bool updateInstalled ;
bool updateIncomplete ;
int index ;
{
QMutexLocker locker ( & m_mutex ) ;
2023-07-01 11:34:21 -04:00
if ( ! m_modelMap . contains ( id ) ) {
qWarning ( ) < < " ERROR: cannot update as model map does not contain " < < id ;
2023-06-25 20:22:38 -04:00
return ;
}
2023-06-22 15:44:49 -04:00
2023-07-01 11:34:21 -04:00
ModelInfo * info = m_modelMap . value ( id ) ;
2023-06-25 20:22:38 -04:00
index = m_models . indexOf ( info ) ;
if ( index = = - 1 ) {
2023-07-01 11:34:21 -04:00
qWarning ( ) < < " ERROR: cannot update as model list does not contain " < < id ;
2023-06-25 20:22:38 -04:00
return ;
}
2023-06-22 15:44:49 -04:00
2023-06-25 20:22:38 -04:00
switch ( role ) {
2023-07-01 11:34:21 -04:00
case IdRole :
info - > setId ( value . toString ( ) ) ; break ;
2023-06-25 20:22:38 -04:00
case NameRole :
2023-07-01 11:34:21 -04:00
info - > setName ( value . toString ( ) ) ; break ;
2023-06-25 20:22:38 -04:00
case FilenameRole :
2023-07-01 11:34:21 -04:00
info - > setFilename ( value . toString ( ) ) ; break ;
2023-06-25 20:22:38 -04:00
case DirpathRole :
info - > dirpath = value . toString ( ) ; break ;
case FilesizeRole :
info - > filesize = value . toString ( ) ; break ;
case Md5sumRole :
info - > md5sum = value . toByteArray ( ) ; break ;
case CalcHashRole :
info - > calcHash = value . toBool ( ) ; break ;
case InstalledRole :
info - > installed = value . toBool ( ) ; break ;
case DefaultRole :
info - > isDefault = value . toBool ( ) ; break ;
case ChatGPTRole :
info - > isChatGPT = value . toBool ( ) ; break ;
case DisableGUIRole :
info - > disableGUI = value . toBool ( ) ; break ;
case DescriptionRole :
info - > description = value . toString ( ) ; break ;
case RequiresVersionRole :
info - > requiresVersion = value . toString ( ) ; break ;
case DeprecatedVersionRole :
info - > deprecatedVersion = value . toString ( ) ; break ;
case UrlRole :
info - > url = value . toString ( ) ; break ;
case BytesReceivedRole :
info - > bytesReceived = value . toLongLong ( ) ; break ;
case BytesTotalRole :
info - > bytesTotal = value . toLongLong ( ) ; break ;
case TimestampRole :
info - > timestamp = value . toLongLong ( ) ; break ;
case SpeedRole :
info - > speed = value . toString ( ) ; break ;
case DownloadingRole :
info - > isDownloading = value . toBool ( ) ; break ;
case IncompleteRole :
info - > isIncomplete = value . toBool ( ) ; break ;
case DownloadErrorRole :
info - > downloadError = value . toString ( ) ; break ;
case OrderRole :
info - > order = value . toString ( ) ; break ;
case RamrequiredRole :
info - > ramrequired = value . toInt ( ) ; break ;
case ParametersRole :
info - > parameters = value . toString ( ) ; break ;
case QuantRole :
info - > quant = value . toString ( ) ; break ;
case TypeRole :
info - > type = value . toString ( ) ; break ;
2023-07-01 11:34:21 -04:00
case IsCloneRole :
info - > isClone = value . toBool ( ) ; break ;
case TemperatureRole :
info - > setTemperature ( value . toDouble ( ) ) ; break ;
case TopPRole :
info - > setTopP ( value . toDouble ( ) ) ; break ;
case TopKRole :
info - > setTopK ( value . toInt ( ) ) ; break ;
case MaxLengthRole :
info - > setMaxLength ( value . toInt ( ) ) ; break ;
case PromptBatchSizeRole :
info - > setPromptBatchSize ( value . toInt ( ) ) ; break ;
case RepeatPenaltyRole :
info - > setRepeatPenalty ( value . toDouble ( ) ) ; break ;
case RepeatPenaltyTokensRole :
info - > setRepeatPenaltyTokens ( value . toInt ( ) ) ; break ;
case PromptTemplateRole :
info - > setPromptTemplate ( value . toString ( ) ) ; break ;
case SystemPromptRole :
info - > setSystemPrompt ( value . toString ( ) ) ; break ;
2023-06-25 20:22:38 -04:00
}
2023-06-22 15:44:49 -04:00
2023-06-25 20:22:38 -04:00
// Extra guarantee that these always remains in sync with filesystem
2023-07-01 11:34:21 -04:00
QFileInfo fileInfo ( info - > dirpath + info - > filename ( ) ) ;
2023-06-25 20:22:38 -04:00
if ( info - > installed ! = fileInfo . exists ( ) ) {
info - > installed = fileInfo . exists ( ) ;
updateInstalled = true ;
}
2023-07-01 11:34:21 -04:00
QFileInfo incompleteInfo ( incompleteDownloadPath ( info - > filename ( ) ) ) ;
2023-06-25 20:22:38 -04:00
if ( info - > isIncomplete ! = incompleteInfo . exists ( ) ) {
info - > isIncomplete = incompleteInfo . exists ( ) ;
updateIncomplete = true ;
}
2023-06-22 15:44:49 -04:00
2023-06-25 20:22:38 -04:00
std : : stable_sort ( m_models . begin ( ) , m_models . end ( ) , ModelList : : lessThan ) ;
modelSize = m_models . size ( ) ;
}
emit dataChanged ( createIndex ( 0 , 0 ) , createIndex ( modelSize - 1 , 0 ) ) ;
2023-06-22 15:44:49 -04:00
emit userDefaultModelListChanged ( ) ;
}
2023-07-05 20:12:37 -04:00
void ModelList : : updateDataByFilename ( const QString & filename , int role , const QVariant & value )
{
QVector < QString > modelsById ;
{
QMutexLocker locker ( & m_mutex ) ;
for ( ModelInfo * info : m_models )
if ( info - > filename ( ) = = filename )
modelsById . append ( info - > id ( ) ) ;
}
if ( modelsById . isEmpty ( ) ) {
qWarning ( ) < < " ERROR: cannot update model as list does not contain file " < < filename ;
return ;
}
for ( const QString & id : modelsById )
updateData ( id , role , value ) ; ;
}
2023-07-01 11:34:21 -04:00
ModelInfo ModelList : : modelInfo ( const QString & id ) const
2023-06-22 15:44:49 -04:00
{
QMutexLocker locker ( & m_mutex ) ;
2023-07-01 11:34:21 -04:00
if ( ! m_modelMap . contains ( id ) )
2023-06-22 15:44:49 -04:00
return ModelInfo ( ) ;
2023-07-01 11:34:21 -04:00
return * m_modelMap . value ( id ) ;
}
ModelInfo ModelList : : modelInfoByFilename ( const QString & filename ) const
{
QMutexLocker locker ( & m_mutex ) ;
for ( ModelInfo * info : m_models )
if ( info - > filename ( ) = = filename )
return * info ;
return ModelInfo ( ) ;
}
bool ModelList : : isUniqueName ( const QString & name ) const
{
QMutexLocker locker ( & m_mutex ) ;
for ( const ModelInfo * info : m_models ) {
if ( info - > name ( ) = = name )
return false ;
}
return true ;
}
QString ModelList : : clone ( const ModelInfo & model )
{
const QString id = Network : : globalInstance ( ) - > generateUniqueId ( ) ;
addModel ( id ) ;
updateData ( id , ModelList : : IsCloneRole , true ) ;
updateData ( id , ModelList : : NameRole , uniqueModelName ( model ) ) ;
updateData ( id , ModelList : : FilenameRole , model . filename ( ) ) ;
updateData ( id , ModelList : : DirpathRole , model . dirpath ) ;
updateData ( id , ModelList : : InstalledRole , model . installed ) ;
updateData ( id , ModelList : : ChatGPTRole , model . isChatGPT ) ;
updateData ( id , ModelList : : TemperatureRole , model . temperature ( ) ) ;
updateData ( id , ModelList : : TopPRole , model . topP ( ) ) ;
updateData ( id , ModelList : : TopKRole , model . topK ( ) ) ;
updateData ( id , ModelList : : MaxLengthRole , model . maxLength ( ) ) ;
updateData ( id , ModelList : : PromptBatchSizeRole , model . promptBatchSize ( ) ) ;
updateData ( id , ModelList : : RepeatPenaltyRole , model . repeatPenalty ( ) ) ;
updateData ( id , ModelList : : RepeatPenaltyTokensRole , model . repeatPenaltyTokens ( ) ) ;
updateData ( id , ModelList : : PromptTemplateRole , model . promptTemplate ( ) ) ;
updateData ( id , ModelList : : SystemPromptRole , model . systemPrompt ( ) ) ;
return id ;
}
void ModelList : : remove ( const ModelInfo & model )
{
Q_ASSERT ( model . isClone ) ;
if ( ! model . isClone )
return ;
const bool hasModel = contains ( model . id ( ) ) ;
Q_ASSERT ( hasModel ) ;
if ( ! hasModel ) {
qWarning ( ) < < " ERROR: model list does not contain " < < model . id ( ) ;
return ;
}
int indexOfModel = 0 ;
int modelSizeAfter = 0 ;
{
QMutexLocker locker ( & m_mutex ) ;
ModelInfo * info = m_modelMap . value ( model . id ( ) ) ;
indexOfModel = m_models . indexOf ( info ) ;
}
beginRemoveRows ( QModelIndex ( ) , indexOfModel , indexOfModel ) ;
{
QMutexLocker locker ( & m_mutex ) ;
ModelInfo * info = m_models . takeAt ( indexOfModel ) ;
m_modelMap . remove ( info - > id ( ) ) ;
delete info ;
modelSizeAfter = m_models . size ( ) ;
}
endRemoveRows ( ) ;
emit dataChanged ( index ( 0 , 0 ) , index ( modelSizeAfter - 1 , 0 ) ) ;
emit userDefaultModelListChanged ( ) ;
MySettings : : globalInstance ( ) - > eraseModel ( model ) ;
}
QString ModelList : : uniqueModelName ( const ModelInfo & model ) const
{
QMutexLocker locker ( & m_mutex ) ;
QRegularExpression re ( " ^(.*) ~ ( \ \ d + ) $ " ) ;
QRegularExpressionMatch match = re . match ( model . name ( ) ) ;
QString baseName ;
if ( match . hasMatch ( ) )
baseName = match . captured ( 1 ) ;
else
baseName = model . name ( ) ;
int maxSuffixNumber = 0 ;
bool baseNameExists = false ;
for ( const ModelInfo * info : m_models ) {
if ( info - > name ( ) = = baseName )
baseNameExists = true ;
QRegularExpressionMatch match = re . match ( info - > name ( ) ) ;
if ( match . hasMatch ( ) ) {
QString currentBaseName = match . captured ( 1 ) ;
int currentSuffixNumber = match . captured ( 2 ) . toInt ( ) ;
if ( currentBaseName = = baseName & & currentSuffixNumber > maxSuffixNumber )
maxSuffixNumber = currentSuffixNumber ;
}
}
if ( baseNameExists )
return baseName + " ~ " + QString : : number ( maxSuffixNumber + 1 ) ;
return baseName ;
2023-06-22 15:44:49 -04:00
}
QString ModelList : : modelDirPath ( const QString & modelName , bool isChatGPT )
{
QVector < QString > possibleFilePaths ;
if ( isChatGPT )
possibleFilePaths < < " / " + modelName + " .txt " ;
else {
possibleFilePaths < < " /ggml- " + modelName + " .bin " ;
possibleFilePaths < < " / " + modelName + " .bin " ;
}
for ( const QString & modelFilename : possibleFilePaths ) {
QString appPath = QCoreApplication : : applicationDirPath ( ) + modelFilename ;
QFileInfo infoAppPath ( appPath ) ;
if ( infoAppPath . exists ( ) )
return QCoreApplication : : applicationDirPath ( ) ;
2023-06-28 16:05:35 -04:00
QString downloadPath = MySettings : : globalInstance ( ) - > modelPath ( ) + modelFilename ;
2023-06-22 15:44:49 -04:00
QFileInfo infoLocalPath ( downloadPath ) ;
if ( infoLocalPath . exists ( ) )
2023-06-28 16:05:35 -04:00
return MySettings : : globalInstance ( ) - > modelPath ( ) ;
2023-06-22 15:44:49 -04:00
}
return QString ( ) ;
}
void ModelList : : updateModelsFromDirectory ( )
{
const QString exePath = QCoreApplication : : applicationDirPath ( ) + QDir : : separator ( ) ;
2023-06-28 16:05:35 -04:00
const QString localPath = MySettings : : globalInstance ( ) - > modelPath ( ) ;
2023-06-22 15:44:49 -04:00
2023-06-26 10:11:33 -04:00
auto processDirectory = [ & ] ( const QString & path ) {
QDirIterator it ( path , QDirIterator : : Subdirectories ) ;
while ( it . hasNext ( ) ) {
it . next ( ) ;
if ( ! it . fileInfo ( ) . isDir ( ) ) {
QString filename = it . fileName ( ) ;
2023-06-22 15:44:49 -04:00
2023-06-26 10:11:33 -04:00
// All files that end with .bin and have 'ggml' somewhere in the name
if ( ( filename . endsWith ( " .bin " ) & & filename . contains ( " ggml " ) & & ! filename . startsWith ( " incomplete " ) )
| | ( filename . endsWith ( " .txt " ) & & filename . startsWith ( " chatgpt- " ) ) ) {
QString filePath = it . filePath ( ) ;
QFileInfo info ( filePath ) ;
if ( ! info . exists ( ) )
continue ;
2023-07-01 11:34:21 -04:00
QVector < QString > modelsById ;
{
QMutexLocker locker ( & m_mutex ) ;
for ( ModelInfo * info : m_models )
if ( info - > filename ( ) = = filename )
modelsById . append ( info - > id ( ) ) ;
}
2023-06-26 10:11:33 -04:00
2023-07-01 11:34:21 -04:00
if ( modelsById . isEmpty ( ) ) {
addModel ( filename ) ;
modelsById . append ( filename ) ;
}
for ( const QString & id : modelsById ) {
updateData ( id , FilenameRole , filename ) ;
updateData ( id , ChatGPTRole , filename . startsWith ( " chatgpt- " ) ) ;
2023-07-12 14:12:41 -04:00
updateData ( id , DirpathRole , info . dir ( ) . absolutePath ( ) + " / " ) ;
2023-07-01 11:34:21 -04:00
updateData ( id , FilesizeRole , toFileSize ( info . size ( ) ) ) ;
}
2023-06-26 10:11:33 -04:00
}
2023-06-22 15:44:49 -04:00
}
}
2023-06-26 10:11:33 -04:00
} ;
2023-06-22 15:44:49 -04:00
2023-06-26 10:11:33 -04:00
processDirectory ( exePath ) ;
if ( localPath ! = exePath )
processDirectory ( localPath ) ;
2023-06-22 15:44:49 -04:00
}
2023-06-28 11:13:33 -04:00
2023-07-01 11:34:21 -04:00
void ModelList : : updateModelsFromJson ( )
2023-06-28 11:13:33 -04:00
{
# if defined(USE_LOCAL_MODELSJSON)
QUrl jsonUrl ( " file:// " + QDir : : homePath ( ) + " /dev/large_language_models/gpt4all/gpt4all-chat/metadata/models.json " ) ;
# else
QUrl jsonUrl ( " http://gpt4all.io/models/models.json " ) ;
# endif
QNetworkRequest request ( jsonUrl ) ;
QSslConfiguration conf = request . sslConfiguration ( ) ;
conf . setPeerVerifyMode ( QSslSocket : : VerifyNone ) ;
request . setSslConfiguration ( conf ) ;
QNetworkReply * jsonReply = m_networkManager . get ( request ) ;
2023-07-11 12:37:21 -04:00
connect ( qApp , & QCoreApplication : : aboutToQuit , jsonReply , & QNetworkReply : : abort ) ;
2023-06-28 11:13:33 -04:00
QEventLoop loop ;
connect ( jsonReply , & QNetworkReply : : finished , & loop , & QEventLoop : : quit ) ;
QTimer : : singleShot ( 1500 , & loop , & QEventLoop : : quit ) ;
loop . exec ( ) ;
if ( jsonReply - > error ( ) = = QNetworkReply : : NoError & & jsonReply - > isFinished ( ) ) {
QByteArray jsonData = jsonReply - > readAll ( ) ;
jsonReply - > deleteLater ( ) ;
2023-07-10 16:14:57 -04:00
parseModelsJsonFile ( jsonData , true ) ;
2023-06-28 11:13:33 -04:00
} else {
2023-07-10 16:14:57 -04:00
qWarning ( ) < < " WARNING: Could not download models.json synchronously " ;
updateModelsFromJsonAsync ( ) ;
QSettings settings ;
QFileInfo info ( settings . fileName ( ) ) ;
QString dirPath = info . canonicalPath ( ) ;
const QString modelsConfig = dirPath + " /models.json " ;
QFile file ( modelsConfig ) ;
if ( ! file . open ( QIODeviceBase : : ReadOnly ) ) {
qWarning ( ) < < " ERROR: Couldn't read models config file: " < < modelsConfig ;
} else {
QByteArray jsonData = file . readAll ( ) ;
file . close ( ) ;
parseModelsJsonFile ( jsonData , false ) ;
}
2023-06-28 11:13:33 -04:00
}
delete jsonReply ;
}
2023-07-10 16:14:57 -04:00
void ModelList : : updateModelsFromJsonAsync ( )
{
2023-07-12 11:46:40 -04:00
m_asyncModelRequestOngoing = true ;
emit asyncModelRequestOngoingChanged ( ) ;
2023-07-10 16:14:57 -04:00
# if defined(USE_LOCAL_MODELSJSON)
QUrl jsonUrl ( " file:// " + QDir : : homePath ( ) + " /dev/large_language_models/gpt4all/gpt4all-chat/metadata/models.json " ) ;
# else
QUrl jsonUrl ( " http://gpt4all.io/models/models.json " ) ;
# endif
QNetworkRequest request ( jsonUrl ) ;
QSslConfiguration conf = request . sslConfiguration ( ) ;
conf . setPeerVerifyMode ( QSslSocket : : VerifyNone ) ;
request . setSslConfiguration ( conf ) ;
QNetworkReply * jsonReply = m_networkManager . get ( request ) ;
2023-07-11 12:37:21 -04:00
connect ( qApp , & QCoreApplication : : aboutToQuit , jsonReply , & QNetworkReply : : abort ) ;
2023-07-10 16:14:57 -04:00
connect ( jsonReply , & QNetworkReply : : finished , this , & ModelList : : handleModelsJsonDownloadFinished ) ;
2023-07-12 11:46:40 -04:00
connect ( jsonReply , & QNetworkReply : : errorOccurred , this , & ModelList : : handleModelsJsonDownloadErrorOccurred ) ;
2023-07-10 16:14:57 -04:00
}
void ModelList : : handleModelsJsonDownloadFinished ( )
{
QNetworkReply * jsonReply = qobject_cast < QNetworkReply * > ( sender ( ) ) ;
2023-07-12 11:46:40 -04:00
if ( ! jsonReply ) {
m_asyncModelRequestOngoing = false ;
emit asyncModelRequestOngoingChanged ( ) ;
2023-07-10 16:14:57 -04:00
return ;
2023-07-12 11:46:40 -04:00
}
2023-07-10 16:14:57 -04:00
QByteArray jsonData = jsonReply - > readAll ( ) ;
jsonReply - > deleteLater ( ) ;
parseModelsJsonFile ( jsonData , true ) ;
2023-07-12 11:46:40 -04:00
m_asyncModelRequestOngoing = false ;
emit asyncModelRequestOngoingChanged ( ) ;
}
void ModelList : : handleModelsJsonDownloadErrorOccurred ( QNetworkReply : : NetworkError code )
{
2023-07-12 12:47:09 -04:00
// TODO: Show what error occurred in the GUI
2023-07-12 11:46:40 -04:00
m_asyncModelRequestOngoing = false ;
emit asyncModelRequestOngoingChanged ( ) ;
QNetworkReply * reply = qobject_cast < QNetworkReply * > ( sender ( ) ) ;
if ( ! reply )
return ;
qWarning ( ) < < QString ( " ERROR: Modellist download failed with error code \" %1-%2 \" " )
. arg ( code ) . arg ( reply - > errorString ( ) ) . toStdString ( ) ;
2023-07-10 16:14:57 -04:00
}
void ModelList : : handleSslErrors ( QNetworkReply * reply , const QList < QSslError > & errors )
{
QUrl url = reply - > request ( ) . url ( ) ;
for ( const auto & e : errors )
qWarning ( ) < < " ERROR: Received ssl error: " < < e . errorString ( ) < < " for " < < url ;
}
2023-07-01 11:34:21 -04:00
void ModelList : : updateDataForSettings ( )
{
emit dataChanged ( index ( 0 , 0 ) , index ( m_models . size ( ) - 1 , 0 ) ) ;
2023-06-28 11:13:33 -04:00
}
static bool compareVersions ( const QString & a , const QString & b ) {
QStringList aParts = a . split ( ' . ' ) ;
QStringList bParts = b . split ( ' . ' ) ;
for ( int i = 0 ; i < std : : min ( aParts . size ( ) , bParts . size ( ) ) ; + + i ) {
int aInt = aParts [ i ] . toInt ( ) ;
int bInt = bParts [ i ] . toInt ( ) ;
if ( aInt > bInt ) {
return true ;
} else if ( aInt < bInt ) {
return false ;
}
}
return aParts . size ( ) > bParts . size ( ) ;
}
2023-07-10 16:14:57 -04:00
void ModelList : : parseModelsJsonFile ( const QByteArray & jsonData , bool save )
2023-06-28 11:13:33 -04:00
{
QJsonParseError err ;
QJsonDocument document = QJsonDocument : : fromJson ( jsonData , & err ) ;
if ( err . error ! = QJsonParseError : : NoError ) {
qWarning ( ) < < " ERROR: Couldn't parse: " < < jsonData < < err . errorString ( ) ;
return ;
}
2023-07-10 16:14:57 -04:00
if ( save ) {
QSettings settings ;
QFileInfo info ( settings . fileName ( ) ) ;
QString dirPath = info . canonicalPath ( ) ;
const QString modelsConfig = dirPath + " /models.json " ;
QFile file ( modelsConfig ) ;
if ( ! file . open ( QIODeviceBase : : WriteOnly ) ) {
qWarning ( ) < < " ERROR: Couldn't write models config file: " < < modelsConfig ;
} else {
file . write ( jsonData . constData ( ) ) ;
file . close ( ) ;
}
}
2023-06-28 11:13:33 -04:00
QJsonArray jsonArray = document . array ( ) ;
const QString currentVersion = QCoreApplication : : applicationVersion ( ) ;
for ( const QJsonValue & value : jsonArray ) {
QJsonObject obj = value . toObject ( ) ;
QString modelName = obj [ " name " ] . toString ( ) ;
QString modelFilename = obj [ " filename " ] . toString ( ) ;
QString modelFilesize = obj [ " filesize " ] . toString ( ) ;
QString requiresVersion = obj [ " requires " ] . toString ( ) ;
QString deprecatedVersion = obj [ " deprecated " ] . toString ( ) ;
QString url = obj [ " url " ] . toString ( ) ;
QByteArray modelMd5sum = obj [ " md5sum " ] . toString ( ) . toLatin1 ( ) . constData ( ) ;
bool isDefault = obj . contains ( " isDefault " ) & & obj [ " isDefault " ] = = QString ( " true " ) ;
bool disableGUI = obj . contains ( " disableGUI " ) & & obj [ " disableGUI " ] = = QString ( " true " ) ;
QString description = obj [ " description " ] . toString ( ) ;
QString order = obj [ " order " ] . toString ( ) ;
int ramrequired = obj [ " ramrequired " ] . toString ( ) . toInt ( ) ;
QString parameters = obj [ " parameters " ] . toString ( ) ;
QString quant = obj [ " quant " ] . toString ( ) ;
QString type = obj [ " type " ] . toString ( ) ;
// If the currentVersion version is strictly less than required version, then continue
if ( ! requiresVersion . isEmpty ( )
& & requiresVersion ! = currentVersion
& & compareVersions ( requiresVersion , currentVersion ) ) {
continue ;
}
// If the current version is strictly greater than the deprecated version, then continue
if ( ! deprecatedVersion . isEmpty ( )
& & compareVersions ( currentVersion , deprecatedVersion ) ) {
continue ;
}
modelFilesize = ModelList : : toFileSize ( modelFilesize . toULongLong ( ) ) ;
2023-07-01 11:34:21 -04:00
const QString id = modelName ;
Q_ASSERT ( ! id . isEmpty ( ) ) ;
2023-07-10 16:14:57 -04:00
if ( contains ( modelFilename ) )
changeId ( modelFilename , id ) ;
2023-07-01 11:34:21 -04:00
if ( ! contains ( id ) )
addModel ( id ) ;
updateData ( id , ModelList : : NameRole , modelName ) ;
updateData ( id , ModelList : : FilenameRole , modelFilename ) ;
updateData ( id , ModelList : : FilesizeRole , modelFilesize ) ;
updateData ( id , ModelList : : Md5sumRole , modelMd5sum ) ;
updateData ( id , ModelList : : DefaultRole , isDefault ) ;
updateData ( id , ModelList : : DescriptionRole , description ) ;
updateData ( id , ModelList : : RequiresVersionRole , requiresVersion ) ;
updateData ( id , ModelList : : DeprecatedVersionRole , deprecatedVersion ) ;
updateData ( id , ModelList : : UrlRole , url ) ;
updateData ( id , ModelList : : DisableGUIRole , disableGUI ) ;
updateData ( id , ModelList : : OrderRole , order ) ;
updateData ( id , ModelList : : RamrequiredRole , ramrequired ) ;
updateData ( id , ModelList : : ParametersRole , parameters ) ;
updateData ( id , ModelList : : QuantRole , quant ) ;
updateData ( id , ModelList : : TypeRole , type ) ;
if ( obj . contains ( " temperature " ) )
updateData ( id , ModelList : : TemperatureRole , obj [ " temperature " ] . toDouble ( ) ) ;
if ( obj . contains ( " topP " ) )
updateData ( id , ModelList : : TopPRole , obj [ " topP " ] . toDouble ( ) ) ;
if ( obj . contains ( " topK " ) )
updateData ( id , ModelList : : TopKRole , obj [ " topK " ] . toInt ( ) ) ;
if ( obj . contains ( " maxLength " ) )
updateData ( id , ModelList : : MaxLengthRole , obj [ " maxLength " ] . toInt ( ) ) ;
if ( obj . contains ( " promptBatchSize " ) )
updateData ( id , ModelList : : PromptBatchSizeRole , obj [ " promptBatchSize " ] . toInt ( ) ) ;
if ( obj . contains ( " repeatPenalty " ) )
updateData ( id , ModelList : : RepeatPenaltyRole , obj [ " repeatPenalty " ] . toDouble ( ) ) ;
if ( obj . contains ( " repeatPenaltyTokens " ) )
updateData ( id , ModelList : : RepeatPenaltyTokensRole , obj [ " repeatPenaltyTokens " ] . toInt ( ) ) ;
if ( obj . contains ( " promptTemplate " ) )
updateData ( id , ModelList : : PromptTemplateRole , obj [ " promptTemplate " ] . toString ( ) ) ;
if ( obj . contains ( " systemPrompt " ) )
updateData ( id , ModelList : : SystemPromptRole , obj [ " systemPrompt " ] . toString ( ) ) ;
2023-06-28 11:13:33 -04:00
}
const QString chatGPTDesc = tr ( " <ul><li>Requires personal OpenAI API key.</li><li>WARNING: Will send "
" your chats to OpenAI!</li><li>Your API key will be stored on disk</li><li>Will only be used "
" to communicate with OpenAI</li><li>You can apply for an API key "
" <a href= \" https://platform.openai.com/account/api-keys \" >here.</a></li> " ) ;
{
2023-07-01 11:34:21 -04:00
const QString modelName = " ChatGPT-3.5 Turbo " ;
const QString id = modelName ;
2023-06-28 11:13:33 -04:00
const QString modelFilename = " chatgpt-gpt-3.5-turbo.txt " ;
2023-07-10 16:14:57 -04:00
if ( contains ( modelFilename ) )
changeId ( modelFilename , id ) ;
2023-07-01 11:34:21 -04:00
if ( ! contains ( id ) )
addModel ( id ) ;
updateData ( id , ModelList : : NameRole , modelName ) ;
updateData ( id , ModelList : : FilenameRole , modelFilename ) ;
updateData ( id , ModelList : : FilesizeRole , " minimal " ) ;
updateData ( id , ModelList : : ChatGPTRole , true ) ;
updateData ( id , ModelList : : DescriptionRole ,
2023-06-28 11:13:33 -04:00
tr ( " <strong>OpenAI's ChatGPT model GPT-3.5 Turbo</strong><br> " ) + chatGPTDesc ) ;
2023-07-01 11:34:21 -04:00
updateData ( id , ModelList : : RequiresVersionRole , " 2.4.2 " ) ;
updateData ( id , ModelList : : OrderRole , " ca " ) ;
updateData ( id , ModelList : : RamrequiredRole , 0 ) ;
updateData ( id , ModelList : : ParametersRole , " ? " ) ;
updateData ( id , ModelList : : QuantRole , " NA " ) ;
updateData ( id , ModelList : : TypeRole , " GPT " ) ;
2023-06-28 11:13:33 -04:00
}
{
2023-07-11 15:35:10 -04:00
const QString chatGPT4Warn = tr ( " <br><br><i>* Even if you pay OpenAI for ChatGPT-4 this does not guarantee API key access. Contact OpenAI for more info. " ) ;
2023-07-01 11:34:21 -04:00
const QString modelName = " ChatGPT-4 " ;
const QString id = modelName ;
2023-06-28 11:13:33 -04:00
const QString modelFilename = " chatgpt-gpt-4.txt " ;
2023-07-10 16:14:57 -04:00
if ( contains ( modelFilename ) )
changeId ( modelFilename , id ) ;
2023-07-01 11:34:21 -04:00
if ( ! contains ( id ) )
addModel ( id ) ;
updateData ( id , ModelList : : NameRole , modelName ) ;
updateData ( id , ModelList : : FilenameRole , modelFilename ) ;
updateData ( id , ModelList : : FilesizeRole , " minimal " ) ;
updateData ( id , ModelList : : ChatGPTRole , true ) ;
updateData ( id , ModelList : : DescriptionRole ,
2023-07-11 15:35:10 -04:00
tr ( " <strong>OpenAI's ChatGPT model GPT-4</strong><br> " ) + chatGPTDesc + chatGPT4Warn ) ;
2023-07-01 11:34:21 -04:00
updateData ( id , ModelList : : RequiresVersionRole , " 2.4.2 " ) ;
updateData ( id , ModelList : : OrderRole , " cb " ) ;
updateData ( id , ModelList : : RamrequiredRole , 0 ) ;
updateData ( id , ModelList : : ParametersRole , " ? " ) ;
updateData ( id , ModelList : : QuantRole , " NA " ) ;
updateData ( id , ModelList : : TypeRole , " GPT " ) ;
2023-06-28 11:13:33 -04:00
}
}
2023-07-01 11:34:21 -04:00
void ModelList : : updateModelsFromSettings ( )
{
QSettings settings ;
settings . sync ( ) ;
QStringList groups = settings . childGroups ( ) ;
for ( const QString g : groups ) {
if ( ! g . startsWith ( " model- " ) )
continue ;
const QString id = g . sliced ( 6 ) ;
if ( contains ( id ) )
continue ;
if ( ! settings . contains ( g + " /isClone " ) )
continue ;
Q_ASSERT ( settings . contains ( g + " /name " ) ) ;
const QString name = settings . value ( g + " /name " ) . toString ( ) ;
Q_ASSERT ( settings . contains ( g + " /filename " ) ) ;
const QString filename = settings . value ( g + " /filename " ) . toString ( ) ;
Q_ASSERT ( settings . contains ( g + " /temperature " ) ) ;
const double temperature = settings . value ( g + " /temperature " ) . toDouble ( ) ;
Q_ASSERT ( settings . contains ( g + " /topP " ) ) ;
const double topP = settings . value ( g + " /topP " ) . toDouble ( ) ;
Q_ASSERT ( settings . contains ( g + " /topK " ) ) ;
const int topK = settings . value ( g + " /topK " ) . toInt ( ) ;
Q_ASSERT ( settings . contains ( g + " /maxLength " ) ) ;
const int maxLength = settings . value ( g + " /maxLength " ) . toInt ( ) ;
Q_ASSERT ( settings . contains ( g + " /promptBatchSize " ) ) ;
const int promptBatchSize = settings . value ( g + " /promptBatchSize " ) . toInt ( ) ;
Q_ASSERT ( settings . contains ( g + " /repeatPenalty " ) ) ;
const double repeatPenalty = settings . value ( g + " /repeatPenalty " ) . toDouble ( ) ;
Q_ASSERT ( settings . contains ( g + " /repeatPenaltyTokens " ) ) ;
const int repeatPenaltyTokens = settings . value ( g + " /repeatPenaltyTokens " ) . toInt ( ) ;
Q_ASSERT ( settings . contains ( g + " /promptTemplate " ) ) ;
const QString promptTemplate = settings . value ( g + " /promptTemplate " ) . toString ( ) ;
Q_ASSERT ( settings . contains ( g + " /systemPrompt " ) ) ;
const QString systemPrompt = settings . value ( g + " /systemPrompt " ) . toString ( ) ;
addModel ( id ) ;
updateData ( id , ModelList : : IsCloneRole , true ) ;
updateData ( id , ModelList : : NameRole , name ) ;
updateData ( id , ModelList : : FilenameRole , filename ) ;
updateData ( id , ModelList : : TemperatureRole , temperature ) ;
updateData ( id , ModelList : : TopPRole , topP ) ;
updateData ( id , ModelList : : TopKRole , topK ) ;
updateData ( id , ModelList : : MaxLengthRole , maxLength ) ;
updateData ( id , ModelList : : PromptBatchSizeRole , promptBatchSize ) ;
updateData ( id , ModelList : : RepeatPenaltyRole , repeatPenalty ) ;
updateData ( id , ModelList : : RepeatPenaltyTokensRole , repeatPenaltyTokens ) ;
updateData ( id , ModelList : : PromptTemplateRole , promptTemplate ) ;
updateData ( id , ModelList : : SystemPromptRole , systemPrompt ) ;
}
}