File size: 7,334 Bytes
e8b5344 01e9501 01e2d92 aac02fe 01e2d92 2168f52 01e2d92 5853d12 39318e7 01e9501 189caf9 e8b5344 01e2d92 aac02fe 9662103 aac02fe 2168f52 aac02fe 6a8491a aac02fe 6a8491a 01e2d92 2168f52 01e2d92 39318e7 5acf3a4 b2170a7 01e2d92 1a8c098 b2170a7 01e2d92 b2170a7 01e2d92 52803ee 01e2d92 1a8c098 2168f52 1a8c098 01e2d92 5acf3a4 01e2d92 2168f52 1a8c098 01e2d92 6a8491a 5acf3a4 6a8491a aac02fe 6a8491a aac02fe 6a8491a aac02fe 6a8491a aac02fe 6a8491a aac02fe 6a8491a b2170a7 aac02fe 6a8491a 01e2d92 aac02fe 1a8c098 654f56a b2170a7 654f56a b2170a7 |
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 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 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 |
import { defaultGenerationConfig } from "$lib/components/InferencePlayground/generationConfigSettings";
import { models } from "$lib/stores/models";
import {
PipelineTag,
type Conversation,
type ConversationMessage,
type DefaultProject,
type ModelWithTokenizer,
type Project,
type Session,
} from "$lib/types";
import { safeParse } from "$lib/utils/json";
import { getTrending } from "$lib/utils/model";
import { get, writable } from "svelte/store";
import typia from "typia";
const LOCAL_STORAGE_KEY = "hf_inference_playground_session";
const startMessageUser: ConversationMessage = { role: "user", content: "" };
const systemMessage: ConversationMessage = {
role: "system",
content: "",
};
const emptyModel: ModelWithTokenizer = {
_id: "",
inferenceProviderMapping: [],
pipeline_tag: PipelineTag.TextGeneration,
trendingScore: 0,
tags: ["text-generation"],
id: "",
tokenizerConfig: {},
config: {
architectures: [] as string[],
model_type: "",
tokenizer_config: {},
},
};
function getDefaults() {
const $models = get(models);
const featured = getTrending($models);
const defaultModel = featured[0] ?? $models[0] ?? emptyModel;
const defaultConversation: Conversation = {
model: defaultModel,
config: { ...defaultGenerationConfig },
messages: [{ ...startMessageUser }],
systemMessage,
streaming: true,
};
const defaultProject: DefaultProject = {
name: "Default",
id: "default",
conversations: [defaultConversation],
};
return { defaultProject, defaultConversation };
}
function createSessionStore() {
const store = writable<Session>(undefined, set => {
const { defaultConversation, defaultProject } = getDefaults();
// Get saved session from localStorage if available
let savedSession: Session = {
projects: [defaultProject],
activeProjectId: defaultProject.id,
};
const savedData = localStorage.getItem(LOCAL_STORAGE_KEY);
if (savedData) {
const parsed = safeParse(savedData);
const res = typia.validate<Session>(parsed);
if (res.success) {
savedSession = parsed;
} else {
localStorage.setItem(LOCAL_STORAGE_KEY, JSON.stringify(savedSession));
}
}
// Merge query params with savedSession's default project
// Query params models and providers take precedence over savedSession's.
// In any case, we try to merge the two, and the amount of conversations
// is the maximum between the two.
const dp = savedSession.projects.find(p => p.id === "default");
if (typia.is<DefaultProject>(dp)) {
const $models = get(models);
// Parse URL query parameters
const searchParams = new URLSearchParams(window.location.search);
const searchProviders = searchParams.getAll("provider");
const searchModelIds = searchParams.getAll("modelId");
const modelsFromSearch = searchModelIds.map(id => $models.find(model => model.id === id)).filter(Boolean);
if (modelsFromSearch.length > 0) savedSession.activeProjectId = "default";
const max = Math.max(dp.conversations.length, modelsFromSearch.length, searchProviders.length);
for (let i = 0; i < max; i++) {
const conversation = dp.conversations[i] ?? defaultConversation;
dp.conversations[i] = {
...conversation,
model: modelsFromSearch[i] ?? conversation.model,
provider: searchProviders[i] ?? conversation.provider,
};
}
}
set(savedSession);
});
// Override update method to sync with localStorage and URL params
const update: typeof store.update = cb => {
// const prevQuery = window.location.search;
// const query = new URLSearchParams(window.location.search);
// query.delete("modelId");
// query.delete("provider");
store.update($s => {
const s = cb($s);
// Save to localStorage
try {
localStorage.setItem(LOCAL_STORAGE_KEY, JSON.stringify(s));
} catch (e) {
console.error("Failed to save session to localStorage:", e);
}
// Update URL query parameters
// const modelIds = s.conversations.map(c => c.model.id);
// modelIds.forEach(m => query.append("modelId", m));
//
// const providers = s.conversations.map(c => c.provider ?? "hf-inference");
// providers.forEach(p => query.append("provider", p));
// const newQuery = query.toString();
// if (newQuery !== prevQuery.slice(1)) {
// window.parent.postMessage(
// {
// queryString: query.toString(),
// },
// "https://huggingface.co"
// );
// goto(`?${query}`, { replaceState: true });
// }
return s;
});
};
const set: typeof store.set = (...args) => {
update(_ => args[0]);
};
// Add a method to clear localStorage
function clearSavedSession() {
localStorage.removeItem(LOCAL_STORAGE_KEY);
}
/**
* @deprecated - Use `saveProject` instead
*/
function addProject(name: string) {
const { defaultConversation } = getDefaults();
update(s => {
const project: Project = {
name,
id: crypto.randomUUID(),
conversations: [defaultConversation],
};
return { ...s, projects: [...s.projects, project], activeProjectId: project.id };
});
}
/**
* Saves a new project with the data inside the default project
*/
function saveProject(name: string) {
update(s => {
const defaultProject = s.projects.find(p => p.id === "default");
if (!defaultProject) return s;
const project: Project = {
...defaultProject,
name,
id: crypto.randomUUID(),
};
return { ...s, projects: [...s.projects, project], activeProjectId: project.id };
});
}
function deleteProject(id: string) {
// Can't delete default project!
if (id === "default") return;
update(s => {
const projects = s.projects.filter(p => p.id !== id);
if (projects.length === 0) {
const { defaultProject } = getDefaults();
const newSession = { ...s, projects: [defaultProject], activeProjectId: defaultProject.id };
return typia.is<Session>(newSession) ? newSession : s;
}
const currProject = projects.find(p => p.id === s.activeProjectId);
const newSession = { ...s, projects, activeProjectId: currProject?.id ?? projects[0]?.id };
return typia.is<Session>(newSession) ? newSession : s;
});
}
function updateProject(id: string, data: Partial<Project>) {
update(s => {
const projects = s.projects.map(p => (p.id === id ? { ...p, ...data } : p));
const newSession = { ...s, projects };
return typia.is<Session>(newSession) ? newSession : s;
});
}
return { ...store, set, update, clearSavedSession, addProject, deleteProject, saveProject, updateProject };
}
export const session = createSessionStore();
export function getActiveProject(s: Session) {
return s.projects.find(p => p.id === s.activeProjectId) ?? s.projects[0];
}
function createProjectStore() {
const store = writable<Project>(undefined, set => {
return session.subscribe(s => {
set(getActiveProject(s));
});
});
const update: (typeof store)["update"] = cb => {
session.update(s => {
const project = getActiveProject(s);
const newProject = cb(project);
const projects = s.projects.map(p => (p.id === project.id ? newProject : p));
const newSession = { ...s, projects };
return typia.is<Session>(newSession) ? newSession : s;
});
};
const set: typeof store.set = (...args) => {
update(_ => args[0]);
};
return { ...store, update, set };
}
export const project = createProjectStore();
|