add website config.

This commit is contained in:
2023-04-03 13:31:00 +08:00
parent 70a4b67b58
commit b6ae429052
6 changed files with 286 additions and 5 deletions

View File

@@ -4,8 +4,10 @@ import {
BlankContest,
BlankProblem,
Contest,
Server,
TestcaseUploadedReturns,
User,
WebsiteConfig,
} from "~/utils/types"
export function getBaseInfo() {
@@ -149,3 +151,27 @@ export function addProblemForContest(
display_id: displayID,
})
}
export function getWebsite() {
return http.get<WebsiteConfig>("admin/website")
}
export function editWebsite(data: WebsiteConfig) {
return http.post("admin/website", data)
}
export function listInvalidTestcases() {
return http.get("admin/prune_test_case")
}
export function pruneInvalidTestcases(id?: string) {
return http.delete("admin/prune_test_case", { params: { id } })
}
export function getJudgeServer() {
return http.get<{ token: string; servers: Server[] }>("admin/judge_server")
}
export function deleteJudgeServer(hostname: string) {
return http.delete("admin/judge_server", { params: { hostname } })
}

View File

@@ -1,7 +1,6 @@
<script setup lang="ts">
import { formatISO } from "date-fns"
import TextEditor from "~/shared/TextEditor.vue"
import { CONTEST_TYPE } from "~/utils/constants"
import { BlankContest } from "~/utils/types"
import { createContest, editContest, getContest } from "../api"

View File

@@ -118,7 +118,11 @@ watch(query, listProblems, { deep: true })
<n-button v-if="isContestProblemList" @click="createContestProblem">
新建比赛题目
</n-button>
<n-button v-if="isContestProblemList" @click="selectProblems">
<n-button
v-if="isContestProblemList"
type="primary"
@click="selectProblems"
>
从题库中选择
</n-button>
<n-input v-model:value="query.keyword" placeholder="输入标题关键字" />

View File

