C# 微信小程序获取群id

news2025/7/16 2:54:46

前提

有个需求,需要限制小程序的抽奖只能在某个群内,需要知道谁在群里面,但是微信并没有提供谁在群里面的方法,不过提供了获取群id的方法,这样加上限制分享就能保证群里的参加,即时分享出去了,判断来源的时候也不是来自于群了,就不允许用户参与。

需要的方法

wx.login
在这里插入图片描述

wx.getGroupEnterInfo
在这里插入图片描述
加密解密数据
在这里插入图片描述

解密解密官方代码c++、php、node、python

没有java和c#的代码,目前需要c#代码

解密流程

在这里插入图片描述

需要的nuget包

skit.flurlhttpclient.wechat.api.2.35.0

实现代码

微信小程序

wxml

<view style="width: 80%;margin: 0 auto;margin-top: 30rpx;">
  <text>微信群id:</text>
  <text>{{groupId}}</text>
  <view>
  <button bind:tap="copyGroupId">复制</button>
  </view>
</view>

js

// pages/wechatgroupinfo/wechatgroupinfo.js
Page({

  /**
   * 页面的初始数据
   */
  data: {
    groupId: ""
  },

  /**
   * 生命周期函数--监听页面加载
   */
  onLoad(options) {
    wx.setNavigationBarTitle({
      title: '获取微信群信息',
    })
  },

  /**
   * 生命周期函数--监听页面初次渲染完成
   */
  onReady() {
    let that = this;
    wx.login({
      success: (res) => {
        if (res.errMsg == "login:ok") {
          const code = res.code;
          wx.getGroupEnterInfo({
            success(resData) {
              if (resData.errMsg === "getGroupEnterInfo:ok") {
                const iv = resData.iv;
                const encryptedData = resData.encryptedData;
                that.getWechatGroupId(code, iv, encryptedData);
              } else {
                wx.showToast({
                  title: 'wx.getGroupEnterInfo出错',
                })
              }
            },
            fail() {
              wx.showToast({
                title: 'wx.getGroupEnterInfo:' + resData.errMsg,
              })
            }
          })
        } else {
          wx.showToast({
            title: 'wx.login:' + res.errMsg,
          })
        }
      },
      fail: () => {
        wx.showToast({
          title: 'wx.login出错',
        })
      }
    })

  },

  /**
   * 获取微信群id
   * @param {String} code 登录的id
   * @param {String} iv 偏移量
   * @param {String} encryptData 加密的数据
   * @returns 微信群id
   */
  getWechatGroupId(code, iv, encryptData) {
    let that = this;
    wx.request({
      method: "POST",
      url: 'http://localhost:5000/v1/user/GetWechatGroupId',
      data: JSON.stringify({
        code,
        iv,
        encryptData
      }),
      success: (response) => {
        console.log(response);
        if (response.data && response.data.success) {
          that.setData({
            groupId: response.data.response
          })
        } else {
          wx.showToast({
            title: response.data.msg,
          })
        }
      },
      fail: () => {
        wx.showToast({
          title: '后台获取微信群id失败',
        })
      }
    })
  },
  /**
   * 复制微信群组
   */
  copyGroupId() {
    wx.setClipboardData({
      data: this.data.groupId,
      success(res) {
        wx.showToast({
          title: '复制成功',
        })
      }
    })
  },

  /**
   * 生命周期函数--监听页面显示
   */
  onShow() {

  },

  /**
   * 生命周期函数--监听页面隐藏
   */
  onHide() {

  },

  /**
   * 生命周期函数--监听页面卸载
   */
  onUnload() {

  },

  /**
   * 页面相关事件处理函数--监听用户下拉动作
   */
  onPullDownRefresh() {

  },

  /**
   * 页面上拉触底事件的处理函数
   */
  onReachBottom() {

  },

  /**
   * 用户点击右上角分享
   */
  onShareAppMessage() {

  }
})

C#后台

获取微信群id接口

