Selector.svelte 16 KB
Newer Older
1
<script lang="ts">
2
	import { DropdownMenu } from 'bits-ui';
Timothy J. Baek's avatar
Timothy J. Baek committed
3
	import { marked } from 'marked';
4
	import Fuse from 'fuse.js';
5
6

	import { flyAndScale } from '$lib/utils/transitions';
7
	import { createEventDispatcher, onMount, getContext, tick } from 'svelte';
8
9
10
11
12

	import ChevronDown from '$lib/components/icons/ChevronDown.svelte';
	import Check from '$lib/components/icons/Check.svelte';
	import Search from '$lib/components/icons/Search.svelte';

13
	import { deleteModel, getOllamaVersion, pullModel } from '$lib/apis/ollama';
14

15
	import { user, MODEL_DOWNLOAD_POOL, models, mobile } from '$lib/stores';
16
	import { toast } from 'svelte-sonner';
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
17
18
19
	import { capitalizeFirstLetter, sanitizeResponseContent, splitStream } from '$lib/utils';
	import { getModels } from '$lib/apis';

20
21
22
23
24
25
26
27
	import Tooltip from '$lib/components/common/Tooltip.svelte';

	const i18n = getContext('i18n');
	const dispatch = createEventDispatcher();

	export let value = '';
	export let placeholder = 'Select a model';
	export let searchEnabled = true;
28
	export let searchPlaceholder = $i18n.t('Search a model');
29

30
31
32
33
34
35
	export let items: {
		label: string;
		value: string;
		// eslint-disable-next-line @typescript-eslint/no-explicit-any
		[key: string]: any;
	} = [];
36

Timothy J. Baek's avatar
Timothy J. Baek committed
37
	export let className = 'w-[32rem]';
38

Timothy J. Baek's avatar
Timothy J. Baek committed
39
	let show = false;
40
41
42
43

	let selectedModel = '';
	$: selectedModel = items.find((item) => item.value === value) ?? '';

44
45
46
	let searchValue = '';
	let ollamaVersion = null;

47
	let selectedModelIdx = 0;
48

49
50
51
52
53
54
55
56
57
58
59
60
61
62
	const fuse = new Fuse(
		items
			.filter((item) => !item.model?.info?.meta?.hidden)
			.map((item) => {
				const _item = {
					...item,
					modelName: item.model?.name,
					tags: item.model?.info?.meta?.tags?.map((tag) => tag.name).join(' ')
				};
				return _item;
			}),
		{
			keys: ['value', 'label', 'tags', 'modelName']
		}
63
	);
64

65
66
67
68
69
70
	$: filteredItems = searchValue
		? fuse.search(searchValue).map((e) => {
				return e.item;
		  })
		: items.filter((item) => !item.model?.info?.meta?.hidden);