@@ -1,7 +1,233 @@
<script setup lang="ts"></script>
<script setup lang="ts">
import { DataTableColumn, NButton, NTag } from "naive-ui"
import {
editWebsite,
getJudgeServer,
deleteJudgeServer,
getWebsite,
listInvalidTestcases,
pruneInvalidTestcases,
} from "../api"
import { Server } from "~/utils/types"
import { parseTime } from "~/utils/functions"
const message = useMessage()
const columns: DataTableColumn<any>[] = [
{ title: "上次修改时间", key: "create_time" },
{ title: "ID", key: "id" },
{
title: "选项",
key: "delete",
render: (row) =>
h(
NButton,
{ size: "small", onClick: () => deleteTestcase(row.id) },
() => "删除"
),
},
]
const statusMap: {
[key in "normal" | "abnormal"]: { color: "primary" | "error"; label: string }
} = {
normal: { color: "primary", label: "正常" },
abnormal: { color: "error", label: "异常" },
}
const serverColumns: DataTableColumn<Server>[] = [
{
title: "状态",
key: "status",
render: (row) =>
h(
NTag,
{ type: statusMap[row.status].color, size: "small" },
() => statusMap[row.status].label
),
},
{
title: "选项",
key: "options",
render: (row) =>
h(
NButton,
{
type: "primary",
size: "small",
disabled: row.status === "normal",
onClick: () => delJudgeServer(row.hostname),
},
() => "删除"
),
},
{ title: "主机", key: "hostname", width: 130 },
{
title: "内存占用",
key: "memory_usage",
render: (row) => row.memory_usage + "%",
width: 80,
},
{ title: "IP", key: "ip", width: 100 },
{ title: "判题机版本", key: "judger_version" },
{ title: "服务器 URL", key: "service_url" },
{
title: "上一次心跳",
key: "last_heartbeat",
render: (row) => parseTime(row.last_heartbeat, "YYYY-MM-DD hh:mm:ss"),
width: 120,
},
{
title: "创建时间",
key: "create_time",
render: (row) => parseTime(row.create_time, "YYYY-MM-DD hh:mm:ss"),
width: 120,
},
]
const testcases = ref([])
const token = ref("")
const servers = ref<Server[]>([])
const websiteConfig = reactive({
website_base_url: "https://oj.hyyz.izhai.net",
website_name: "徐越的判题狗",
website_name_shortcut: "徐越的判题狗",
website_footer: "所有权归属于徐越,感谢青岛大学开源 OJ 系统,感谢开源社区",
allow_register: true,
submission_list_show_all: true,
})
async function getWebsiteConfig() {
const res = await getWebsite()
websiteConfig.website_base_url = res.data.website_base_url
websiteConfig.website_name = res.data.website_name
websiteConfig.website_name_shortcut = res.data.website_name_shortcut
websiteConfig.website_footer = res.data.website_footer
websiteConfig.allow_register = res.data.allow_register
websiteConfig.submission_list_show_all = res.data.submission_list_show_all
}
async function saveWebsiteConfig() {
await editWebsite(websiteConfig)
message.success("网站配置保存成功")
}
async function deleteTestcase(id?: string) {
await pruneInvalidTestcases(id)
message.success("删除成功")
}
async function getTestcases() {
const res = await listInvalidTestcases()
testcases.value = res.data
}
async function getJudgeServerData() {
const res = await getJudgeServer()
token.value = res.data.token
servers.value = res.data.servers
}
async function delJudgeServer(hostname: string) {
await deleteJudgeServer(hostname)
message.success("删除成功")
}
onMounted(() => {
getWebsiteConfig()
getTestcases()
getJudgeServerData()
})
</script>
<template>
<div>conf</div>
<n-card class="box">
<template #header>
<n-space align="center">
网站设置
<n-button type="primary" size="small" @click="saveWebsiteConfig">
保存
</n-button>
</n-space>
</template>
<n-form inline label-placement="left">
<n-form-item label="网站 URL">
<n-input class="url" v-model:value="websiteConfig.website_base_url" />
</n-form-item>
<n-form-item label="网站名">
<n-input v-model:value="websiteConfig.website_name" />
</n-form-item>
<n-form-item label="网站简称">
<n-input v-model:value="websiteConfig.website_name_shortcut" />
</n-form-item>
</n-form>
<n-form label-placement="left">
<n-form-item label="页脚">
<n-input v-model:value="websiteConfig.website_footer" />
</n-form-item>
</n-form>
<n-space align="center">
<n-space align="center">
<span>是否允许注册</span>
<n-switch v-model:value="websiteConfig.allow_register" />
</n-space>
<n-space align="center">
<span>显示全部题目的提交</span>
<n-switch v-model:value="websiteConfig.submission_list_show_all" />
</n-space>
</n-space>
</n-card>
<n-card class="box" title="判题服务器">
<div class="serverToken">
接口凭证 <n-tag size="small">{{ token }}</n-tag>
</div>
<n-data-table
:single-line="false"
striped
size="small"
:columns="serverColumns"
:data="servers"
/>
</n-card>
<n-card class="box">
<template #header>
<n-space align="center">
无效的测试用例
<n-button
v-if="testcases.length"
size="small"
type="primary"
@click="() => deleteTestcase()"
>
全部删除
</n-button>
</n-space>
</template>
<n-data-table
striped
size="small"
class="table"
:columns="columns"
:data="testcases"
/>
</n-card>
</template>
<style scoped></style>
<style scoped>
.url {
width: 200px;
}
.box {
margin-bottom: 16px;
}
.table {
width: 40%;
}
.serverToken {
margin-bottom: 16px;
}
</style>

View File

@@ -37,6 +37,7 @@ export function getTagColor(
}[tag]
}
// 2023-04-03T02:43:28.673156Z
export function parseTime(utc: Date | string, format = "YYYY年M月D日") {
const time = useDateFormat(utc, format, { locales: "zh-CN" })
return time.value

View File

@@ -312,3 +312,28 @@ export interface ContestRank {
submission_info: { [key: string]: SubmissionInfo }
contest: number
}
export interface WebsiteConfig {
website_base_url: string
website_name: string
website_name_shortcut: string
website_footer: string
allow_register: boolean
submission_list_show_all: boolean
}
export interface Server {
id: number
status: "abnormal" | "normal"
hostname: string
ip: string
judger_version: string
cpu_core: number
memory_usage: number
cpu_usage: number
last_heartbeat: Date
create_time: Date
task_number: number
service_url: string
is_disabled: boolean
}