/// <summary>
/// 获取微信的群组id
/// </summary>
/// <returns></returns>
[HttpPost]
public async Task<MessageModel<string>> GetWechatGroupId([FromBody] GetWechatGroupIdParamModel param)
{
    var result = new MessageModel<string>();
    var res = await weChatMiniProgramHelper.GetWechatGroupId(param.Code, param.Iv, param.EncryptData);
    result.Success = !string.IsNullOrEmpty(res);
    result.Msg = result.Success ? "成功" : "失败";
    result.Response = res;
    return result;
}

/// <summary>
/// 单例模式注入
/// </summary>
public class WeChatMiniProgramClient
{
    public WechatApiClient Instance { get; private set; }

    /// <summary>
    /// 构造函数注入
    /// </summary>
    /// <param name="wechatConfig"></param>
    public WeChatMiniProgramClient(IOptions<WechatConfigModel> wechatConfig)
    {
        var options = new WechatApiClientOptions()
        {
            AppId = wechatConfig.Value.MiniProgramAppId,
            AppSecret = wechatConfig.Value.MiniProgramAppSecret,
            MidasAppKey = "",//米大师相关服务 AppKey,不用则不填
            ImmeDeliveryAppKey = "",//即时配送相关服务 AppKey,不用则不填
            ImmeDeliveryAppSecret = ""//即时配送相关服务 AppSecret,不用则不填
        };
        Instance = new WechatApiClient(options);
    }
}

//weChatMiniProgramHelper下的方法

/// <summary>
/// 获取微信群id
/// </summary>
/// <param name="code"></param>
/// <param name="iv"></param>
/// <param name="encryptData"></param>
/// <returns></returns>
public async Task<string> GetWechatGroupId(string code, string iv, string encryptData)
{
    if (string.IsNullOrEmpty(iv) || iv.Length != 24)
    {
        return string.Empty;
    }
    if (string.IsNullOrEmpty(encryptData) || encryptData.Length < 1)
    {
        return string.Empty;
    }
    //WechatApiClient客户端
    var client = weChatMiniProgramClient.Instance;
    try
    {
        var userLoginRequest = new SnsJsCode2SessionRequest();
        userLoginRequest.JsCode = code;
        var loginInfo = await client.ExecuteSnsJsCode2SessionAsync(userLoginRequest);
        if (!loginInfo.IsSuccessful())
        {
            return string.Empty;
        }
        var sessionKey = loginInfo.SessionKey;
        if (string.IsNullOrEmpty(sessionKey) || sessionKey.Length != 24)
        {
            return string.Empty;
        }
        var encryptDataDecode = Convert.FromBase64String(encryptData);
        var jsonStr = AESUtility.DecryptWithCBC(sessionKey, iv, encryptData);
        if (string.IsNullOrEmpty(jsonStr))
        {
            return string.Empty;
        }
        var openGidDic = JsonConvert.DeserializeObject<Dictionary<string, string>>(jsonStr);
        return openGidDic["opengid"];
    }
    catch (Exception ex)
    {
        logger.LogError($"获取微信群id失败\r\n{ex.Message}");
        throw new Exception(ex.Message, ex.InnerException);
    }
}

测试
在这里插入图片描述
在这里插入图片描述
如果直接加密解密会报错Specified initialization vector (IV) does not match the block size for this algorithm,需要先base64解码一下
AESUtility.cs源代码,来自skit.flurlhttpclient.wechat

