1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060 |
- import { WEBUI_API_BASE_URL, WEBUI_BASE_URL } from '$lib/constants';
- export const getModels = async (token: string = '') => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/models`, {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- let models = res?.data ?? [];
- models = models
- .filter((models) => models)
- // Sort the models
- .sort((a, b) => {
- // Check if models have position property
- const aHasPosition = a.info?.meta?.position !== undefined;
- const bHasPosition = b.info?.meta?.position !== undefined;
- // If both a and b have the position property
- if (aHasPosition && bHasPosition) {
- return a.info.meta.position - b.info.meta.position;
- }
- // If only a has the position property, it should come first
- if (aHasPosition) return -1;
- // If only b has the position property, it should come first
- if (bHasPosition) return 1;
- // Compare case-insensitively by name for models without position property
- const lowerA = a.name.toLowerCase();
- const lowerB = b.name.toLowerCase();
- if (lowerA < lowerB) return -1;
- if (lowerA > lowerB) return 1;
- // If same case-insensitively, sort by original strings,
- // lowercase will come before uppercase due to ASCII values
- if (a.name < b.name) return -1;
- if (a.name > b.name) return 1;
- return 0; // They are equal
- });
- console.log(models);
- return models;
- };
- type ChatCompletedForm = {
- model: string;
- messages: string[];
- chat_id: string;
- session_id: string;
- };
- export const chatCompleted = async (token: string, body: ChatCompletedForm) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/chat/completed`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify(body)
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- type ChatActionForm = {
- model: string;
- messages: string[];
- chat_id: string;
- };
- export const chatAction = async (token: string, action_id: string, body: ChatActionForm) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/chat/actions/${action_id}`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify(body)
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getTaskConfig = async (token: string = '') => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/config`, {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const updateTaskConfig = async (token: string, config: object) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/config/update`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify(config)
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const generateTitle = async (
- token: string = '',
- model: string,
- prompt: string,
- chat_id?: string
- ) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/title/completions`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- model: model,
- prompt: prompt,
- ...(chat_id && { chat_id: chat_id })
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res?.choices[0]?.message?.content.replace(/["']/g, '') ?? 'New Chat';
- };
- export const generateTags = async (
- token: string = '',
- model: string,
- messages: string,
- chat_id?: string
- ) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/tags/completions`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- model: model,
- messages: messages,
- ...(chat_id && { chat_id: chat_id })
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- try {
- // Step 1: Safely extract the response string
- const response = res?.choices[0]?.message?.content ?? '';
- // Step 2: Attempt to fix common JSON format issues like single quotes
- const sanitizedResponse = response.replace(/['‘’`]/g, '"'); // Convert single quotes to double quotes for valid JSON
- // Step 3: Find the relevant JSON block within the response
- const jsonStartIndex = sanitizedResponse.indexOf('{');
- const jsonEndIndex = sanitizedResponse.lastIndexOf('}');
- // Step 4: Check if we found a valid JSON block (with both `{` and `}`)
- if (jsonStartIndex !== -1 && jsonEndIndex !== -1) {
- const jsonResponse = sanitizedResponse.substring(jsonStartIndex, jsonEndIndex + 1);
- // Step 5: Parse the JSON block
- const parsed = JSON.parse(jsonResponse);
- // Step 6: If there's a "tags" key, return the tags array; otherwise, return an empty array
- if (parsed && parsed.tags) {
- return Array.isArray(parsed.tags) ? parsed.tags : [];
- } else {
- return [];
- }
- }
- // If no valid JSON block found, return an empty array
- return [];
- } catch (e) {
- // Catch and safely return empty array on any parsing errors
- console.error('Failed to parse response: ', e);
- return [];
- }
- };
- export const generateEmoji = async (
- token: string = '',
- model: string,
- prompt: string,
- chat_id?: string
- ) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/emoji/completions`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- model: model,
- prompt: prompt,
- ...(chat_id && { chat_id: chat_id })
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- const response = res?.choices[0]?.message?.content.replace(/["']/g, '') ?? null;
- if (response) {
- if (/\p{Extended_Pictographic}/u.test(response)) {
- return response.match(/\p{Extended_Pictographic}/gu)[0];
- }
- }
- return null;
- };
- export const generateSearchQuery = async (
- token: string = '',
- model: string,
- messages: object[],
- prompt: string
- ) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/query/completions`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- model: model,
- messages: messages,
- prompt: prompt
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res?.choices[0]?.message?.content.replace(/["']/g, '') ?? prompt;
- };
- export const generateMoACompletion = async (
- token: string = '',
- model: string,
- prompt: string,
- responses: string[]
- ) => {
- const controller = new AbortController();
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/task/moa/completions`, {
- signal: controller.signal,
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- model: model,
- prompt: prompt,
- responses: responses,
- stream: true
- })
- }).catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return [res, controller];
- };
- export const getPipelinesList = async (token: string = '') => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/pipelines/list`, {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- let pipelines = res?.data ?? [];
- return pipelines;
- };
- export const uploadPipeline = async (token: string, file: File, urlIdx: string) => {
- let error = null;
- // Create a new FormData object to handle the file upload
- const formData = new FormData();
- formData.append('file', file);
- formData.append('urlIdx', urlIdx);
- const res = await fetch(`${WEBUI_BASE_URL}/api/pipelines/upload`, {
- method: 'POST',
- headers: {
- ...(token && { authorization: `Bearer ${token}` })
- // 'Content-Type': 'multipart/form-data' is not needed as Fetch API will set it automatically
- },
- body: formData
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const downloadPipeline = async (token: string, url: string, urlIdx: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/pipelines/add`, {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify({
- url: url,
- urlIdx: urlIdx
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const deletePipeline = async (token: string, id: string, urlIdx: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/pipelines/delete`, {
- method: 'DELETE',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify({
- id: id,
- urlIdx: urlIdx
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getPipelines = async (token: string, urlIdx?: string) => {
- let error = null;
- const searchParams = new URLSearchParams();
- if (urlIdx !== undefined) {
- searchParams.append('urlIdx', urlIdx);
- }
- const res = await fetch(`${WEBUI_BASE_URL}/api/pipelines?${searchParams.toString()}`, {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- let pipelines = res?.data ?? [];
- return pipelines;
- };
- export const getPipelineValves = async (token: string, pipeline_id: string, urlIdx: string) => {
- let error = null;
- const searchParams = new URLSearchParams();
- if (urlIdx !== undefined) {
- searchParams.append('urlIdx', urlIdx);
- }
- const res = await fetch(
- `${WEBUI_BASE_URL}/api/pipelines/${pipeline_id}/valves?${searchParams.toString()}`,
- {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- }
- )
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getPipelineValvesSpec = async (token: string, pipeline_id: string, urlIdx: string) => {
- let error = null;
- const searchParams = new URLSearchParams();
- if (urlIdx !== undefined) {
- searchParams.append('urlIdx', urlIdx);
- }
- const res = await fetch(
- `${WEBUI_BASE_URL}/api/pipelines/${pipeline_id}/valves/spec?${searchParams.toString()}`,
- {
- method: 'GET',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- }
- }
- )
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const updatePipelineValves = async (
- token: string = '',
- pipeline_id: string,
- valves: object,
- urlIdx: string
- ) => {
- let error = null;
- const searchParams = new URLSearchParams();
- if (urlIdx !== undefined) {
- searchParams.append('urlIdx', urlIdx);
- }
- const res = await fetch(
- `${WEBUI_BASE_URL}/api/pipelines/${pipeline_id}/valves/update?${searchParams.toString()}`,
- {
- method: 'POST',
- headers: {
- Accept: 'application/json',
- 'Content-Type': 'application/json',
- ...(token && { authorization: `Bearer ${token}` })
- },
- body: JSON.stringify(valves)
- }
- )
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- if ('detail' in err) {
- error = err.detail;
- } else {
- error = err;
- }
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getBackendConfig = async () => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/config`, {
- method: 'GET',
- credentials: 'include',
- headers: {
- 'Content-Type': 'application/json'
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getChangelog = async () => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/changelog`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json'
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getVersionUpdates = async () => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/version/updates`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json'
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getModelFilterConfig = async (token: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/config/model/filter`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const updateModelFilterConfig = async (
- token: string,
- enabled: boolean,
- models: string[]
- ) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/config/model/filter`, {
- method: 'POST',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- enabled: enabled,
- models: models
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getWebhookUrl = async (token: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/webhook`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res.url;
- };
- export const updateWebhookUrl = async (token: string, url: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/webhook`, {
- method: 'POST',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- url: url
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res.url;
- };
- export const getCommunitySharingEnabledStatus = async (token: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/community_sharing`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const toggleCommunitySharingEnabledStatus = async (token: string) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/community_sharing/toggle`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err.detail;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
- export const getModelConfig = async (token: string): Promise<GlobalModelConfig> => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/config/models`, {
- method: 'GET',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- }
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res.models;
- };
- export interface ModelConfig {
- id: string;
- name: string;
- meta: ModelMeta;
- base_model_id?: string;
- params: ModelParams;
- }
- export interface ModelMeta {
- description?: string;
- capabilities?: object;
- profile_image_url?: string;
- }
- export interface ModelParams {}
- export type GlobalModelConfig = ModelConfig[];
- export const updateModelConfig = async (token: string, config: GlobalModelConfig) => {
- let error = null;
- const res = await fetch(`${WEBUI_BASE_URL}/api/config/models`, {
- method: 'POST',
- headers: {
- 'Content-Type': 'application/json',
- Authorization: `Bearer ${token}`
- },
- body: JSON.stringify({
- models: config
- })
- })
- .then(async (res) => {
- if (!res.ok) throw await res.json();
- return res.json();
- })
- .catch((err) => {
- console.log(err);
- error = err;
- return null;
- });
- if (error) {
- throw error;
- }
- return res;
- };
|