首页
友链
导航
影视
壁纸
统计
留言板
Search
1
el-upload自定义触发按钮及触发上传前判断
896 阅读
2
vue配置二级目录以及nginx多网站部署
710 阅读
3
el-cascader选择任意一级搭配懒加载使用,单选框radio不会触发懒加载
592 阅读
4
joe主题自定义导航页面
580 阅读
5
js获取昨天今天明天日期
496 阅读
web前端
vue
react
javascript
nuxt
typescript
indexDB数据库
微信小程序
美文欣赏
心情随笔
技术分享
其他
PHP
nodejs
博客api实战项目
typecho
登录
Search
标签搜索
web
vue
node项目实战
js
javascript
typecho
css
vuex
router
nginx
git
element
joe
utils
leaflet
dateFormat
map
date
axios
reg
辰漪
累计撰写
66
篇文章
累计收到
122
条评论
首页
栏目
web前端
vue
react
javascript
nuxt
typescript
indexDB数据库
微信小程序
美文欣赏
心情随笔
技术分享
其他
PHP
nodejs
博客api实战项目
typecho
页面
友链
导航
影视
壁纸
统计
留言板
搜索到
66
篇与
辰漪
的结果
2022-01-25
图片转base64
1. 使用FileReader对象FileReader对象可以异步读取File对象或者是Blob对象FileReader是一个构造函数,返回一个FileReader对象FileReader对象通过readAsDataUrl方法读取File对象或者Blob对象,一旦读取完成,result属性中会包含一个data: URL格式的Base64字符串以表示所读取文件的内容读取完成会触发FileReader对象的onload事件,可以从FileReader对象的result属性上获取返回的结果函数封装/** * 通过file对象/blob对象获取base64 * @param {Object} data file对象或者blob对象 */ const getBase64 = function (data) { return new Promise((resolve, reject) => { let imgRes = '' const reader = new FileReader() reader.readAsDataURL(data) reader.onerror = () => reject(reader.error) reader.onload = function (e) { // imgRes = e.target.result imgRes = reader.result } reader.onloadend = () => resolve(imgRes) }) }文件上传案例,并实现预览(file对象)<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> </head> <body> <input type="file" onchange="inputChange()"> <img src="" alt="预览图片"> <script> function inputChange (e) { const file = document.querySelector('input[type=file]').files[0] getBase64(file).then((res) => { const url = res document.querySelector('img').src = url }) } /** * 通过file对象/blob对象获取base64 * @param {Object} data file对象或者blob对象 */ const getBase64 = function (data) { return new Promise((resolve, reject) => { let imgRes = '' const reader = new FileReader() reader.readAsDataURL(data) reader.onerror = () => reject(reader.error) reader.onload = function (e) { // imgRes = e.target.result imgRes = reader.result } reader.onloadend = () => resolve(imgRes) }) } </script> </body> </html>通过图片url获取文件流blob对象案例(blob对象)<!DOCTYPE html> <html lang="en"> <head> <meta charset="UTF-8"> <meta http-equiv="X-UA-Compatible" content="IE=edge"> <meta name="viewport" content="width=device-width, initial-scale=1.0"> <title>Document</title> <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.0/jquery.min.js"></script> </head> <body> <img src="" alt="预览图片"> <button onclick="getImgByUrl()">通过url获取图片blob</button> <script> function getImgByUrl() { $.ajax({ // 可能会存在跨域问题 url: 'https://p5.qhimg.com/bdr/__85/t01409d4a62e0210249.jpg', xhrFields: { responseType: 'blob' }, success: (data, status) => { getBase64(data).then((res) => { const url = res document.querySelector('img').src = url }) } }) } </script> </body> </html> 2. 使用canvas.toDataURL()封装函数/** * 通过url获取图片base64 * @param {String} url 图片地址 */ const getBase64ByUrl = function (url) { return new Promise((resolve, reject) => { let base64Url const img = new Image() // 如果图片url存在跨域需要设置该属性,并且后端也要允许跨域 img.setAttribute('crossOrigin', 'Anonymous') img.src = url img.onload = function () { const c = document.createElement('canvas') c.width = img.width c.height = img.height const ctx = c.getContext('2d') ctx.drawImage(img, 0, 0, c.width, c.height) base64Url = c.toDataURL() resolve(base64Url) } img.onerror = function () { reject(base64Url) } }) }
2022年01月25日
118 阅读
1 评论
1 点赞
2022-01-21
git仓库迁移后本地git如何修改远程git地址
修改git远程仓库地址第一种:使用修改命令git remote origin set-url [url]第二种:先删除,再添加git remote rm origin git remote add origin [url]第三种:修改config文件push时要注意先拉取远程地址的分支合并到本地,不然可能会出现一系列问题,报错原因忘记复制了就不贴了git pull origin master // 拉取远程master分支到本地master分支 git pull origin dev_wrz // 拉取远程dev_wrz分支到本地dev_wrz分支
2022年01月21日
258 阅读
2 评论
1 点赞
2022-01-20
10-博客api接口项目实战之接口文档
title: 个人博客 v1.0.0language_tabs:shell: Shellhttp: HTTPjavascript: JavaScriptruby: Rubypython: Pythonphp: PHPjava: Javago: Gotoc_footers: []includes: []search: truehighlight_theme: darkulaheadingLevel: 2个人博客v1.0.0DefaultPOST 登录POST /api/loginBody 请求参数type: object properties: username: type: string description: 用户名 example: admin password: type: string description: 密码 example: "1314521999" required: - username - password 请求参数名称位置类型必选说明bodybodyobjectfalsenone» usernamebodystringtrue用户名» passwordbodystringtrue密码返回示例成功{ "status": 0, "message": "登陆成功!", "userInfo": { "user_id": 15, "username": "admin", "password": "", "nickname": "辰漪", "user_pic": "https://wrz521.top:8080/uploads/art_cover-2021-09-13 16-56-36.png", "is_admin": 1, "is_use": 1, "reg_time": "2021-08-26T08:08:08.000Z" }, "token": "Bearer eyJhbGciOiJIUzI1NiIsInR5cCI6IkpXVCJ9.eyJ1c2VyX2lkIjoxNSwidXNlcm5hbWUiOiJhZG1pbiIsInBhc3N3b3JkIjoiIiwibmlja25hbWUiOiLovrDmvKoiLCJ1c2VyX3BpYyI6Imh0dHBzOi8vd3J6NTIxLnRvcDo4MDgwL3VwbG9hZHMvYXJ0X2NvdmVyLTIwMjEtMDktMTMgMTYtNTYtMzYucG5nIiwiaXNfYWRtaW4iOjEsImlzX3VzZSI6MSwicmVnX3RpbWUiOiIyMDIxLTA4LTI2VDA4OjA4OjA4LjAwMFoiLCJpYXQiOjE2NDAxNDk4MjcsImV4cCI6MTY0MDE4NTgyN30.bXcGc-iMvVhICfF64EWpBZcQxETbkCFqWORnRgiCnCg" }返回结果状态码状态码含义说明数据模型200OK成功Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenone是否成功 0 成功 1 失败» messagestringtruenone说明信息» userInfoobjecttruenone用户信息»» user_idintegertruenone用户id»» usernamestringtruenone用户名»» passwordstringtruenone密码»» nicknamestringtruenone昵称»» user_picstringtruenone用户头像»» is_adminintegertruenone是否管理员»» is_useintegertruenone能否使用»» reg_timestringtruenone注册时间» tokenstringtruenonetokenblogGET 获取用户信息列表分页GET /my/getUserListPage请求参数名称位置类型必选说明fuzzyquerystringfalse查询字符串(通过username 或者 nickname)可以传空 获取全部pageNumquerystringfalse当前页码(默认 1)pageSizequerystringfalse每页显示条数(默认 10)sortKeyquerystringfalse根据哪个字段排序sortValuequerystringfalsedesc降序 asc升序Authorizationheaderstringtruenone返回示例成功{ "status": 0, "msg": "获取用户列表成功!", "total": 6, "data": [ { "user_id": 44, "username": "admin3", "nickname": "辰漪3", "user_pic": "3", "is_admin": 0, "is_use": 1, "reg_time": "2022-01-13T05:44:04.000Z" }, { "user_id": 43, "username": "admin2", "nickname": "辰漪1", "user_pic": "1", "is_admin": 0, "is_use": 1, "reg_time": "2022-01-12T06:22:14.000Z" }, { "user_id": 39, "username": "admin11561", "nickname": null, "user_pic": null, "is_admin": 0, "is_use": 1, "reg_time": "2022-01-11T07:25:21.000Z" }, { "user_id": 30, "username": "admin1", "nickname": null, "user_pic": null, "is_admin": 0, "is_use": 1, "reg_time": "2022-01-11T06:44:05.000Z" }, { "user_id": 26, "username": "admin", "nickname": null, "user_pic": null, "is_admin": 0, "is_use": 1, "reg_time": "2022-01-11T06:35:04.000Z" }, { "user_id": 16, "username": "123", "nickname": null, "user_pic": null, "is_admin": 0, "is_use": 1, "reg_time": "2022-01-11T06:06:24.000Z" } ] }返回结果状态码状态码含义说明数据模型200OK成功Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenonenone» msgstringtruenonenone» totalintegertruenonenone» data[object]truenonenone»» user_idintegertruenonenone»» usernamestringtruenonenone»» nicknamestring¦nulltruenonenone»» user_picstring¦nulltruenonenone»» is_adminintegertruenonenone»» is_useintegertruenonenone»» reg_timestringtruenonenonePOST 用户注册POST /api/reguserBody 请求参数type: object properties: username: type: string description: 用户名 password: type: string description: 密码 required: - username - password 请求参数名称位置类型必选说明bodybodyobjectfalsenone» usernamebodystringtrue用户名» passwordbodystringtrue密码返回示例成功{ "status": 0, "msg": "注册成功!" }返回结果状态码状态码含义说明数据模型200OK成功Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenonenone» msgstringtruenonenonePOST 文件上传POST /my/uploadsBody 请求参数type: object properties: avatar: type: string description: 要上传的文件信息 format: binary 请求参数名称位置类型必选说明Authorizationheaderstringtrue身份认证bodybodyobjectfalsenone» avatarbodystring(binary)false要上传的文件信息返回示例成功{ "status": 0, "fileInfo": { "fieldname": "avatar", "originalname": "g.png", "encoding": "7bit", "mimetype": "image/png", "destination": "./uploads", "filename": "avatar-2022-01-20 14-43-33.png", "path": "/uploads/avatar-2022-01-20 14-43-33.png", "size": 19531 }, "msg": "上传成功!" }返回结果状态码状态码含义说明数据模型200OK成功Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenonenone» fileInfoobjecttruenonenone»» fieldnamestringtruenonenone»» originalnamestringtruenonenone»» encodingstringtruenonenone»» mimetypestringtruenonenone»» destinationstringtruenonenone»» filenamestringtruenonenone»» pathstringtruenonenone»» sizeintegertruenonenone» msgstringtruenonenonePUT 更新用户基本资料PUT /my/updateInfoBody 请求参数type: object properties: nickname: type: string description: 昵称 user_pic: type: string description: 头像 is_admin: type: string description: 是否管理员 is_use: type: string description: 账号是否可用 请求参数名称位置类型必选说明user_idquerystringtrue用户idAuthorizationheaderstringtruetokenbodybodyobjectfalsenone» nicknamebodystringfalse昵称» user_picbodystringfalse头像» is_adminbodystringfalse是否管理员» is_usebodystringfalse账号是否可用返回示例成功{ "status": 0, "msg": "修改成功!" }返回结果状态码状态码含义说明数据模型200OK成功Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenonenone» msgstringtruenonenonePUT 修改密码PUT /my/updatePwdBody 请求参数type: object properties: currentPwd: type: string description: 当前密码 editPwd: type: string description: 要修改的密码 repeatPwd: type: string description: 重复密码 required: - currentPwd - editPwd - repeatPwd 请求参数名称位置类型必选说明user_idquerystringtrue用户idAuthorizationheaderstringtruetokenbodybodyobjectfalsenone» currentPwdbodystringtrue当前密码» editPwdbodystringtrue要修改的密码» repeatPwdbodystringtrue重复密码返回示例返回结果状态码状态码含义说明数据模型200OK失败Inline返回数据结构状态码 200名称类型必选约束说明» statusintegertruenonenone» msgstringtruenonenone数据模型
2022年01月20日
102 阅读
4 评论
1 点赞
2022-01-20
09-博客api接口项目实战之自动化导入路由
使用自动化导入路由使用自动化导入路由文件,就不需要自己一个一个导入了更改路由文件命名格式,以api开头 或者 以 my开头,用于判断是否需要token认证比如 router/user.js 改成 apiUser.js文件 upload.js 改成 myUpload.js封装自动化导入函数其实就是用到node的fs模块,使用fs.readdirSync()方法获取目录下的所有文件名称组成的数组,拼接路径,然后使用require()导入,再使用app.use()进行路由注册// utils/createRouter.js const fs = require('fs') /** * 挂载路由函数 * @param {object} app serve实例 * @param {string} path 路由文件所在目录绝对路径 * @param {string} dir 路由文件所在目录 */ const cerateRouter = (app, path, dir) => { const routerDir = fs.readdirSync(dir) // 所有的路由文件名称组成的数组 routerDir.forEach(item => { const url = path + '\\' + item const router = require(url) item.startsWith('api') ? app.use('/api', router) : app.use('/my', router) }) } module.exports = cerateRouter 如何使用?在app.js中 导入函数并执行const createRouter = require('./utils/createRouter') // 挂载路由函数 // __dirname是一个node变量,表示当前执行文件所在目录 createRouter(app, __dirname + '\\router', 'router') // 调用,并传参
2022年01月20日
35 阅读
0 评论
0 点赞
2022-01-20
08-博客api接口项目实战之文件上传
实现上传文件功能需要字段 avatar 为文件信息对象限定,只能上传 jpg jpeg png gif 格式的图片有必要的话可以设置下文件上传的文件大小需要用到的npm包:multer 用来上传文件multer官网文档:https://github.com/expressjs/multer/blob/master/doc/README-zh-cn.md实现步骤1. 安装multer包npm install multer2. 创建路由在router下新建upload.js文件const { Router } = require('express') const router = Router() const dateFormat = require('../utils/dateFormat') const uploadHandler = require('../router_handler/upload') const multer = require('multer') // 上传文件所需要的包 // 设置文件存储的格式,以及位置 const storage = multer.diskStorage({ destination: function (req, file, cb) { // 设置文件存储的位置 cb(null, './uploads') }, filename: function (req, file, cb) { // 设置文件名的格式 // 获取文件后缀名 const prefix = file.originalname.split('.')[file.originalname.split('.').length - 1] cb(null, file.fieldname + '-' + dateFormat(new Date(), 'yyyy-MM-dd hh-mm-ss') + '.' + prefix) } }) // 设置文件过滤器 用来过滤不符合条件的文件 const fileFilter = (req, file, cb) => { // 设置文件格式 const type = ['image/png', 'image/jpeg', 'image/jpg', 'image/gif'] if (type.includes(file.mimetype)) return cb(null, true) cb(new Error('上传失败,不支持的文件格式'+file.mimetype), false) } const upload = multer({ storage, fileFilter, }) // 上传文件 router.post('/uploads', upload.single('avatar'), uploadHandler) module.exports = router 3. 创建路由处理函数在router_handler下创建upload.js // 上传文件 const uploadHandler = (req, res) => { if (!req.file) return res.cc('field:avatar is file and required!') const fileInfo = req.file fileInfo.path = `/uploads/${fileInfo.filename}` res.send({ status: 0, fileInfo, msg: '上传成功!' }) } module.exports = uploadHandler4. 测试接口: http://127.0.0.1:8086/my/uploads请求路径:/my/uploads请求方法:post请求头:Authorization: 登录token请求参数:请求参数参数类型参数说明备注avatarfile文件对象信息必填
2022年01月20日
34 阅读
0 评论
0 点赞
2022-01-20
07-博客api接口项目实战之获取用户列表
实现用户列表获取功能接口所需字段:fuzzy 查询字符串 pageNum 页码 pageSize 每页显示条数查询字符串查的是username用户名和nickname昵称,允许为空页码默认为1,不允许传空,非必填每页条数默认为10,不允许传空,非必填实现步骤1. 创建字段校验对象// schema/user.js // 获取用户列表 const getUser = { query: { fuzzy: joi.string().allow(''), pageNum: joi.number().integer().min(1).disallow(''), pageSize: joi.number().integer().min(1).disallow(''), sortKey: joi.string().disallow(''), sortValue: joi.string().disallow('') } } module.exports = { reg_login, updatePwd, updateInfo, getUser }2. 封装自定义sql执行函数// utils/db.js /** * 通过sql语句查询 * @param {String} sql sql语句 **/ const selectDataBySql = function (sql) { return new Promise((resolve, reject) => { db.query(sql, (err, res) => { if (err) { resolve({status: 1, msg: 'sql语句查询失败', data: null, err}) } else { resolve({status: 0, msg: 'sql语句查询成功', data: res, err: null}) } }) }) } module.exports = { addData, updateData, selectData, selectDataBySql // 向外共享sql语句查询函数 }3. 创建路由// router/userInfo.js const { updatePwd, updateInfo, getUserList } = require('../router_handler/userinfo') // 获取用户列表 router.get('/getUserListPage', validateJoi(userSchema.getUser), getUserList)4. 创建路由处理函数如果传了pageNum 就用 pageNum 否则默认1如果传了pageSize 就用pageSize 否则默认10fuzzy 不传或空 查询所有用户列表,传了查询username或者nickname为fuzzy的用户列表sortKey排序字段,不传不排序sortValue排序规则,升序还是降序// router_handler/userInfo.js // 获取用户列表 const getUserList = async (req, res) => { let {fuzzy, pageNum, pageSize, sortKey, sortValue} = req.query if (!pageNum) pageNum = 1 if (!pageSize) pageSize = 10 let totalSql = `select count(*) AS total from users` // 查询所有数据总条数 let selectSql = `select user_id,username,nickname,user_pic,is_admin,is_use,reg_time from users` // 查询表数据 let total = 0 // 总条数 if (fuzzy) { // 查询用户名或昵称为fuzzy的用户列表 totalSql += ` where username like '%${fuzzy}%' or nickname like '%${fuzzy}%'` selectSql += ` where username like '%${fuzzy}%' or nickname like '%${fuzzy}%'` } // 添加分页和排序 根据注册时间 添加limit分页 if (sortKey && sortValue) selectSql += ` order by ${sortKey} ${sortValue} limit ${(pageNum - 1) * pageSize},${pageSize}` selectSql += ` limit ${(pageNum - 1) * pageSize},${pageSize}` // 不排序 // 执行sql语句 const totalRes = await db.selectDataBySql(totalSql) if (totalRes.status !== 0) return res.cc(totalRes.err) total = totalRes.data[0].total // 总条数 const sqlectRes = await db.selectDataBySql(selectSql) if (sqlectRes.status !== 0) return res.cc(sqlectRes.err) const userList = sqlectRes.data res.send({ status: 0, msg: '获取用户列表成功!', total, data: userList }) } module.exports = { updatePwd, updateInfo, getUserList }5. 测试接口:http://127.0.0.1:8086/my/getUserListPage请求路径:/my/updateInfo请求方法:get请求头:Authorization: 登录token请求参数:请求参数参数类型参数说明备注fuzzyString查询字符串非必填 允许为空pageNumNumber页码非必填 非空 默认 1pageSizeNumber每页显示条数非必填 非空 默认 10sortKeyString排序字段非必填 非空sortValueString排序值非必填 非空 (desc 降序、asc升序)
2022年01月20日
41 阅读
0 评论
0 点赞
2022-01-20
06-博客api接口项目实战之数据库方法封装
使用promise封装数据库操作方法在utils文件夹中创建db.js1. 根据字段查询数据// db.js /** * 根据某个字段查询数据 * @param {String} storeName 表名 * @param {String} key 字段 * @param {String} value 字段值 **/ const selectData = function (storeName, key, value) { return new Promise((resolve, reject) => { const sql = `select * from ${storeName} where ${key} = ${value}` console.log(sql); db.query(sql, (err, res) => { if (err) { resolve({status: 1, msg: '查询失败', data: null, err}) } else { resolve({status: 0, msg: '查询成功', data: res, err: null}) } }) }) }2. 根据字段更新数据// db.js /** * 根据某个字段更新数据 * @param {String} storeName 表名 * @param {Object} data 要更新的数据 * @param {String} key 字段 * @param {String} value 字段值 **/ const updateData = function (storeName, data, key, value) { return new Promise((resolve, reject) => { const sql = `update ${storeName} set ? where ${key} = ${value}` db.query(sql, data, (err, res) => { if (err) { resolve({status: 1, msg: '更新数据失败', data: null, err}) } else { resolve({status: 0, msg: '更新数据成功', data: res, err: null}) } }) }) }3. 给某个表中添加数据// db.js /** * 添加数据 * @param {String} storeName 表名 * @param {Object} data 要插入的数据 * @param **/ const addData = function (storeName, data) { return new Promise((resolve, reject) => { const sql = `insert into ${storeName} set ?` db.query(sql, data, (err, res) => { if (err) { resolve({status: 1, msg: '插入数据失败', data: null, err}) } else { resolve({status: 0, msg: '插入数据成功', data: res, err: null}) } }) }) }我在这里都用的是resolve,你也可以用reject,reject抛出,需要使用.catch捕获使用实例:修改用户基本信息处理函数 router_handler/userInfo.js// 更新用户基本资料 const updateInfo = async (req, res) => { const user_id = req.query.user_id const userInfo = req.body if (JSON.stringify(userInfo) === '{}') return res.cc('请填写要修改的参数!') // 查询 const selectData = await db.selectData('users', 'user_id', user_id) if (selectData.status !== 0) return res.cc(selectData.err) if (selectData.data.length !== 1) return res.cc('该用户不存在!') // 更新 const updateData = await db.updateData('users', userInfo, 'user_id', user_id) if (updateData.status !== 0) return res.cc(updateData.err) const {data: result} = updateData if (result.affectedRows !== 1) return res.cc('更新用户信息失败,请重试!') res.send({ status: 0, msg: '修改成功!' }) }其他路由处理函数,自己看着调整,后续就直接使用promise操作数据库,根据实际情况封装对应的数据库操作方法
2022年01月20日
36 阅读
0 评论
0 点赞
2022-01-20
05-博客api接口项目实战之基本信息修改
实现用户基本信息修改功能允许修改 昵称(nickname) 头像(user_pic) 是否管理员(is_admin) 账号是否可用(is_use),需要进行token认证实现步骤:1. 创建字段校验对象// schema/user.js // 修改用户基本信息 const updateInfo = { query: { user_id }, body: { nickname: joi.string().max(10).disallow(''), user_pic: joi.string().allow(''), is_admin: joi.number().integer().valid(0, 1), is_use: joi.number().integer().valid(0, 1) } } module.exports = { reg_login, updatePwd, updateInfo }2. 创建路由// router/userInfo.js const { updatePwd, updateInfo } = require('../router_handler/userinfo') // 更新基本用户信息 router.put('/updateInfo', validateJoi(userSchema.updateInfo), updateInfo)3. 创建路由处理函数检查数据库是否有该用户拿到更新的参数数据,更新该用户的数据库信息// router_handler/userInfo.js // 更新用户基本资料 const updateInfo = (req, res) => { const user_id = req.query.user_id const userInfo = req.body // console.log(req.user); if (JSON.stringify(userInfo) === '{}') return res.cc('请填写要修改的参数!') const sql = `select * from users where user_id = ?` const sql2 = `update users set ? where user_id = ?` db.query(sql, user_id, (err, result) => { if (err) return res.cc(err) if (result.length !== 1) return res.cc('用户不存在!') db.query(sql2, [userInfo, user_id], (err, result) => { if (err) { const arr = err.sqlMessage.split(' ') if (err.code === 'ER_DUP_ENTRY' && arr[arr.length -1] === "'nickname'") return res.cc('昵称重复,换一个叭!') return res.cc(err) } if (result.affectedRows !== 1) return res.cc('更新用户信息失败,请重试!') res.send({ status: 0, msg: '修改成功' }) }) }) } module.exports = { updatePwd, updateInfo }4. 测试接口:http://127.0.0.1:8086/my/updateInfo请求路径:/my/updateInfo请求方法:put请求头:Authorization: 登录token请求参数:请求参数参数类型参数说明备注user_idNumber用户id以query参数携带必填nicknameString用户昵称非必填 非空 10位user_picString用户头像非必填 允许为空is_adminNumber是否管理员非必填 只能是0 或 1is_useNumber账号是否可用非必填 只能是0 或 1
2022年01月20日
24 阅读
0 评论
0 点赞
2022-01-20
04-博客api接口项目实战之密码修改
实现密码修改功能在router下新建userinfo.js存放用户信息相关的路由在router_handler下新建userinfo.js存放用户信息相关的路由处理函数密码修改需要进行token认证,需要在请求头携带Authorization字段app.js中在所有路由之前,使用express-jwt中间件解析token1. app.js中添加token解析中间件,在错误级别中间件中添加token解析报错处理... // 在路由之前解析token 排除 api和uploads开头的路由 const expressJWT = require('express-jwt') // 解析token const { jwtConfig } = require('./config') // token配置 记录了 秘钥 有效期 app.use(expressJWT({ secret: jwtConfig.jwtKey, algorithms: ['HS256'] }).unless({ path: [/^\/api/, /^\/uploads/] })) // 托管静态资源 以后上传文件要用 app.use('/uploads', express.static('./uploads')) // 定义错误级别中间件 app.use((err, req, res, next) => { // 字段校验错误捕获 if (err instanceof joi.ValidationError) return res.cc(err) // 捕获身份认证错误 if (err.name === 'UnauthorizedError') { if (err.message === 'jwt expired') return res.cc('token过期,请重新登录!', 401) return res.cc('身份认证失败!', 401) } // 其他错误 res.cc(err) }) ...2. 创建路由// router/userinfo.js const { Router } = require('express') const router = Router() const validateJoi = require('../utils/validate') // 校验数据 const userSchema = require('../schema/user') const { updatePwd } = require('../router_handler/userinfo') // 修改密码 router.put('/updatePwd', validateJoi(userSchema.updatePwd), updatePwd) module.exports = router3. 创建路由处理函数检查数据库中是否有这个人校验当前密码是否正确校验要修改的密码和重复的密码是否相同校验要修改的密码是否和原先密码一致全部通过执行更新密码// router_handler/userinfo.js const bcrypt = require('bcryptjs') // 密码加密和解密 const db = require('../db/index') // 导入数据库对象 // 修改密码 const updatePwd = (req, res) => { const user_id = req.query.user_id const { currentPwd, editPwd, repeatPwd } = req.body const sql = `select * from users where user_id = ?` db.query(sql, user_id, (err, result) => { if (err) return res.cc(err) if (result.length !== 1) return res.cc('用户不存在!') console.log(currentPwd); const compareResult = bcrypt.compareSync(currentPwd, result[0].password) if (!compareResult) return res.cc('当前密码错误!') if (currentPwd === editPwd) return res.cc('不能与原先密码一致!') const sql1 = `update users set ? where user_id = '${user_id}'` const password = bcrypt.hashSync(repeatPwd, 10) db.query(sql1, {password: password}, (err, result) => { if (err) return res.cc(err) if (result.affectedRows !== 1) return res.cc('修改失败,请重试!') res.send({ status: 0, msg: '修改成功!' }) }) }) }4. 测试密码修改接口:http://127.0.0.1:8086/my/updatePwd请求路径:/my/updatePwd请求方法:put请求头:Authorization: 登录token请求参数:请求参数参数类型参数说明备注user_idNumber用户id以query参数携带currentPwdString当前密码 editPwdString要修改的密码 repeatPwdString重复修改的密码
2022年01月20日
30 阅读
0 评论
0 点赞
2022-01-20
03-博客api接口项目实战之用户登录
1. 实现登录功能所需字段:username password所需npm包:jsonwebtoken 生成token express-jwt 解析tokennpm install jsonwebtoken express-jwtexpress-jwt官方文档:https://www.npmjs.com/package/express-jwtjsonwebtoken官方文档:https://www.npmjs.com/package/jsonwebtoken实现步骤定义登录字段校验规则对象// schema/user.js const joi = require('joi') const reg_login = { body: { username: joi.string().alphanum().min(1).max(10).required(), password: joi.string().pattern(/^[\S]{6,12}$/).required() } } module.exports = { reg_login }登录和注册实际上都是校验username 和 password,可以共用,注意修改 用户注册路由验证// 用户注册 router.post('/reguser', validateJoi(userSchema.reg_login), user_handler.regUser)创建路由// router/user.js // 添加登录路由 // 用户登录 router.post('/login', validateJoi(userSchema.reg_login), user_handler.login)创建路由处理函数登录实现:拿到用户名查询数据库有没有此人,没有,返回查无此人,有则继续下一步拿到当前用户在数据库的密码,使用bcrypt和当前输入的密码进行对比,密码错误,返回密码错误,登陆失败,否则进行下一步使用jsonwebtoken创建token,返回登陆成功以及token,再加上用户的基本信息// router_handler/user.js // 登录 const login = (req, res) => { const userInfo = req.body const sql = `select * from users where username = ?` // 查询用户在不在数据库中 db.query(sql, userInfo.username, (err, result) => { if (err) return res.cc(err) if (result.length !== 1) return res.cc('登陆失败,查无此人!') // TODO:判断用户输入的登录密码是否和数据库中的密码一致 const compareResult = bcrypt.compareSync(userInfo.password, result[0].password) if (!compareResult) return res.cc('密码错误,登录失败!') const user = { ...result[0], password: '' } // 将查询到的用户信息的密码剔除 // 创建token jwt.sign(加密信息, 加密秘钥, 有效期) const token = jwt.sign(user, 'cjaliwbfsia1sa', {expiresIn: '24h'}) res.send({ status: 0, msg: '登陆成功!', userInfo: user, token: 'Bearer ' + token }) }) }可以将jwt加密秘钥和有效期抽离成一个文件 config.js,在根目录创建// config.js const jwtConfig = { jwtKey: 'cjaliwbfsia1sa', // jwt秘钥 expiresIn: '24h' // token有效期 } module.exports = { jwtConfig }哪里要用就引入// 修改 登录路由 const { jwtConfig } = require('../config') const token = jwt.sign(user, jwtConfig.jwtKey, {expiresIn: jwtConfig.expiresIn})测试登录接口: http://127.0.0.1:8086/api/login请求路径:/api/login请求方法:post请求参数:请求参数参数类型参数说明备注usernameString用户名不能为空 长度1-10位passwordString密码不能为空 长度6-12位
2022年01月20日
31 阅读
0 评论
0 点赞
1
2
3
4
...
7