/// <summary>
/// 基于 CBC 模式解密数据。
/// </summary>
/// <param name="keyBytes">AES 密钥字节数组。</param>
/// <param name="ivBytes">加密使用的初始化向量字节数组。</param>
/// <param name="cipherBytes">待解密数据字节数组。</param>
/// <returns>解密后的数据字节数组。</returns>
public static byte[] DecryptWithCBC(byte[] keyBytes, byte[] ivBytes, byte[] cipherBytes)
{
    if (keyBytes == null) throw new ArgumentNullException(nameof(keyBytes));
    if (ivBytes == null) throw new ArgumentNullException(nameof(ivBytes));
    if (cipherBytes == null) throw new ArgumentNullException(nameof(cipherBytes));

    using (SymmetricAlgorithm aes = Aes.Create())
    {
        aes.Mode = CipherMode.CBC;
        aes.Padding = PaddingMode.PKCS7;
        aes.Key = keyBytes;
        aes.IV = ivBytes;

        using ICryptoTransform transform = aes.CreateDecryptor();
        return transform.TransformFinalBlock(cipherBytes, 0, cipherBytes.Length);
    }
}

/// <summary>
/// 基于 CBC 模式解密数据。
/// </summary>
/// <param name="encodingKey">经 Base64 编码后的 AES 密钥。</param>
/// <param name="encodingIV">经 Base64 编码后的 AES 初始化向量。</param>
/// <param name="encodingCipherText">经 Base64 编码后的待解密数据。</param>
/// <returns>解密后的文本数据。</returns>
public static string DecryptWithCBC(string encodingKey, string encodingIV, string encodingCipherText)
{
    if (encodingKey == null) throw new ArgumentNullException(nameof(encodingKey));
    if (encodingCipherText == null) throw new ArgumentNullException(nameof(encodingCipherText));

    byte[] plainBytes = DecryptWithCBC(
        keyBytes: Convert.FromBase64String(encodingKey),
        ivBytes: Convert.FromBase64String(encodingIV),
        cipherBytes: Convert.FromBase64String(encodingCipherText)
    );
    return Encoding.UTF8.GetString(plainBytes);
}

/// <summary>
/// 基于 CBC 模式加密数据。
/// </summary>
/// <param name="keyBytes">AES 密钥字节数组。</param>
/// <param name="ivBytes">加密使用的初始化向量字节数组。</param>
/// <param name="plainBytes">待加密数据字节数组。</param>
/// <returns>加密后的数据字节数组。</returns>
public static byte[] EncryptWithCBC(byte[] keyBytes, byte[] ivBytes, byte[] plainBytes)
{
    if (keyBytes == null) throw new ArgumentNullException(nameof(keyBytes));
    if (ivBytes == null) throw new ArgumentNullException(nameof(ivBytes));
    if (plainBytes == null) throw new ArgumentNullException(nameof(plainBytes));

    using (SymmetricAlgorithm aes = Aes.Create())
    {
        aes.Mode = CipherMode.CBC;
        aes.Padding = PaddingMode.PKCS7;
        aes.Key = keyBytes;
        aes.IV = ivBytes;

        using ICryptoTransform transform = aes.CreateEncryptor();
        return transform.TransformFinalBlock(plainBytes, 0, plainBytes.Length);
    }
}

/// <summary>
/// 基于 CBC 模式加密数据。
/// </summary>
/// <param name="encodingKey">经 Base64 编码后的 AES 密钥。</param>
/// <param name="encodingIV">经 Base64 编码后的 AES 初始化向量。</param>
/// <param name="plainText">待加密文本。</param>
/// <returns>经 Base64 编码的加密后的数据。</returns>
public static string EncryptWithCBC(string encodingKey, string encodingIV, string plainText)
{
    if (encodingKey == null) throw new ArgumentNullException(nameof(encodingKey));
    if (plainText == null) throw new ArgumentNullException(nameof(plainText));

    byte[] plainBytes = EncryptWithCBC(
        keyBytes: Convert.FromBase64String(encodingKey),
        ivBytes: Convert.FromBase64String(encodingIV),
        plainBytes: Encoding.UTF8.GetBytes(plainText)
    );
    return Convert.ToBase64String(plainBytes);
}

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

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

相关文章

网络层详解

目录 前言 一、IP协议 1、IP协议报头 2、协议字段理解 &#xff08;1&#xff09;4位版本 &#xff08;2&#xff09;4位首部长度 &#xff08;3&#xff09;8位服务类型 &#xff08;4&#xff09;16位总长度 &#xff08;5&#xff09;标识、标志与片偏移 &#xf…

