Compare commits
11 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
bb17cdd123 | ||
|
|
4cfc9f4aea | ||
|
|
cd50086c1e | ||
|
|
7e5498f779 | ||
|
|
d933236a5d | ||
|
|
0be2d45cd5 | ||
|
|
e24ad26d99 | ||
|
|
052f5299a0 | ||
|
|
8340edbf40 | ||
|
|
7bff84638e | ||
|
|
54660706e3 |
@@ -1,5 +1,5 @@
|
||||
<p align="center">
|
||||
<img alt="demo" src="./demos/demo.png?v=1">
|
||||
<img alt="demo" src="./demos/demo.gif?v=1">
|
||||
</p>
|
||||
|
||||
[English](./README.md) | [中文](./docs/zh/README.md)
|
||||
@@ -26,15 +26,15 @@ Version 2 is a major update that separates the backend functionality as an indep
|
||||
|
||||
If you still wish to use the old version, please visit the [v1 branch](https://github.com/WongSaang/chatgpt-ui/tree/v1).
|
||||
|
||||
Version 2 introduces the following new features:
|
||||
</details>
|
||||
|
||||
## Version 2 introduces the following new features:
|
||||
|
||||
- 😉 Separation of the frontend and backend, with the backend now using the Python-based Django framework.
|
||||
- 😘 User authentication, supporting multiple users.
|
||||
- 😀 Ability to store data in an external database (defaulting to Sqlite).
|
||||
- 😎 Session persistence, allowing the API to answer questions based on your context.
|
||||
|
||||
</details>
|
||||
|
||||
## 🚀 One-click deployment <a name="one-click-depolyment"></a>
|
||||
|
||||
Note: This script has only been tested on Ubuntu Server 22.04 LTS.
|
||||
|
||||
@@ -1,34 +1,36 @@
|
||||
<script setup>
|
||||
import { marked } from "marked"
|
||||
import hljs from "highlight.js"
|
||||
import MarkdownIt from 'markdown-it'
|
||||
import copy from 'copy-to-clipboard'
|
||||
|
||||
// Part of the code comes from this project https://github.com/arronhunt/highlightjs-copy, thanks to the author's contribution
|
||||
|
||||
hljs.addPlugin({
|
||||
'after:highlightElement': ({ el, result, text }) => {
|
||||
let header = el.parentElement.querySelector(".hljs-code-header");
|
||||
if (header) {
|
||||
header.remove();
|
||||
}
|
||||
const md = new MarkdownIt({
|
||||
linkify: true,
|
||||
highlight(code, lang) {
|
||||
const language = hljs.getLanguage(lang) ? lang : 'plaintext'
|
||||
return `<pre class="hljs-code-container my-3"><div class="hljs-code-header d-flex align-center justify-space-between bg-grey-darken-3 pa-1"><span class="pl-2 text-caption">${language}</span><button class="hljs-copy-button" data-copied="false">Copy</button></div><code class="hljs language-${language}">${hljs.highlight(code, { language: language, ignoreIllegals: true }).value}</code></pre>`
|
||||
},
|
||||
})
|
||||
|
||||
header = Object.assign(document.createElement("div"), {
|
||||
className: "hljs-code-header d-flex align-center justify-space-between bg-black pa-1",
|
||||
innerHTML: `<div class="pl-2 text-caption">${result.language}</div>`
|
||||
});
|
||||
const props = defineProps(['content'])
|
||||
|
||||
let copyButton = Object.assign(document.createElement("button"), {
|
||||
innerHTML: "Copy",
|
||||
className: "hljs-copy-button",
|
||||
});
|
||||
copyButton.dataset.copied = false;
|
||||
const contentHtml = ref('')
|
||||
|
||||
header.append(copyButton);
|
||||
//
|
||||
el.parentElement.classList.add("d-flex","flex-column", "my-3");
|
||||
el.parentElement.prepend(header);
|
||||
const contentElm = ref(null)
|
||||
|
||||
watchEffect(() => {
|
||||
contentHtml.value = props.content ? md.render(props.content) : ''
|
||||
})
|
||||
|
||||
const bindCopyCodeToButtons = () => {
|
||||
if (!contentElm.value) {
|
||||
return
|
||||
}
|
||||
contentElm.value.querySelectorAll('.hljs-code-container').forEach((codeContainer) => {
|
||||
const copyButton = codeContainer.querySelector('.hljs-copy-button');
|
||||
const codeBody = codeContainer.querySelector('code');
|
||||
copyButton.onclick = function () {
|
||||
copy(text);
|
||||
copy(codeBody.textContent ?? '');
|
||||
|
||||
copyButton.innerHTML = "Copied!";
|
||||
copyButton.dataset.copied = 'true';
|
||||
@@ -38,34 +40,15 @@ hljs.addPlugin({
|
||||
copyButton.dataset.copied = 'false';
|
||||
}, 2000);
|
||||
};
|
||||
}
|
||||
});
|
||||
|
||||
marked.setOptions({
|
||||
// highlight: function (code, lang) {
|
||||
// const language = hljs.getLanguage(lang) ? lang : 'plaintext'
|
||||
// return hljs.highlight(code, { language }).value
|
||||
// },
|
||||
langPrefix: 'hljs language-', // highlight.js css class prefix
|
||||
})
|
||||
|
||||
const props = defineProps(['content'])
|
||||
|
||||
const contentHtml = ref('')
|
||||
|
||||
const contentElm = ref(null)
|
||||
|
||||
const highlightCode = () => {
|
||||
contentElm.value.querySelectorAll('pre code').forEach((block) => {
|
||||
hljs.highlightElement(block)
|
||||
})
|
||||
}
|
||||
|
||||
watchEffect(() => {
|
||||
contentHtml.value = props.content ? marked(props.content) : ''
|
||||
nextTick(() => {
|
||||
highlightCode()
|
||||
})
|
||||
onMounted(() => {
|
||||
bindCopyCodeToButtons()
|
||||
})
|
||||
|
||||
onUpdated(() => {
|
||||
bindCopyCodeToButtons()
|
||||
})
|
||||
|
||||
</script>
|
||||
@@ -74,7 +57,7 @@ watchEffect(() => {
|
||||
<div
|
||||
ref="contentElm"
|
||||
v-html="contentHtml"
|
||||
class="chat-msg-content text-justify"
|
||||
class="chat-msg-content"
|
||||
></div>
|
||||
</template>
|
||||
|
||||
@@ -82,6 +65,10 @@ watchEffect(() => {
|
||||
.chat-msg-content ol {
|
||||
list-style-position: inside;
|
||||
}
|
||||
.hljs-code-container {
|
||||
border-radius: 3px;
|
||||
overflow: hidden;
|
||||
}
|
||||
.hljs-copy-button{
|
||||
width:2rem;height:2rem;text-indent:-9999px;color:#fff;
|
||||
border-radius:.25rem;border:1px solid #ffffff22;
|
||||
|
||||
@@ -2,13 +2,12 @@
|
||||
<v-textarea
|
||||
v-model="message"
|
||||
:label="$t('writeAMessage')"
|
||||
:placeholder="$t('writeAMessage') + '...'"
|
||||
:placeholder="hint"
|
||||
rows="1"
|
||||
:auto-grow="autoGrow"
|
||||
:disabled="disabled"
|
||||
:loading="loading"
|
||||
:hint="hint"
|
||||
:hide-details="loading"
|
||||
:hide-details="true"
|
||||
append-inner-icon="send"
|
||||
@keyup.enter.exact="enterOnly"
|
||||
@click:appendInner="clickSendBtn"
|
||||
@@ -60,6 +59,9 @@ export default {
|
||||
}
|
||||
this.message = ""
|
||||
},
|
||||
usePrompt(prompt) {
|
||||
this.message = prompt
|
||||
},
|
||||
clickSendBtn () {
|
||||
this.send()
|
||||
},
|
||||
|
||||
224
components/Prompt.vue
Normal file
224
components/Prompt.vue
Normal file
@@ -0,0 +1,224 @@
|
||||
<script setup>
|
||||
const menu = ref(false)
|
||||
const prompts = ref([])
|
||||
const editingPrompt = ref(null)
|
||||
const newPrompt = ref('')
|
||||
const submittingNewPrompt = ref(false)
|
||||
const promptInputErrorMessage = ref('')
|
||||
const loadingPrompts = ref(false)
|
||||
const deletingPromptIndex = ref(null)
|
||||
|
||||
const props = defineProps({
|
||||
usePrompt: {
|
||||
type: Function,
|
||||
required: true
|
||||
}
|
||||
})
|
||||
|
||||
const addPrompt = async () => {
|
||||
if (!newPrompt.value) {
|
||||
promptInputErrorMessage.value = 'Please enter a prompt'
|
||||
return
|
||||
}
|
||||
submittingNewPrompt.value = true
|
||||
const { data, error } = await useAuthFetch('/api/chat/prompts/', {
|
||||
method: 'POST',
|
||||
body: JSON.stringify({
|
||||
prompt: newPrompt.value
|
||||
})
|
||||
})
|
||||
if (!error.value) {
|
||||
prompts.value.push(data.value)
|
||||
newPrompt.value = ''
|
||||
}
|
||||
submittingNewPrompt.value = false
|
||||
}
|
||||
|
||||
const editPrompt = (index) => {
|
||||
editingPrompt.value = Object.assign({}, prompts.value[index])
|
||||
}
|
||||
|
||||
const updatePrompt = async (index) => {
|
||||
editingPrompt.value.updating = true
|
||||
const { data, error } = await useAuthFetch(`/api/chat/prompts/${editingPrompt.value.id}/`, {
|
||||
method: 'PUT',
|
||||
body: JSON.stringify({
|
||||
prompt: editingPrompt.value.prompt
|
||||
})
|
||||
})
|
||||
if (!error.value) {
|
||||
prompts.value[index] = editingPrompt.value
|
||||
}
|
||||
editingPrompt.value.updating = false
|
||||
editingPrompt.value = null
|
||||
}
|
||||
|
||||
const cancelEditPrompt = () => {
|
||||
editingPrompt.value = null
|
||||
}
|
||||
|
||||
const deletePrompt = async (index) => {
|
||||
deletingPromptIndex.value = index
|
||||
const { data, error } = await useAuthFetch(`/api/chat/prompts/${prompts.value[index].id}/`, {
|
||||
method: 'DELETE'
|
||||
})
|
||||
deletingPromptIndex.value = null
|
||||
if (!error.value) {
|
||||
prompts.value.splice(index, 1)
|
||||
}
|
||||
}
|
||||
|
||||
const loadPrompts = async () => {
|
||||
loadingPrompts.value = true
|
||||
const { data, error } = await useAuthFetch('/api/chat/prompts/')
|
||||
if (!error.value) {
|
||||
prompts.value = data.value
|
||||
}
|
||||
loadingPrompts.value = false
|
||||
}
|
||||
|
||||
const selectPrompt = (prompt) => {
|
||||
props.usePrompt(prompt.prompt)
|
||||
menu.value = false
|
||||
}
|
||||
|
||||
onMounted( () => {
|
||||
loadPrompts()
|
||||
})
|
||||
</script>
|
||||
|
||||
<template>
|
||||
<div>
|
||||
<v-menu
|
||||
v-model="menu"
|
||||
:close-on-content-click="false"
|
||||
>
|
||||
<template v-slot:activator="{ props }">
|
||||
<v-btn
|
||||
v-bind="props"
|
||||
icon="speaker_notes"
|
||||
title="Common prompts"
|
||||
class="mr-3"
|
||||
></v-btn>
|
||||
</template>
|
||||
|
||||
<v-container>
|
||||
<v-card
|
||||
min-width="300"
|
||||
max-width="500"
|
||||
>
|
||||
<v-card-title>
|
||||
<span class="headline">Frequently prompts</span>
|
||||
</v-card-title>
|
||||
|
||||
<v-divider></v-divider>
|
||||
|
||||
<v-list>
|
||||
<v-list-item v-show="loadingPrompts">
|
||||
<v-list-item-title class="d-flex justify-center">
|
||||
<v-progress-circular indeterminate></v-progress-circular>
|
||||
</v-list-item-title>
|
||||
</v-list-item>
|
||||
<template
|
||||
v-for="(prompt, idx) in prompts"
|
||||
:key="prompt.id"
|
||||
>
|
||||
<v-list-item
|
||||
active-color="primary"
|
||||
rounded="xl"
|
||||
v-if="editingPrompt && editingPrompt.id === prompt.id"
|
||||
>
|
||||
<v-textarea
|
||||
rows="2"
|
||||
v-model="editingPrompt.prompt"
|
||||
:loading="editingPrompt.updating"
|
||||
variant="underlined"
|
||||
hide-details
|
||||
density="compact"
|
||||
>
|
||||
<template v-slot:append>
|
||||
<div class="d-flex flex-column">
|
||||
<v-btn
|
||||
icon="done"
|
||||
variant="text"
|
||||
:loading="editingPrompt.updating"
|
||||
@click="updatePrompt(idx)"
|
||||
>
|
||||
</v-btn>
|
||||
<v-btn
|
||||
icon="close"
|
||||
variant="text"
|
||||
@click="cancelEditPrompt()"
|
||||
>
|
||||
</v-btn>
|
||||
</div>
|
||||
</template>
|
||||
</v-textarea>
|
||||
</v-list-item>
|
||||
<v-list-item
|
||||
v-if="!editingPrompt || editingPrompt.id !== prompt.id"
|
||||
rounded="xl"
|
||||
active-color="primary"
|
||||
@click="selectPrompt(prompt)"
|
||||
>
|
||||
<v-list-item-title>{{ prompt.prompt }}</v-list-item-title>
|
||||
<template v-slot:append>
|
||||
<v-btn
|
||||
icon="edit"
|
||||
size="small"
|
||||
variant="text"
|
||||
@click="editPrompt(idx)"
|
||||
>
|
||||
</v-btn>
|
||||
<v-btn
|
||||
icon="delete"
|
||||
size="small"
|
||||
variant="text"
|
||||
:loading="deletingPromptIndex === idx"
|
||||
@click="deletePrompt(idx)"
|
||||
>
|
||||
</v-btn>
|
||||
</template>
|
||||
</v-list-item>
|
||||
</template>
|
||||
|
||||
<v-list-item
|
||||
active-color="primary"
|
||||
>
|
||||
<div
|
||||
class="pt-3"
|
||||
>
|
||||
<v-textarea
|
||||
rows="2"
|
||||
v-model="newPrompt"
|
||||
label="Add a new prompt"
|
||||
variant="outlined"
|
||||
density="compact"
|
||||
:error-messages="promptInputErrorMessage"
|
||||
@update:modelValue="promptInputErrorMessage = ''"
|
||||
clearable
|
||||
>
|
||||
</v-textarea>
|
||||
</div>
|
||||
</v-list-item>
|
||||
<v-list-item>
|
||||
<v-btn
|
||||
variant="text"
|
||||
block
|
||||
:loading="submittingNewPrompt"
|
||||
@click="addPrompt()"
|
||||
>
|
||||
<v-icon icon="add"></v-icon>
|
||||
Add prompt
|
||||
</v-btn>
|
||||
</v-list-item>
|
||||
</v-list>
|
||||
</v-card>
|
||||
</v-container>
|
||||
</v-menu>
|
||||
</div>
|
||||
</template>
|
||||
|
||||
<style scoped>
|
||||
|
||||
</style>
|
||||
BIN
demos/demo.gif
Normal file
BIN
demos/demo.gif
Normal file
Binary file not shown.
|
After Width: | Height: | Size: 144 KiB |
BIN
demos/demo.png
BIN
demos/demo.png
Binary file not shown.
|
Before Width: | Height: | Size: 71 KiB After Width: | Height: | Size: 47 KiB |
@@ -1,8 +1,8 @@
|
||||
<p align="center">
|
||||
<img alt="demo" src="./demos/demo.png?v=1">
|
||||
<img alt="demo" src="../../demos/demo.gif?v=1">
|
||||
</p>
|
||||
|
||||
[English](./README.md) | [中文](./docs/zh/README.md)
|
||||
[English](../../README.md) | [中文](./docs/zh/README.md)
|
||||
|
||||
# ChatGPT UI
|
||||
|
||||
|
||||
@@ -18,7 +18,7 @@
|
||||
"copy-to-clipboard": "^3.3.3",
|
||||
"highlight.js": "^11.7.0",
|
||||
"is-mobile": "^3.1.1",
|
||||
"marked": "^4.2.12",
|
||||
"markdown-it": "^13.0.1",
|
||||
"nanoid": "^4.0.1",
|
||||
"vuetify": "^3.0.6"
|
||||
},
|
||||
|
||||
@@ -1,4 +1,6 @@
|
||||
<script setup>
|
||||
import Prompt from "~/components/Prompt.vue";
|
||||
|
||||
definePageMeta({
|
||||
middleware: ["auth"]
|
||||
})
|
||||
@@ -10,6 +12,32 @@ const runtimeConfig = useRuntimeConfig()
|
||||
const currentModel = useCurrentModel()
|
||||
const openaiApiKey = useApiKey()
|
||||
const fetchingResponse = ref(false)
|
||||
const messageQueue = []
|
||||
let isProcessingQueue = false
|
||||
|
||||
const processMessageQueue = () => {
|
||||
if (isProcessingQueue || messageQueue.length === 0) {
|
||||
return
|
||||
}
|
||||
if (!currentConversation.value.messages[currentConversation.value.messages.length - 1].is_bot) {
|
||||
currentConversation.value.messages.push({id: null, is_bot: true, message: ''})
|
||||
}
|
||||
isProcessingQueue = true
|
||||
const nextMessage = messageQueue.shift()
|
||||
currentConversation.value.messages[currentConversation.value.messages.length - 1].message += nextMessage
|
||||
isProcessingQueue = false
|
||||
processMessageQueue()
|
||||
// let wordIndex = 0;
|
||||
// const intervalId = setInterval(() => {
|
||||
// currentConversation.value.messages[currentConversation.value.messages.length - 1].message += nextMessage[wordIndex]
|
||||
// wordIndex++
|
||||
// if (wordIndex === nextMessage.length) {
|
||||
// clearInterval(intervalId)
|
||||
// isProcessingQueue = false
|
||||
// processMessageQueue()
|
||||
// }
|
||||
// }, 50)
|
||||
}
|
||||
|
||||
let ctrl
|
||||
const abortFetch = () => {
|
||||
@@ -69,11 +97,8 @@ const fetchReply = async (message, parentMessageId) => {
|
||||
return;
|
||||
}
|
||||
|
||||
if (currentConversation.value.messages[currentConversation.value.messages.length - 1].is_bot) {
|
||||
currentConversation.value.messages[currentConversation.value.messages.length - 1].message += data.content
|
||||
} else {
|
||||
currentConversation.value.messages.push({id: null, is_bot: true, message: data.content})
|
||||
}
|
||||
messageQueue.push(data.content)
|
||||
processMessageQueue()
|
||||
|
||||
scrollChatWindow()
|
||||
},
|
||||
@@ -120,6 +145,10 @@ const showSnackbar = (text) => {
|
||||
snackbar.value = true
|
||||
}
|
||||
|
||||
const editor = ref(null)
|
||||
const usePrompt = (prompt) => {
|
||||
editor.value.usePrompt(prompt)
|
||||
}
|
||||
|
||||
</script>
|
||||
|
||||
@@ -168,6 +197,7 @@ const showSnackbar = (text) => {
|
||||
<Welcome v-else />
|
||||
<v-footer app class="d-flex flex-column">
|
||||
<div class="px-md-16 w-100 d-flex align-center">
|
||||
<Prompt v-show="!fetchingResponse" :use-prompt="usePrompt" />
|
||||
<v-btn
|
||||
v-show="fetchingResponse"
|
||||
icon="close"
|
||||
@@ -175,7 +205,7 @@ const showSnackbar = (text) => {
|
||||
class="mr-3"
|
||||
@click="stop"
|
||||
></v-btn>
|
||||
<MsgEditor :send-message="send" :disabled="fetchingResponse" :loading="fetchingResponse" />
|
||||
<MsgEditor ref="editor" :send-message="send" :disabled="fetchingResponse" :loading="fetchingResponse" />
|
||||
</div>
|
||||
|
||||
<div class="px-4 py-2 text-disabled text-caption font-weight-light text-center w-100">
|
||||
|
||||
33
yarn.lock
33
yarn.lock
@@ -2005,6 +2005,11 @@ entities@^2.0.0:
|
||||
resolved "https://registry.npmmirror.com/entities/-/entities-2.2.0.tgz#098dc90ebb83d8dffa089d55256b351d34c4da55"
|
||||
integrity sha512-p92if5Nz619I0w+akJrLZH0MX0Pb5DX39XOwQTtXSdQQOaYH03S1uIQp4mhOZtAXrxq4ViO67YTiLBo2638o9A==
|
||||
|
||||
entities@~3.0.1:
|
||||
version "3.0.1"
|
||||
resolved "https://registry.npmmirror.com/entities/-/entities-3.0.1.tgz#2b887ca62585e96db3903482d336c1006c3001d4"
|
||||
integrity sha512-WiyBqoomrwMdFG1e0kqvASYfnlb0lp8M5o5Fw2OFq1hNZxxcNk8Ik0Xm7LxzBhuidnZB/UtBqVCgUz3kBOP51Q==
|
||||
|
||||
errno@^0.1.3:
|
||||
version "0.1.8"
|
||||
resolved "https://registry.npmmirror.com/errno/-/errno-0.1.8.tgz#8bb3e9c7d463be4976ff888f76b4809ebc2e811f"
|
||||
@@ -2801,6 +2806,13 @@ lilconfig@^2.0.3:
|
||||
resolved "https://registry.npmmirror.com/lilconfig/-/lilconfig-2.0.6.tgz#32a384558bd58af3d4c6e077dd1ad1d397bc69d4"
|
||||
integrity sha512-9JROoBW7pobfsx+Sq2JsASvCo6Pfo6WWoUW79HuB1BCoBXD4PLWJPqDF6fNj67pqBYTbAHkE57M1kS/+L1neOg==
|
||||
|
||||
linkify-it@^4.0.1:
|
||||
version "4.0.1"
|
||||
resolved "https://registry.npmmirror.com/linkify-it/-/linkify-it-4.0.1.tgz#01f1d5e508190d06669982ba31a7d9f56a5751ec"
|
||||
integrity sha512-C7bfi1UZmoj8+PQx22XyeXCuBlokoyWQL5pWSP+EI6nzRylyThouddufc2c1NDIcP9k5agmN9fLpA7VNJfIiqw==
|
||||
dependencies:
|
||||
uc.micro "^1.0.1"
|
||||
|
||||
listhen@^1.0.2:
|
||||
version "1.0.2"
|
||||
resolved "https://registry.npmmirror.com/listhen/-/listhen-1.0.2.tgz#3332af0cf77dd914e12d125c70a9c6aed9537033"
|
||||
@@ -2950,6 +2962,17 @@ make-dir@^3.1.0, make-dir@~3.1.0:
|
||||
dependencies:
|
||||
semver "^6.0.0"
|
||||
|
||||
markdown-it@^13.0.1:
|
||||
version "13.0.1"
|
||||
resolved "https://registry.npmmirror.com/markdown-it/-/markdown-it-13.0.1.tgz#c6ecc431cacf1a5da531423fc6a42807814af430"
|
||||
integrity sha512-lTlxriVoy2criHP0JKRhO2VDG9c2ypWCsT237eDiLqi09rmbKoUetyGHq2uOIRoRS//kfoJckS0eUzzkDR+k2Q==
|
||||
dependencies:
|
||||
argparse "^2.0.1"
|
||||
entities "~3.0.1"
|
||||
linkify-it "^4.0.1"
|
||||
mdurl "^1.0.1"
|
||||
uc.micro "^1.0.5"
|
||||
|
||||
marked@^4.2.12:
|
||||
version "4.2.12"
|
||||
resolved "https://registry.npmmirror.com/marked/-/marked-4.2.12.tgz#d69a64e21d71b06250da995dcd065c11083bebb5"
|
||||
@@ -2965,6 +2988,11 @@ mdn-data@2.0.14:
|
||||
resolved "https://registry.npmmirror.com/mdn-data/-/mdn-data-2.0.14.tgz#7113fc4281917d63ce29b43446f701e68c25ba50"
|
||||
integrity sha512-dn6wd0uw5GsdswPFfsgMp5NSB0/aDe6fK94YJV/AJDYXL6HVLWBsxeq7js7Ad+mU2K9LAlwpk6kN2D5mwCPVow==
|
||||
|
||||
mdurl@^1.0.1:
|
||||
version "1.0.1"
|
||||
resolved "https://registry.npmmirror.com/mdurl/-/mdurl-1.0.1.tgz#fe85b2ec75a59037f2adfec100fd6c601761152e"
|
||||
integrity sha512-/sKlQJCBYVY9Ers9hqzKou4H6V5UWc/M59TH2dvkt+84itfnq7uFOMLpOiOS4ujvHP4etln18fmIxA5R5fll0g==
|
||||
|
||||
memory-fs@^0.5.0:
|
||||
version "0.5.0"
|
||||
resolved "https://registry.npmmirror.com/memory-fs/-/memory-fs-0.5.0.tgz#324c01288b88652966d161db77838720845a8e3c"
|
||||
@@ -4266,6 +4294,11 @@ type-fest@^3.0.0:
|
||||
resolved "https://registry.npmmirror.com/type-fest/-/type-fest-3.5.7.tgz#1ee9efc9a172f4002c40b896689928a7bba537f2"
|
||||
integrity sha512-6J4bYzb4sdkcLBty4XW7F18VPI66M4boXNE+CY40532oq2OJe6AVMB5NmjOp6skt/jw5mRjz/hLRpuglz0U+FA==
|
||||
|
||||
uc.micro@^1.0.1, uc.micro@^1.0.5:
|
||||
version "1.0.6"
|
||||
resolved "https://registry.npmmirror.com/uc.micro/-/uc.micro-1.0.6.tgz#9c411a802a409a91fc6cf74081baba34b24499ac"
|
||||
integrity sha512-8Y75pvTYkLJW2hWQHXxoqRgV7qb9B+9vFEtidML+7koHUFapnVJAZ6cKs+Qjz5Aw3aZWHMC6u0wJE3At+nSGwA==
|
||||
|
||||
ufo@^1.0.0, ufo@^1.0.1:
|
||||
version "1.0.1"
|
||||
resolved "https://registry.npmmirror.com/ufo/-/ufo-1.0.1.tgz#64ed43b530706bda2e4892f911f568cf4cf67d29"
|
||||
|
||||
Reference in New Issue
Block a user