搭建一个可以发送邮箱验证码的接口,内含前端处理 接口返回、请求处理

环境搭建

在node安装好的情况下(一般vue环境有的node也有 没有可以使用win+r回车输入node -v 有版本号则已经安装好
找一个空文件夹作为此项目文件夹 点击上面的地址栏输入cmd回车
输入npm init -y
再输入npm install nodemailer安装发送邮件的插件

在这里插入图片描述

环境配置

使用vscode打开这个程序
在vscode中的终端输入npm install
命令完成后项目会多个json文件,里面是插件还有一些基础环境的版本配置
在文件夹根目录新建servers.js

const express = require('express');
const nodemailer = require('nodemailer');const app = express();
const port = 3000;
const content = 123456;// 创建发送邮件的接口
app.get('/send-email', (req, res) => {console.log(req)console.log(bodyParser)console.log(123123);// 创建邮件传输对象let transporter = nodemailer.createTransport({service: '163', // 运营商选择   qq   网易port: 465,secure: true, // true for 465, false for other portsauth: {user: "123123@163.com", // 发送方的邮箱pass: 'ABCDSD' //   pop3 授权码 **百度如何获取吧**}});// 生成六位随机数作为验证码const code = Math.floor(100000 + Math.random() * 900000);let mailOptions = {from: '"发送人" <123123@163.com>', // 发送方邮箱to: "123123@qq.com", // list of receiverssubject: '欢迎注册√', // Subject linetext: `${content}`, // plain text bodyhtml: `欢迎注册xxx系统,验证码为:<h1 style="background-color:white;">${code}<h1>,有效期为五分钟.` // html body   }// 发送邮件transporter.sendMail(mailOptions, (error, info) => {if (error) {console.log(error);res.status(500).send({success: false,data: {}});} else {console.log(mailOptions)console.log('Email sent: ' + info.response);res.send({//返回值success: true,data: {code: 123123}});}});
});// 启动服务器
app.listen(port, () => {console.log(`Server running on port ${port}`);
});

是的,我把后面的代码也贴上了 其实具体只需要添加启动服务器 端口设置,使用一个测试接口即可 设置完之后先配置自己的邮箱

配置邮箱

打开自己的网易邮箱或者是qq邮箱
点击设置
在这里插入图片描述
打开这个
在这里插入图片描述
在里面打开这两个设置
在这里插入图片描述
通过验证之后保存好自己的pop3授权码,只显示一次 ,找不到了自行百度解决

配置好之后将授权码、自己的邮箱 、要发送的邮箱都进行配置
在这里插入图片描述

配置好之后启动

启动

终端中输入
node server.js
可以看到终端显示
{//返回值
success: true,
data: {
code: code
}
});
这样的,这就是返回值。
这里面的code就是生成的六位数随机码,通过接口传给前端
尝试使用浏览器
里边输入http://localhost:3000/send-email回车能看到同样的有返回值
并且发现收到了验证码

前端配置

前端如果使用ajax直接键入接口api即可,即http://localhost:3000/send-email

思路

实现邮箱验证码是从你的数据库获得id对应的邮箱并填充(不建议),或者用户输入邮箱之后先进行校验 校验结束之后用户可以点击发送验证码 ,点击按钮开始60秒倒计时 倒计时结束之后才能再次点击,这是最简单的防抖动
点击之后调用接口,发送邮箱号,然后咱们刚刚写好的接口对这个邮箱号进行获取,获取之后配置好邮箱 随机生成六位验证码,其实就是生成一个100000到999999的数而已 ,配置到邮箱之后发送邮箱,接口返回一个验证码给前端 前端获取验证码之后放入pinia中,然后和用户键入的六位验证码对比,如果用户输入的不是六位验证码或者是其他数字,则提醒输入六位数字

待解决问题

1、接口返回的验证码很容易被截取,也就是数据安全收到威胁 ,现在还没有更好的办法
2、用户输入id的时候,是直接获取其邮箱i地址还是由用户直接输入存在争议,因为邮箱和账号并没有绑定,如果绑定了必然要在注册页面对邮箱进行绑定,绑定的时候又要发送验证码。算是我之前考虑不周

