项目经验 | Vue+webpack4的项目总结

一、目录结构

  • assets放置了静态文件
  • components放置了通用组件
  • config放置了一些常量配置
  • helper放置了主要的功能模块比如ajax.js、api.js
  • mixins放置了混入
  • router放置了路由文件
  • store放置了vuex的文件
  • views放置了页面文件
目录结构

二、 鉴权

项目的登陆状态由Vuex+LocalStorage维持,项目的鉴权主要靠 路由鉴权 和 Ajax鉴权。

store结构

store结构

state,localStorage维持了登陆信息的持久化

const state = {
  userInfo: $storage.getLocalStorage('userInfo') || {}
}

设置和删除userInfo的都放在mutation里

import { $storage } from '@/helper'

export default {
  setUserInfo (state, info) {
    state.userInfo = info
    $storage.setLocalStorage('userInfo', info)
  },
  deleteUserInfo (state) {
    state.userInfo = null
    $storage.removeLocalStorage('userInfo')
  }
}

向服务器获取userInfo放在action里(异步类型的)

import { $apis } from '@/helper'
export default {
  async getUserInfo ({ commit, state }) {
    let userId = state.userInfo.id
    if (userId) {
      $apis.getUserInfo(userId).then(res => {
        commit('setUserInfo', res)
      })
    }
  }
}

auth文件

实际上router鉴定权限调用的都是这两个方法,把它们分离出一个文件,用来检查有没有userInfo和userInfo是否是Admin的权限。

import { $apis, $storage } from '.'

export default {
  checkSession () {
    return $storage.getLocalStorage('userInfo') || false
  },

  async checkAdmin () {
    let userInfo = $storage.getLocalStorage('userInfo') || {}
    let userId = userInfo.id || 0
    if (!userId) return false
    userInfo = userInfo || (await $apis.getUserInfo(userId))
    let isAdmin = userInfo.permission < 2
    return userInfo.id && isAdmin
  }
}

router结构

router结构

鉴权这里用的是全局前置守卫,所以在beforeEachHooks.js写上鉴定权限的规则。
规则:在需要admin权限的地方无权限访问,则会跳到404页面;在需要登陆权限的地方无权限访问,则会跳到login界面。

import { $auth } from '@/helper'
export default {
  checkVisitAuth (to, from, next) {
    if (to.meta.isNeedAdmin) {
      $auth.checkAdmin().then(result => {
        return result ? next() : next({ path: '/404' })
      })
    } else if (to.meta.isNotNeedLogin) {
      next()
    } else {
      $auth.checkSession() ? next() : next({ path: '/login' })
    }
  }
}

在index.js处挂载该规则

Object.values(beforeEachHooks).forEach(hook => {
  routerInstance.beforeEach(hook)
})

Ajax

用了一层requestHandle来封装 axios,以后可能会改用interceptors来定制,比较符合通用的做法。
后端返回 401 就是没有登陆或者登陆信息过期,这时候清空登陆缓存状态,然后重定向到登陆页面。
后端返回 403 就是没有权限访问,由于本项目403也是跳到404页面所以和404同用一个处理方式。
后端返回 415 是访问api过多被限制。

/**
 * @param url
 * @param method get|post|put|delete...
 * @param params like queryString. if a url is index?a=1&b=2, params = {a: '1', b: '2'}
 * @param data post data, use for method put|post
 * @returns {Promise}
 */
function requestHandle (url, method, options) {
  if (options !== undefined) {
    var { params = {}, data = {} } = options
  } else {
    options = {}
  }
  return new Promise((resolve, reject) => {
    axios({
      url,
      method,
      params,
      data
    })
      .then(
        res => {
          if (res.status === 200 || res.status === 201 || res.status === 204) {
            resolve(res.data)
          } else {
            reject(res)
          }
        },
        err => {
          if (err.response.status === 401) {
            store.commit('deleteUserInfo')
            Vue.prototype.$error('请登陆')
            window.location.href = '/login'
            reject(err)
          } else if (
            err.response.status === 403 ||
            err.response.status === 404
          ) {
            reject(err.response.data)
          } else if (err.response.status === 415) {
            Vue.prototype.$error('您访问太过频繁, 已被限速')
            reject(err.response.data)
          }
        }
      )
      .catch(err => {
        Vue.prototype.$error(err)
        reject(err)
      })
  })
}

总体结构

实际上该项目的鉴权就是路由拦截那些正常的权限缺失,减少后端压力,而在登陆信息过期或者恶意修改登陆信息的会遭到后端的铁拳制裁😄,后端必须要有完整的鉴权系统才能是健壮的。

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 205,033评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 87,725评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 151,473评论 0 338
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,846评论 1 277
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,848评论 5 368
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,691评论 1 282
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 38,053评论 3 399
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,700评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 42,856评论 1 300
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,676评论 2 323
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,787评论 1 333
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,430评论 4 321
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 39,034评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,990评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,218评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 45,174评论 2 352
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,526评论 2 343

推荐阅读更多精彩内容