【自控实验】2. 采样控制系统特性

本科课程实验报告&#xff0c;有太多公式和图片了&#xff0c;干脆直接转成图片了 仅分享和记录&#xff0c;不保证全对 通过对二阶连续系统、二阶采样系统和具有保持器的二阶采样系统仿真实验&#xff0c;比较三种系统的特性&#xff0c;加深对采样控制系统的了解 使用matl…

阿里云OSS上传视频,可分片上传

uniappH5实现 阿里云OSS上传视频 示例图&#xff1a; 上传视频完整示例代码&#xff1a; 使用npm安装SDK开发包&#xff0c;安装命令为 npm install ali-oss --save accessKeyId 和 accessKeySecret 还有 bucket 替换成你的就行。 multipartUpload 的第一个入参是&#x…

leetcode 动态规划(最后一块石头的重量II、目标和、一和零)

1049.最后一块石头的重量II 力扣题目链接(opens new window) 题目难度&#xff1a;中等 有一堆石头&#xff0c;每块石头的重量都是正整数。 每一回合&#xff0c;从中选出任意两块石头&#xff0c;然后将它们一起粉碎。假设石头的重量分别为 x 和 y&#xff0c;且 x < …

Redis的设计、实现

数据结构和内部编码 type命令实际返回的就是当前键的数据结构类型,它们分别是:string(字符串)hash(哈希)、list(列表)、set(集合)、zset (有序集合),但这些只是Redis对外的数据结构。 实际上每种数据结构都有自己底层的内部编码实现,而且是多种实现,这样Redis会在合适的…

动手学深度学习-卷积神经网络

卷积神经网络 在前面的章节中&#xff0c;我们遇到过图像数据。这种数据的每个样本都由一个二维像素网格组成&#xff0c;每个像素可能是一个或者多个数值&#xff0c;取决于是黑白还是彩色图像。到目前为止&#xff0c;我们处理这类结构丰富的数据方式还不够有效。我们仅仅通…

[Kubernetes]7. K8s包管理工具Helm、使用Helm部署mongodb集群(主从数据库集群)

上一节讲解了[Kubernetes]6. k8s Pod配置管理ConfigMap & Secret以及传递环境变量的使用,k8s的命名空间以及使用kubens管理命名空间的使用,这里来介绍一下Helm的使用 一.Helm相关介绍 1.介绍 在 kubernetes 系统上部署容器化应用时需要事 先手动编写资源配置清单文件 以…

页面跳转后,默认选中tree节点并高亮显示

1.场景 操作步骤&#xff1a; 1.点击数据连接数&#xff0c;打开弹窗 2.点击连接状态跳转到数据连接模块 3.默认选中tree的数据源id节点 2.代码 参数解释&#xff1a; 3.实现逻辑 首先将id通过组件传参的方式传过去&#xff0c;数据连接接收后&#xff0c;在tree里设置…

Vue+element-china-area-data实现省市区三级联动

安装依赖 npm install element-china-area-data -S cnpm install element-china-area-data -S 引用 import { provinceAndCityData, regionData, provinceAndCityDataPlus, regionDataPlus, CodeToText, TextToCode } from element-china-area-data&#xff1b; provinceAnd…

【UE Niagara学习笔记】06 - 制作火焰喷射过程中飞舞的火星

在上一篇博客&#xff08;【UE Niagara学习笔记】05 - 喷射火焰顶部的蓝色火焰&#xff09;的基础上继续实现喷射火焰的火星的效果。 目录 效果 步骤 一、创建材质实例 二、添加新的发射器 2.1 设置粒子材质 2.2 设置发射器持续生成粒子 2.3 设置粒子生成数量 2.4 设…

Windows7共享文档—开启方法及用户权限设置