71
	const pullModelHandler = async () => {
Timothy J. Baek's avatar
Timothy J. Baek committed
72
		const sanitizedModelTag = searchValue.trim().replace(/^ollama\s+(run|pull)\s+/, '');
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89

		console.log($MODEL_DOWNLOAD_POOL);
		if ($MODEL_DOWNLOAD_POOL[sanitizedModelTag]) {
			toast.error(
				$i18n.t(`Model '{{modelTag}}' is already in queue for downloading.`, {
					modelTag: sanitizedModelTag
				})
			);
			return;
		}
		if (Object.keys($MODEL_DOWNLOAD_POOL).length === 3) {
			toast.error(
				$i18n.t('Maximum of 3 models can be downloaded simultaneously. Please try again later.')
			);
			return;
		}

90
91
92
93
94
95
		const [res, controller] = await pullModel(localStorage.token, sanitizedModelTag, '0').catch(
			(error) => {
				toast.error(error);
				return null;
			}
		);
96
97
98
99
100
101
102

		if (res) {
			const reader = res.body
				.pipeThrough(new TextDecoderStream())
				.pipeThrough(splitStream('\n'))
				.getReader();

103
104
105
106
107
108
109
110
111
112
			MODEL_DOWNLOAD_POOL.set({
				...$MODEL_DOWNLOAD_POOL,
				[sanitizedModelTag]: {
					...$MODEL_DOWNLOAD_POOL[sanitizedModelTag],
					abortController: controller,
					reader,
					done: false
				}
			});

113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
			while (true) {
				try {
					const { value, done } = await reader.read();
					if (done) break;

					let lines = value.split('\n');

					for (const line of lines) {
						if (line !== '') {
							let data = JSON.parse(line);
							console.log(data);
							if (data.error) {
								throw data.error;
							}
							if (data.detail) {
								throw data.detail;
							}

							if (data.status) {
								if (data.digest) {
									let downloadProgress = 0;
									if (data.completed) {
										downloadProgress = Math.round((data.completed / data.total) * 1000) / 10;
									} else {
										downloadProgress = 100;
									}

									MODEL_DOWNLOAD_POOL.set({
										...$MODEL_DOWNLOAD_POOL,
										[sanitizedModelTag]: {
											...$MODEL_DOWNLOAD_POOL[sanitizedModelTag],
											pullProgress: downloadProgress,
											digest: data.digest
										}
									});
								} else {
									toast.success(data.status);

									MODEL_DOWNLOAD_POOL.set({
										...$MODEL_DOWNLOAD_POOL,
										[sanitizedModelTag]: {
											...$MODEL_DOWNLOAD_POOL[sanitizedModelTag],
											done: data.status === 'success'
										}
									});
								}
							}
						}
					}
				} catch (error) {
					console.log(error);
					if (typeof error !== 'string') {
						error = error.message;
					}

					toast.error(error);
					// opts.callback({ success: false, error, modelName: opts.modelName });
170
					break;
171
172
173
174
175
176
177
178
179
				}
			}

			if ($MODEL_DOWNLOAD_POOL[sanitizedModelTag].done) {
				toast.success(
					$i18n.t(`Model '{{modelName}}' has been successfully downloaded.`, {
						modelName: sanitizedModelTag
					})
				);
180

Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
181
				models.set(await getModels(localStorage.token));
182
			} else {
Jannik Streidl's avatar
Jannik Streidl committed
183
				toast.error($i18n.t('Download canceled'));
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
			}

			delete $MODEL_DOWNLOAD_POOL[sanitizedModelTag];

			MODEL_DOWNLOAD_POOL.set({
				...$MODEL_DOWNLOAD_POOL
			});
		}
	};

	onMount(async () => {
		ollamaVersion = await getOllamaVersion(localStorage.token).catch((error) => false);
	});

	const cancelModelPullHandler = async (model: string) => {
199
200
201
202
		const { reader, abortController } = $MODEL_DOWNLOAD_POOL[model];
		if (abortController) {
			abortController.abort();
		}
203
204
205
206
207
208
209
210
211
212
213
214
		if (reader) {
			await reader.cancel();
			delete $MODEL_DOWNLOAD_POOL[model];
			MODEL_DOWNLOAD_POOL.set({
				...$MODEL_DOWNLOAD_POOL
			});
			await deleteModel(localStorage.token, model);
			toast.success(`${model} download has been canceled`);
		}
	};
</script>

215
<DropdownMenu.Root
Timothy J. Baek's avatar
Timothy J. Baek committed
216
	bind:open={show}
217
	onOpenChange={async () => {
218
		searchValue = '';
219
		selectedModelIdx = 0;
220
		window.setTimeout(() => document.getElementById('model-search-input')?.focus(), 0);
221
	}}
Timothy J. Baek's avatar
Timothy J. Baek committed
222
	closeFocus={false}
223
>
Timothy J. Baek's avatar
Timothy J. Baek committed
224
	<DropdownMenu.Trigger class="relative w-full font-primary" aria-label={placeholder}>
225
226
227
228
229
230
231
232
233
234
235
		<div
			class="flex w-full text-left px-0.5 outline-none bg-transparent truncate text-lg font-semibold placeholder-gray-400 focus:outline-none"
		>
			{#if selectedModel}
				{selectedModel.label}
			{:else}
				{placeholder}
			{/if}
			<ChevronDown className=" self-center ml-2 size-3" strokeWidth="2.5" />
		</div>
	</DropdownMenu.Trigger>
236

237
	<DropdownMenu.Content
Timothy J. Baek's avatar
Timothy J. Baek committed
238
239
		class=" z-40 {$mobile
			? `w-full`
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
240
			: `${className}`} max-w-[calc(100vw-1rem)] justify-start rounded-xl  bg-white dark:bg-gray-850 dark:text-white shadow-lg border border-gray-300/30 dark:border-gray-700/40  outline-none"
241
		transition={flyAndScale}
242
		side={$mobile ? 'bottom' : 'bottom-start'}
