detemine root cause of download issue.
This commit is contained in:
parent
87446784ae
commit
4549442bd8
@ -136,12 +136,32 @@ export function whisperFileExists(whisper_model : whisper_model_tag_t) {
|
|||||||
return target.exists
|
return target.exists
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function downloadWhisperModel(
|
export type DownloadCallback = (arg0 : FileSystem.DownloadProgressData) => any;
|
||||||
|
|
||||||
|
async function updateModelSize(model_label : string, size : number) {
|
||||||
|
const db = await getDb();
|
||||||
|
const query = "INSERT OR REPLACE INTO whisper_models (model, bytes_total) VALUES (?, ?)"
|
||||||
|
const stmt = db.prepareSync(query);
|
||||||
|
stmt.executeSync(model_label, size);
|
||||||
|
}
|
||||||
|
|
||||||
|
async function getExpectedModelSize(model_label : string) : Promise<number | undefined> {
|
||||||
|
const db = await getDb();
|
||||||
|
const query = "SELECT bytes_total FROM whisper_models WHERE model = ?"
|
||||||
|
const stmt = db.prepareSync(query);
|
||||||
|
const curs = stmt.executeSync(model_label);
|
||||||
|
const row = curs.getFirstSync()
|
||||||
|
return row ? row.bytes_total : undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export async function initiateWhisperDownload(
|
||||||
whisper_model: whisper_model_tag_t,
|
whisper_model: whisper_model_tag_t,
|
||||||
options: {
|
options: {
|
||||||
force_redownload: boolean;
|
force_redownload?: boolean;
|
||||||
|
onDownload?: DownloadCallback | undefined;
|
||||||
} = {
|
} = {
|
||||||
force_redownload: false,
|
force_redownload: false,
|
||||||
|
onDownload: undefined,
|
||||||
}
|
}
|
||||||
) {
|
) {
|
||||||
|
|
||||||
@ -161,34 +181,33 @@ export async function downloadWhisperModel(
|
|||||||
if (options.force_redownload) {
|
if (options.force_redownload) {
|
||||||
whisperTarget.delete()
|
whisperTarget.delete()
|
||||||
} else {
|
} else {
|
||||||
console.warn("Whisper model for %s already exists", whisper_model);
|
const expected = await getExpectedModelSize(whisper_model);
|
||||||
return;
|
if (whisperTarget.size === expected) {
|
||||||
|
console.warn("Whisper model for %s already exists", whisper_model);
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// Initiate a new resumable download.
|
// Initiate a new resumable download.
|
||||||
const spec = WHISPER_MODELS[whisper_model];
|
const spec = WHISPER_MODELS[whisper_model];
|
||||||
|
|
||||||
|
console.log("Downloading %s", spec.source);
|
||||||
|
|
||||||
const resumable = FileSystem.createDownloadResumable(
|
const resumable = FileSystem.createDownloadResumable(
|
||||||
spec.source,
|
spec.source,
|
||||||
whisperTarget.uri,
|
whisperTarget.uri,
|
||||||
undefined,
|
{},
|
||||||
// On each data write, update the whisper model download status.
|
// On each data write, update the whisper model download status.
|
||||||
// Note that since createDownloadResumable callback only works in the foreground,
|
// Note that since createDownloadResumable callback only works in the foreground,
|
||||||
// a background process will also be updating the file size.
|
// a background process will also be updating the file size.
|
||||||
async (data) => {
|
async (data) => {
|
||||||
const db = await getDb();
|
console.log("%s: %d bytes of %d", whisperTarget.uri, data.totalBytesWritten, data.totalBytesExpectedToWrite);
|
||||||
const args = [
|
await updateModelSize(whisper_model, data.totalBytesExpectedToWrite)
|
||||||
whisper_model,
|
if (options.onDownload) await options.onDownload(data);
|
||||||
data.totalBytesWritten,
|
},
|
||||||
data.totalBytesExpectedToWrite,
|
whisperTarget.exists ? whisperTarget.base64() : undefined,
|
||||||
];
|
|
||||||
console.log("%s, %s of %s", whisper_model, data.totalBytesWritten, data.totalBytesExpectedToWrite);
|
|
||||||
await db.runAsync(
|
|
||||||
`INSERT OR REPLACE INTO whisper_models (model, bytes_done, bytes_remaining) VALUES (?, ?, ?)`,
|
|
||||||
args
|
|
||||||
);
|
|
||||||
}
|
|
||||||
);
|
);
|
||||||
|
|
||||||
await resumable.downloadAsync();
|
return resumable;
|
||||||
}
|
}
|
@ -6,11 +6,12 @@ import { Settings } from "@/app/lib/settings";
|
|||||||
import { LanguageServer, fetchWithTimeout } from "@/app/i18n/api";
|
import { LanguageServer, fetchWithTimeout } from "@/app/i18n/api";
|
||||||
import { Picker } from "@react-native-picker/picker";
|
import { Picker } from "@react-native-picker/picker";
|
||||||
import { longLang } from "@/app/i18n/lang";
|
import { longLang } from "@/app/i18n/lang";
|
||||||
|
import FileSystem, { DownloadResumable } from "expo-file-system";
|
||||||
import { LIBRETRANSLATE_BASE_URL } from "@/constants/api";
|
import { LIBRETRANSLATE_BASE_URL } from "@/constants/api";
|
||||||
import {
|
import {
|
||||||
WHISPER_MODELS,
|
WHISPER_MODELS,
|
||||||
WHISPER_MODEL_DIR,
|
WHISPER_MODEL_DIR,
|
||||||
downloadWhisperModel,
|
initiateWhisperDownload,
|
||||||
download_status,
|
download_status,
|
||||||
getWhisperDownloadStatus,
|
getWhisperDownloadStatus,
|
||||||
getWhisperTarget,
|
getWhisperTarget,
|
||||||
@ -53,6 +54,10 @@ const SettingsComponent: React.FC = () => {
|
|||||||
const [langServerConn, setLangServerConn] = useState<
|
const [langServerConn, setLangServerConn] = useState<
|
||||||
undefined | connection_test_t
|
undefined | connection_test_t
|
||||||
>();
|
>();
|
||||||
|
const [whisperDownloadProgress, setWhisperDownloadProgress] = useState<
|
||||||
|
FileSystem.DownloadProgressData | undefined
|
||||||
|
>();
|
||||||
|
const [downloader, setDownloader] = useState<DownloadResumable | undefined>();
|
||||||
|
|
||||||
const fillHostLanguageOptions = async () => {
|
const fillHostLanguageOptions = async () => {
|
||||||
const settings = await Settings.getDefault();
|
const settings = await Settings.getDefault();
|
||||||
@ -73,7 +78,7 @@ const SettingsComponent: React.FC = () => {
|
|||||||
error: `Could not connect to ${libretranslateBaseUrl}: ${err}`,
|
error: `Could not connect to ${libretranslateBaseUrl}: ${err}`,
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
}
|
};
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
(async () => {
|
(async () => {
|
||||||
@ -147,12 +152,25 @@ const SettingsComponent: React.FC = () => {
|
|||||||
|
|
||||||
const doReadownload = async () => {
|
const doReadownload = async () => {
|
||||||
if (!whisperModel) return;
|
if (!whisperModel) return;
|
||||||
await downloadWhisperModel(whisperModel, { force_redownload: true });
|
await initiateWhisperDownload(whisperModel, {
|
||||||
|
force_redownload: true,
|
||||||
|
onDownload: setWhisperDownloadProgress,
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
const doDownload = async () => {
|
const doDownload = async () => {
|
||||||
if (!whisperModel) return;
|
if (!whisperModel) return;
|
||||||
await downloadWhisperModel(whisperModel);
|
try {
|
||||||
|
setDownloader(
|
||||||
|
await initiateWhisperDownload(whisperModel, {
|
||||||
|
onDownload: setWhisperDownloadProgress,
|
||||||
|
})
|
||||||
|
);
|
||||||
|
await downloader?.downloadAsync();
|
||||||
|
console.log("completed download");
|
||||||
|
} catch (err) {
|
||||||
|
console.error(err);
|
||||||
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const handleHostLanguageChange = async (value: string) => {
|
const handleHostLanguageChange = async (value: string) => {
|
||||||
@ -217,28 +235,18 @@ const SettingsComponent: React.FC = () => {
|
|||||||
<Picker.Item key={key} label={label} value={key} />
|
<Picker.Item key={key} label={label} value={key} />
|
||||||
))}
|
))}
|
||||||
</Picker>
|
</Picker>
|
||||||
{whisperModel && (
|
<View>
|
||||||
<View>
|
{ /* If there's a downloader, that means we're in the middle of a download */}
|
||||||
{downloadStatus?.status === "complete" ? (
|
{downloader && whisperDownloadProgress && (
|
||||||
<Pressable onPress={doReadownload}>Re-Download</Pressable>
|
<Text>
|
||||||
) : downloadStatus?.status === "in_progress" ? (
|
{whisperDownloadProgress.totalBytesWritten} of {whisperDownloadProgress.totalBytesExpectedToWrite}
|
||||||
<Text>
|
</Text>
|
||||||
{(downloadStatus.bytes.done / downloadStatus.bytes.total) * 100.0}{" "}
|
)
|
||||||
% complete
|
}
|
||||||
{downloadStatus.bytes.done} bytes of {downloadStatus.bytes.total}
|
<Pressable onPress={doDownload} style={styles.button}>
|
||||||
</Text>
|
<Text style={styles.buttonText}>Download</Text>
|
||||||
) : (
|
</Pressable>
|
||||||
<View>
|
</View>
|
||||||
<Pressable onPress={doDownload} style={styles.button}>
|
|
||||||
<Text style={styles.buttonText}>Download</Text>
|
|
||||||
</Pressable>
|
|
||||||
<Text>
|
|
||||||
This will download to {Paths.join(WHISPER_MODEL_DIR, WHISPER_MODELS[whisperModel].target)}
|
|
||||||
</Text>
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
</View>
|
|
||||||
)}
|
|
||||||
</View>
|
</View>
|
||||||
) : (
|
) : (
|
||||||
<View>
|
<View>
|
||||||
|
0
components/ui/WhisperDownloadButton.tsx
Normal file
0
components/ui/WhisperDownloadButton.tsx
Normal file
33
components/ui/WhisperDownloader.tsx
Normal file
33
components/ui/WhisperDownloader.tsx
Normal file
@ -0,0 +1,33 @@
|
|||||||
|
import { Settings } from "@/app/lib/settings"
|
||||||
|
import { WHISPER_MODELS } from "@/app/lib/whisper"
|
||||||
|
import { Picker } from "@react-native-picker/picker"
|
||||||
|
import { useEffect, useState } from "react"
|
||||||
|
import { Pressable, View } from "react-native"
|
||||||
|
|
||||||
|
const WhisperDownloader = () => {
|
||||||
|
|
||||||
|
const [whisperModel, setWhisperModel] = useState<string|undefined>();
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const settings = await Settings.getDefault();
|
||||||
|
setWhisperModel((await settings.getWhisperModel()) || "small");
|
||||||
|
}, [])
|
||||||
|
|
||||||
|
return (
|
||||||
|
<View>
|
||||||
|
<Picker
|
||||||
|
selectedValue={whisperModel || ""}
|
||||||
|
style={{ height: 50, width: "100%" }}
|
||||||
|
onValueChange={setWhisperModel}
|
||||||
|
accessibilityHint="whisper model"
|
||||||
|
>
|
||||||
|
{Object.entries(WHISPER_MODELS).map(([key, { label }]) => (
|
||||||
|
<Picker.Item key={key} label={label} value={key} />
|
||||||
|
))}
|
||||||
|
</Picker>
|
||||||
|
|
||||||
|
<WhisperDownloadButton whisperModel={whisperModel} />
|
||||||
|
|
||||||
|
<WhisperDownloadInfo whisperModel={whisperModel} />
|
||||||
|
</View>
|
||||||
|
}
|
Loading…
x
Reference in New Issue
Block a user