前端配置60秒倒数

设置验证码和邮箱校验

/** 6位数字验证码正则 */
export const REGEXP_SIX = /^\d{6}KaTeX parse error: Undefined control sequence: \w at position 40: …nst EMAIL = /^[\̲w̲-]+(\.[\w-]+)*@…/;

/** 忘记密码校验 */
const updateRules = reactive({

verifyCode: [
{
validator: (rule, value, callback) => {
if (value === “”) {
callback(new Error(“请输入验证码”));
} else if (!REGEXP_SIX.test(value)) {
callback(new Error(“请输入六位数字”));
} else {
callback();
}
},
trigger: “blur”
}
],
email: [
{
validator: (rule, value, callback) => {
if (value === “”) {
callback(new Error(“请输入邮箱”));
} else if (!EMAIL.test(value)) {
callback(new Error(“不正确的邮箱格式”));
} else {
callback();
}
},
trigger: “blur”
}
],
password: [
{
validator: (rule, value, callback) => {
if (value === “”) {
callback(new Error(“请输入密码”));
} else if (!REGEXP_PWD.test(value)) {
callback(new Error(“密码格式应为8-18位数字、字母、符号的任意两种组合”));
} else {
callback();
}
},
trigger: “blur”
}
]
});

整个ts文件代码

import { reactive } from "vue";
import type { FormRules } from "element-plus";import { useUserStoreHook } from "@/store/modules/user";
/** 密码正则(密码格式应为8-18位数字、字母、符号的任意两种组合) */
export const REGEXP_PWD =/^(?![0-9]+$)(?![a-z]+$)(?![A-Z]+$)(?!([^(0-9a-zA-Z)]|[()])+$)(?!^.*[\u4E00-\u9FA5].*$)([^(0-9a-zA-Z)]|[()]|[a-z]|[A-Z]|[0-9]){8,18}$/;
/** 6位数字验证码正则 */
export const REGEXP_SIX = /^\d{6}$/;
/** 邮箱正则 */
export const EMAIL = /^[\w-]+(\.[\w-]+)*@([\w-]+\.)+[a-zA-Z]{2,7}$/;/** 登录校验 */
const loginRules = reactive<FormRules>({password: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入密码"));}//  else if (!REGEXP_PWD.test(value)) {//   callback(new Error(" 密码格式应为8-18位数字、字母、符号的任意两种组合"));// }else {callback();}},trigger: "blur"}],email: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入邮箱"));} else if (!EMAIL.test(value)) {callback(new Error("不正确的邮箱格式"));} else {callback();}},trigger: "blur"}],verifyCode: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入验证码"));} else if (useUserStoreHook().verifyCode !== value) {callback(new Error("请输入正确的验证码"));} else {callback();}},trigger: "blur"}]
});/** 忘记密码校验 */
const updateRules = reactive<FormRules>({verifyCode: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入验证码"));} else if (!REGEXP_SIX.test(value)) {callback(new Error("请输入六位数字"));} else {callback();}},trigger: "blur"}],email: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入邮箱"));} else if (!EMAIL.test(value)) {callback(new Error("不正确的邮箱格式"));} else {callback();}},trigger: "blur"}],password: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入密码"));} else if (!REGEXP_PWD.test(value)) {callback(new Error("密码格式应为8-18位数字、字母、符号的任意两种组合"));} else {callback();}},trigger: "blur"}]
});// else if (REGEXP_PWD.test(value)) {//!REGEXP_PWD.test(value)
//   callback(
//     new Error("密码格式应为8-18位数字、字母、符号的任意两种组合")
//   );
export { loginRules, updateRules };

整个vue文件

