Flutter实现短信验证码监控与转发

验证码提取转发应用

1. 前言

前段时间,我基于deepseek制作了一个基于小红书的自动推文生成发送工作流。然而,先前制作的windows端的工作流到小红书发布时显得异常繁琐,原先的思路是在手机接收到验证码后进入系统进行人为输入,这显然太麻烦了。同时,这一问题当部署到linux服务器上时显得尤为突出,这与自动化的理念显然有些背道而驰。因此,我决定基于flutter制作一个验证码提取转发应用,将手机短信验证码提取出来,通过http接口转发给工作流,从而实现自动化的工作流。

2.开发环境

  • IDE:VSCode
  • 语言:Dart 3.7.0
  • 框架:Flutter 3.29.0

3. 实现思路

3.1 验证码提取

flutter中存在大量不错的第三方短信处理库,例如flutter_sms_inbox, sms_v2, sms_receiver等,但经过测试,许多库在当前开发环境下存在许多问题,因此我最终选择了sms_advanced库进行短信处理。

sms_advanced提供了querySms()这样一个方法,这个方法可以根据条件进行短信查询。以下是方法源码:

dart 复制代码
/// Query a list of SMS
Future<List<SmsMessage>> querySms({
    int? start,
    int? count,
    String? address,
    int? threadId,
    List<SmsQueryKind> kinds = const [SmsQueryKind.Inbox],
    bool sort = true}) async {
    List<SmsMessage> result = [];
    for (var kind in kinds) {
        result.addAll(await _querySmsWrapper(
        start: start,
        count: count,
        address: address,
        threadId: threadId,
        kind: kind,
    ));
}
if (sort == true) {
    result.sort((a, b) => a.compareTo(b));
}
return (result);
}

可以看到,querySms()方法可以接受多个参数,其中address参数可以指定短信发送者的手机号,kinds参数默认为SmsQueryKind.Inbox,即从收件箱获取短信,从而实现短信提取。然后使用正则表达式对短信内容进行匹配,提取出验证码。

dart 复制代码
if (messages.isNotEmpty) {
    // 获取第一条短信
    SmsMessage firstMessage = messages.first;
    String? messageBody = firstMessage.body;

    // 使用正则表达式匹配验证码,假设验证码是 6 位数字
    RegExp regex = RegExp(r'\d{6}');
    Match? match = regex.firstMatch(messageBody!);

    if (match != null) {
        String smsCode = match.group(0)!;
        // 发送验证码到 API
        result = await _sendCodeToAPI(smsCode);
    } else {
        result = '未在短信中找到验证码';
    }
} else {
    result = '未找到短信';
}

但后面我发现小红书的验证码发送者手机号并非固定,因此我选择制作一个多条件筛选器。在条件筛选中,我选择先根据手机号做一次短信筛选,如果没有找到,则根据短信内容做一次筛选,如果还是没有找到,则返回未找到短信。这样用户就可以在仅知道验证码发送应用名称的情况下,不填写发送者手机号,获取到短信并提取到验证码。实现代码如下:

dart 复制代码
SmsQuery query = SmsQuery();
List<SmsMessage>? messages = await query.querySms(
    address: _phoneNumber,
    kinds: [SmsQueryKind.Inbox],
); // 获取收件箱中的短信
if (messages.isEmpty) {
    List<SmsMessage>? messages = await query.querySms(
    kinds: [SmsQueryKind.Inbox],
    ); // 根据条件二进行查询
    for (SmsMessage message in messages) {
        if (message.body?.contains(_targetApp) ?? false) {
            final code = _extractCode(message.body);
            if (code != null) {
                return await _sendCodeToAPI(code);
            }
        }
    }
}

3.2 验证码转发

验证码转发是将提取到的验证码通过http接口转发给工作流。这里我选择使用http库进行http请求,实现代码如下:

dart 复制代码
Future<String?> _sendCodeToAPI(String code) async {
    try {
        final response = await http.post(
            Uri.parse(_apiEndpoint),
            headers: {'Content-Type': 'application/json'},
            body: jsonEncode({'code': code}),
        );
        if (response.statusCode != 200) {
            return ('Failed to send code: ${response.statusCode}');
        } else {
            return 'Successfully Code sent ';
        }
    } catch (e) {
        return ('Error sending code: $e');
    }
}

3.3 线程通信

