450 lines
15 KiB
Vue
450 lines
15 KiB
Vue
<template>
|
||
<div class="p-16-24">
|
||
<h2 class="mb-16">{{ $t('views.applicationOverview.title') }}</h2>
|
||
|
||
<el-scrollbar>
|
||
<div class="main-calc-height">
|
||
<el-card style="--el-card-padding: 24px">
|
||
<h4 class="title-decoration-1 mb-16">
|
||
{{ $t('views.applicationOverview.appInfo.header') }}
|
||
</h4>
|
||
<el-card shadow="never" class="overview-card" v-loading="loading">
|
||
<div class="title flex align-center">
|
||
<div
|
||
class="edit-avatar mr-12"
|
||
@mouseenter="showEditIcon = true"
|
||
@mouseleave="showEditIcon = false"
|
||
>
|
||
<el-avatar
|
||
v-if="isAppIcon(detail?.icon)"
|
||
shape="square"
|
||
:size="32"
|
||
style="background: none"
|
||
>
|
||
<img :src="detail?.icon" alt="" />
|
||
</el-avatar>
|
||
|
||
<LogoIcon v-else height="28px" style="width: 28px; height: 28px; display: block" />
|
||
<el-avatar
|
||
v-if="showEditIcon"
|
||
shape="square"
|
||
class="edit-mask"
|
||
:size="32"
|
||
@click="openEditAvatar"
|
||
>
|
||
<el-icon><EditPen /></el-icon>
|
||
</el-avatar>
|
||
</div>
|
||
|
||
<h4>{{ detail?.name || '-' }}</h4>
|
||
</div>
|
||
|
||
<el-row :gutter="12">
|
||
<el-col :span="12" class="mt-16">
|
||
<div class="flex">
|
||
<el-text type="info">{{
|
||
$t('views.applicationOverview.appInfo.publicAccessLink')
|
||
}}</el-text>
|
||
<el-switch
|
||
v-model="accessToken.is_active"
|
||
class="ml-8"
|
||
size="small"
|
||
inline-prompt
|
||
:active-text="$t('views.applicationOverview.appInfo.openText')"
|
||
:inactive-text="$t('views.applicationOverview.appInfo.closeText')"
|
||
:before-change="() => changeState(accessToken.is_active)"
|
||
v-hasPermission="[RoleConst.WORKSPACE_MANAGE.getWorkspaceRole,PermissionConst.APPLICATION_EDIT.getWorkspacePermission]"
|
||
/>
|
||
</div>
|
||
|
||
<div class="mt-4 mb-16 url-height flex align-center" style="margin-bottom: 37px">
|
||
<span class="vertical-middle lighter break-all ellipsis-1">
|
||
{{ shareUrl }}
|
||
</span>
|
||
<el-tooltip effect="dark" :content="$t('common.copy')" placement="top">
|
||
<el-button type="primary" text @click="copyClick(shareUrl)">
|
||
<AppIcon iconName="app-copy"></AppIcon>
|
||
</el-button>
|
||
</el-tooltip>
|
||
<el-tooltip effect="dark" :content="$t('common.refresh')" placement="top">
|
||
<el-button
|
||
@click="refreshAccessToken"
|
||
type="primary"
|
||
text
|
||
style="margin-left: 1px"
|
||
>
|
||
<el-icon><RefreshRight /></el-icon>
|
||
</el-button>
|
||
</el-tooltip>
|
||
</div>
|
||
<div>
|
||
<el-button
|
||
v-if="accessToken?.is_active"
|
||
:disabled="!accessToken?.is_active"
|
||
tag="a"
|
||
:href="shareUrl"
|
||
target="_blank"
|
||
>
|
||
<AppIcon iconName="app-create-chat" class="mr-4"></AppIcon>
|
||
{{ $t('views.application.operation.toChat') }}
|
||
</el-button>
|
||
<el-button v-else :disabled="!accessToken?.is_active">
|
||
<AppIcon iconName="app-create-chat" class="mr-4"></AppIcon>
|
||
{{ $t('views.application.operation.toChat') }}
|
||
</el-button>
|
||
<el-button :disabled="!accessToken?.is_active" @click="openDialog"
|
||
v-hasPermission="[RoleConst.WORKSPACE_MANAGE.getWorkspaceRole,PermissionConst.APPLICATION_OVERVIEW_EMBEDDED.getWorkspacePermission]"
|
||
>
|
||
<AppIcon iconName="app-export" class="mr-4"></AppIcon>
|
||
{{ $t('views.applicationOverview.appInfo.embedInWebsite') }}
|
||
</el-button>
|
||
<el-button @click="openLimitDialog"
|
||
v-hasPermission="[RoleConst.WORKSPACE_MANAGE.getWorkspaceRole,PermissionConst.APPLICATION_OVERVIEW_ACCESS.getWorkspacePermission]"
|
||
>
|
||
<el-icon class="mr-4"><Lock /></el-icon>
|
||
{{ $t('views.applicationOverview.appInfo.accessControl') }}
|
||
</el-button>
|
||
<el-button @click="openDisplaySettingDialog"
|
||
v-hasPermission="[RoleConst.WORKSPACE_MANAGE.getWorkspaceRole,PermissionConst.APPLICATION_OVERVIEW_DISPLAY.getWorkspacePermission]"
|
||
>
|
||
<el-icon class="mr-4"><Setting /></el-icon>
|
||
{{ $t('views.applicationOverview.appInfo.displaySetting') }}
|
||
</el-button>
|
||
</div>
|
||
</el-col>
|
||
<el-col :span="12" class="mt-16">
|
||
<div class="flex">
|
||
<el-text type="info"
|
||
>{{ $t('views.applicationOverview.appInfo.apiAccessCredentials') }}
|
||
</el-text>
|
||
</div>
|
||
<div class="mt-4 mb-16 url-height">
|
||
<div>
|
||
<el-text>API {{ $t('common.fileUpload.document') }}:</el-text
|
||
><el-button
|
||
type="primary"
|
||
link
|
||
@click="toUrl(apiUrl)"
|
||
class="vertical-middle lighter break-all"
|
||
>
|
||
{{ apiUrl }}
|
||
</el-button>
|
||
</div>
|
||
<div class="flex align-center">
|
||
<span class="flex">
|
||
<el-text style="width: 80px">Base URL:</el-text>
|
||
</span>
|
||
|
||
<span class="vertical-middle lighter break-all ellipsis-1">{{
|
||
baseUrl + id
|
||
}}</span>
|
||
<el-tooltip effect="dark" :content="$t('common.copy')" placement="top">
|
||
<el-button type="primary" text @click="copyClick(baseUrl + id)">
|
||
<AppIcon iconName="app-copy"></AppIcon>
|
||
</el-button>
|
||
</el-tooltip>
|
||
</div>
|
||
</div>
|
||
<div>
|
||
<el-button @click="openAPIKeyDialog"
|
||
v-hasPermission="[RoleConst.WORKSPACE_MANAGE.getWorkspaceRole,PermissionConst.APPLICATION_OVERVIEW_API_KEY.getWorkspacePermission]"
|
||
>
|
||
<el-icon class="mr-4"><Key /></el-icon>
|
||
{{ $t('views.applicationOverview.appInfo.apiKey') }}</el-button
|
||
>
|
||
</div>
|
||
</el-col>
|
||
</el-row>
|
||
</el-card>
|
||
</el-card>
|
||
<el-card style="--el-card-padding: 24px" class="mt-16">
|
||
<h4 class="title-decoration-1 mb-16">
|
||
{{ $t('views.applicationOverview.monitor.monitoringStatistics') }}
|
||
</h4>
|
||
<div class="mb-16">
|
||
<el-select
|
||
v-model="history_day"
|
||
class="mr-12"
|
||
@change="changeDayHandle"
|
||
style="width: 180px"
|
||
>
|
||
<el-option
|
||
v-for="item in dayOptions"
|
||
:key="item.value"
|
||
:label="item.label"
|
||
:value="item.value"
|
||
/>
|
||
</el-select>
|
||
<el-date-picker
|
||
v-if="history_day === 'other'"
|
||
v-model="daterangeValue"
|
||
type="daterange"
|
||
:start-placeholder="$t('views.applicationOverview.monitor.startDatePlaceholder')"
|
||
:end-placeholder="$t('views.applicationOverview.monitor.endDatePlaceholder')"
|
||
format="YYYY-MM-DD"
|
||
value-format="YYYY-MM-DD"
|
||
@change="changeDayRangeHandle"
|
||
/>
|
||
</div>
|
||
<div v-loading="statisticsLoading">
|
||
<StatisticsCharts :data="statisticsData" />
|
||
</div>
|
||
</el-card>
|
||
</div>
|
||
</el-scrollbar>
|
||
|
||
<EmbedDialog
|
||
ref="EmbedDialogRef"
|
||
:data="detail"
|
||
:api-input-params="mapToUrlParams(apiInputParams)"
|
||
/>
|
||
<APIKeyDialog ref="APIKeyDialogRef" />
|
||
<LimitDialog ref="LimitDialogRef" @refresh="refresh" />
|
||
<EditAvatarDialog ref="EditAvatarDialogRef" @refresh="refreshIcon" />
|
||
<XPackDisplaySettingDialog ref="XPackDisplaySettingDialogRef" @refresh="refresh" />
|
||
<!-- v-if="user.isEnterprise()" -->
|
||
<DisplaySettingDialog ref="DisplaySettingDialogRef" @refresh="refresh" />
|
||
</div>
|
||
</template>
|
||
<script setup lang="ts">
|
||
import { ref, computed, onMounted, watch } from 'vue'
|
||
import { useRoute } from 'vue-router'
|
||
import EmbedDialog from './component/EmbedDialog.vue'
|
||
import APIKeyDialog from './component/APIKeyDialog.vue'
|
||
import LimitDialog from './component/LimitDialog.vue'
|
||
import DisplaySettingDialog from './component/DisplaySettingDialog.vue'
|
||
import XPackDisplaySettingDialog from './component/XPackDisplaySettingDialog.vue'
|
||
import EditAvatarDialog from './component/EditAvatarDialog.vue'
|
||
import StatisticsCharts from './component/StatisticsCharts.vue'
|
||
import applicationApi from '@/api/application/application'
|
||
import overviewApi from '@/api/application/application-key'
|
||
import { nowDate, beforeDay } from '@/utils/time'
|
||
import { MsgSuccess, MsgConfirm } from '@/utils/message'
|
||
import { copyClick } from '@/utils/clipboard'
|
||
import { isAppIcon } from '@/utils/common'
|
||
import useStore from '@/stores'
|
||
import { t } from '@/locales'
|
||
import { PermissionConst, RoleConst } from '@/utils/permission/data'
|
||
import { hasPermission } from '@/utils/permission/index'
|
||
const { user, application } = useStore()
|
||
const route = useRoute()
|
||
const {
|
||
params: { id },
|
||
} = route as any
|
||
|
||
const apiUrl = window.location.origin + '/doc/chat/'
|
||
|
||
const baseUrl = window.location.origin + '/api/application/'
|
||
|
||
const DisplaySettingDialogRef = ref()
|
||
const XPackDisplaySettingDialogRef = ref()
|
||
const EditAvatarDialogRef = ref()
|
||
const LimitDialogRef = ref()
|
||
const APIKeyDialogRef = ref()
|
||
const EmbedDialogRef = ref()
|
||
|
||
const accessToken = ref<any>({})
|
||
const detail = ref<any>(null)
|
||
|
||
const loading = ref(false)
|
||
|
||
const urlParams = computed(() =>
|
||
mapToUrlParams(apiInputParams.value) ? '?' + mapToUrlParams(apiInputParams.value) : '',
|
||
)
|
||
const shareUrl = computed(
|
||
() => application.location + accessToken.value.access_token + urlParams.value,
|
||
)
|
||
|
||
const dayOptions = [
|
||
{
|
||
value: 7,
|
||
// @ts-ignore
|
||
label: t('views.applicationOverview.monitor.pastDayOptions.past7Days'),
|
||
},
|
||
{
|
||
value: 30,
|
||
label: t('views.applicationOverview.monitor.pastDayOptions.past30Days'),
|
||
},
|
||
{
|
||
value: 90,
|
||
label: t('views.applicationOverview.monitor.pastDayOptions.past90Days'),
|
||
},
|
||
{
|
||
value: 183,
|
||
label: t('views.applicationOverview.monitor.pastDayOptions.past183Days'),
|
||
},
|
||
{
|
||
value: 'other',
|
||
label: t('views.applicationOverview.monitor.pastDayOptions.other'),
|
||
},
|
||
]
|
||
|
||
const history_day = ref<number | string>(7)
|
||
|
||
// 日期组件时间
|
||
const daterangeValue = ref('')
|
||
|
||
// 提交日期时间
|
||
const daterange = ref({
|
||
start_time: '',
|
||
end_time: '',
|
||
})
|
||
|
||
const statisticsLoading = ref(false)
|
||
const statisticsData = ref([])
|
||
|
||
const showEditIcon = ref(false)
|
||
const apiInputParams = ref([])
|
||
|
||
function toUrl(url: string) {
|
||
window.open(url, '_blank')
|
||
}
|
||
function openDisplaySettingDialog() {
|
||
// if (user.isEnterprise()) {
|
||
XPackDisplaySettingDialogRef.value?.open(accessToken.value, detail.value)
|
||
// } else {
|
||
// DisplaySettingDialogRef.value?.open(accessToken.value, detail.value)
|
||
// }
|
||
}
|
||
function openEditAvatar() {
|
||
EditAvatarDialogRef.value.open(detail.value)
|
||
}
|
||
|
||
function changeDayHandle(val: number | string) {
|
||
if (val !== 'other') {
|
||
daterange.value.start_time = beforeDay(val)
|
||
daterange.value.end_time = nowDate
|
||
getAppStatistics()
|
||
}
|
||
}
|
||
|
||
function changeDayRangeHandle(val: string) {
|
||
daterange.value.start_time = val[0]
|
||
daterange.value.end_time = val[1]
|
||
getAppStatistics()
|
||
}
|
||
|
||
function getAppStatistics() {
|
||
applicationApi.getStatistics(id, daterange.value, statisticsLoading).then((res: any) => {
|
||
statisticsData.value = res.data
|
||
})
|
||
}
|
||
|
||
function refreshAccessToken() {
|
||
MsgConfirm(
|
||
t('views.applicationOverview.appInfo.refreshToken.msgConfirm1'),
|
||
t('views.applicationOverview.appInfo.refreshToken.msgConfirm2'),
|
||
{
|
||
confirmButtonText: t('common.confirm'),
|
||
cancelButtonText: t('common.cancel'),
|
||
},
|
||
)
|
||
.then(() => {
|
||
const obj = {
|
||
access_token_reset: true,
|
||
}
|
||
// @ts-ignore
|
||
const str = t('views.applicationOverview.appInfo.refreshToken.refreshSuccess')
|
||
updateAccessToken(obj, str)
|
||
})
|
||
.catch(() => {})
|
||
}
|
||
function changeState(bool: boolean) {
|
||
const obj = {
|
||
is_active: !bool,
|
||
}
|
||
const str = obj.is_active ? t('common.status.enableSuccess') : t('common.status.disableSuccess')
|
||
updateAccessToken(obj, str)
|
||
.then(() => {
|
||
return true
|
||
})
|
||
.catch(() => {
|
||
return false
|
||
})
|
||
}
|
||
|
||
async function updateAccessToken(obj: any, str: string) {
|
||
applicationApi.putAccessToken(id as string, obj, loading).then((res) => {
|
||
accessToken.value = res?.data
|
||
MsgSuccess(str)
|
||
})
|
||
}
|
||
|
||
function openLimitDialog() {
|
||
LimitDialogRef.value.open(accessToken.value)
|
||
}
|
||
|
||
function openAPIKeyDialog() {
|
||
APIKeyDialogRef.value.open()
|
||
}
|
||
function openDialog() {
|
||
EmbedDialogRef.value.open(accessToken.value?.access_token)
|
||
}
|
||
function getAccessToken() {
|
||
application.asyncGetAccessToken(id, loading).then((res: any) => {
|
||
accessToken.value = res?.data
|
||
})
|
||
}
|
||
|
||
function getDetail() {
|
||
application.asyncGetApplicationDetail(id, loading).then((res: any) => {
|
||
detail.value = res.data
|
||
detail.value.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,
|
||
}
|
||
})
|
||
: []
|
||
})
|
||
})
|
||
}
|
||
|
||
function refresh() {
|
||
getAccessToken()
|
||
}
|
||
|
||
function refreshIcon() {
|
||
getDetail()
|
||
}
|
||
|
||
function mapToUrlParams(map: any[]) {
|
||
const params = new URLSearchParams()
|
||
|
||
map.forEach((item: any) => {
|
||
params.append(encodeURIComponent(item.name), encodeURIComponent(item.value))
|
||
})
|
||
|
||
return params.toString() // 返回 URL 查询字符串
|
||
}
|
||
|
||
onMounted(() => {
|
||
getDetail()
|
||
getAccessToken()
|
||
changeDayHandle(history_day.value)
|
||
})
|
||
</script>
|
||
<style lang="scss" scoped>
|
||
.overview-card {
|
||
position: relative;
|
||
.active-button {
|
||
position: absolute;
|
||
right: 16px;
|
||
top: 21px;
|
||
}
|
||
}
|
||
</style>
|