<script setup lang="ts">
import { ref, reactive, watch, onMounted } from "vue";
import Motion from "../utils/motion";
import { message } from "@/utils/message";
import { updateRules } from "../utils/rule";
import type { FormInstance } from 'element-plus';
import { useVerifyCode } from "../utils/verifyCode";
import { useUserStoreHook } from "@/store/modules/user";
import { useRenderIcon } from "@/components/ReIcon/src/hooks";
import Lock from "@iconify-icons/ri/lock-fill";
import Iphone from "@iconify-icons/ep/iphone";
import email from "@iconify-icons/ep/message";
import User from "@iconify-icons/ri/user-3-fill";
import { ReImageVerify } from "@/components/ReImageVerify";
import sendMail from '../utils/send';// sendMail(mail, 123)
//   .then(() => {
//     console.log('邮件发送成功');
//   })
//   .catch(() => {
//     console.log('邮件发送失败');
//   });
import nodemailer from 'nodemailer'
const store = useUserStoreHook();const checked = ref(false);
const loading = ref(false);
const ruleForm = reactive({username: "",email: "",verifyCode: "",password: "",repeatPassword: ""
});
const ruleFormRef = ref<FormInstance>();
const { isDisabled, text } = useVerifyCode();
const repeatPasswordRule = [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入确认密码"));} else if (ruleForm.password !== value) {callback(new Error("两次密码不一致!"));} else {callback();}},trigger: "blur"}
];const onUpdate = async (formEl: FormInstance | undefined) => {loading.value = true;if (!formEl) return;await formEl.validate((valid, fields) => {if (valid) {if (checked.value) {// 模拟请求,需根据实际开发进行修改setTimeout(() => {message("密码修改成功,即将返回登录界面", {type: "success"});loading.value = false;}, 2000);setTimeout(() => {useUserStoreHook().SET_CURRENTPAGE(0);}, 3000);}} else {loading.value = false;return fields;}});
};function onBack() {useVerifyCode().end();useUserStoreHook().SET_CURRENTPAGE(0);}const imgCode = ref("");
watch(imgCode, value => {useUserStoreHook().SET_VERIFYCODE(value);
});onMounted(() => {});
</script><template><el-form ref="ruleFormRef" :model="ruleForm" :rules="updateRules" size="large"><Motion><el-form-item :rules="[{required: true,message: '请输入账号',trigger: 'blur'}]" prop="username"><el-input clearable v-model="ruleForm.username" :placeholder='"账号"' :prefix-icon="useRenderIcon(User)" /></el-form-item></Motion><Motion><el-form-item prop="email"><el-inputclearablev-model="ruleForm.email":placeholder="'邮箱'":prefix-icon="useRenderIcon(email)"/></el-form-item></Motion><Motion :delay="100"><el-form-item prop="verifyCode"><div class="w-full flex justify-between" style="display: flex; justify-content: space-between;"><el-inputclearablev-model="ruleForm.verifyCode":placeholder="'邮箱验证码'"style="width: 19vw; margin-right: 10px;":prefix-icon="useRenderIcon('ri:shield-keyhole-line')"/><el-button:disabled="isDisabled"class="ml-2 align-right"@click="useVerifyCode().start(ruleFormRef, 'email')">{{text.length > 0? text + "秒后重新获取": "获取验证码"}}</el-button></div></el-form-item>
</Motion><Motion :delay="200"><el-form-item prop="password"><el-input clearable show-password v-model="ruleForm.password" :placeholder='"新密码"':prefix-icon="useRenderIcon(Lock)" /></el-form-item></Motion><Motion :delay="250"><el-form-item :rules="repeatPasswordRule" prop="repeatPassword"><el-input clearable show-password v-model="ruleForm.repeatPassword" :placeholder='"确认密码"':prefix-icon="useRenderIcon(Lock)" /></el-form-item></Motion><Motion :delay="350"><el-form-item><div style="display: flex; justify-content: space-between;"><el-button class="w-full" size="default" type="primary" :loading="loading" @click="onUpdate(ruleFormRef)">确定</el-button><el-button class="w-full" size="default" @click="onBack">返回</el-button></div></el-form-item></Motion></el-form>
</template>
<style>
.scroll-container {/* width: 300px; *//* 设置容器的宽度 *//* height: 200px; *//* 设置容器的高度 */overflow: auto;/* 开启滚动功能 */
}.scroll-container::-webkit-scrollbar {width: 0;/* 隐藏滚动条 */
}.align-right {margin-left: auto;
}
</style>

无关的引入报错了就删掉就行

配置发送按钮的方法

点击发送,先校验邮箱 通过之后则传入接口

按钮点击校验邮箱

  <el-button:disabled="isDisabled"class="ml-2 align-right"@click="useVerifyCode().start(ruleFormRef, 'email'),getcode()">{{text.length > 0? text + "秒后重新获取": "获取验证码"}}</el-button>

添加一个getcode()方法
function getcode() {
const email = ruleForm.verifyCode.trim(); // 去除输入的空格

if (ruleForm.verifyCode === “”) {
ElMessage.error(‘请输入邮箱地址’);
return;
}

const emailRegex = /1+(.[\w-]+)*@([\w-]+.)+[a-zA-Z]{2,7}$/;
if (!emailRegex.test(ruleForm.verifyCode)) {
ElMessage.error(‘请输入正确的邮箱格式’);
return;
}

}
还是先校验一下 这次用弹窗的方式提醒用户
弹窗引入的插件是这个
import { ElMessage, formContextKey } from ‘element-plus’;

注意使用了trim()方法 去除空格

配置接口

我之前尝试的直接在地址栏传入信息 ,成功了 ,但是我觉得还是不太妥
const toemail = req.query.toemail; // 获取name参数的值
地址栏输入的样子
http://localhost:3000/send-email?toemail=123123@qq.com
这样能获取到,但是我还是想使用接口里面的请求体发送请求数据
最后还是没有找到放置的地方 只能放在地址栏传输了 ,没关系 邮箱又不是什么私密的东西
我使用的是pure-admin的框架 里面配置的api基本地址
在utils.ts中加一行
export const baseUrlMyApi = (url: string) => /myapi/${url}
在vite.config.ts中加上
“^/myapi/.*”: {
target: “http://localhost:3000”,
changeOrigin: true,
rewrite: path => path.replace(/^/myapi/, “”)
}
这里面的target就是配置的后端接口

前端接口编写

import { getToken } from "@/utils/auth";
import { http } from "@/utils/http";
import { baseUrlMyApi } from "./utils";import { useUserStore } from "@/store/modules/user";
import { useselectedStore } from "@/store/modules/selected";//发送邮箱验证码
export const postemail = (myemail) => {const data = myemailreturn http.request<any>(//baseUrlApi("/Analysis/ResultTempCurrent"),"post",baseUrlMyApi("send-email?toemail=" + data),);
};

这个api接口需要一个参数 参数传的就是用户输入的邮箱地址

校验

在rule.ts中添加校验


/** 忘记密码校验 */
const updateRules = reactive<FormRules>({emailCode: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入验证码"));} else if (!REGEXP_SIX.test(value)) {callback(new Error("请输入六位数字"));} else if (useUserStoreHook().emailcode != value) {console.log(useUserStoreHook().emailcode)console.log(value)callback(new Error("请输入正确的验证码"));}else {callback();}},trigger: "blur"}],email: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入邮箱"));} else if (!EMAIL.test(value)) {callback(new Error("不正确的邮箱格式"));} else {callback();}},trigger: "blur"}],password: [{validator: (rule, value, callback) => {if (value === "") {callback(new Error("请输入密码"));} else if (!REGEXP_PWD.test(value)) {callback(new Error("密码格式应为8-18位数字、字母、符号的任意两种组合"));} else {callback();}},trigger: "blur"}]
});

如果没有这个文件可以就按照我之前写的校验类型编写

测试接口是否能使用

import {postemail} from '@/api/email'postemail(email).then((response) => { 
console.log(response)code.value = response.data.code useUserStoreHook().SET_EMAILCODE(code.value);})

然后可以看到我使用response获取到了接口返回来的code
useUserStoreHook().SET_EMAILCODE(code.value);是我将code赋值给pinia里面的emailcode便于在另一个页面校验

也就是如果鼠标失去这个焦点的时候就开始校验 然后在注册按钮可以再次校验一次 ,如果和用户输入的不一样就报错


  1. \w- ↩︎

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/178013.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

oracle基础系统学习文章目录

oracle基础系统学习——点击标题可跳转对应文章 01.CentOS7静默安装oracle11g02.Oracle的启动过程03.从简单的sql开始04.Oracle的体系架构05.Oracle数据库对象06.Oracle数据备份与恢复07.用户和权限管理08.Oracle的表09.Oracle表的分区10.Oracle的同义词与序列11.Oracle的视图1…

vue3中toRef创建一个ref对象

为源响应式对象上的某个属性创建一个 ref对象, 二者内部操作的是同一个数据值, 更新时二者是同步的 区别ref: 拷贝了一份新的数据值单独操作, 更新时相互不影响 应用: 当要将 某个prop 的 ref 传递给复合函数时&#xff0c;toRef 很有用 父组件代码: <template><…

