forked from AkkomaGang/admin-fe
Merge branch 'feature/update-login-process' into 'master'
Update login process Closes #11 See merge request pleroma/admin-fe!9
This commit is contained in:
commit
fea370c2a8
25 changed files with 469 additions and 288 deletions
|
@ -1,5 +1,4 @@
|
||||||
module.exports = {
|
module.exports = {
|
||||||
NODE_ENV: '"development"',
|
NODE_ENV: '"development"',
|
||||||
ENV_CONFIG: '"dev"',
|
ENV_CONFIG: '"dev"'
|
||||||
BASE_API: '"http://localhost:4000"'
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -92,6 +92,7 @@
|
||||||
"eslint-loader": "2.0.0",
|
"eslint-loader": "2.0.0",
|
||||||
"eslint-plugin-vue": "4.7.1",
|
"eslint-plugin-vue": "4.7.1",
|
||||||
"file-loader": "1.1.11",
|
"file-loader": "1.1.11",
|
||||||
|
"flush-promises": "^1.0.2",
|
||||||
"friendly-errors-webpack-plugin": "1.7.0",
|
"friendly-errors-webpack-plugin": "1.7.0",
|
||||||
"hash-sum": "1.0.2",
|
"hash-sum": "1.0.2",
|
||||||
"html-webpack-plugin": "^3.2.0",
|
"html-webpack-plugin": "^3.2.0",
|
||||||
|
|
59
src/api/__mocks__/login.js
Normal file
59
src/api/__mocks__/login.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
const users = [
|
||||||
|
{ username: 'bob', password: '123456' }
|
||||||
|
]
|
||||||
|
|
||||||
|
export async function loginByUsername(username, password) {
|
||||||
|
const user = users.find(user => user.username === username)
|
||||||
|
const verifyPassword = user.password === password
|
||||||
|
const data = {
|
||||||
|
'token_type': 'Bearer',
|
||||||
|
'scope': 'read write follow',
|
||||||
|
'refresh_token': 'foo123',
|
||||||
|
'me': 'bob',
|
||||||
|
'expires_in': 600,
|
||||||
|
'access_token': 'bar123'
|
||||||
|
}
|
||||||
|
|
||||||
|
return verifyPassword
|
||||||
|
? Promise.resolve({ data })
|
||||||
|
: Promise.reject({ message: 'Invalid credentials' })
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getUserInfo(token, authHost) {
|
||||||
|
const userInfo = {
|
||||||
|
'name_html': 'bob',
|
||||||
|
'background_image': null,
|
||||||
|
'friends_count': 0,
|
||||||
|
'description_html': '',
|
||||||
|
'followers_count': 0,
|
||||||
|
'locked': false,
|
||||||
|
'follows_you': true,
|
||||||
|
'statusnet_blocking': false,
|
||||||
|
'statusnet_profile_url': '',
|
||||||
|
'following': true,
|
||||||
|
'id': '10',
|
||||||
|
'is_local': true,
|
||||||
|
'profile_image_url': '',
|
||||||
|
'role': 'admin',
|
||||||
|
'profile_image_url_profile_size': '',
|
||||||
|
'rights': { 'admin': true, 'delete_others_notice': true },
|
||||||
|
'token': 'foo123456',
|
||||||
|
'no_rich_text': false,
|
||||||
|
'statuses_count': 0,
|
||||||
|
'cover_photo': '',
|
||||||
|
'hide_follows': false,
|
||||||
|
'pleroma': { 'confirmation_pending': false, 'deactivated': false, 'tags': ['force_nsfw'] },
|
||||||
|
'profile_image_url_original': '',
|
||||||
|
'created_at': 'Fri Mar 01 15:15:19 +0000 2019',
|
||||||
|
'fields': [],
|
||||||
|
'name': 'bob',
|
||||||
|
'description': '',
|
||||||
|
'favourites_count': 0,
|
||||||
|
'default_scope': 'public',
|
||||||
|
'profile_image_url_https': '',
|
||||||
|
'hide_followers': false,
|
||||||
|
'show_role': true,
|
||||||
|
'screen_name': 'bob' }
|
||||||
|
|
||||||
|
return Promise.resolve({ data: userInfo })
|
||||||
|
}
|
|
@ -4,7 +4,7 @@ const users = [
|
||||||
{ deactivated: true, id: 'abc', nickname: 'john', local: true, roles: { admin: false, moderator: false }, tags: ['strip_media'] }
|
{ deactivated: true, id: 'abc', nickname: 'john', local: true, roles: { admin: false, moderator: false }, tags: ['strip_media'] }
|
||||||
]
|
]
|
||||||
|
|
||||||
export async function fetchUsers(showLocalUsersOnly, token, page = 1) {
|
export async function fetchUsers(showLocalUsersOnly, authHost, token, page = 1) {
|
||||||
const filteredUsers = showLocalUsersOnly ? users.filter(user => user.local) : users
|
const filteredUsers = showLocalUsersOnly ? users.filter(user => user.local) : users
|
||||||
return Promise.resolve({ data: {
|
return Promise.resolve({ data: {
|
||||||
users: filteredUsers,
|
users: filteredUsers,
|
||||||
|
@ -13,12 +13,12 @@ export async function fetchUsers(showLocalUsersOnly, token, page = 1) {
|
||||||
}})
|
}})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function toggleUserActivation(nickname, token) {
|
export async function toggleUserActivation(nickname, authHost, token) {
|
||||||
const response = users.find(user => user.nickname === nickname)
|
const response = users.find(user => user.nickname === nickname)
|
||||||
return Promise.resolve({ data: { ...response, deactivated: !response.deactivated }})
|
return Promise.resolve({ data: { ...response, deactivated: !response.deactivated }})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function searchUsers(query, showLocalUsersOnly, token, page = 1) {
|
export async function searchUsers(query, showLocalUsersOnly, authHost, token, page = 1) {
|
||||||
const filteredUsers = showLocalUsersOnly ? users.filter(user => user.local) : users
|
const filteredUsers = showLocalUsersOnly ? users.filter(user => user.local) : users
|
||||||
const response = filteredUsers.filter(user => user.nickname === query)
|
const response = filteredUsers.filter(user => user.nickname === query)
|
||||||
return Promise.resolve({ data: {
|
return Promise.resolve({ data: {
|
||||||
|
@ -28,28 +28,28 @@ export async function searchUsers(query, showLocalUsersOnly, token, page = 1) {
|
||||||
}})
|
}})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function addRight(nickname, right, token) {
|
export async function addRight(nickname, right, authHost, token) {
|
||||||
return Promise.resolve({ data:
|
return Promise.resolve({ data:
|
||||||
{ [`is_${right}`]: true }
|
{ [`is_${right}`]: true }
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteRight(nickname, right, token) {
|
export async function deleteRight(nickname, right, authHost, token) {
|
||||||
return Promise.resolve({ data:
|
return Promise.resolve({ data:
|
||||||
{ [`is_${right}`]: false }
|
{ [`is_${right}`]: false }
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteUser(nickname, token) {
|
export async function deleteUser(nickname, authHost, token) {
|
||||||
return Promise.resolve({ data:
|
return Promise.resolve({ data:
|
||||||
nickname
|
nickname
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function tagUser(nickname, tag, token) {
|
export async function tagUser(nickname, tag, authHost, token) {
|
||||||
return Promise.resolve()
|
return Promise.resolve()
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function untagUser(nickname, tag, token) {
|
export async function untagUser(nickname, tag, authHost, token) {
|
||||||
return Promise.resolve()
|
return Promise.resolve()
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,7 +1,9 @@
|
||||||
import request from '@/utils/request'
|
import request from '@/utils/request'
|
||||||
|
import { baseName } from './utils'
|
||||||
|
|
||||||
export async function loginByUsername(username, password) {
|
export async function loginByUsername(username, password, authHost) {
|
||||||
const appsRequest = await request({
|
const appsRequest = await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: '/api/v1/apps',
|
url: '/api/v1/apps',
|
||||||
method: 'post',
|
method: 'post',
|
||||||
data: {
|
data: {
|
||||||
|
@ -14,6 +16,7 @@ export async function loginByUsername(username, password) {
|
||||||
const app = appsRequest.data
|
const app = appsRequest.data
|
||||||
|
|
||||||
return request({
|
return request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: '/oauth/token',
|
url: '/oauth/token',
|
||||||
method: 'post',
|
method: 'post',
|
||||||
data: {
|
data: {
|
||||||
|
@ -26,8 +29,9 @@ export async function loginByUsername(username, password) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export function getUserInfo(token) {
|
export function getUserInfo(token, authHost) {
|
||||||
return request({
|
return request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: '/api/account/verify_credentials',
|
url: '/api/account/verify_credentials',
|
||||||
method: 'post',
|
method: 'post',
|
||||||
headers: token ? { 'Authorization': `Bearer ${token}` } : {}
|
headers: token ? { 'Authorization': `Bearer ${token}` } : {}
|
||||||
|
|
|
@ -1,56 +1,64 @@
|
||||||
import request from '@/utils/request'
|
import request from '@/utils/request'
|
||||||
import { getToken } from '@/utils/auth'
|
import { getToken } from '@/utils/auth'
|
||||||
|
import { baseName } from './utils'
|
||||||
|
|
||||||
export async function fetchUsers(showLocalUsersOnly, token, page = 1) {
|
export async function fetchUsers(showLocalUsersOnly, authHost, token, page = 1) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/users?page=${page}&local_only=${showLocalUsersOnly}`,
|
url: `/api/pleroma/admin/users?page=${page}&local_only=${showLocalUsersOnly}`,
|
||||||
method: 'get',
|
method: 'get',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function toggleUserActivation(nickname, token) {
|
export async function toggleUserActivation(nickname, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/users/${nickname}/toggle_activation`,
|
url: `/api/pleroma/admin/users/${nickname}/toggle_activation`,
|
||||||
method: 'patch',
|
method: 'patch',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function searchUsers(query, showLocalUsersOnly, token, page = 1) {
|
export async function searchUsers(query, showLocalUsersOnly, authHost, token, page = 1) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/users?query=${query}&page=${page}&local_only=${showLocalUsersOnly}`,
|
url: `/api/pleroma/admin/users?query=${query}&page=${page}&local_only=${showLocalUsersOnly}`,
|
||||||
method: 'get',
|
method: 'get',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function addRight(nickname, right, token) {
|
export async function addRight(nickname, right, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/permission_group/${nickname}/${right}`,
|
url: `/api/pleroma/admin/permission_group/${nickname}/${right}`,
|
||||||
method: 'post',
|
method: 'post',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteRight(nickname, right, token) {
|
export async function deleteRight(nickname, right, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/permission_group/${nickname}/${right}`,
|
url: `/api/pleroma/admin/permission_group/${nickname}/${right}`,
|
||||||
method: 'delete',
|
method: 'delete',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function deleteUser(nickname, token) {
|
export async function deleteUser(nickname, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: `/api/pleroma/admin/user.json?nickname=${nickname}`,
|
url: `/api/pleroma/admin/user.json?nickname=${nickname}`,
|
||||||
method: 'delete',
|
method: 'delete',
|
||||||
headers: authHeaders(token)
|
headers: authHeaders(token)
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function tagUser(nickname, tag, token) {
|
export async function tagUser(nickname, tag, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: '/api/pleroma/admin/users/tag',
|
url: '/api/pleroma/admin/users/tag',
|
||||||
method: 'put',
|
method: 'put',
|
||||||
headers: authHeaders(token),
|
headers: authHeaders(token),
|
||||||
|
@ -58,8 +66,9 @@ export async function tagUser(nickname, tag, token) {
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
export async function untagUser(nickname, tag, token) {
|
export async function untagUser(nickname, tag, authHost, token) {
|
||||||
return await request({
|
return await request({
|
||||||
|
baseURL: baseName(authHost),
|
||||||
url: '/api/pleroma/admin/users/tag',
|
url: '/api/pleroma/admin/users/tag',
|
||||||
method: 'delete',
|
method: 'delete',
|
||||||
headers: authHeaders(token),
|
headers: authHeaders(token),
|
||||||
|
|
5
src/api/utils.js
Normal file
5
src/api/utils.js
Normal file
|
@ -0,0 +1,5 @@
|
||||||
|
const isLocalhost = (instanceName) =>
|
||||||
|
instanceName.startsWith('localhost:') || instanceName.startsWith('127.0.0.1:')
|
||||||
|
|
||||||
|
export const baseName = (instanceName) =>
|
||||||
|
isLocalhost(instanceName) ? `http://${instanceName}` : `https://${instanceName}`
|
|
@ -76,8 +76,9 @@ export default {
|
||||||
login: {
|
login: {
|
||||||
title: 'Login Form',
|
title: 'Login Form',
|
||||||
logIn: 'Log in',
|
logIn: 'Log in',
|
||||||
username: 'Username',
|
username: 'Username@Host',
|
||||||
password: 'Password',
|
password: 'Password',
|
||||||
|
errorMessage: 'Username must contain username and host, e.g. john@pleroma.social',
|
||||||
any: 'any',
|
any: 'any',
|
||||||
thirdparty: 'Or connect with',
|
thirdparty: 'Or connect with',
|
||||||
thirdpartyTips: 'Can not be simulated on local, so please combine you own business simulation! ! !'
|
thirdpartyTips: 'Can not be simulated on local, so please combine you own business simulation! ! !'
|
||||||
|
|
|
@ -16,7 +16,7 @@ function hasPermission(roles, permissionRoles) {
|
||||||
|
|
||||||
const whiteList = ['/login', '/auth-redirect']// no redirect whitelist
|
const whiteList = ['/login', '/auth-redirect']// no redirect whitelist
|
||||||
|
|
||||||
router.beforeEach((to, from, next) => {
|
export const beforeEachRoute = (to, from, next) => {
|
||||||
NProgress.start() // start progress bar
|
NProgress.start() // start progress bar
|
||||||
if (getToken()) { // determine if there has token
|
if (getToken()) { // determine if there has token
|
||||||
/* has token*/
|
/* has token*/
|
||||||
|
@ -24,12 +24,12 @@ router.beforeEach((to, from, next) => {
|
||||||
next({ path: '/' })
|
next({ path: '/' })
|
||||||
NProgress.done() // if current page is dashboard will not trigger afterEach hook, so manually handle it
|
NProgress.done() // if current page is dashboard will not trigger afterEach hook, so manually handle it
|
||||||
} else {
|
} else {
|
||||||
if (store.getters.roles.length === 0) { // 判断当前用户是否已拉取完user_info信息
|
if (store.getters.roles.length === 0) {
|
||||||
store.dispatch('GetUserInfo').then(res => { // 拉取user_info
|
store.dispatch('GetUserInfo').then(res => {
|
||||||
const roles = res.data.rights.admin ? ['admin'] : []
|
const roles = res.data.rights.admin ? ['admin'] : []
|
||||||
store.dispatch('GenerateRoutes', { roles }).then(() => { // 根据roles权限生成可访问的路由表
|
store.dispatch('GenerateRoutes', { roles }).then(() => {
|
||||||
router.addRoutes(store.getters.addRouters) // 动态添加可访问路由表
|
router.addRoutes(store.getters.addRouters)
|
||||||
next({ ...to, replace: true }) // hack方法 确保addRoutes已完成 ,set the replace: true so the navigation will not leave a history record
|
next({ ...to, replace: true })
|
||||||
})
|
})
|
||||||
}).catch((err) => {
|
}).catch((err) => {
|
||||||
store.dispatch('FedLogOut').then(() => {
|
store.dispatch('FedLogOut').then(() => {
|
||||||
|
@ -38,25 +38,24 @@ router.beforeEach((to, from, next) => {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
} else {
|
} else {
|
||||||
// 没有动态改变权限的需求可直接next() 删除下方权限判断 ↓
|
|
||||||
if (hasPermission(store.getters.roles, to.meta.roles)) {
|
if (hasPermission(store.getters.roles, to.meta.roles)) {
|
||||||
next()
|
next()
|
||||||
} else {
|
} else {
|
||||||
next({ path: '/401', replace: true, query: { noGoBack: true }})
|
next({ path: '/401', replace: true, query: { noGoBack: true }})
|
||||||
}
|
}
|
||||||
// 可删 ↑
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
/* has no token*/
|
/* has no token*/
|
||||||
if (whiteList.indexOf(to.path) !== -1) { // 在免登录白名单,直接进入
|
if (whiteList.indexOf(to.path) !== -1) {
|
||||||
next()
|
next()
|
||||||
} else {
|
} else {
|
||||||
next(`/login?redirect=${to.path}`) // 否则全部重定向到登录页
|
next(`/login?redirect=${to.path}`)
|
||||||
NProgress.done() // if current page is login will not trigger afterEach hook, so manually handle it
|
NProgress.done() // if current page is login will not trigger afterEach hook, so manually handle it
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
})
|
}
|
||||||
|
router.beforeEach(beforeEachRoute)
|
||||||
|
|
||||||
router.afterEach(() => {
|
router.afterEach(() => {
|
||||||
NProgress.done() // finish progress bar
|
NProgress.done() // finish progress bar
|
||||||
|
|
|
@ -15,6 +15,7 @@ const getters = {
|
||||||
permission_routers: state => state.permission.routers,
|
permission_routers: state => state.permission.routers,
|
||||||
addRouters: state => state.permission.addRouters,
|
addRouters: state => state.permission.addRouters,
|
||||||
errorLogs: state => state.errorLog.logs,
|
errorLogs: state => state.errorLog.logs,
|
||||||
users: state => state.users.fetchedUsers
|
users: state => state.users.fetchedUsers,
|
||||||
|
authHost: state => state.user.authHost
|
||||||
}
|
}
|
||||||
export default getters
|
export default getters
|
||||||
|
|
|
@ -1,5 +1,5 @@
|
||||||
import { loginByUsername, getUserInfo } from '@/api/login'
|
import { loginByUsername, getUserInfo } from '@/api/login'
|
||||||
import { getToken, setToken, removeToken } from '@/utils/auth'
|
import { getToken, setToken, removeToken, getAuthHost, setAuthHost, removeAuthHost } from '@/utils/auth'
|
||||||
|
|
||||||
const user = {
|
const user = {
|
||||||
state: {
|
state: {
|
||||||
|
@ -8,6 +8,7 @@ const user = {
|
||||||
status: '',
|
status: '',
|
||||||
code: '',
|
code: '',
|
||||||
token: getToken(),
|
token: getToken(),
|
||||||
|
authHost: getAuthHost(),
|
||||||
name: '',
|
name: '',
|
||||||
avatar: '',
|
avatar: '',
|
||||||
introduction: '',
|
introduction: '',
|
||||||
|
@ -44,19 +45,24 @@ const user = {
|
||||||
},
|
},
|
||||||
SET_ID: (state, id) => {
|
SET_ID: (state, id) => {
|
||||||
state.id = id
|
state.id = id
|
||||||
|
},
|
||||||
|
SET_AUTH_HOST: (state, authHost) => {
|
||||||
|
state.authHost = authHost
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
|
||||||
actions: {
|
actions: {
|
||||||
LoginByUsername({ commit }, userInfo) {
|
LoginByUsername({ commit, dispatch }, { username, authHost, password }) {
|
||||||
const username = userInfo.username.trim()
|
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
loginByUsername(username, userInfo.password).then(response => {
|
loginByUsername(username, password, authHost).then(response => {
|
||||||
const data = response.data
|
const data = response.data
|
||||||
commit('SET_TOKEN', data.access_token)
|
commit('SET_TOKEN', data.access_token)
|
||||||
|
commit('SET_AUTH_HOST', authHost)
|
||||||
setToken(data.access_token)
|
setToken(data.access_token)
|
||||||
|
setAuthHost(authHost)
|
||||||
resolve()
|
resolve()
|
||||||
}).catch(error => {
|
}).catch(error => {
|
||||||
|
dispatch('addErrorLog', { message: error.message })
|
||||||
reject(error)
|
reject(error)
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
@ -64,7 +70,7 @@ const user = {
|
||||||
|
|
||||||
GetUserInfo({ commit, state }) {
|
GetUserInfo({ commit, state }) {
|
||||||
return new Promise((resolve, reject) => {
|
return new Promise((resolve, reject) => {
|
||||||
getUserInfo(state.token).then(response => {
|
getUserInfo(state.token, state.authHost).then(response => {
|
||||||
const data = response.data
|
const data = response.data
|
||||||
|
|
||||||
if (!data) {
|
if (!data) {
|
||||||
|
@ -91,11 +97,13 @@ const user = {
|
||||||
commit('SET_TOKEN', '')
|
commit('SET_TOKEN', '')
|
||||||
commit('SET_ROLES', [])
|
commit('SET_ROLES', [])
|
||||||
removeToken()
|
removeToken()
|
||||||
|
removeAuthHost()
|
||||||
},
|
},
|
||||||
FedLogOut({ commit }) {
|
FedLogOut({ commit }) {
|
||||||
return new Promise(resolve => {
|
return new Promise(resolve => {
|
||||||
commit('SET_TOKEN', '')
|
commit('SET_TOKEN', '')
|
||||||
removeToken()
|
removeToken()
|
||||||
|
removeAuthHost()
|
||||||
resolve()
|
resolve()
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -43,14 +43,14 @@ const users = {
|
||||||
},
|
},
|
||||||
actions: {
|
actions: {
|
||||||
async FetchUsers({ commit, state, getters }, { page }) {
|
async FetchUsers({ commit, state, getters }, { page }) {
|
||||||
const response = await fetchUsers(state.showLocalUsersOnly, getters.token, page)
|
const response = await fetchUsers(state.showLocalUsersOnly, getters.authHost, getters.token, page)
|
||||||
|
|
||||||
commit('SET_LOADING', true)
|
commit('SET_LOADING', true)
|
||||||
|
|
||||||
loadUsers(commit, page, response.data)
|
loadUsers(commit, page, response.data)
|
||||||
},
|
},
|
||||||
async ToggleUserActivation({ commit, getters }, nickname) {
|
async ToggleUserActivation({ commit, getters }, nickname) {
|
||||||
const response = await toggleUserActivation(nickname, getters.token)
|
const response = await toggleUserActivation(nickname, getters.authHost, getters.token)
|
||||||
|
|
||||||
commit('SWAP_USER', response.data)
|
commit('SWAP_USER', response.data)
|
||||||
},
|
},
|
||||||
|
@ -62,7 +62,7 @@ const users = {
|
||||||
commit('SET_LOADING', true)
|
commit('SET_LOADING', true)
|
||||||
commit('SET_SEARCH_QUERY', query)
|
commit('SET_SEARCH_QUERY', query)
|
||||||
|
|
||||||
const response = await searchUsers(query, state.showLocalUsersOnly, getters.token, page)
|
const response = await searchUsers(query, state.showLocalUsersOnly, getters.authHost, getters.token, page)
|
||||||
|
|
||||||
loadUsers(commit, page, response.data)
|
loadUsers(commit, page, response.data)
|
||||||
}
|
}
|
||||||
|
@ -73,24 +73,24 @@ const users = {
|
||||||
},
|
},
|
||||||
async ToggleRight({ commit, getters }, { user, right }) {
|
async ToggleRight({ commit, getters }, { user, right }) {
|
||||||
user.roles[right]
|
user.roles[right]
|
||||||
? await deleteRight(user.nickname, right, getters.token)
|
? await deleteRight(user.nickname, right, getters.authHost, getters.token)
|
||||||
: await addRight(user.nickname, right, getters.token)
|
: await addRight(user.nickname, right, getters.authHost, getters.token)
|
||||||
|
|
||||||
const updatedUser = { ...user, roles: { ...user.roles, [right]: !user.roles[right] }}
|
const updatedUser = { ...user, roles: { ...user.roles, [right]: !user.roles[right] }}
|
||||||
commit('SWAP_USER', updatedUser)
|
commit('SWAP_USER', updatedUser)
|
||||||
},
|
},
|
||||||
async DeleteUser({ commit, getters }, user) {
|
async DeleteUser({ commit, getters }, user) {
|
||||||
await deleteUser(user.nickname, getters.token)
|
await deleteUser(user.nickname, getters.authHost, getters.token)
|
||||||
const updatedUser = { ...user, deactivated: true }
|
const updatedUser = { ...user, deactivated: true }
|
||||||
commit('SWAP_USER', updatedUser)
|
commit('SWAP_USER', updatedUser)
|
||||||
},
|
},
|
||||||
async ToggleTag({ commit, getters }, { user, tag }) {
|
async ToggleTag({ commit, getters }, { user, tag }) {
|
||||||
if (user.tags.includes(tag)) {
|
if (user.tags.includes(tag)) {
|
||||||
await untagUser(user.nickname, tag, getters.token)
|
await untagUser(user.nickname, tag, getters.authHost, getters.token)
|
||||||
const updatedUser = { ...user, tags: user.tags.filter(userTag => userTag !== tag) }
|
const updatedUser = { ...user, tags: user.tags.filter(userTag => userTag !== tag) }
|
||||||
commit('SWAP_USER', updatedUser)
|
commit('SWAP_USER', updatedUser)
|
||||||
} else {
|
} else {
|
||||||
await tagUser(user.nickname, tag, getters.token)
|
await tagUser(user.nickname, tag, getters.authHost, getters.token)
|
||||||
const updatedUser = { ...user, tags: [...user.tags, tag] }
|
const updatedUser = { ...user, tags: [...user.tags, tag] }
|
||||||
commit('SWAP_USER', updatedUser)
|
commit('SWAP_USER', updatedUser)
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
import Cookies from 'js-cookie'
|
import Cookies from 'js-cookie'
|
||||||
|
|
||||||
const TokenKey = 'Admin-Token'
|
const TokenKey = 'Admin-Token'
|
||||||
|
const AuthHostKey = 'Auth-Host'
|
||||||
|
|
||||||
export function getToken() {
|
export function getToken() {
|
||||||
return Cookies.get(TokenKey)
|
return Cookies.get(TokenKey)
|
||||||
|
@ -13,3 +14,15 @@ export function setToken(token) {
|
||||||
export function removeToken() {
|
export function removeToken() {
|
||||||
return Cookies.remove(TokenKey)
|
return Cookies.remove(TokenKey)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
export function getAuthHost() {
|
||||||
|
return Cookies.get(AuthHostKey)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function setAuthHost(token) {
|
||||||
|
return Cookies.set(AuthHostKey, token)
|
||||||
|
}
|
||||||
|
|
||||||
|
export function removeAuthHost() {
|
||||||
|
return Cookies.remove(AuthHostKey)
|
||||||
|
}
|
||||||
|
|
|
@ -3,7 +3,6 @@ import { Message } from 'element-ui'
|
||||||
|
|
||||||
// create an axios instance
|
// create an axios instance
|
||||||
const service = axios.create({
|
const service = axios.create({
|
||||||
baseURL: process.env.BASE_API, // api 的 base_url
|
|
||||||
timeout: 5000 // request timeout
|
timeout: 5000 // request timeout
|
||||||
})
|
})
|
||||||
|
|
||||||
|
|
|
@ -1,143 +0,0 @@
|
||||||
<template>
|
|
||||||
<div>
|
|
||||||
<el-card class="box-card" style="margin-top:40px;">
|
|
||||||
<div slot="header" class="clearfix">
|
|
||||||
<svg-icon icon-class="international" />
|
|
||||||
<span style="margin-left:10px;">{{ $t('i18nView.title') }}</span>
|
|
||||||
</div>
|
|
||||||
<div>
|
|
||||||
<el-radio-group v-model="lang" size="small">
|
|
||||||
<el-radio label="zh" border>简体中文</el-radio>
|
|
||||||
<el-radio label="en" border>English</el-radio>
|
|
||||||
<el-radio label="es" border>Español</el-radio>
|
|
||||||
</el-radio-group>
|
|
||||||
<el-tag style="margin-top:15px;display:block;" type="info">{{ $t('i18nView.note') }}</el-tag>
|
|
||||||
</div>
|
|
||||||
</el-card>
|
|
||||||
|
|
||||||
<el-row :gutter="20" style="margin:100px 15px 50px;">
|
|
||||||
<el-col :span="12" :xs="24">
|
|
||||||
<div class="block">
|
|
||||||
<el-date-picker v-model="date" :placeholder="$t('i18nView.datePlaceholder')" type="date"/>
|
|
||||||
</div>
|
|
||||||
<div class="block">
|
|
||||||
<el-select v-model="value" :placeholder="$t('i18nView.selectPlaceholder')">
|
|
||||||
<el-option
|
|
||||||
v-for="item in options"
|
|
||||||
:key="item.value"
|
|
||||||
:label="item.label"
|
|
||||||
:value="item.value"/>
|
|
||||||
</el-select>
|
|
||||||
</div>
|
|
||||||
<div class="block">
|
|
||||||
<el-button class="item-btn" size="small">{{ $t('i18nView.default') }}</el-button>
|
|
||||||
<el-button class="item-btn" size="small" type="primary">{{ $t('i18nView.primary') }}</el-button>
|
|
||||||
<el-button class="item-btn" size="small" type="success">{{ $t('i18nView.success') }}</el-button>
|
|
||||||
<el-button class="item-btn" size="small" type="info">{{ $t('i18nView.info') }}</el-button>
|
|
||||||
<el-button class="item-btn" size="small" type="warning">{{ $t('i18nView.warning') }}</el-button>
|
|
||||||
<el-button class="item-btn" size="small" type="danger">{{ $t('i18nView.danger') }}</el-button>
|
|
||||||
</div>
|
|
||||||
</el-col>
|
|
||||||
<el-col :span="12" :xs="24">
|
|
||||||
<el-table :data="tableData" fit highlight-current-row border style="width: 100%">
|
|
||||||
<el-table-column :label="$t('i18nView.tableName')" prop="name" width="100" align="center"/>
|
|
||||||
<el-table-column :label="$t('i18nView.tableDate')" prop="date" width="120" align="center"/>
|
|
||||||
<el-table-column :label="$t('i18nView.tableAddress')" prop="address"/>
|
|
||||||
</el-table>
|
|
||||||
</el-col>
|
|
||||||
</el-row>
|
|
||||||
</div>
|
|
||||||
</template>
|
|
||||||
|
|
||||||
<script>
|
|
||||||
import local from './local'
|
|
||||||
const viewName = 'i18nView'
|
|
||||||
|
|
||||||
export default {
|
|
||||||
name: 'I18n',
|
|
||||||
data: function() {
|
|
||||||
return {
|
|
||||||
date: '',
|
|
||||||
tableData: [{
|
|
||||||
date: '2016-05-03',
|
|
||||||
name: 'Tom',
|
|
||||||
address: 'No. 189, Grove St, Los Angeles'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
date: '2016-05-02',
|
|
||||||
name: 'Tom',
|
|
||||||
address: 'No. 189, Grove St, Los Angeles'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
date: '2016-05-04',
|
|
||||||
name: 'Tom',
|
|
||||||
address: 'No. 189, Grove St, Los Angeles'
|
|
||||||
},
|
|
||||||
{
|
|
||||||
date: '2016-05-01',
|
|
||||||
name: 'Tom',
|
|
||||||
address: 'No. 189, Grove St, Los Angeles'
|
|
||||||
}],
|
|
||||||
options: [],
|
|
||||||
value: ''
|
|
||||||
}
|
|
||||||
},
|
|
||||||
computed: {
|
|
||||||
lang: {
|
|
||||||
get() {
|
|
||||||
return this.$store.state.app.language
|
|
||||||
},
|
|
||||||
set(lang) {
|
|
||||||
this.$i18n.locale = lang
|
|
||||||
this.$store.dispatch('setLanguage', lang)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
},
|
|
||||||
watch: {
|
|
||||||
lang() {
|
|
||||||
this.setOptions()
|
|
||||||
}
|
|
||||||
},
|
|
||||||
created() {
|
|
||||||
if (!this.$i18n.getLocaleMessage('en')[viewName]) {
|
|
||||||
this.$i18n.mergeLocaleMessage('en', local.en)
|
|
||||||
this.$i18n.mergeLocaleMessage('zh', local.zh)
|
|
||||||
this.$i18n.mergeLocaleMessage('es', local.es)
|
|
||||||
}
|
|
||||||
this.setOptions() // set default select options
|
|
||||||
},
|
|
||||||
methods: {
|
|
||||||
setOptions() {
|
|
||||||
this.options = [
|
|
||||||
{
|
|
||||||
value: '1',
|
|
||||||
label: this.$t('i18nView.one')
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: '2',
|
|
||||||
label: this.$t('i18nView.two')
|
|
||||||
},
|
|
||||||
{
|
|
||||||
value: '3',
|
|
||||||
label: this.$t('i18nView.three')
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
</script>
|
|
||||||
|
|
||||||
<style scoped>
|
|
||||||
.box-card {
|
|
||||||
width: 600px;
|
|
||||||
max-width: 100%;
|
|
||||||
margin: 20px auto;
|
|
||||||
}
|
|
||||||
.item-btn{
|
|
||||||
margin-bottom: 15px;
|
|
||||||
margin-left: 0px;
|
|
||||||
}
|
|
||||||
.block {
|
|
||||||
padding: 25px;
|
|
||||||
}
|
|
||||||
</style>
|
|
|
@ -1,63 +0,0 @@
|
||||||
|
|
||||||
export default {
|
|
||||||
zh: {
|
|
||||||
i18nView: {
|
|
||||||
title: '切换语言',
|
|
||||||
note: '本项目国际化基于 vue-i18n',
|
|
||||||
datePlaceholder: '请选择日期',
|
|
||||||
selectPlaceholder: '请选择',
|
|
||||||
tableDate: '日期',
|
|
||||||
tableName: '姓名',
|
|
||||||
tableAddress: '地址',
|
|
||||||
default: '默认按钮',
|
|
||||||
primary: '主要按钮',
|
|
||||||
success: '成功按钮',
|
|
||||||
info: '信息按钮',
|
|
||||||
warning: '警告按钮',
|
|
||||||
danger: '危险按钮',
|
|
||||||
one: '一',
|
|
||||||
two: '二',
|
|
||||||
three: '三'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
en: {
|
|
||||||
i18nView: {
|
|
||||||
title: 'Switch Language',
|
|
||||||
note: 'The internationalization of this project is based on vue-i18n',
|
|
||||||
datePlaceholder: 'Pick a day',
|
|
||||||
selectPlaceholder: 'Select',
|
|
||||||
tableDate: 'tableDate',
|
|
||||||
tableName: 'tableName',
|
|
||||||
tableAddress: 'tableAddress',
|
|
||||||
default: 'default:',
|
|
||||||
primary: 'primary',
|
|
||||||
success: 'success',
|
|
||||||
info: 'info',
|
|
||||||
warning: 'warning',
|
|
||||||
danger: 'danger',
|
|
||||||
one: 'One',
|
|
||||||
two: 'Two',
|
|
||||||
three: 'Three'
|
|
||||||
}
|
|
||||||
},
|
|
||||||
es: {
|
|
||||||
i18nView: {
|
|
||||||
title: 'Switch Language',
|
|
||||||
note: 'The internationalization of this project is based on vue-i18n',
|
|
||||||
datePlaceholder: 'Pick a day',
|
|
||||||
selectPlaceholder: 'Select',
|
|
||||||
tableDate: 'tableDate',
|
|
||||||
tableName: 'tableName',
|
|
||||||
tableAddress: 'tableAddress',
|
|
||||||
default: 'default:',
|
|
||||||
primary: 'primary',
|
|
||||||
success: 'success',
|
|
||||||
info: 'info',
|
|
||||||
warning: 'warning',
|
|
||||||
danger: 'danger',
|
|
||||||
one: 'One',
|
|
||||||
two: 'Two',
|
|
||||||
three: 'Three'
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
|
@ -45,8 +45,13 @@
|
||||||
</template>
|
</template>
|
||||||
|
|
||||||
<script>
|
<script>
|
||||||
|
import { Message } from 'element-ui'
|
||||||
|
import SvgIcon from '@/components/SvgIcon'
|
||||||
|
import i18n from '@/lang'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
name: 'Login',
|
name: 'Login',
|
||||||
|
components: { 'svg-icon': SvgIcon },
|
||||||
data: function() {
|
data: function() {
|
||||||
return {
|
return {
|
||||||
loginForm: {
|
loginForm: {
|
||||||
|
@ -77,12 +82,35 @@ export default {
|
||||||
},
|
},
|
||||||
handleLogin() {
|
handleLogin() {
|
||||||
this.loading = true
|
this.loading = true
|
||||||
this.$store.dispatch('LoginByUsername', this.loginForm).then(() => {
|
if (this.checkUsername()) {
|
||||||
|
const loginData = this.getLoginData()
|
||||||
|
this.$store.dispatch('LoginByUsername', loginData).then(() => {
|
||||||
|
this.loading = false
|
||||||
|
this.$router.push({ path: this.redirect || '/users/index' })
|
||||||
|
}).catch(() => {
|
||||||
|
this.loading = false
|
||||||
|
})
|
||||||
|
} else {
|
||||||
|
Message({
|
||||||
|
message: i18n.t('login.errorMessage'),
|
||||||
|
type: 'error',
|
||||||
|
duration: 7000
|
||||||
|
})
|
||||||
|
this.$store.dispatch('addErrorLog', { message: i18n.t('login.errorMessage') })
|
||||||
this.loading = false
|
this.loading = false
|
||||||
this.$router.push({ path: this.redirect || '/users/index' })
|
}
|
||||||
}).catch(() => {
|
},
|
||||||
this.loading = false
|
checkUsername() {
|
||||||
})
|
return this.loginForm.username.includes('@')
|
||||||
|
},
|
||||||
|
getLoginData() {
|
||||||
|
const [username, authHost] = this.loginForm.username.split('@')
|
||||||
|
|
||||||
|
return {
|
||||||
|
username: username.trim(),
|
||||||
|
authHost: authHost.trim(),
|
||||||
|
password: this.loginForm.password
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
47
test/views/layout/index.test.js
Normal file
47
test/views/layout/index.test.js
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
import Vuex from 'vuex'
|
||||||
|
import VueRouter from 'vue-router'
|
||||||
|
import { mount, createLocalVue, config } from '@vue/test-utils'
|
||||||
|
import Element from 'element-ui'
|
||||||
|
import Layout from '@/views/layout/Layout'
|
||||||
|
import storeConfig from './store.conf'
|
||||||
|
import routerConfig from './router.conf'
|
||||||
|
import { cloneDeep } from 'lodash'
|
||||||
|
import { beforeEachRoute } from '@/permission'
|
||||||
|
|
||||||
|
config.mocks["$t"] = () => {}
|
||||||
|
|
||||||
|
const localVue = createLocalVue()
|
||||||
|
localVue.use(Vuex)
|
||||||
|
localVue.use(VueRouter)
|
||||||
|
localVue.use(Element)
|
||||||
|
|
||||||
|
describe('Log out', () => {
|
||||||
|
let store
|
||||||
|
let router
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
store = new Vuex.Store(cloneDeep(storeConfig))
|
||||||
|
router = new VueRouter(cloneDeep(routerConfig))
|
||||||
|
router.beforeEach(beforeEachRoute)
|
||||||
|
window.location.reload = jest.fn()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('logs out user', async (done) => {
|
||||||
|
const wrapper = mount(Layout, {
|
||||||
|
store,
|
||||||
|
router,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
|
||||||
|
const logoutButton = wrapper.find('span')
|
||||||
|
expect(store.state.user.roles.length).toBe(1)
|
||||||
|
expect(store.state.user.token).toBe('foo123')
|
||||||
|
|
||||||
|
logoutButton.trigger('click')
|
||||||
|
await wrapper.vm.$nextTick()
|
||||||
|
|
||||||
|
expect(store.state.user.roles.length).toBe(0)
|
||||||
|
expect(store.state.user.token).toBe('')
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
40
test/views/layout/router.conf.js
Normal file
40
test/views/layout/router.conf.js
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import Layout from '@/views/layout/Layout'
|
||||||
|
|
||||||
|
export const constantRouterMap = [
|
||||||
|
{
|
||||||
|
path: '/redirect',
|
||||||
|
component: Layout,
|
||||||
|
hidden: true,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: '/redirect/:path*',
|
||||||
|
component: () => import('@/views/redirect/index')
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
component: () => import('@/views/login/index'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/auth-redirect',
|
||||||
|
component: () => import('@/views/login/authredirect'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/404',
|
||||||
|
component: () => import('@/views/errorPage/404'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/401',
|
||||||
|
component: () => import('@/views/errorPage/401'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '',
|
||||||
|
component: Layout,
|
||||||
|
redirect: '/users/index'
|
||||||
|
}
|
||||||
|
]
|
34
test/views/layout/store.conf.js
Normal file
34
test/views/layout/store.conf.js
Normal file
|
@ -0,0 +1,34 @@
|
||||||
|
import app from '@/store/modules/app'
|
||||||
|
import errorLog from '@/store/modules/errorLog'
|
||||||
|
import permission from '@/store/modules/permission'
|
||||||
|
import tagsView from '@/store/modules/tagsView'
|
||||||
|
import user from '@/store/modules/user'
|
||||||
|
import users from '@/store/modules/users'
|
||||||
|
import getters from '@/store/getters'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
modules: {
|
||||||
|
app,
|
||||||
|
errorLog,
|
||||||
|
permission,
|
||||||
|
tagsView,
|
||||||
|
user: {
|
||||||
|
...user,
|
||||||
|
state: {
|
||||||
|
id:"10",
|
||||||
|
status: '',
|
||||||
|
code: '',
|
||||||
|
token: "foo123",
|
||||||
|
authHost:"apple",
|
||||||
|
name:"bob",
|
||||||
|
avatar: '',
|
||||||
|
introduction: '',
|
||||||
|
roles: ['admin'],
|
||||||
|
setting: {
|
||||||
|
articlePlatform: []
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
getters
|
||||||
|
}
|
106
test/views/login/index.test.js
Normal file
106
test/views/login/index.test.js
Normal file
|
@ -0,0 +1,106 @@
|
||||||
|
import Vuex from 'vuex'
|
||||||
|
import VueRouter from 'vue-router'
|
||||||
|
import { mount, createLocalVue, config } from '@vue/test-utils'
|
||||||
|
import flushPromises from 'flush-promises'
|
||||||
|
import Element from 'element-ui'
|
||||||
|
import Login from '@/views/login/index'
|
||||||
|
import storeConfig from './store.conf'
|
||||||
|
import routerConfig from './router.conf'
|
||||||
|
import { cloneDeep } from 'lodash'
|
||||||
|
import { beforeEachRoute } from '@/permission'
|
||||||
|
|
||||||
|
config.mocks["$t"] = () => {}
|
||||||
|
|
||||||
|
const localVue = createLocalVue()
|
||||||
|
localVue.use(Vuex)
|
||||||
|
localVue.use(VueRouter)
|
||||||
|
localVue.use(Element)
|
||||||
|
|
||||||
|
jest.mock('@/api/login')
|
||||||
|
|
||||||
|
describe('Login', () => {
|
||||||
|
let store
|
||||||
|
let router
|
||||||
|
|
||||||
|
const usernameInput = 'input[name="username"]'
|
||||||
|
const passwordInput = 'input[name="password"]'
|
||||||
|
|
||||||
|
beforeEach(() => {
|
||||||
|
store = new Vuex.Store(cloneDeep(storeConfig))
|
||||||
|
router = new VueRouter(cloneDeep(routerConfig))
|
||||||
|
router.beforeEach(beforeEachRoute)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws error if username does not contain authHost', () => {
|
||||||
|
const wrapper = mount(Login, {
|
||||||
|
store,
|
||||||
|
router,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
|
||||||
|
const errorLog = store.state.errorLog.logs
|
||||||
|
expect(errorLog.length).toBe(0)
|
||||||
|
const submitButton = wrapper.find('button')
|
||||||
|
|
||||||
|
wrapper.find(usernameInput).element.value = 'bob'
|
||||||
|
wrapper.find(usernameInput).trigger('input')
|
||||||
|
wrapper.find(passwordInput).element.value = '1234'
|
||||||
|
wrapper.find(passwordInput).trigger('input')
|
||||||
|
submitButton.trigger('click')
|
||||||
|
|
||||||
|
const updatedErrorLog = store.state.errorLog.logs
|
||||||
|
expect(updatedErrorLog.length).toBe(1)
|
||||||
|
expect(updatedErrorLog[0].message).toEqual(
|
||||||
|
'Username must contain username and host, e.g. john@pleroma.social'
|
||||||
|
)
|
||||||
|
})
|
||||||
|
|
||||||
|
it('throws error if password is incorrect', async (done) => {
|
||||||
|
const wrapper = mount(Login, {
|
||||||
|
store,
|
||||||
|
router,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
|
||||||
|
const errorLog = store.state.errorLog.logs
|
||||||
|
expect(errorLog.length).toBe(0)
|
||||||
|
const submitButton = wrapper.find('button')
|
||||||
|
|
||||||
|
wrapper.find(usernameInput).element.value = 'bob@apple'
|
||||||
|
wrapper.find(usernameInput).trigger('input')
|
||||||
|
wrapper.find(passwordInput).element.value = '1234'
|
||||||
|
wrapper.find(passwordInput).trigger('input')
|
||||||
|
submitButton.trigger('click')
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
const updatedErrorLog = store.state.errorLog.logs
|
||||||
|
expect(updatedErrorLog.length).toBe(1)
|
||||||
|
expect(updatedErrorLog[0].message).toEqual(
|
||||||
|
'Invalid credentials'
|
||||||
|
)
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
|
||||||
|
it('logs user in', async (done) => {
|
||||||
|
const wrapper = mount(Login, {
|
||||||
|
store,
|
||||||
|
router,
|
||||||
|
localVue
|
||||||
|
})
|
||||||
|
|
||||||
|
const errorLog = store.state.errorLog.logs
|
||||||
|
const submitButton = wrapper.find('button')
|
||||||
|
expect(wrapper.vm.$route.path).toBe('/login')
|
||||||
|
|
||||||
|
wrapper.find(usernameInput).element.value = 'bob@apple'
|
||||||
|
wrapper.find(usernameInput).trigger('input')
|
||||||
|
wrapper.find(passwordInput).element.value = '123456'
|
||||||
|
wrapper.find(passwordInput).trigger('input')
|
||||||
|
submitButton.trigger('click')
|
||||||
|
await flushPromises()
|
||||||
|
|
||||||
|
expect(errorLog.length).toBe(0)
|
||||||
|
expect(wrapper.vm.$route.path).toBe('/')
|
||||||
|
done()
|
||||||
|
})
|
||||||
|
})
|
40
test/views/login/router.conf.js
Normal file
40
test/views/login/router.conf.js
Normal file
|
@ -0,0 +1,40 @@
|
||||||
|
import Layout from '@/views/layout/Layout'
|
||||||
|
|
||||||
|
export const constantRouterMap = [
|
||||||
|
{
|
||||||
|
path: '/redirect',
|
||||||
|
component: Layout,
|
||||||
|
hidden: true,
|
||||||
|
children: [
|
||||||
|
{
|
||||||
|
path: '/redirect/:path*',
|
||||||
|
component: () => import('@/views/redirect/index')
|
||||||
|
}
|
||||||
|
]
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/login',
|
||||||
|
component: () => import('@/views/login/index'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/auth-redirect',
|
||||||
|
component: () => import('@/views/login/authredirect'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/404',
|
||||||
|
component: () => import('@/views/errorPage/404'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '/401',
|
||||||
|
component: () => import('@/views/errorPage/401'),
|
||||||
|
hidden: true
|
||||||
|
},
|
||||||
|
{
|
||||||
|
path: '',
|
||||||
|
component: Layout,
|
||||||
|
redirect: '/users/index'
|
||||||
|
}
|
||||||
|
]
|
16
test/views/login/store.conf.js
Normal file
16
test/views/login/store.conf.js
Normal file
|
@ -0,0 +1,16 @@
|
||||||
|
import app from '@/store/modules/app'
|
||||||
|
import errorLog from '@/store/modules/errorLog'
|
||||||
|
import user from '@/store/modules/user'
|
||||||
|
import users from '@/store/modules/users'
|
||||||
|
import getters from '@/store/getters'
|
||||||
|
|
||||||
|
export default {
|
||||||
|
modules: {
|
||||||
|
app,
|
||||||
|
errorLog,
|
||||||
|
user,
|
||||||
|
users
|
||||||
|
},
|
||||||
|
getters
|
||||||
|
}
|
||||||
|
|
|
@ -1,28 +1,13 @@
|
||||||
import getters from '@/store/getters'
|
|
||||||
import app from '@/store/modules/app'
|
import app from '@/store/modules/app'
|
||||||
import user from '@/store/modules/user'
|
import user from '@/store/modules/user'
|
||||||
import users from '@/store/modules/users'
|
import users from '@/store/modules/users'
|
||||||
|
import getters from '@/store/getters'
|
||||||
|
|
||||||
export default {
|
export default {
|
||||||
modules: {
|
modules: {
|
||||||
app,
|
app,
|
||||||
users,
|
user,
|
||||||
user: {
|
users
|
||||||
state: {
|
|
||||||
user: '',
|
|
||||||
id: '1',
|
|
||||||
status: '',
|
|
||||||
code: '',
|
|
||||||
token: 'MmwSkMgBW6lAkWCFspIjX9icmLfqSCohSi-GReAZrQw',
|
|
||||||
name: 'john',
|
|
||||||
avatar: '',
|
|
||||||
introduction: '',
|
|
||||||
roles: ["admin"],
|
|
||||||
setting: {
|
|
||||||
articlePlatform: []
|
|
||||||
}
|
|
||||||
}, ...user
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
getters
|
getters
|
||||||
}
|
}
|
||||||
|
|
17
yarn.lock
17
yarn.lock
|
@ -4181,6 +4181,11 @@ flat-cache@^1.2.1:
|
||||||
rimraf "~2.6.2"
|
rimraf "~2.6.2"
|
||||||
write "^0.2.1"
|
write "^0.2.1"
|
||||||
|
|
||||||
|
flush-promises@^1.0.2:
|
||||||
|
version "1.0.2"
|
||||||
|
resolved "https://registry.yarnpkg.com/flush-promises/-/flush-promises-1.0.2.tgz#4948fd58f15281fed79cbafc86293d5bb09b2ced"
|
||||||
|
integrity sha512-G0sYfLQERwKz4+4iOZYQEZVpOt9zQrlItIxQAAYAWpfby3gbHrx0osCHz5RLl/XoXevXk0xoN4hDFky/VV9TrA==
|
||||||
|
|
||||||
flush-write-stream@^1.0.0:
|
flush-write-stream@^1.0.0:
|
||||||
version "1.1.1"
|
version "1.1.1"
|
||||||
resolved "https://registry.yarnpkg.com/flush-write-stream/-/flush-write-stream-1.1.1.tgz#8dd7d873a1babc207d94ead0c2e0e44276ebf2e8"
|
resolved "https://registry.yarnpkg.com/flush-write-stream/-/flush-write-stream-1.1.1.tgz#8dd7d873a1babc207d94ead0c2e0e44276ebf2e8"
|
||||||
|
@ -10164,18 +10169,6 @@ vue-jest@4.0.0-beta.2:
|
||||||
source-map "^0.5.6"
|
source-map "^0.5.6"
|
||||||
ts-jest "^23.10.5"
|
ts-jest "^23.10.5"
|
||||||
|
|
||||||
vue-jest@4.0.0-beta.2:
|
|
||||||
version "4.0.0-beta.2"
|
|
||||||
resolved "https://registry.yarnpkg.com/vue-jest/-/vue-jest-4.0.0-beta.2.tgz#f2120ea9d24224aad3a100c2010b0760d47ee6fe"
|
|
||||||
integrity sha512-SywBIciuIfqsCb8Eb9UQ02s06+NV8Ry8KnbyhAfnvnyFFulIuh7ujtga9eJYq720nCS4Hz4TpVtS4pD1ZbUILQ==
|
|
||||||
dependencies:
|
|
||||||
"@babel/plugin-transform-modules-commonjs" "^7.2.0"
|
|
||||||
"@vue/component-compiler-utils" "^2.4.0"
|
|
||||||
chalk "^2.1.0"
|
|
||||||
extract-from-css "^0.4.4"
|
|
||||||
source-map "^0.5.6"
|
|
||||||
ts-jest "^23.10.5"
|
|
||||||
|
|
||||||
vue-loader@15.3.0:
|
vue-loader@15.3.0:
|
||||||
version "15.3.0"
|
version "15.3.0"
|
||||||
resolved "https://registry.yarnpkg.com/vue-loader/-/vue-loader-15.3.0.tgz#b474d10a4e93d934a78c147fc3e314b370e9fc54"
|
resolved "https://registry.yarnpkg.com/vue-loader/-/vue-loader-15.3.0.tgz#b474d10a4e93d934a78c147fc3e314b370e9fc54"
|
||||||
|
|
Loading…
Reference in a new issue