feat: users management

Signed-off-by: Ryan Wang <i@ryanc.cc>
pull/588/head
Ryan Wang 2022-06-23 14:28:29 +08:00
parent 62412028b0
commit 4ef9f0cf5d
11 changed files with 242 additions and 103 deletions

View File

@ -27,10 +27,7 @@ const handleRouteToProfile = () => {
<template> <template>
<div class="flex h-full"> <div class="flex h-full">
<aside <aside class="navbar fixed hidden h-full overflow-y-auto md:block">
class="navbar fixed hidden h-full overflow-y-auto md:block"
style="background: #fff"
>
<div class="logo flex justify-center py-5"> <div class="logo flex justify-center py-5">
<img :src="logo" alt="Halo Logo" style="width: 78px" /> <img :src="logo" alt="Halo Logo" style="width: 78px" />
</div> </div>
@ -174,6 +171,7 @@ const handleRouteToProfile = () => {
<style lang="scss"> <style lang="scss">
.navbar { .navbar {
@apply w-64; @apply w-64;
@apply bg-white;
z-index: 999; z-index: 999;
box-shadow: 0 4px 4px #f6c6ce; box-shadow: 0 4px 4px #f6c6ce;
padding-bottom: 70px; padding-bottom: 70px;

View File

@ -3,8 +3,9 @@ import { BasicLayout } from "@/layouts";
import { IconUpload, VButton, VTabbar } from "@halo-dev/components"; import { IconUpload, VButton, VTabbar } from "@halo-dev/components";
import { onMounted, provide, ref } from "vue"; import { onMounted, provide, ref } from "vue";
import { useRoute, useRouter } from "vue-router"; import { useRoute, useRouter } from "vue-router";
import { users } from "@/modules/system/users/users-mock";
import { Starport } from "vue-starport"; import { Starport } from "vue-starport";
import axiosInstance from "@/utils/api-client";
import type { User } from "@/types/extension";
const tabs = [ const tabs = [
{ {
@ -29,15 +30,20 @@ const tabs = [
}, },
]; ];
const user = ref(); const user = ref<User>();
const route = useRoute(); const { params } = useRoute();
if (route.params.username) { const handleFetchUser = async () => {
user.value = users.find((u) => u.username === route.params.username); try {
} else { const { data } = await axiosInstance.get(
user.value = users[0]; `/api/v1alpha1/users/${params.name}`
} );
user.value = data;
} catch (e) {
console.error(e);
}
};
provide("user", user); provide("user", user);
@ -48,6 +54,7 @@ const router = useRouter();
// set default active tab // set default active tab
onMounted(() => { onMounted(() => {
handleFetchUser();
const tab = tabs.find((tab) => tab.routeName === currentRouteName); const tab = tabs.find((tab) => tab.routeName === currentRouteName);
activeTab.value = tab ? tab.id : tabs[0].id; activeTab.value = tab ? tab.id : tabs[0].id;
}); });
@ -62,19 +69,19 @@ const handleTabChange = (id: string) => {
<template> <template>
<BasicLayout> <BasicLayout>
<header class="bg-white"> <header class="bg-white">
<div :class="user.cover" class="h-48 bg-gradient-to-r"></div> <div class="h-48 bg-gradient-to-r from-gray-800 to-red-500"></div>
<div class="px-4 sm:px-6 lg:px-8"> <div class="px-4 sm:px-6 lg:px-8">
<div class="-mt-12 flex items-end space-x-5 sm:-mt-16"> <div class="-mt-12 flex items-end space-x-5 sm:-mt-16">
<div class="flex"> <div class="flex">
<Starport <Starport
:port="`user-profile-${user.name}`"
class="h-24 w-24 sm:h-32 sm:w-32"
:duration="400" :duration="400"
:port="`user-profile-${user.metadata?.name}`"
class="h-24 w-24 sm:h-32 sm:w-32"
> >
<img <img
:src="user.avatar" :src="user.spec?.avatar"
alt="Avatar" alt="Avatar"
class="rounded-full ring-4 ring-white drop-shadow-lg" class="h-full w-full rounded-full ring-4 ring-white drop-shadow-lg"
/> />
</Starport> </Starport>
</div> </div>
@ -83,7 +90,7 @@ const handleTabChange = (id: string) => {
> >
<div class="mt-6 block min-w-0 flex-1"> <div class="mt-6 block min-w-0 flex-1">
<h1 class="truncate text-xl font-bold text-gray-900"> <h1 class="truncate text-xl font-bold text-gray-900">
<span class="mr-1">{{ user.name }}</span> <span class="mr-1">{{ user.spec?.displayName }}</span>
</h1> </h1>
</div> </div>
<div <div

View File

@ -11,7 +11,7 @@ import {
} from "@halo-dev/components"; } from "@halo-dev/components";
import { useRoute } from "vue-router"; import { useRoute } from "vue-router";
import { computed, ref } from "vue"; import { computed, ref } from "vue";
import type { Plugin } from "./types"; import type { Plugin } from "@/types/extension";
import axiosInstance from "@/utils/api-client"; import axiosInstance from "@/utils/api-client";
const pluginActiveId = ref("detail"); const pluginActiveId = ref("detail");
@ -31,7 +31,9 @@ const handleFetchPlugin = async () => {
}; };
const isStarted = computed(() => { const isStarted = computed(() => {
return plugin.value?.status.phase === "STARTED" && plugin.value?.spec.enabled; return (
plugin.value?.status?.phase === "STARTED" && plugin.value?.spec.enabled
);
}); });
const handleChangePluginStatus = async () => { const handleChangePluginStatus = async () => {

View File

@ -14,7 +14,7 @@ import {
} from "@halo-dev/components"; } from "@halo-dev/components";
import { onMounted, ref } from "vue"; import { onMounted, ref } from "vue";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
import type { Plugin } from "./types"; import type { Plugin } from "@/types/extension";
import axiosInstance from "@/utils/api-client"; import axiosInstance from "@/utils/api-client";
const checkAll = ref(false); const checkAll = ref(false);
@ -30,7 +30,7 @@ const handleRouteToDetail = (plugin: Plugin) => {
}; };
function isStarted(plugin: Plugin) { function isStarted(plugin: Plugin) {
return plugin.status.phase === "STARTED" && plugin.spec.enabled; return plugin.status?.phase === "STARTED" && plugin.spec.enabled;
} }
const handleFetchPlugins = async () => { const handleFetchPlugins = async () => {

View File

@ -1,47 +0,0 @@
// eslint-disable-next-line @typescript-eslint/no-empty-interface
export interface PluginDependencies {}
export interface License {
name: string;
url: string;
}
export interface Spec {
displayName: string;
version: string;
author: string;
logo: string;
pluginDependencies: PluginDependencies;
homepage: string;
description: string;
license: License[];
requires: string;
pluginClass: string;
enabled: boolean;
}
export interface Metadata {
name: string;
version: number;
creationTimestamp: Date;
}
export interface Status {
phase: string;
entry?: string;
stylesheet?: string;
}
export interface Plugin {
spec: Spec;
apiVersion: string;
kind: string;
metadata: Metadata;
status: Status;
extensions: Extension[];
}
export interface Extension {
name: string;
fields: string[];
}

View File

@ -1,8 +1,9 @@
<script lang="ts" setup> <script lang="ts" setup>
import { VButton, VInput, VTextarea } from "@halo-dev/components"; import { VButton, VInput, VTextarea } from "@halo-dev/components";
import { inject } from "vue"; import { inject } from "vue";
import type { User } from "@/types/extension";
const user = inject("user"); const user = inject<User>("user");
</script> </script>
<template> <template>
<form class="space-y-8 divide-y divide-gray-200 sm:space-y-5"> <form class="space-y-8 divide-y divide-gray-200 sm:space-y-5">
@ -13,7 +14,7 @@ const user = inject("user");
</label> </label>
<div class="mt-1 sm:col-span-3 sm:mt-0"> <div class="mt-1 sm:col-span-3 sm:mt-0">
<div class="flex max-w-lg shadow-sm"> <div class="flex max-w-lg shadow-sm">
<VInput :modelValue="user.username" /> <VInput :modelValue="user?.metadata?.name" />
</div> </div>
</div> </div>
</div> </div>
@ -23,7 +24,7 @@ const user = inject("user");
</label> </label>
<div class="mt-1 sm:col-span-3 sm:mt-0"> <div class="mt-1 sm:col-span-3 sm:mt-0">
<div class="flex max-w-lg shadow-sm"> <div class="flex max-w-lg shadow-sm">
<VInput :modelValue="user.name" /> <VInput :modelValue="user?.spec?.displayName" />
</div> </div>
</div> </div>
</div> </div>
@ -34,7 +35,7 @@ const user = inject("user");
</label> </label>
<div class="mt-1 sm:col-span-3 sm:mt-0"> <div class="mt-1 sm:col-span-3 sm:mt-0">
<div class="flex max-w-lg shadow-sm"> <div class="flex max-w-lg shadow-sm">
<VInput :modelValue="user.email" /> <VInput :modelValue="user?.spec?.email" />
</div> </div>
</div> </div>
</div> </div>
@ -44,7 +45,7 @@ const user = inject("user");
</label> </label>
<div class="mt-1 sm:col-span-3 sm:mt-0"> <div class="mt-1 sm:col-span-3 sm:mt-0">
<div class="flex max-w-lg shadow-sm"> <div class="flex max-w-lg shadow-sm">
<VTextarea modelValue="Halo" /> <VTextarea :modelValue="user?.spec?.bio" />
</div> </div>
</div> </div>
</div> </div>
@ -52,7 +53,7 @@ const user = inject("user");
<div class="pt-5"> <div class="pt-5">
<div class="flex justify-start"> <div class="flex justify-start">
<VButton type="secondary"> 保存</VButton> <VButton type="secondary">保存</VButton>
</div> </div>
</div> </div>
</form> </form>

View File

@ -2,8 +2,9 @@
import { IconUserSettings, VTag } from "@halo-dev/components"; import { IconUserSettings, VTag } from "@halo-dev/components";
import { inject } from "vue"; import { inject } from "vue";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
import type { User } from "@/types/extension";
const user = inject("user"); const user = inject<User>("user");
const router = useRouter(); const router = useRouter();
</script> </script>
@ -15,7 +16,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">显示名称</dt> <dt class="text-sm font-medium text-gray-900">显示名称</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
{{ user.name }} {{ user?.spec?.displayName }}
</dd> </dd>
</div> </div>
<div <div
@ -23,7 +24,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">用户名</dt> <dt class="text-sm font-medium text-gray-900">用户名</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
{{ user.username }} {{ user?.metadata?.name }}
</dd> </dd>
</div> </div>
<div <div
@ -31,7 +32,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">电子邮箱</dt> <dt class="text-sm font-medium text-gray-900">电子邮箱</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
{{ user.email || "未设置" }} {{ user?.spec?.email || "未设置" }}
</dd> </dd>
</div> </div>
<div <div
@ -43,7 +44,7 @@ const router = useRouter();
<template #leftIcon> <template #leftIcon>
<IconUserSettings /> <IconUserSettings />
</template> </template>
{{ user.role }} {{ user?.metadata?.name }}
</VTag> </VTag>
</dd> </dd>
</div> </div>
@ -52,7 +53,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">描述</dt> <dt class="text-sm font-medium text-gray-900">描述</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
Hello Halo {{ user?.spec?.bio }}
</dd> </dd>
</div> </div>
<div <div
@ -66,7 +67,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">注册时间</dt> <dt class="text-sm font-medium text-gray-900">注册时间</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
2022-01-01 00:00:00 {{ user?.metadata?.creationTimestamp }}
</dd> </dd>
</div> </div>
<div <div
@ -74,7 +75,7 @@ const router = useRouter();
> >
<dt class="text-sm font-medium text-gray-900">最近登录时间</dt> <dt class="text-sm font-medium text-gray-900">最近登录时间</dt>
<dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0"> <dd class="mt-1 text-sm text-gray-900 sm:col-span-3 sm:mt-0">
2022-05-30 12:00:00 {{ user?.metadata?.creationTimestamp }}
</dd> </dd>
</div> </div>
</dl> </dl>

View File

@ -12,18 +12,24 @@ import {
VSpace, VSpace,
VTag, VTag,
} from "@halo-dev/components"; } from "@halo-dev/components";
import { users } from "./users-mock";
import { useRouter } from "vue-router";
import { ref } from "vue"; import { ref } from "vue";
import { Starport } from "vue-starport"; import { Starport } from "vue-starport";
import axiosInstance from "@/utils/api-client";
import type { User } from "@/types/extension";
const checkAll = ref(false); const checkAll = ref(false);
const users = ref<User[]>([]);
const router = useRouter(); const handleFetchUsers = async () => {
try {
const handleRouteToDetail = (username: string) => { const { data } = await axiosInstance.get("/api/v1alpha1/users");
router.push({ name: "UserDetail", params: { username } }); users.value = data;
} catch (e) {
console.error(e);
}
}; };
handleFetchUsers();
</script> </script>
<template> <template>
<VPageHeader title="用户"> <VPageHeader title="用户">
@ -172,7 +178,12 @@ const handleRouteToDetail = (username: string) => {
<li <li
v-for="(user, index) in users" v-for="(user, index) in users"
:key="index" :key="index"
@click="handleRouteToDetail(user.username)" @click="
$router.push({
name: 'UserDetail',
params: { name: user.metadata.name },
})
"
> >
<div <div
:class="{ :class="{
@ -192,15 +203,15 @@ const handleRouteToDetail = (username: string) => {
type="checkbox" type="checkbox"
/> />
</div> </div>
<div v-if="user.avatar" class="mr-4"> <div v-if="user.spec.avatar" class="mr-4">
<Starport <Starport
:duration="400" :duration="400"
:port="`user-profile-${user.name}`" :port="`user-profile-${user.metadata.name}`"
class="h-12 w-12" class="h-12 w-12"
> >
<img <img
:alt="user.name" :alt="user.spec.displayName"
:src="user.avatar" :src="user.spec.avatar"
class="h-full w-full overflow-hidden rounded border bg-white hover:shadow-sm" class="h-full w-full overflow-hidden rounded border bg-white hover:shadow-sm"
/> />
</Starport> </Starport>
@ -208,14 +219,14 @@ const handleRouteToDetail = (username: string) => {
<div class="flex-1"> <div class="flex-1">
<div class="flex flex-row items-center"> <div class="flex flex-row items-center">
<span class="mr-2 truncate text-sm font-medium text-gray-900"> <span class="mr-2 truncate text-sm font-medium text-gray-900">
{{ user.name }} {{ user.spec.displayName }}
</span> </span>
<VTag class="sm:hidden">{{ user.role }}</VTag> <VTag class="sm:hidden">{{ user.metadata.name }}</VTag>
</div> </div>
<div class="mt-1 flex"> <div class="mt-1 flex">
<VSpace align="start" direction="column" spacing="xs"> <VSpace align="start" direction="column" spacing="xs">
<span class="text-xs text-gray-500"> <span class="text-xs text-gray-500">
{{ user.username }} {{ user.metadata.name }}
</span> </span>
</VSpace> </VSpace>
</div> </div>
@ -224,13 +235,13 @@ const handleRouteToDetail = (username: string) => {
<div <div
class="inline-flex flex-col flex-col-reverse items-end gap-4 sm:flex-row sm:items-center sm:gap-6" class="inline-flex flex-col flex-col-reverse items-end gap-4 sm:flex-row sm:items-center sm:gap-6"
> >
<span class="hidden sm:block"> <div class="hidden items-center sm:flex">
<VTag> <VTag>
{{ user.role }} {{ user.metadata.name }}
</VTag> </VTag>
</span> </div>
<time class="text-sm text-gray-500" datetime="2020-01-07"> <time class="text-sm text-gray-500" datetime="2020-01-07">
2020-01-07 {{ user.metadata.creationTimestamp }}
</time> </time>
<span class="cursor-pointer"> <span class="cursor-pointer">
<IconSettings /> <IconSettings />

View File

@ -27,7 +27,7 @@ export default definePlugin({
], ],
}, },
{ {
path: ":username", path: ":name",
component: UserProfileLayout, component: UserProfileLayout,
children: [ children: [
{ {

166
src/types/extension.d.ts vendored Normal file
View File

@ -0,0 +1,166 @@
export interface License {
name?: string;
url?: string;
}
export interface Metadata {
name: string;
labels?: {
[key: string]: string | null;
} | null;
annotations?: {
[key: string]: string | null;
} | null;
version?: number | null;
creationTimestamp?: string | null;
deletionTimestamp?: string | null;
}
export interface Plugin {
spec: PluginSpec;
status?: PluginStatus;
apiVersion: string;
kind: string;
metadata: Metadata;
extensions: Extension[];
}
export interface Extension {
name: string;
fields: string[];
}
export interface PluginSpec {
displayName?: string;
version?: string;
author?: string;
logo?: string;
pluginDependencies?: {
[key: string]: string;
};
homepage?: string;
description?: string;
license?: License[];
requires?: string;
pluginClass?: string;
enabled?: boolean;
}
export interface PluginStatus {
phase?:
| "CREATED"
| "DISABLED"
| "RESOLVED"
| "STARTED"
| "STOPPED"
| "FAILED";
reason?: string;
message?: string;
lastStartTime?: string;
lastTransitionTime?: string;
entry?: string;
stylesheet?: string;
}
export interface PersonalAccessToken {
spec?: PersonalAccessTokenSpec;
apiVersion: string;
kind: string;
metadata: Metadata;
}
export interface PersonalAccessTokenSpec {
userName?: string;
displayName?: string;
revoked?: boolean;
expiresAt?: string;
scopes?: string;
tokenDigest?: string;
}
export interface RoleBinding {
subjects?: Subject[];
roleRef?: RoleRef;
apiVersion: string;
kind: string;
metadata: Metadata;
}
export interface RoleRef {
kind?: string;
name?: string;
apiGroup?: string;
}
export interface Subject {
kind?: string;
name?: string;
apiGroup?: string;
}
export interface PolicyRule {
apiGroups?: string[];
resources?: string[];
resourceNames?: string[];
nonResourceURLs?: string[];
verbs?: string[];
pluginName?: string;
}
export interface Role {
rules?: PolicyRule[];
apiVersion: string;
kind: string;
metadata: Metadata;
}
export interface LoginHistory {
loginAt: string;
sourceIp: string;
userAgent: string;
successful: boolean;
reason?: string;
}
export interface User {
spec: UserSpec;
status?: UserStatus;
apiVersion: string;
kind: string;
metadata: Metadata;
}
export interface UserSpec {
displayName: string;
avatar?: string;
email: string;
phone?: string;
password?: string;
bio?: string;
registeredAt?: string;
twoFactorAuthEnabled?: boolean;
disabled?: boolean;
loginHistoryLimit?: number;
}
export interface UserStatus {
lastLoginAt?: string;
loginHistories?: LoginHistory[];
}
export interface FileReverseProxyProvider {
directory?: string;
filename?: string;
}
export interface ReverseProxy {
rules?: ReverseProxyRule[];
apiVersion: string;
kind: string;
metadata: Metadata;
}
export interface ReverseProxyRule {
path?: string;
file?: FileReverseProxyProvider;
}

View File

@ -1,7 +1,7 @@
import axios from "axios"; import axios from "axios";
const token = const token =
"eyJhbGciOiJSUzUxMiJ9.eyJpc3MiOiJIYWxvIE93bmVyIiwic3ViIjoiYWRtaW4iLCJleHAiOjE2NTU4OTQxNTQsImlhdCI6MTY1NTgwNzc1NCwic2NvcGUiOlsiUk9MRV9zdXBlci1yb2xlIl19.Gnj0rM8DU2bP1KcgKBUVaKf6zs1pDqGxYvii9zxG4lFv4rVZ_uNGXyfhi9V10vRK0GM4v4NEuMtX9-DYnqAV0wR2JcoFevPrJnHHWsvnFrOQm32qeMpew3PsZ5-YAwi9n8Y9GpAcQz_6aWsEuRwm9w5CC3A67CrYPfCK5qwuR5FFLfiMRqPAqNNuZ4r2IfoSZUvXy4HxhUS-01J2BCqP3-hbdN_-tFHCDxtIO637a51EsCmRItY5wSVNmwYPaPOYV7lbHxzBIKXw5RNXg6SrQCSLTVaaJCXsZjwIirk02RQACr6oqTHPbriBVuu-SIgPXS5PJ9i4VaMCn-z8t-oZlQ"; "eyJhbGciOiJSUzUxMiJ9.eyJpc3MiOiJIYWxvIE93bmVyIiwic3ViIjoiYWRtaW4iLCJleHAiOjE2NTYwNDM3NTYsImlhdCI6MTY1NTk1NzM1Niwic2NvcGUiOlsiUk9MRV9zdXBlci1yb2xlIl19.XHmq5q9-HWkWQsPdcuvldeiKOQxbKHEd9qP33ZWaLSFVgj5D-8QvfLjuLreMWUBLvZXvsqBuDHpib70gO6V2c4VtUbnAQnzr8oQx4E5ypMnWH4Gdbs8UlSpMGjTPzSk-QNFKB48nMo8wgTcq2oyhBsMEIArKFm7v2pa5dSX1LbWTRRpNfJpHPVwrAPzaNkOs_qasS8QzSTHU1C3wCf_A4lEILVhbrHq_mv9yeMQZL0enD-gpbGXEQzHE59zwxFC7kfgb_YhzYZfzuXAv2BIKn4TU14W9aW4HySymsqM0ItO5RT3GmJgurbX9USHhIKfGdTFEG1cfgZ0ZJNNOOLEndA";
const axiosInstance = axios.create({ const axiosInstance = axios.create({
headers: { headers: {