搜索的剪枝

1.可行性剪枝:如果继续搜下去已经不能得到答案,就return 2.排除等效冗余:在搜索的几个分支中具有完全相同的效果时,选择其中一个走即可 3.最优性剪枝:如果题目要求是最大&#xff0c;最小之类的,没搜到一个解&#xff0c;和之前的解作对比&#xff0c;如果不如之前搜到的&…

js 深度学习(八)

原型及原型链 prototype是function对象的一个属性 它也是一个对象 prototype是定义构造函数构造出的每个对象的公共祖先 所以被该构造函数构造出来的对象 都可以继承原型上的属性和方法 自己有的属性不会去原型上找 方法写在原型上 属性写在构造函数内部 __proto__是实例化以后…

关于easy-es的聚合问题-已解决

es实体类&#xff1a; public class ChemicalES {IndexId(type IdType.CUSTOMIZE)private Long id;HighLightIndexField(fieldType FieldType.TEXT, analyzer "ik_max_word")private String name;IndexField(fieldType FieldType.KEYWORD)private List<Stri…

nginx 配置跨域(小皮面板)

本地开发的时候&#xff0c;前端请求后端&#xff0c;后端不能用域名请求&#xff0c;只能用端口模式&#xff0c;在小皮面板的话就是如下配置&#xff1a; 我的测试项目部署&#xff1a; 前端&#xff1a;http://localhost:8082 后端&#xff1a;http://localhost:8081 前端…

二百零八、Hive——HiveSQL异常:Select查询数据正常,但SQL语句加上group by查询数据为空

一、目的 在HiveSQL的DWD层中&#xff0c;需要对原始数据进行去重在内的清洗&#xff0c;结果一开始其他数据类型的清洗工作都正常&#xff0c;直到碰到转向比数据。 一般的SQL查询有数据&#xff0c;但是加上group by以后就没数据&#xff1b; 一般的SQL查询有数据&#xf…

Python实现WOA智能鲸鱼优化算法优化XGBoost分类模型(XGBClassifier算法)项目实战

说明&#xff1a;这是一个机器学习实战项目&#xff08;附带数据代码文档视频讲解&#xff09;&#xff0c;如需数据代码文档视频讲解可以直接到文章最后获取。 1.项目背景 鲸鱼优化算法 (whale optimization algorithm,WOA)是 2016 年由澳大利亚格里菲斯大学的Mirjalili 等提…

uniapp基础-教程之HBuilderX基础常识篇02

uniapp创建项目时属性多为vue后缀&#xff1b;其中每个文件中都包含了三段式结构分别是template&#xff1b;script&#xff1b;style形势&#xff0c;分别是前端显示的画面以及js和css样式。 template&#xff1a;说大白话就是给别人看的&#xff0c;我们打开页面就可以看到的…

商城系统通过Kafka消息队列,实现订单的处理和状态更新springboot例子解决并发处理、数据一致性等问题

在商城系统中&#xff0c;订单的处理和状态更新是非常关键的部分&#xff0c;需要保证并发处理和数据一致性。使用Kafka消息队列可以很好地解决这些问题。 下面是一个使用Kafka消息队列实现订单处理和状态更新的Spring Boot例子&#xff1a; 1. 添加Kafka依赖 在pom.xml文件…

