node.js提供接口,vue展示頁面,先後端分離,出於編輯器功能和編輯習慣,vue用HbuilderX,node.js用VScode。(PS:僅做爲學習筆記,若有不當之處歡迎指出,在此先謝爲敬~~~)html
首先須要有node.js環境,安裝教程 在這裏,最好下載較新的版本,對es六、es7有更好的支持,再裝個 淘寶鏡像,完畢!
vue
解壓到安裝位置,修改環境變量,win10編輯環境變量很方便了,win7的話記得以 ; 分割開
node
在mysql的bin目錄下,新建my.ini文件(若是沒有),打開my.ini文件,寫入如下配置內容mysql
[mysqld] # 設置3306端口 port=3306 # 設置mysql的安裝目錄 basedir=D:\\myInstalls\\mysql-8.0.11 # 設置mysql數據庫的數據的存放目錄 datadir=D:\\myInstalls\\mysql-8.0.11\\Data # 容許最大鏈接數 max_connections=200 # 容許鏈接失敗的次數。這是爲了防止有人從該主機試圖攻擊數據庫系統 max_connect_errors=10 # 服務端使用的字符集默認爲UTF8 character-set-server=utf8 # 建立新表時將使用的默認存儲引擎 default-storage-engine=INNODB # 默認使用「mysql_native_password」插件認證 default_authentication_plugin=mysql_native_password [mysql] # 設置mysql客戶端默認字符集 default-character-set=utf8 [client] # 設置mysql客戶端鏈接服務端時默認使用的端口 port=3306 default-character-set=utf8
以管理員身份運行cmd,進入mysql的bin目錄下,不進入也行,由於咱們已經配置了環境變量ios
初始化數據庫,運行 mysqld --initialize --console,記住紅色框內的初始密碼es6
安裝mysql服務,運行 mysqld --install [服務名] ,服務名能夠不寫,安裝完畢 net start mysql 啓動mysqlweb
啓動成果,mysql中止指令 net stop mysqlajax
默認密碼太複雜,改個簡單的,首先運行 mysql -u root -p 進入mysql,密碼是剛纔記住的初始密碼sql
修改密碼指令:ALTER USER 'root'@'localhost' IDENTIFIED WITH mysql_native_password BY '新密碼'; vue-cli
OK,mysql咱們已經有了,接下來搭建koa2!
(你可使用系統自帶的cmd窗口,也能夠用編輯器自帶的。我這裏用VScode的命令行終端,看起來特別虛浮~~~)
安裝指令:cnpm install koa-generator -g
成功,根據上面提示走~~~
cd paopao 進入項目目錄
cnpm install 安裝項目依賴
cnpm start paopao 運行項目(cnpm是淘寶鏡像)
有個報錯大概意思是這個包再也不維護了,cnpm uninstall koa-onerror 卸載,從新裝最新的版本 cnpm install koa-onerror --save
在瀏覽器輸入:localhost:3000,瀏覽器運行結果(左),項目結構(右)
cnpm install sequelize mysql mysql2 --save
全部安裝的依賴能夠在package.json裏查看:
注意:我在使用時發現koa-static(處理靜態文件的中間件),默認3.0.0版本會報錯,因而更新成了最新版本
使用cnpm install koa-static@5.0.0 --save更新,再查看package.json,版本變成了5.0.0便可
在項目根目錄下建一個config文件夾,在該文件夾建一個js文件,取名db.js,用來配置數據庫鏈接
config-->db.js
var Sequelize = require("sequelize") var sequelize = new Sequelize('paopao','root','happy',{ host:'localhost', dialect:'mysql', operatorsAliases:false, dialectOptions:{ //字符集 charset:'utf8mb4', collate:'utf8mb4_unicode_ci', supportBigNumbers: true, bigNumberStrings: true }, pool:{ max: 5, min: 0, acquire: 30000, idle: 10000 }, timezone: '+08:00' //東八時區 }); module.exports = { sequelize };
paopao是個人數據庫表名,root數據庫用戶名,happy數據庫用戶密碼
在根目錄建一個module文件夾,在module文件下面建一個user.js,用來定義數據模型,告訴sequelize怎麼跟數據庫的數據一一對應
module-->user.js
module.exports = function(sequelize,DataTypes){ return sequelize.define( 'user', { userId:{ type: DataTypes.INTEGER, primaryKey: true, allowNull: true, autoIncrement: true }, mobileNo:{ type: DataTypes.STRING, allowNull: false, field: 'mobileNo' }, password:{ type: DataTypes.STRING, allowNull: false, field: 'password' } }, { timestamps: false } ); }
controller-->user.js 添加如下代碼
//引入db配置 const db = require('../config/db') //引入sequelize對象 const Sequelize = db.sequelize //引入數據表模型 const user = Sequelize.import('../module/user') //自動建立表 user.sync({ force: false }); //數據庫操做類 class userModule { static async userRegist(data) { return await user.create({ password: data.password, mobileNo: data.mobileNo }) } static async getUserInfo(mobileNo) { return await user.findOne({ where: { mobileNo } }) } }
數據庫操做有了,接下來進行功能處理,仍是在該文件添加
controller-->user.js 裏添加該userController 類,並將之exports出去
//功能處理 class userController { } module.exports = userController;
用戶註冊:
在 userController 類裏添加用戶註冊邏輯
//註冊用戶 static async create(ctx) { const req = ctx.request.body; if (req.mobileNo && req.password) { try { const query = await userModule.getUserInfo(req.mobileNo); if (query) { ctx.response.status = 200; ctx.body = { code: -1, desc: '用戶已存在' } } else { const param = { password: req.password, mobileNo: req.mobileNo, userName: req.mobileNo } const data = await userModule.userRegist(param); ctx.response.status = 200; ctx.body = { code: 0, desc: '用戶註冊成功', userInfo: { mobileNo: req.mobileNo } } } } catch (error) { ctx.response.status = 416; ctx.body = { code: -1, desc: '參數不齊全' } } } }
由於還要作登陸超時token驗證,用戶登陸成功還要返回token,爲了生成token,咱們須要安裝幾個中間件
cnpm install jsonwebtoken --save 導入jwt模塊
cnpm install koa-jwt --save koa提供的jwt中間件
在app.js裏添加以下代碼:
unless()表示裏面的regist、login不作token驗證
const koajwt = require('koa-jwt') // logger app.use(async (ctx, next) => { return next().catch((err) => { if(err.status === 401){ ctx.status = 401; ctx.body = { code: '-2000', desc: '登錄過時,請從新登錄' }; }else{ throw err; } }) }) app.use(koajwt({ secret: '123456' }).unless({ path: [/^\/user\/regist/,/^\/user\/login/] }))
爲了解析token,在public目錄下新建tool.js,加入解析token的代碼
const getToken = require('jsonwebtoken') exports.verToken = function(token){ return new Promise((resolve,rejece) => { const info = getToken.verify(token.split(' ')[1],"123456"); resolve(info); }) }
返回controller-->user.js,添加
//引入jwt作token驗證 const jwt = require('jsonwebtoken') //解析token const tools = require('../public/tool') //統一設置token有效時間 爲了方便觀察,設爲10s const expireTime = '10s'
用戶登陸:
以後就能夠寫用戶登陸邏輯了
controller-->user.js-->userController 類裏添加
經過 jwt.asign() 方法生成token,這裏的123456跟app.js裏的123456相同,就理解爲一個祕鑰吧~~
//密碼登錄 static async login(ctx) { const req = ctx.request.body; if (!req.mobileNo || !req.password) { return ctx.body = { code: '-1', msg: '用戶名或密碼不能爲空' } } else { const data = await userModule.getUserInfo(req.mobileNo); if (data) { if (data.password === req.passWord) { //生成token,驗證登陸有效期 const token = jwt.sign({ user: req.mobileNo, passWord: req.password }, '123456', { expiresIn: expireTime }); const info = { createdAt: data.createdAt, updatedAt: data.updatedAt, mobileNo: data.mobileNo, userId: data.userId } return ctx.body = { code: '0', token: token, userInfo: JSON.stringify(info), desc: '登錄成功' } } else { return ctx.body = { code: '-1', desc: '用戶密碼錯誤' } } } else { return ctx.body = { code: '-1', desc: '該用戶還沒有註冊' } } }; }
爲了驗證token是否過時,咱們再定義一個獲取用戶信息的邏輯,登錄10s後獲取用戶信息,驗證token是否過時
獲取用戶信息:
controller-->user.js-->userController 類裏添加
//獲取用戶信息(除密碼外) static async getUserInfo(ctx){ const req = ctx.request.body; const token = ctx.headers.authorization; if(token){ try { const result = await tools.verToken(token); if (!req.mobileNo) { return ctx.body = { code: '-1', desc: '參數錯誤' } } else { let data = await userModule.getUserInfo(req.mobileNo); if (req.mobileNo == data.mobileNo) { const info = { createdAt: data.createdAt, updatedAt: data.updatedAt, mobileNo: data.mobileNo, userId: data.userId }; return ctx.body = { code: '0', userInfo: JSON.stringify(info), desc: '獲取用戶信息成功' } } } } catch (error) { ctx.status = 401; return ctx.body = { code: '-1', desc: '登錄過時,請從新登錄' } } }else{ ctx.status = 401; return ctx.body = { code: '-1', desc: '登錄過時,請從新登錄' } } }
不用從新導入,koa-generator已經幫咱們導入了,直接使用
在routes目錄下新建文件 user.js
寫入如下代碼:
routes-->user.js
const Router = require('koa-router'); const userController = require('../controller/user') const router = new Router({ prefix: '/user' }); //用戶註冊 router.post('/regist',userController.create) //密碼登錄 router.post('/login',userController.login) //獲取用戶信息 router.post('/getUserInfo',userController.getUserInfo) module.exports = router;
而後在入口文件app.js引入
使用
完成這些之後,cnpm run dev 啓動項目(依賴nodemon,package.json裏面有,這樣每次更改代碼之後不用手動從新啓動)
啓動正常以下:
若是有報錯,提示缺乏這包那包的,不用着急!
把根目錄下的node_modules目錄刪除
檢查一遍package.json
確認無誤後從新cnpm install
再次啓動 cnpm run dev ~~~
補充一點,若是想在其餘端口啓動,在app.js裏添加 app.listen(3333),修改成3333端口,自動熱刷新~~~蛋是此時接口仍然不可調試,由於存在跨域問題
koa一樣提供瞭解決跨域的依賴包
cnpm install koa-cors --save
在app.js添加:
如今能夠測試接口了,隨便寫個ajax或者使用postman,postman測試結果:
註冊:
登陸:
查看數據庫結果(使用的是破解版Navicat圖形化數據庫管理工具):
到此爲止,API就完成了,最後一步,驗證token過時有沒有效果
寫到太晚了,想起來今天還沒給泡泡鏟屎,VUE就不寫那麼詳細了,有空再補上 ~.~
我就貼一下代碼和驗證結果
vue項目裏,在接口文件裏:
import axios from 'axios'; import qs from 'qs'; import route from '../router'; import { message } from 'ant-design-vue' axios.interceptors.request.use(function(config) { // 處理請求參數 config.data = qs.stringify(config.data) //將token寫入請求頭 if (window.localStorage.getItem('token')) { config.headers.Authorization = `Bearer ${window.localStorage.getItem('token')}`; } return config; }, function(error) { // 對請求錯誤作些什麼 return Promise.reject(error); }); axios.interceptors.response.use( response => { return response }, error => { if (error.response) { switch (error.response.status) { case 401: message.error("登陸過時,請從新登陸!", ()=>{ window.localStorage.removeItem("token"); //多是token過時,清除它 route.replace({ //跳轉到登陸頁面 path: '/login', query: { // 將跳轉的路由path做爲參數,登陸成功後跳轉到該路由 redirect: route.currentRoute.fullPath } }); }) } } return Promise.reject(error) // 返回接口返回的錯誤信息 } ); //註冊 export const regist = params => { return axios.post('http://localhost:3333/user/regist', params, {}).then(res => res.data) } //登陸 export const login = params => { return axios.post('http://localhost:3333/user/login', params, {}).then(res => res.data) } //獲取用戶信息 export const getUserInfo = params => { return axios.post('http://localhost:3333/user/getUserInfo', params, {}).then(res => res.data) }
axios.interceptors.request.use攔截請求,給請求頭加上token
axios.interceptors.response.use攔截響應,若是返回401,token過時,跳回login路由
登陸後10s再請求用戶數據,返回登陸過時: