feat: application

This commit is contained in:
wangdan-fit2cloud 2025-06-10 15:03:47 +08:00
parent e6e3acdfea
commit 649e2ce1f6
9 changed files with 324 additions and 116 deletions

View File

@ -18,6 +18,9 @@
</slot> </slot>
<slot name="subTitle"> </slot> <slot name="subTitle"> </slot>
</div> </div>
<div class="status-tag">
<slot name="tag"> </slot>
</div>
</div> </div>
</slot> </slot>
</div> </div>
@ -109,5 +112,10 @@ function subHoveredEnter() {
width: 100%; width: 100%;
box-sizing: border-box; box-sizing: border-box;
} }
.status-tag {
position: absolute;
right: 16px;
top: 15px;
}
} }
</style> </style>

View File

@ -14,7 +14,10 @@ export default {
searchBar: { searchBar: {
placeholder: 'Search by name', placeholder: 'Search by name',
}, },
status: {
published: 'Published',
unpublished: 'Unpublished',
},
setting: { setting: {
demo: 'Demo', demo: 'Demo',
}, },

View File

@ -13,6 +13,10 @@ export default {
searchBar: { searchBar: {
placeholder: '按名称搜索', placeholder: '按名称搜索',
}, },
status: {
published: '已发布',
unpublished: '未发布',
},
setting: { setting: {
demo: '演示', demo: '演示',
}, },

View File

@ -12,6 +12,10 @@ export default {
copy: '副本', copy: '副本',
searchBar: { searchBar: {
placeholder: '按名稱搜尋', placeholder: '按名稱搜尋',
},
status: {
published: '已发布',
unpublished: '未发布',
}, },
setting: { setting: {
demo: '示範', demo: '示範',

View File

@ -9,6 +9,7 @@ import usePromptStore from './modules/prompt'
import useProblemStore from './modules/problem' import useProblemStore from './modules/problem'
import useParagraphStore from './modules/paragraph' import useParagraphStore from './modules/paragraph'
import useDocumentStore from './modules/document' import useDocumentStore from './modules/document'
import useApplicationStore from './modules/application'
const useStore = () => ({ const useStore = () => ({
common: useCommonStore(), common: useCommonStore(),
@ -22,6 +23,7 @@ const useStore = () => ({
problem: useProblemStore(), problem: useProblemStore(),
paragraph: useParagraphStore(), paragraph: useParagraphStore(),
document: useDocumentStore(), document: useDocumentStore(),
application: useApplicationStore(),
}) })
export default useStore export default useStore

View File

@ -0,0 +1,139 @@
import { defineStore } from 'pinia'
import applicationApi from '@/api/application/application'
import applicationXpackApi from '@/api/application/application-xpack'
import { type Ref } from 'vue'
import { getBrowserLang } from '@/locales/index'
import useUserStore from './user'
const useApplicationStore = defineStore('application', {
state: () => ({
location: `${window.location.origin}/ui/chat/`,
}),
actions: {
async asyncGetAllApplication() {
return new Promise((resolve, reject) => {
applicationApi
.getAllAppilcation()
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
})
},
async asyncGetApplicationDetail(id: string, loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
applicationApi
.getApplicationDetail(id, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
})
},
async asyncGetApplicationDataset(id: string, loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
applicationApi
.getApplicationDataset(id, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
})
},
async asyncGetAccessToken(id: string, loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
const user = useUserStore()
if (user.isEnterprise()) {
applicationXpackApi
.getAccessToken(id, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
} else {
applicationApi
.getAccessToken(id, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
}
})
},
async asyncGetAppProfile(loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
applicationApi
.getAppProfile(loading)
.then((res) => {
sessionStorage.setItem('language', res.data?.language || getBrowserLang())
resolve(res)
})
.catch((error) => {
reject(error)
})
})
},
async asyncAppAuthentication(
token: string,
loading?: Ref<boolean>,
authentication_value?: any,
) {
return new Promise((resolve, reject) => {
applicationApi
.postAppAuthentication(token, loading, authentication_value)
.then((res) => {
localStorage.setItem(`${token}-accessToken`, res.data)
sessionStorage.setItem(`${token}-accessToken`, res.data)
resolve(res)
})
.catch((error) => {
reject(error)
})
})
},
async refreshAccessToken(token: string) {
this.asyncAppAuthentication(token)
},
// 修改应用
async asyncPutApplication(id: string, data: any, loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
applicationApi
.putApplication(id, data, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
})
},
async validatePassword(id: string, password: string, loading?: Ref<boolean>) {
return new Promise((resolve, reject) => {
applicationApi
.validatePassword(id, password, loading)
.then((data) => {
resolve(data)
})
.catch((error) => {
reject(error)
})
})
},
},
})
export default useApplicationStore

View File

@ -61,9 +61,8 @@
</el-avatar> </el-avatar>
<div class="pre-wrap ml-8"> <div class="pre-wrap ml-8">
<div class="lighter">{{ $t('views.application.simple') }}</div> <div class="lighter">{{ $t('views.application.simple') }}</div>
<el-text type="info" size="small">{{ <el-text type="info" size="small"
$t('views.application.simplePlaceholder') >{{ $t('views.application.simplePlaceholder') }}
}}
</el-text> </el-text>
</div> </div>
</div> </div>
@ -79,9 +78,8 @@
</el-avatar> </el-avatar>
<div class="pre-wrap ml-8"> <div class="pre-wrap ml-8">
<div class="lighter">{{ $t('views.application.workflow') }}</div> <div class="lighter">{{ $t('views.application.workflow') }}</div>
<el-text type="info" size="small">{{ <el-text type="info" size="small"
$t('views.application.workflowPlaceholder') >{{ $t('views.application.workflowPlaceholder') }}
}}
</el-text> </el-text>
</div> </div>
</div> </div>
@ -93,8 +91,16 @@
</template> </template>
<div> <div>
<el-row v-if="applicationList.length > 0" :gutter="15"> <el-row v-if="applicationList.length > 0" :gutter="15">
<!-- <template v-for="(item, index) in datasetFolderList" :key="index"> <template v-for="(item, index) in applicationList" :key="index">
<el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="6" class="mb-16"> <el-col
v-if="item.resource_type === 'folder'"
:xs="24"
:sm="12"
:md="12"
:lg="8"
:xl="6"
class="mb-16"
>
<CardBox <CardBox
:title="item.name" :title="item.name"
:description="item.desc || $t('common.noData')" :description="item.desc || $t('common.noData')"
@ -112,9 +118,7 @@
</template> </template>
</CardBox> </CardBox>
</el-col> </el-col>
</template> --> <el-col v-else :xs="24" :sm="12" :md="12" :lg="8" :xl="6" class="mb-16">
<template v-for="(item, index) in applicationList" :key="index">
<el-col :xs="24" :sm="12" :md="12" :lg="8" :xl="6" class="mb-16">
<CardBox <CardBox
:title="item.name" :title="item.name"
:description="item.desc" :description="item.desc"
@ -131,58 +135,31 @@
</auto-tooltip> </auto-tooltip>
</el-text> </el-text>
</template> </template>
<div class="status-tag"> <template #tag>
<el-tag type="warning" v-if="isWorkFlow(item.type)" style="height: 22px"> <el-tag type="warning" v-if="isWorkFlow(item.type)" style="height: 22px">
{{ $t('views.application.workflow') }} {{ $t('views.application.workflow') }}
</el-tag> </el-tag>
<el-tag class="blue-tag" v-else style="height: 22px"> <el-tag class="blue-tag" v-else style="height: 22px">
{{ $t('views.application.simple') }} {{ $t('views.application.simple') }}
</el-tag> </el-tag>
</div> </template>
<template #footer> <template #footer>
<div class="footer-content"> <div v-if="item.is_publish" class="flex align-center">
<el-tooltip <el-icon class="color-success mr-8" style="font-size: 16px">
effect="dark" <SuccessFilled />
:content="$t('views.application.setting.demo')" </el-icon>
placement="top" <span class="color-secondary">
> {{ $t('views.application.status.published') }}
<el-button text @click.stop @click="getAccessToken(item.id)"> </span>
<AppIcon iconName="app-view"></AppIcon>
</el-button>
</el-tooltip>
<el-divider direction="vertical" /> <el-divider direction="vertical" />
<el-tooltip effect="dark" :content="$t('common.setting')" placement="top"> <el-icon class="mr-8"><Clock /></el-icon>
<el-button text @click.stop="settingApplication(item)"> <span class="color-secondary">{{ dateFormat(item.update_time) }}</span>
<AppIcon iconName="Setting"></AppIcon> </div>
</el-button> <div v-else class="flex align-center">
</el-tooltip> <AppIcon iconName="app-disabled" class="color-secondary mr-8"></AppIcon>
<el-divider direction="vertical"/> <span class="color-secondary">
<span @click.stop> {{ $t('views.application.status.unpublished') }}
<el-dropdown trigger="click">
<el-button text @click.stop>
<el-icon><MoreFilled/></el-icon>
</el-button>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item
v-if="is_show_copy_button(item)"
@click="copyApplication(item)"
>
<AppIcon iconName="app-copy"></AppIcon>
{{ $t('common.copy') }}
</el-dropdown-item>
<el-dropdown-item @click.stop="exportApplication(item)">
<AppIcon iconName="app-export"></AppIcon>
{{ $t('common.export') }}
</el-dropdown-item>
<el-dropdown-item icon="Delete" @click.stop="deleteApplication(item)">{{
$t('common.delete')
}}</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</span> </span>
</div> </div>
</template> </template>
@ -196,41 +173,17 @@
</el-button> </el-button>
<template #dropdown> <template #dropdown>
<el-dropdown-menu> <el-dropdown-menu>
<el-dropdown-item <!-- <el-dropdown-item
icon="Refresh" icon="Refresh"
@click.stop="syncDataset(item)" @click.stop="syncDataset(item)"
v-if="item.type === 1" v-if="item.type === 1"
>{{ $t('views.knowledge.setting.sync') }} >{{ $t('views.knowledge.setting.sync') }}
</el-dropdown-item </el-dropdown-item>
>
<el-dropdown-item @click.stop="reEmbeddingDataset(item)"> <el-dropdown-item @click.stop="reEmbeddingDataset(item)">
<AppIcon iconName="app-vectorization"></AppIcon> <AppIcon iconName="app-vectorization"></AppIcon>
{{ $t('views.knowledge.setting.vectorization') }} {{ $t('views.knowledge.setting.vectorization') }}
</el-dropdown-item> </el-dropdown-item>
<!-- -->
<el-dropdown-item
icon="Connection"
@click.stop="openGenerateDialog(item)"
>{{ $t('views.document.generateQuestion.title') }}</el-dropdown-item
>
<el-dropdown-item
icon="Setting"
@click.stop="router.push({ path: `/knowledge/${item.id}/setting` })"
>
{{ $t('common.setting') }}</el-dropdown-item
>
<el-dropdown-item @click.stop="export_dataset(item)">
<AppIcon iconName="app-export"></AppIcon
>{{ $t('views.document.setting.export') }} Excel</el-dropdown-item
>
<el-dropdown-item @click.stop="export_zip_dataset(item)">
<AppIcon iconName="app-export"></AppIcon
>{{ $t('views.document.setting.export') }} ZIP</el-dropdown-item
>
<el-dropdown-item icon="Delete" @click.stop="deleteDataset(item)">{{
$t('common.delete')
}}</el-dropdown-item> -->
</el-dropdown-menu> </el-dropdown-menu>
</template> </template>
</el-dropdown> </el-dropdown>
@ -244,23 +197,24 @@
</div> </div>
</ContentContainer> </ContentContainer>
<CreateApplicationDialog ref="CreateApplicationDialogRef" /> <CreateApplicationDialog ref="CreateApplicationDialogRef" />
<CopyApplicationDialog ref="CopyApplicationDialogRef" />
</LayoutContainer> </LayoutContainer>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { onMounted, ref, reactive, computed } from 'vue' import { onMounted, ref, reactive, computed } from 'vue'
import CreateApplicationDialog from '@/views/application/component/CreateApplicationDialog.vue' import CreateApplicationDialog from '@/views/application/component/CreateApplicationDialog.vue'
import GenerateRelatedDialog from '@/components/generate-related-dialog/index.vue' import CopyApplicationDialog from '@/views/application/component/CopyApplicationDialog.vue'
import ApplicaitonApi from '@/api/application/application' import ApplicaitonApi from '@/api/application/application'
import {MsgSuccess, MsgConfirm} from '@/utils/message' import { MsgSuccess, MsgConfirm, MsgError } from '@/utils/message'
import useStore from '@/stores' import useStore from '@/stores'
import {numberFormat} from '@/utils/common'
import { t } from '@/locales' import { t } from '@/locales'
import { useRouter } from 'vue-router' import { useRouter } from 'vue-router'
import { isWorkFlow } from '@/utils/application' import { isWorkFlow } from '@/utils/application'
import { dateFormat } from '@/utils/time'
const router = useRouter() const router = useRouter()
const {folder} = useStore() const { folder, application, user } = useStore()
const loading = ref(false) const loading = ref(false)
@ -284,7 +238,7 @@ const paginationConfig = reactive({
const folderList = ref<any[]>([]) const folderList = ref<any[]>([])
const applicationList = ref<any[]>([]) const applicationList = ref<any[]>([])
const currentFolder = ref<any>({}) const currentFolder = ref<any>({})
const CopyApplicationDialogRef = ref()
const CreateApplicationDialogRef = ref() const CreateApplicationDialogRef = ref()
function openCreateDialog(type?: string) { function openCreateDialog(type?: string) {
@ -334,6 +288,100 @@ function folderClickHandel(row: any) {
getList() getList()
} }
function getAccessToken(id: string) {
applicationList.value
.filter((app) => app.id === id)[0]
?.work_flow?.nodes?.filter((v: any) => v.id === 'base-node')
.map((v: any) => {
apiInputParams.value = v.properties.api_input_field_list
? v.properties.api_input_field_list.map((v: any) => {
return {
name: v.variable,
value: v.default_value,
}
})
: v.properties.input_field_list
? v.properties.input_field_list
.filter((v: any) => v.assignment_method === 'api_input')
.map((v: any) => {
return {
name: v.variable,
value: v.default_value,
}
})
: []
})
const apiParams = mapToUrlParams(apiInputParams.value)
? '?' + mapToUrlParams(apiInputParams.value)
: ''
application.asyncGetAccessToken(id, loading).then((res: any) => {
window.open(application.location + res?.data?.access_token + apiParams)
})
}
const apiInputParams = ref([])
function copyApplication(row: any) {
application.asyncGetApplicationDetail(row.id, loading).then((res: any) => {
if (res?.data) {
CopyApplicationDialogRef.value.open({ ...res.data, model_id: res.data.model })
}
})
}
const is_show_copy_button = (row: any) => {
return user.userInfo ? user.userInfo.id == row.user_id : false
}
function settingApplication(row: any) {
if (isWorkFlow(row.type)) {
router.push({ path: `/application/${row.id}/workflow` })
} else {
router.push({ path: `/application/${row.id}/${row.type}/setting` })
}
}
function mapToUrlParams(map: any[]) {
const params = new URLSearchParams()
map.forEach((item: any) => {
params.append(encodeURIComponent(item.name), encodeURIComponent(item.value))
})
return params.toString() // URL
}
function deleteApplication(row: any) {
MsgConfirm(
// @ts-ignore
`${t('views.application.delete.confirmTitle')}${row.name} ?`,
t('views.application.delete.confirmMessage'),
{
confirmButtonText: t('common.confirm'),
cancelButtonText: t('common.cancel'),
confirmButtonClass: 'danger',
},
)
.then(() => {
ApplicaitonApi.delApplication(row.id, loading).then(() => {
const index = applicationList.value.findIndex((v) => v.id === row.id)
applicationList.value.splice(index, 1)
MsgSuccess(t('common.deleteSuccess'))
})
})
.catch(() => {})
}
const exportApplication = (application: any) => {
ApplicaitonApi.exportApplication(application.id, application.name, loading).catch((e) => {
if (e.response.status !== 403) {
e.response.data.text().then((res: string) => {
MsgError(`${t('views.application.tip.ExportError')}:${JSON.parse(res).message}`)
})
}
})
}
onMounted(() => { onMounted(() => {
getFolder() getFolder()
}) })

View File

@ -35,15 +35,15 @@
@click="openBatchEditDocument" @click="openBatchEditDocument"
:disabled="multipleSelection.length === 0" :disabled="multipleSelection.length === 0"
> >
{{ $t('common.setting') }}</el-dropdown-item {{ $t('common.setting') }}
> </el-dropdown-item>
<el-dropdown-item <el-dropdown-item
divided divided
@click="syncMulDocument" @click="syncMulDocument"
:disabled="multipleSelection.length === 0" :disabled="multipleSelection.length === 0"
v-if="datasetDetail.type === 1" v-if="datasetDetail.type === 1"
>{{ $t('views.document.syncDocument') }}</el-dropdown-item >{{ $t('views.document.syncDocument') }}
> </el-dropdown-item>
<el-dropdown-item <el-dropdown-item
divided divided
v-if="datasetDetail.type === 2" v-if="datasetDetail.type === 2"
@ -54,22 +54,22 @@
query: { id: id, folder_token: datasetDetail.meta.folder_token }, query: { id: id, folder_token: datasetDetail.meta.folder_token },
}) })
" "
>{{ $t('views.document.importDocument') }}</el-dropdown-item >{{ $t('views.document.importDocument') }}
> </el-dropdown-item>
<el-dropdown-item <el-dropdown-item
divided divided
@click="syncLarkMulDocument" @click="syncLarkMulDocument"
:disabled="multipleSelection.length === 0" :disabled="multipleSelection.length === 0"
v-if="datasetDetail.type === 2" v-if="datasetDetail.type === 2"
>{{ $t('views.document.syncDocument') }}</el-dropdown-item >{{ $t('views.document.syncDocument') }}
> </el-dropdown-item>
<el-dropdown-item <el-dropdown-item
divided divided
@click="deleteMulDocument" @click="deleteMulDocument"
:disabled="multipleSelection.length === 0" :disabled="multipleSelection.length === 0"
>{{ $t('common.delete') }}</el-dropdown-item >{{ $t('common.delete') }}
> </el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</template> </template>
</el-dropdown> </el-dropdown>
@ -249,9 +249,9 @@
</template> </template>
<template #default="{ row }"> <template #default="{ row }">
<div v-if="row.is_active" class="flex align-center"> <div v-if="row.is_active" class="flex align-center">
<el-icon class="color-success mr-8" style="font-size: 16px" <el-icon class="color-success mr-8" style="font-size: 16px">
><SuccessFilled <SuccessFilled />
/></el-icon> </el-icon>
<span class="color-secondary"> <span class="color-secondary">
{{ $t('common.status.enabled') }} {{ $t('common.status.enabled') }}
</span> </span>

View File

@ -106,7 +106,7 @@ function submit() {
} else if (radioType.value === 'custom' && iconFile.value) { } else if (radioType.value === 'custom' && iconFile.value) {
const fd = new FormData() const fd = new FormData()
fd.append('file', iconFile.value.raw) fd.append('file', iconFile.value.raw)
toolApi.putToolIcon(detail.value.id, fd, loading).then((res: any) => { ToolApi.putToolIcon(detail.value.id, fd, loading).then((res: any) => {
emit('refresh', res.data) emit('refresh', res.data)
dialogVisible.value = false dialogVisible.value = false
}) })