243
244
245
246
247
248
249
250
		sideOffset={4}
	>
		<slot>
			{#if searchEnabled}
				<div class="flex items-center gap-2.5 px-5 mt-3.5 mb-3">
					<Search className="size-4" strokeWidth="2.5" />

					<input
251
						id="model-search-input"
252
253
254
						bind:value={searchValue}
						class="w-full text-sm bg-transparent outline-none"
						placeholder={searchPlaceholder}
Timothy J. Baek's avatar
Timothy J. Baek committed
255
						autocomplete="off"
256
257
						on:keydown={(e) => {
							if (e.code === 'Enter' && filteredItems.length > 0) {
258
								value = filteredItems[selectedModelIdx].value;
259
								show = false;
260
261
								return; // dont need to scroll on selection
							} else if (e.code === 'ArrowDown') {
262
								selectedModelIdx = Math.min(selectedModelIdx + 1, filteredItems.length - 1);
263
							} else if (e.code === 'ArrowUp') {
264
								selectedModelIdx = Math.max(selectedModelIdx - 1, 0);
265
266
							} else {
								// if the user types something, reset to the top selection.
267
								selectedModelIdx = 0;
268
							}
269

270
							const item = document.querySelector(`[data-arrow-selected="true"]`);
271
							item?.scrollIntoView({ block: 'center', inline: 'nearest', behavior: 'instant' });
272
						}}
273
274
275
276
277
278
					/>
				</div>

				<hr class="border-gray-100 dark:border-gray-800" />
			{/if}

279
280
			<div class="px-3 my-2 max-h-64 overflow-y-auto scrollbar-hidden group">
				{#each filteredItems as item, index}
Timothy J. Baek's avatar
Timothy J. Baek committed
281
					<button
Timothy J. Baek's avatar
Timothy J. Baek committed
282
						aria-label="model-item"
283
						class="flex w-full text-left font-medium line-clamp-1 select-none items-center rounded-button py-2 pl-3 pr-1.5 text-sm text-gray-700 dark:text-gray-100 outline-none transition-all duration-75 hover:bg-gray-100 dark:hover:bg-gray-800 rounded-lg cursor-pointer data-[highlighted]:bg-muted {index ===
284
						selectedModelIdx
285
286
							? 'bg-gray-100 dark:bg-gray-800 group-hover:bg-transparent'
							: ''}"
287
						data-arrow-selected={index === selectedModelIdx}
288
289
						on:click={() => {
							value = item.value;
290
							selectedModelIdx = index;
Timothy J. Baek's avatar
Timothy J. Baek committed
291
292

							show = false;
293
						}}
294
					>
295
296
297
						<div class="flex flex-col">
							{#if $mobile && (item?.model?.info?.meta?.tags ?? []).length > 0}
								<div class="flex gap-0.5 self-start h-full mb-0.5 -translate-x-1">
Timothy J. Baek's avatar
Timothy J. Baek committed
298
299
									{#each item.model?.info?.meta.tags as tag}
										<div
Timothy J. Baek's avatar
Timothy J. Baek committed
300
											class=" text-xs font-bold px-1 rounded uppercase line-clamp-1 bg-gray-500/20 text-gray-700 dark:text-gray-200"
Timothy J. Baek's avatar
Timothy J. Baek committed
301
302
303
304
305
306
										>
											{tag.name}
										</div>
									{/each}
								</div>
							{/if}
307
							<div class="flex items-center gap-2">
Timothy J. Baek's avatar
Timothy J. Baek committed
308
								<div class="flex items-center min-w-fit">
309
									<div class="line-clamp-1">
310
311
312
										<div class="flex items-center min-w-fit">
											<img
												src={item.model?.info?.meta?.profile_image_url ?? '/static/favicon.png'}
Timothy J. Baek's avatar
refac  
Timothy J. Baek committed
313
314
												alt="Model"
												class="rounded-full size-5 flex items-center mr-2"
315
316
317
											/>
											{item.label}
										</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
318
									</div>
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
									{#if item.model.owned_by === 'ollama' && (item.model.ollama?.details?.parameter_size ?? '') !== ''}
										<div class="flex ml-1 items-center translate-y-[0.5px]">
											<Tooltip
												content={`${
													item.model.ollama?.details?.quantization_level
														? item.model.ollama?.details?.quantization_level + ' '
														: ''
												}${
													item.model.ollama?.size
														? `(${(item.model.ollama?.size / 1024 ** 3).toFixed(1)}GB)`
														: ''
												}`}
												className="self-end"
											>
												<span
													class=" text-xs font-medium text-gray-600 dark:text-gray-400 line-clamp-1"
													>{item.model.ollama?.details?.parameter_size ?? ''}</span
												>
											</Tooltip>
										</div>
									{/if}
								</div>
341

342
343
344
345
								<!-- {JSON.stringify(item.info)} -->

								{#if item.model.owned_by === 'openai'}
									<Tooltip content={`${'External'}`}>
Timothy J. Baek's avatar
Timothy J. Baek committed
346
										<div class="translate-y-[1px]">
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
											<svg
												xmlns="http://www.w3.org/2000/svg"
												viewBox="0 0 16 16"
												fill="currentColor"
												class="size-3"
											>
												<path
													fill-rule="evenodd"
													d="M8.914 6.025a.75.75 0 0 1 1.06 0 3.5 3.5 0 0 1 0 4.95l-2 2a3.5 3.5 0 0 1-5.396-4.402.75.75 0 0 1 1.251.827 2 2 0 0 0 3.085 2.514l2-2a2 2 0 0 0 0-2.828.75.75 0 0 1 0-1.06Z"
													clip-rule="evenodd"
												/>
												<path
													fill-rule="evenodd"
													d="M7.086 9.975a.75.75 0 0 1-1.06 0 3.5 3.5 0 0 1 0-4.95l2-2a3.5 3.5 0 0 1 5.396 4.402.75.75 0 0 1-1.251-.827 2 2 0 0 0-3.085-2.514l-2 2a2 2 0 0 0 0 2.828.75.75 0 0 1 0 1.06Z"
													clip-rule="evenodd"
												/>
											</svg>
										</div>
									</Tooltip>
								{/if}

								{#if item.model?.info?.meta?.description}
									<Tooltip
Timothy J. Baek's avatar
Timothy J. Baek committed
370
371
372
373
374
375
										content={`${marked.parse(
											sanitizeResponseContent(item.model?.info?.meta?.description).replaceAll(
												'\n',
												'<br>'
											)
										)}`}
376
									>
Timothy J. Baek's avatar
Timothy J. Baek committed
377
										<div class=" translate-y-[1px]">
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
											<svg
												xmlns="http://www.w3.org/2000/svg"
												fill="none"
												viewBox="0 0 24 24"
												stroke-width="1.5"
												stroke="currentColor"
												class="w-4 h-4"
											>
												<path
													stroke-linecap="round"
													stroke-linejoin="round"
													d="m11.25 11.25.041-.02a.75.75 0 0 1 1.063.852l-.708 2.836a.75.75 0 0 0 1.063.853l.041-.021M21 12a9 9 0 1 1-18 0 9 9 0 0 1 18 0Zm-9-3.75h.008v.008H12V8.25Z"
												/>
											</svg>
										</div>
									</Tooltip>
								{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
395
396
397
398
399
400
401
402
403
404
405
406
407
408

								{#if !$mobile && (item?.model?.info?.meta?.tags ?? []).length > 0}
									<div class="flex gap-0.5 self-center items-center h-full translate-y-[0.5px]">
										{#each item.model?.info?.meta.tags as tag}
											<Tooltip content={tag.name}>
												<div
													class=" text-xs font-bold px-1 rounded uppercase line-clamp-1 bg-gray-500/20 text-gray-700 dark:text-gray-200"
												>
													{tag.name}
												</div>
											</Tooltip>
										{/each}
									</div>
								{/if}
409
							</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
410
						</div>
411

412
						{#if value === item.value}
413
							<div class="ml-auto pl-2">
414
415
416
								<Check />
							</div>
						{/if}
Timothy J. Baek's avatar
Timothy J. Baek committed
417
					</button>
418
419
420
				{:else}
					<div>
						<div class="block px-3 py-2 text-sm text-gray-700 dark:text-gray-100">
421
							{$i18n.t('No results found')}
422
423
424
425
						</div>
					</div>
				{/each}

426
427
				{#if !(searchValue.trim() in $MODEL_DOWNLOAD_POOL) && searchValue && ollamaVersion && $user.role === 'admin'}
					<button
Timothy J. Baek's avatar
Timothy J. Baek committed
428
						class="flex w-full font-medium line-clamp-1 select-none items-center rounded-button py-2 pl-3 pr-1.5 text-sm text-gray-700 dark:text-gray-100 outline-none transition-all duration-75 hover:bg-gray-100 dark:hover:bg-gray-800 rounded-lg cursor-pointer data-[highlighted]:bg-muted"
429
430
431
432
						on:click={() => {
							pullModelHandler();
						}}
					>
433
						{$i18n.t(`Pull "{{searchValue}}" from Ollama.com`, { searchValue: searchValue })}
434
435
436
					</button>
				{/if}

437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
				{#each Object.keys($MODEL_DOWNLOAD_POOL) as model}
					<div
						class="flex w-full justify-between font-medium select-none rounded-button py-2 pl-3 pr-1.5 text-sm text-gray-700 dark:text-gray-100 outline-none transition-all duration-75 rounded-lg cursor-pointer data-[highlighted]:bg-muted"
					>
						<div class="flex">
							<div class="-ml-2 mr-2.5 translate-y-0.5">
								<svg
									class="size-4"
									viewBox="0 0 24 24"
									fill="currentColor"
									xmlns="http://www.w3.org/2000/svg"
									><style>
										.spinner_ajPY {
											transform-origin: center;
											animation: spinner_AtaB 0.75s infinite linear;
										}
										@keyframes spinner_AtaB {
											100% {
												transform: rotate(360deg);
											}
										}
									</style><path
										d="M12,1A11,11,0,1,0,23,12,11,11,0,0,0,12,1Zm0,19a8,8,0,1,1,8-8A8,8,0,0,1,12,20Z"
										opacity=".25"
									/><path
										d="M10.14,1.16a11,11,0,0,0-9,8.92A1.59,1.59,0,0,0,2.46,12,1.52,1.52,0,0,0,4.11,10.7a8,8,0,0,1,6.66-6.61A1.42,1.42,0,0,0,12,2.69h0A1.57,1.57,0,0,0,10.14,1.16Z"
										class="spinner_ajPY"
									/></svg
								>
							</div>

							<div class="flex flex-col self-start">
								<div class="line-clamp-1">
									Downloading "{model}" {'pullProgress' in $MODEL_DOWNLOAD_POOL[model]
										? `(${$MODEL_DOWNLOAD_POOL[model].pullProgress}%)`
										: ''}
								</div>

								{#if 'digest' in $MODEL_DOWNLOAD_POOL[model] && $MODEL_DOWNLOAD_POOL[model].digest}
									<div class="-mt-1 h-fit text-[0.7rem] dark:text-gray-500 line-clamp-1">
										{$MODEL_DOWNLOAD_POOL[model].digest}
									</div>
								{/if}
							</div>
						</div>

483
						<div class="mr-2 translate-y-0.5">
484
							<Tooltip content={$i18n.t('Cancel')}>
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
								<button
									class="text-gray-800 dark:text-gray-100"
									on:click={() => {
										cancelModelPullHandler(model);
									}}
								>
									<svg
										class="w-4 h-4 text-gray-800 dark:text-white"
										aria-hidden="true"
										xmlns="http://www.w3.org/2000/svg"
										width="24"
										height="24"
										fill="currentColor"
										viewBox="0 0 24 24"
									>
										<path
											stroke="currentColor"
											stroke-linecap="round"
											stroke-linejoin="round"
											stroke-width="2"
											d="M6 18 17.94 6M18 18 6.06 6"
										/>
									</svg>
								</button>
							</Tooltip>
						</div>
					</div>
				{/each}
			</div>
Timothy J. Baek's avatar
Timothy J. Baek committed
514
515
516

			<div class="hidden w-[42rem]" />
			<div class="hidden w-[32rem]" />
517
		</slot>
518
519
520
521
	</DropdownMenu.Content>
</DropdownMenu.Root>

<style>
Timothy J. Baek's avatar
Timothy J. Baek committed
522
523
524
	.scrollbar-hidden:active::-webkit-scrollbar-thumb,
	.scrollbar-hidden:focus::-webkit-scrollbar-thumb,
	.scrollbar-hidden:hover::-webkit-scrollbar-thumb {
525
526
		visibility: visible;
	}
Timothy J. Baek's avatar
Timothy J. Baek committed
527
	.scrollbar-hidden::-webkit-scrollbar-thumb {
528
529
530
		visibility: hidden;
	}
</style>