使用计算机的朋友&#xff0c;在工作中经常需要在局域网中将文件共享给其他用户&#xff0c;这样其他人可以方便的通过局域网查看&#xff0c;甚至修改这些共享文件。当然&#xff0c;根据文件的重要程度&#xff0c;使用等级不同&#xff0c;不同的用户会赋予不同的权限&#…

使用 schema 库,自定义较复杂的校验方法

目录 一、前置说明1、总体目录2、相关回顾3、本节目标 二、操作步骤1、项目目录2、依赖包安装及说明3、代码实现3、测试代码4、日志输出 三、后置说明1、要点小结2、下节准备 一、前置说明 1、总体目录 《 pyparamvalidate 参数校验器&#xff0c;从编码到发布全过程》 2、相…

记录浏览器莫名其妙部分网页无法访问的一个解决办法

问题描述&#xff1a; 不知道什么原因&#xff0c;浏览器无法访问CSDN了&#xff0c;访问其他网站都可以正常加载。 经电脑网络诊断检测&#xff0c;反馈内容大致为&#xff1a; 资源处于联机状态但未对连接尝试做出响应&#xff0c;远程计算机不接受端口443上的连接。 测试…

如何搭建开源知识库软件AFFiNE并实现公网环境远程协作【内网穿透】

目录 前言 1. 使用Docker安装AFFINE 2. 安装cpolar内网穿透工具 3. 配置AFFINE公网访问地址 4. 实现公网远程访问AFFINE 结语 作者简介&#xff1a; 懒大王敲代码&#xff0c;计算机专业应届生 今天给大家聊聊如何搭建开源知识库软件AFFiNE并实现公网环境远程协作【内网穿…

用java实现Client和Server之间的互相通信

概要&#xff1a;看过我之前文章的人都知道&#xff0c;client和server之间的通信必不可少的就是socket。而java已经帮我们做了很多事情。 创建Server端 第一步&#xff0c;创建ServerSocket 这个从名字上就可以看出来&#xff0c;服务器上的socket 0.0 ServerSocket ser…

ROS2学习笔记三:话题Topic

目录 前言 1 话题简介 2 常用指令 3 RCLCPP实现实现话题 3.1 创建工作空间 3.2 代码编写 3.2.1 发布端编写 3.2.2 发布端编写 前言 ROS2中的一个重要概念是话题&#xff08;Topic&#xff09;。话题是一种通过发布者和订阅者之间进行异步通信的机制。发布者&#xff0…

Java常用类---Math类和Random类

Math类 简介 Java中&#xff0c;Math类包含了用于执行基本数学运算的属性和方法。Math类的方法都被定义为static形式(静态方法)&#xff0c;通过Math类可以直接在主函数中直接调用。 如下图所示&#xff0c;Math.PI等于圆周率π、Math.E等于常量e……等属性和方法。 部分Mat…

主动学习基础-贝叶斯神经网络

引言 传统的深度神经网络一般都有过度自信的问题。 即使我给神经网络提供一个从来没有训练过的类别图像&#xff0c;神经网络也会输出一个类别。比如训练猫狗的分类器&#xff0c;如果你抛出一个人的图像&#xff0c;网络也会将其分类为猫或者狗。 在几乎所有现实世界的问题中…

详解java中ArrayList

目录 前言 一、ArrayList是什么 二、ArrayList使用 1、ArrayList的构造 2 、ArrayList常见操作 3、 ArrayList的遍历 4、 ArrayList的扩容机制 三、来个练习 前言 当你看到这篇文章我觉得很好笑&#xff0c;因为我开始也不懂ArrayList现在轮到你了&#xff0c;嘻嘻嘻&am…

深入解析HubSpot在线客户互动工具:提升客户体验的利器

在数字化时代&#xff0c;客户体验成为企业成功的关键因素之一。HubSpot作为一体化的市场营销、销售和服务平台&#xff0c;其在线客户互动工具扮演着提升客户体验的重要角色。本文将深入探讨HubSpot的在线客户互动工具&#xff0c;包括实时聊天、机器人和社交媒体监控&#xf…