Jtti:linux中udp怎么判断是否接收到数据?

在Linux中&#xff0c;使用UDP协议进行通信时&#xff0c;可以通过检查套接字(Socket)接收缓冲区中是否有数据来判断是否接收到数据。以下是一个简单的方法&#xff1a; #include <stdio.h> #include <stdlib.h> #include <string.h> #include <arpa/ine…

oracle查询开始时间和结束时间之间的连续月份

SELECT TO_CHAR(ADD_MONTHS(TO_DATE(2023-01,YYYY-MM), ROWNUM - 1), YYYY-MM) AS fmonth FROM DUALCONNECT BY ROWNUM < CEIL(MONTHS_BETWEEN(TO_DATE(2023-11, YYYY-MM), TO_DATE(2023-01,YYYY-MM))1)

附录11-math.h的常见方法

stdlib.h是做数学计算的头文件 目录 1 数学知识 1.1 弧度值/π 角度值/180 1.2 双曲函数 2 math.h 2.1 反余弦值 acos() 2.2 反正弦值 asin() 2.3 反正切值 atan() 2.4 两个数的反正切值 atan2() 2.5 向上取整 ceil() 2.6 余弦值 cos() 2.7 双曲余弦 c…

c++环形队列

c环形队列 c环形队列 c环形队列 #pragma once#include <iostream> #include <vector>/// <summary> /// - 环形队列 /// - 不是线程安全 /// </summary> /// <typeparam name"T"></typeparam> template <typename T> cla…

应用在触摸式面板中的电容式触摸芯片

触摸屏又称为“触控屏”、“触控面板”&#xff0c;是一种可接收触头等输入讯号的感应式液晶显示装置&#xff1b;当接触了屏幕上的图形按钮时&#xff0c;屏幕上的触觉反馈系统可根据预先编程的程式驱动各种连结装置&#xff0c;可用以取代机械式的按钮面板&#xff0c;并借由…

2947. 统计美丽子字符串 I (前缀和)

Problem: 2947. 统计美丽子字符串 I 文章目录 题目思路Code 题目 给你一个字符串 s 和一个正整数 k 。 用 vowels 和 consonants 分别表示字符串中元音字母和辅音字母的数量。 如果某个字符串满足以下条件&#xff0c;则称其为 美丽字符串 &#xff1a; vowels consonants…

【JavaEE初阶】 HTTP协议和使用Fiddler抓包

文章目录 &#x1f38d;HTTP协议是什么&#xff1f;&#x1f340;应用层协议&#xff08;HTTP&#xff09;存在的意义&#x1f384;HTTP 协议的工作过程&#x1f334;HTTP 协议格式&#x1f333;Fiddler抓包工具的使用&#x1f6a9;如何抓HTTPS的包&#xff1f; &#x1f38b;抓…

npm私仓 verdaccio搭建 发布到私仓 使用

1.安装verdaccio # 全局安装 npm install -g verdaccio npm list -g verdaccio //查看verdaccio版本 2. 修改配置文件 vim /root/.config/verdaccio/config.yaml # 修改页面标题 web: title: CloudWise-Verdaccio # 上传npm包大小限制修改 max_body_size: 100mb # 允许…

Vue 文件压缩与解压缩

Vue 文件压缩与解压缩 文章目录 Vue 文件压缩与解压缩0. 安装1. 压缩2. 解压 0. 安装 安装 npm install jszip引入 import JSZip from jszip;使用 1. 压缩 async compressFiles() {// 创建jszip实例const zip new JSZip();// 添加需要压缩的文件const file1 Hello World…

flutter开发实战-当前界面无操作60s返回主页实现

flutter开发实战-当前界面无操作60s返回主页实现 当前界面无操作60s返回主页实现&#xff0c;主要是通过Timer来控制&#xff0c;当监听界面是否有pointerDown时候&#xff0c;如果超过60s仍没有操作&#xff0c;则返回主页。 一、Listener Listener是用来用于调用回调以响应…