由于验证码提取是一个耗时操作,因此我选择将其放在一个子线程中执行,以避免阻塞主线程。这里我选择使用flutter的Isolate进行线程通信。同时,为了更新监控状态并控制监控开始和停止,设计了两个Port,分别是mainpPort和isolatePort。mainpPort用于向接收子线程的监控状态消息,实现监控状态的实时更新;isolatePort用于接收主线程发来的启停信息,当点击停止监控后,由父线程告知子线程停止作业。实现代码如下:

MainPort:

dart 复制代码
// 在主isolate的接收端口监听中添加状态更新
void _initReceivePort() {
_receivePort = ReceivePort();
_receivePort.listen((message) {
    if (message is String) {
    if (message == 'isolate_stopped') {
        // 处理isolate退出通知
        if (mounted) {
        setState(() {
            _isolate = null;
            isMonitoring = false;
            buttonText = '开始监控';
        });
        }
    } else {
        setState(() => response = message);
        if(message == 'Successfully Code sent') {
        _stopIsolate();
        }
    }
    } else if (message is SendPort) {
    _isolateSendPort = message;
    }
});
}

IsolatePort:

dart 复制代码
static void _monitorSmsInBackground(List<dynamic> args) async {
    final rootIsolateToken = args[4] as RootIsolateToken;
    BackgroundIsolateBinaryMessenger.ensureInitialized(rootIsolateToken);

    final apiEndpoint = args[0] as String;
    final targetApp = args[1] as String;
    final phoneNumber = args[2] as String;
    final mainSendPort = args[3] as SendPort;
    final smsHandler = SMSHandler(apiEndpoint, targetApp, phoneNumber);
    final controlPort = ReceivePort();
    mainSendPort.send(controlPort.sendPort);

    final stopCompleter = Completer<void>();
    controlPort.listen((message) {
      if (message == 'stop') {
        stopCompleter.complete();
      }
    });

    try {
      while (!stopCompleter.isCompleted) {
        final value = await smsHandler.initSMSListener().timeout(
          const Duration(seconds: 1),
          onTimeout: () => null,
        );
        print(value);
        mainSendPort.send(value);

        if (stopCompleter.isCompleted) break;
      }
    } finally {
      controlPort.close();
      mainSendPort.send('isolate_stopped'); // 添加退出通知
    }
}

StopIsolate:

dart 复制代码
// 修改 _stopIsolate 方法,仅发送停止信号,不强制终止Isolate
void _stopIsolate() {
    if (_isolate != null) {
        _isolateSendPort?.send('stop');
    }
}

4. 总结

总体来说,整体项目还是挺简单的。主要就是利用flutter的插件进行短信的监听,然后通过正则表达式提取验证码,最后通过http接口将验证码发送给工作流。但因为初次学习flutter,许多地方没有做详细的优化,仅仅实现了整体功能。工程代码放在github上,有兴趣的可以看看:verify_code_app

相关推荐
匹马夕阳17 分钟前
ollama本地部署DeepSeek-R1大模型使用前端JS调用的详细流程
人工智能·ai·js
修昔底德29 分钟前
费曼学习法12 - 告别 Excel!用 Python Pandas 开启数据分析高效之路 (Pandas 入门篇)
人工智能·python·学习·excel·pandas
歌刎34 分钟前
从 Transformer 到 DeepSeek-R1:大型语言模型的变革之路与前沿突破
人工智能·深度学习·语言模型·aigc·transformer·deepseek
西猫雷婶35 分钟前
神经网络|(十二)|常见激活函数
人工智能·深度学习·神经网络
go546315846535 分钟前
基于深度学习的静态图像穿搭美学评估与优化建议系统的基本实现思路及示例代码
人工智能·深度学习
li1581726041437 分钟前
T41LQ专为人工智能物联网(AIoT)应用设计,适用于智能安防、智能家居、机器视觉等领域 软硬件资料+样品测试
人工智能·物联网·智能家居
liruiqiang0538 分钟前
神经网络 - 激活函数(Swish函数、GELU函数)
人工智能·深度学习·神经网络·机器学习
轻松Ai享生活1 小时前
手把手教大家如何微调DeepSeek-R1模型,让这个原本冷冰冰的AI开始学会察言观色、妙语连珠
人工智能
邹霍梁@开源软件GoodERP1 小时前
【AI+智造】基于阿里云Ubuntu24.04系统,使用Ollama部署开源DeepSeek模型并集成到企业微信
人工智能·数据分析·制造
uesowys1 小时前
阿里云 | 快速在企业微信中集成一个AI助手
人工智能·阿里云·企业微信·智能体应用