import { AddToDownloadList, DeleteFile, DepCheck, GetProxyPort, InstallPyDep, ListDirFiles, OpenOpenFileDialog, ReadFileInfo, ReadJson, SaveJson, UpdateApp } from '../../wailsjs/go/backend_golang/App'; import manifest from '../../../manifest.json'; import commonStore, { ModelStatus } from '../stores/commonStore'; import { toast } from 'react-toastify'; import { t } from 'i18next'; import { ToastOptions } from 'react-toastify/dist/types'; import { Button } from '@fluentui/react-components'; import { BrowserOpenURL, EventsOff, EventsOn, WindowShow } from '../../wailsjs/runtime'; import { NavigateFunction } from 'react-router'; import { ModelConfig, ModelParameters } from '../types/configs'; import { DownloadStatus } from '../types/downloads'; import { ModelSourceItem } from '../types/models'; import { Language, Languages, SettingsType } from '../types/settings'; import { DataProcessParameters, LoraFinetuneParameters } from '../types/train'; import { InstrumentTypeNameMap, MidiMessage, tracksMinimalTotalTime } from '../types/composition'; import logo from '../assets/images/logo.png'; import { Preset } from '../types/presets'; import { botName, Conversation, MessageType, Role, systemName, userName } from '../types/chat'; import { v4 as uuid } from 'uuid'; import { findLastIndex, throttle } from 'lodash-es'; export type Cache = { version: string models: ModelSourceItem[] depComplete: boolean } export type LocalConfig = { modelSourceManifestList: string currentModelConfigIndex: number modelConfigs: ModelConfig[] settings: SettingsType, dataProcessParams: DataProcessParameters, loraFinetuneParams: LoraFinetuneParameters } export async function refreshBuiltInModels(readCache: boolean = false) { let cache: { models: ModelSourceItem[] } = { models: [] }; if (readCache) await ReadJson('cache.json').then((cacheData: Cache) => { if (cacheData.models) cache.models = cacheData.models; else cache.models = manifest.models.slice(); }).catch(() => { cache.models = manifest.models.slice(); }); else cache.models = manifest.models.slice(); commonStore.setModelSourceList(cache.models); await saveCache().catch(() => { }); return cache; } const modelSuffix = ['.pth', '.st', '.safetensors', '.bin']; export async function refreshLocalModels(cache: { models: ModelSourceItem[] }, filter: boolean = true, initUnfinishedModels: boolean = false) { if (filter) cache.models = cache.models.filter(m => !m.isComplete); //TODO BUG cause local but in manifest files to be removed, so currently cache is disabled await ListDirFiles(commonStore.settings.customModelsPath).then((data) => { cache.models.push(...data.flatMap(d => { if (!d.isDir && modelSuffix.some((ext => d.name.endsWith(ext)))) return [{ name: d.name, size: d.size, lastUpdated: d.modTime, isComplete: true, isLocal: true, tags: ['Local'] }] as ModelSourceItem[]; return []; })); }).catch(() => { }); for (let i = 0; i < cache.models.length; i++) { if (!cache.models[i].lastUpdatedMs) cache.models[i].lastUpdatedMs = Date.parse(cache.models[i].lastUpdated); if (!cache.models[i].tags || !Array.isArray(cache.models[i].tags) || cache.models[i].tags?.length === 0) cache.models[i].tags = ['Other']; for (let j = i + 1; j < cache.models.length; j++) { if (!cache.models[j].lastUpdatedMs) cache.models[j].lastUpdatedMs = Date.parse(cache.models[j].lastUpdated); if (cache.models[i].name === cache.models[j].name) { const tags = Array.from(new Set([...cache.models[i].tags as string[], ...cache.models[j].tags as string[]])); if (cache.models[i].size <= cache.models[j].size) { // j is local file if (cache.models[i].lastUpdatedMs! < cache.models[j].lastUpdatedMs!) { cache.models[i] = Object.assign({}, cache.models[i], cache.models[j]); } else { cache.models[i] = Object.assign({}, cache.models[j], cache.models[i]); } } // else is not complete local file cache.models[i].isLocal = true; cache.models[i].localSize = cache.models[j].size; cache.models[i].tags = tags; cache.models.splice(j, 1); j--; } } } commonStore.setModelSourceList(cache.models); if (initUnfinishedModels) initLastUnfinishedModelDownloads(); await saveCache().catch(() => { }); } function initLastUnfinishedModelDownloads() { const list: DownloadStatus[] = []; commonStore.modelSourceList.forEach((item) => { if (item.isLocal && !item.isComplete) { list.push( { name: item.name, path: `${commonStore.settings.customModelsPath}/${item.name}`, url: getHfDownloadUrl(item.downloadUrl!), transferred: item.localSize!, size: item.size, speed: 0, progress: item.localSize! / item.size * 100, downloading: false, done: false } ); } }); commonStore.setLastUnfinishedModelDownloads(list); } export async function refreshRemoteModels(cache: { models: ModelSourceItem[] }, filter: boolean = true, initUnfinishedModels: boolean = false) { const manifestUrls = commonStore.modelSourceManifestList.split(/[,,;;\n]/); const requests = manifestUrls.filter(url => url.endsWith('.json')).map( url => fetch(url, { cache: 'no-cache' }).then(r => r.json())); await Promise.allSettled(requests) .then((data: PromiseSettledResult[]) => { cache.models.push(...data.flatMap(d => { if (d.status === 'fulfilled') return d.value.models; return []; })); }) .catch(() => { }); cache.models = cache.models.filter((model, index, self) => { return modelSuffix.some((ext => model.name.endsWith(ext))) && index === findLastIndex(self, m => m.name === model.name || (!!m.SHA256 && m.SHA256 === model.SHA256 && m.size === model.size)); }); await refreshLocalModels(cache, filter, initUnfinishedModels); } export const refreshModels = async (readCache: boolean = false, initUnfinishedModels: boolean = false) => { const cache = await refreshBuiltInModels(readCache); await refreshLocalModels(cache, false, initUnfinishedModels); await refreshRemoteModels(cache, false, initUnfinishedModels); }; export const getStrategy = (modelConfig: ModelConfig | undefined = undefined) => { let params: ModelParameters; if (modelConfig) params = modelConfig.modelParameters; else params = commonStore.getCurrentModelConfig().modelParameters; const modelName = params.modelName.toLowerCase(); const avoidOverflow = params.precision !== 'fp32' && modelName.includes('world') && (modelName.includes('0.1b') || modelName.includes('0.4b') || modelName.includes('1.5b') || modelName.includes('1b5')); let strategy = ''; switch (params.device) { case 'CPU': if (avoidOverflow) strategy = 'cpu fp32 *1 -> '; strategy += 'cpu '; strategy += params.precision === 'int8' ? 'fp32i8' : 'fp32'; break; case 'WebGPU': case 'WebGPU (Python)': strategy += params.precision === 'nf4' ? 'fp16i4' : params.precision === 'int8' ? 'fp16i8' : 'fp16'; if (params.quantizedLayers) strategy += ` layer${params.quantizedLayers}`; if (params.tokenChunkSize) strategy += ` chunk${params.tokenChunkSize}`; break; case 'CUDA': case 'CUDA-Beta': if (avoidOverflow) strategy = params.useCustomCuda ? 'cuda fp16 *1 -> ' : 'cuda fp32 *1 -> '; strategy += 'cuda '; strategy += params.precision === 'int8' ? 'fp16i8' : params.precision === 'fp32' ? 'fp32' : 'fp16'; if (params.storedLayers < params.maxStoredLayers) strategy += ` *${params.storedLayers}+`; else strategy += ` -> cuda fp16 *1`; break; case 'MPS': if (avoidOverflow) strategy = 'mps fp32 *1 -> '; strategy += 'mps '; strategy += params.precision === 'int8' ? 'fp32i8' : 'fp32'; break; case 'Custom': strategy = params.customStrategy || ''; break; } return strategy; }; export const saveConfigs = throttle(async () => { const data: LocalConfig = { modelSourceManifestList: commonStore.modelSourceManifestList, currentModelConfigIndex: commonStore.currentModelConfigIndex, modelConfigs: commonStore.modelConfigs, settings: commonStore.settings, dataProcessParams: commonStore.dataProcessParams, loraFinetuneParams: commonStore.loraFinetuneParams }; return SaveJson('config.json', data); }, 500, { leading: true, trailing: true }); export const saveCache = async () => { const data: Cache = { version: manifest.version, models: commonStore.modelSourceList, depComplete: commonStore.depComplete }; return SaveJson('cache.json', data); }; export const savePresets = async () => { return SaveJson('presets.json', commonStore.presets); }; export function getUserLanguage(): Language { // const l = navigator.language.toLowerCase(); // if (['zh-hk', 'zh-mo', 'zh-tw', 'zh-cht', 'zh-hant'].includes(l)) return 'zhHant' const l = navigator.language.substring(0, 2); if (l in Languages) return l as Language; return 'dev'; } export function isSystemLightMode() { return window.matchMedia && window.matchMedia('(prefers-color-scheme: light)').matches; } export function downloadProgramFiles() { manifest.programFiles.forEach(({ url, path }) => { if (path) ReadFileInfo(path).then(info => { if (info.size === 0 && url) AddToDownloadList(path, url.replace('@master', '@v' + manifest.version)); }).catch(() => { if (url) AddToDownloadList(path, url.replace('@master', '@v' + manifest.version)); }); }); } export function forceDownloadProgramFiles() { manifest.programFiles.forEach(({ url, path }) => { if (path && url) AddToDownloadList(path, url.replace('@master', '@v' + manifest.version)); }); } export async function deleteDynamicProgramFiles() { let promises: Promise[] = []; manifest.programFiles.forEach(({ path }) => { if ((path.endsWith('.py') && !path.includes('get-pip.py')) || path.includes('requirements') || path.endsWith('.pyd')) promises.push(DeleteFile(path)); }); return await Promise.allSettled(promises).catch(() => { }); } export function bytesToGb(size: number) { return (size / 1024 / 1024 / 1024).toFixed(2); } export function bytesToMb(size: number) { return (size / 1024 / 1024).toFixed(2); } export function bytesToKb(size: number) { return (size / 1024).toFixed(2); } export function bytesToReadable(size: number) { if (size < 1024) return size + ' B'; else if (size < 1024 * 1024) return bytesToKb(size) + ' KB'; else if (size < 1024 * 1024 * 1024) return bytesToMb(size) + ' MB'; else return bytesToGb(size) + ' GB'; } export async function getReqUrl(port: number, path: string, isCore: boolean = false): Promise<{ url: string, headers: { [key: string]: string } }> { const realUrl = getServerRoot(port, isCore) + path; if (commonStore.platform === 'web') return { url: realUrl, headers: {} }; if (!commonStore.proxyPort) await GetProxyPort().then(p => commonStore.setProxyPort(p)); return { url: `http://127.0.0.1:${commonStore.proxyPort}`, headers: { 'Real-Target': realUrl } }; } export function getServerRoot(defaultLocalPort: number, isCore: boolean = false) { const coreCustomApiUrl = commonStore.settings.coreApiUrl.trim().replace(/\/$/, ''); if (isCore && coreCustomApiUrl) return coreCustomApiUrl; const defaultRoot = `http://127.0.0.1:${defaultLocalPort}`; if (commonStore.status.status !== ModelStatus.Offline) return defaultRoot; const customApiUrl = commonStore.settings.apiUrl.trim().replace(/\/$/, ''); if (customApiUrl) return customApiUrl; if (commonStore.platform === 'web') return ''; return defaultRoot; } export function absPathAsset(path: string) { if (commonStore.platform === 'web') return path; if (path === logo) return path; if ((path.length > 0 && path[0] === '/') || (path.length > 1 && path[1] === ':')) { return '=>' + path; } return path; } export async function checkUpdate(notifyEvenLatest: boolean = false) { fetch(!commonStore.settings.giteeUpdatesSource ? 'https://api.github.com/repos/josstorer/RWKV-Runner/releases/latest' : 'https://gitee.com/api/v5/repos/josc146/RWKV-Runner/releases/latest' ).then((r) => { if (r.ok) { r.json().then((data) => { if (data.tag_name) { const versionTag = data.tag_name; if (versionTag.replace('v', '') > manifest.version) { const verifyUrl = !commonStore.settings.giteeUpdatesSource ? `https://api.github.com/repos/josstorer/RWKV-Runner/releases/tags/${versionTag}` : `https://gitee.com/api/v5/repos/josc146/RWKV-Runner/releases/tags/${versionTag}`; fetch(verifyUrl).then((r) => { if (r.ok) { r.json().then((data) => { if (data.assets && data.assets.length > 0) { const asset = data.assets.find((a: any) => a.name.toLowerCase().includes(commonStore.platform.toLowerCase().replace('darwin', 'macos'))); if (asset) { const updateUrl = !commonStore.settings.giteeUpdatesSource ? `https://github.com/josStorer/RWKV-Runner/releases/download/${versionTag}/${asset.name}` : `https://gitee.com/josc146/RWKV-Runner/releases/download/${versionTag}/${asset.name}`; toastWithButton(t('New Version Available') + ': ' + versionTag, t('Update'), () => { DeleteFile('cache.json'); const progressId = 'update_app'; const progressEvent = 'updateApp'; const updateProgress = (ds: DownloadStatus | null) => { const content = t('Downloading update, please wait. If it is not completed, please manually download the program from GitHub and replace the original program.') + (ds ? ` (${ds.progress.toFixed(2)}% ${bytesToReadable(ds.transferred)}/${bytesToReadable(ds.size)})` : ''); const options: ToastOptions = { type: 'info', position: 'bottom-left', autoClose: false, toastId: progressId, hideProgressBar: false, progress: ds ? ds.progress / 100 : 0 }; if (toast.isActive(progressId)) toast.update(progressId, { render: content, ...options }); else toast(content, options); }; updateProgress(null); EventsOn(progressEvent, updateProgress); UpdateApp(updateUrl).then(() => { toast(t('Update completed, please restart the program.'), { type: 'success', position: 'bottom-left', autoClose: false } ); }).catch((e) => { toast(t('Update Error') + ' - ' + (e.message || e), { type: 'error', position: 'bottom-left', autoClose: false }); }).finally(() => { toast.dismiss(progressId); EventsOff(progressEvent); }); }, { autoClose: false, position: 'bottom-left' }); } } }); } else { throw new Error('Verify response was not ok.'); } }); } else { if (notifyEvenLatest) { toast(t('This is the latest version'), { type: 'success', position: 'bottom-left', autoClose: 2000 }); } } } else { throw new Error('Invalid response.'); } }); } else { throw new Error('Network response was not ok.'); } } ).catch((e) => { toast(t('Updates Check Error') + ' - ' + (e.message || e), { type: 'error', position: 'bottom-left' }); }); } export const checkDependencies = async (navigate: NavigateFunction) => { if (!commonStore.depComplete) { let depErrorMsg = ''; await DepCheck(commonStore.settings.customPythonPath).catch((e) => { depErrorMsg = e.message || e; WindowShow(); if (depErrorMsg === 'python zip not found') { toastWithButton(t('Python target not found, would you like to download it?'), t('Download'), () => { toastWithButton(`${t('Downloading')} Python`, t('Check'), () => { navigate({ pathname: '/downloads' }); }, { autoClose: 3000 }); AddToDownloadList('python-3.10.11-embed-amd64.zip', !commonStore.settings.cnMirror ? 'https://www.python.org/ftp/python/3.10.11/python-3.10.11-embed-amd64.zip' : 'https://mirrors.huaweicloud.com/python/3.10.11/python-3.10.11-embed-amd64.zip'); }); } else if (depErrorMsg.includes('DepCheck Error')) { if (depErrorMsg.includes('vc_redist') || depErrorMsg.includes('DLL load failed while importing')) { toastWithButton(t('Microsoft Visual C++ Redistributable is not installed, would you like to download it?'), t('Download'), () => { BrowserOpenURL('https://aka.ms/vs/16/release/vc_redist.x64.exe'); }); } else { toast(depErrorMsg, { type: 'info', position: 'bottom-left' }); if (commonStore.platform !== 'linux') toastWithButton(t('Python dependencies are incomplete, would you like to install them?'), t('Install'), () => { InstallPyDep(commonStore.settings.customPythonPath, commonStore.settings.cnMirror).catch((e) => { const errMsg = e.message || e; toast(t('Error') + ' - ' + errMsg, { type: 'error' }); }); setTimeout(WindowShow, 1000); }, { autoClose: 8000 }); else toastWithButton(t('On Linux system, you must manually install python dependencies.'), t('Check'), () => { BrowserOpenURL('https://github.com/josStorer/RWKV-Runner/blob/master/build/linux/Readme_Install.txt'); }); } } else { toast(depErrorMsg, { type: 'error' }); } }); if (depErrorMsg) { commonStore.setStatus({ status: ModelStatus.Offline }); return false; } commonStore.setDepComplete(true); } return true; }; export function toastWithButton(text: string, buttonText: string, onClickButton: () => void, options?: ToastOptions) { let triggered = false; const id = toast(
{text}
, { type: 'info', ...options }); return id; } export function getHfDownloadUrl(url: string) { if (commonStore.settings.useHfMirror && url.includes('huggingface.co') && url.includes('resolve')) return url.replace('huggingface.co', 'hf-mirror.com'); return url; } export function refreshTracksTotalTime() { if (commonStore.tracks.length === 0) { commonStore.setTrackTotalTime(tracksMinimalTotalTime); commonStore.setTrackCurrentTime(0); commonStore.setTrackPlayStartTime(0); return; } const endTimes = commonStore.tracks.map(t => t.offsetTime + t.contentTime); const totalTime = Math.max(...endTimes) + tracksMinimalTotalTime; if (commonStore.trackPlayStartTime > totalTime) commonStore.setTrackPlayStartTime(totalTime); commonStore.setTrackTotalTime(totalTime); } export function getMidiRawContentTime(rawContent: MidiMessage[]) { return rawContent.reduce((sum, current) => sum + (current.messageType === 'ElapsedTime' ? current.value : 0) , 0); } export function getMidiRawContentMainInstrument(rawContent: MidiMessage[]) { const sortedInstrumentFrequency = Object.entries(rawContent .filter(c => c.messageType === 'NoteOn') .map(c => c.instrument) .reduce((frequencyCount, current) => (frequencyCount[current] = (frequencyCount[current] || 0) + 1, frequencyCount) , {} as { [key: string]: number })) .sort((a, b) => b[1] - a[1]); let mainInstrument: string = ''; if (sortedInstrumentFrequency.length > 0) mainInstrument = InstrumentTypeNameMap[Number(sortedInstrumentFrequency[0][0])]; return mainInstrument; } export function flushMidiRecordingContent() { const recordingTrackIndex = commonStore.tracks.findIndex(t => t.id === commonStore.recordingTrackId); if (recordingTrackIndex >= 0) { const recordingTrack = commonStore.tracks[recordingTrackIndex]; const tracks = commonStore.tracks.slice(); tracks[recordingTrackIndex] = { ...recordingTrack, content: commonStore.recordingContent, rawContent: commonStore.recordingRawContent, contentTime: getMidiRawContentTime(commonStore.recordingRawContent), mainInstrument: getMidiRawContentMainInstrument(commonStore.recordingRawContent) }; commonStore.setTracks(tracks); refreshTracksTotalTime(); } commonStore.setRecordingContent(''); commonStore.setRecordingRawContent([]); } export async function getSoundFont() { let soundUrl: string; if (commonStore.compositionParams.useLocalSoundFont) soundUrl = 'assets/sound-font'; else soundUrl = !commonStore.settings.giteeUpdatesSource ? `https://raw.githubusercontent.com/josStorer/sgm_plus/master` : `https://cdn.jsdelivr.net/gh/josstorer/sgm_plus`; const fallbackUrl = 'https://cdn.jsdelivr.net/gh/josstorer/sgm_plus'; await fetch(soundUrl + '/soundfont.json').then(r => { if (!r.ok) soundUrl = fallbackUrl; }).catch(() => soundUrl = fallbackUrl); return soundUrl; } export const setActivePreset = (preset: Preset | null, index: number) => { commonStore.setActivePreset(preset); commonStore.setActivePresetIndex(index); //TODO if (preset.displayPresetMessages) { const { pushMessage, saveConversation } = newChatConversation(); if (preset) for (const message of preset.messages) { pushMessage(message.role, message.content); } saveConversation(); //} }; export function getSupportedCustomCudaFile(isBeta: boolean) { if ([' 10', ' 16', ' 20', ' 30', 'MX', 'Tesla P', 'Quadro P', 'NVIDIA P', 'TITAN X', 'TITAN RTX', 'RTX A', 'Quadro RTX 4000', 'Quadro RTX 5000', 'Tesla T4', 'NVIDIA A10', 'NVIDIA A40'].some(v => commonStore.status.device_name.includes(v))) return isBeta ? './backend-python/wkv_cuda_utils/beta/wkv_cuda10_30.pyd' : './backend-python/wkv_cuda_utils/wkv_cuda10_30.pyd'; else if ([' 40', 'RTX 5000 Ada', 'RTX 6000 Ada', 'RTX TITAN Ada', 'NVIDIA L40'].some(v => commonStore.status.device_name.includes(v))) return isBeta ? './backend-python/wkv_cuda_utils/beta/wkv_cuda40.pyd' : './backend-python/wkv_cuda_utils/wkv_cuda40.pyd'; else return ''; } // a wrapper for webOpenOpenFileDialog and OpenOpenFileDialog export function OpenFileDialog(filterPattern: string): Promise { return new Promise((resolve) => { OpenOpenFileDialog(filterPattern).then(async filePath => { if (!filePath) return; let blob: Blob; if (commonStore.platform === 'web') blob = (filePath as unknown as { blob: Blob }).blob; else blob = await fetch(absPathAsset(filePath)).then(r => r.blob()); resolve(blob); }).catch(e => { toast(t('Error') + ' - ' + (e.message || e), { type: 'error', autoClose: 2500 }); }); } ); } export function newChatConversation() { const conversation: Conversation = {}; const conversationOrder: string[] = []; const pushMessage = (role: Role, content: string) => { const newUuid = uuid(); conversationOrder.push(newUuid); conversation[newUuid] = { sender: role === 'user' ? userName : role === 'assistant' ? botName : systemName, type: MessageType.Normal, color: role === 'user' ? 'brand' : 'neutral', avatarImg: role === 'user' ? undefined : logo, time: new Date().toISOString(), content: content, side: role === 'user' ? 'right' : role === 'assistant' ? 'left' : 'center', done: true }; }; const saveConversation = () => { commonStore.setConversation(conversation); commonStore.setConversationOrder(conversationOrder); }; return { pushMessage, saveConversation }; }