C# 微信小程序获取群id

前提

有个需求,需要限制小程序的抽奖只能在某个群内,需要知道谁在群里面,但是微信并没有提供谁在群里面的方法,不过提供了获取群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);
}

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

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

Vue3 中使用 Vuex 和 Pinia 对比之 Vuex的用法

本文基于 Vue3 的 composition API 来展开 Vuex 和 Pinia 的用法比较 Pinia传送门 Vuex传送门 Vuex 状态管理的核心概念 状态- 驱动应用的数据源&#xff1b;视图 - 以声明方式将状态映射到视图&#xff1b;操作 - 响应在视图上的用户输入导致的状态变化 下面是源自Vuex 官…

每日一题——LeetCode1154.一年中的第几天

方法一 列举法&#xff1a; 用一个数组把每个月份的天数都列举出来 判断闰年&#xff0c;是闰年2月份有29天 循环对当前月份之前的月份天数求和 加上当天月份的天数 var dayOfYear function(date) {let year date.slice(0, 4);let month date.slice(5, 7);let day dat…

第十二章Session

第十二章Session 1.什么是Session2.Session的创建与获取3.session域中数据的存取4.Session超时的控制5.浏览器和session之间关联的技术内幕 1.什么是Session 注意&#xff1a;前面的Cookie是保存在客户端&#xff0c;而session是在服务端的 2.Session的创建与获取 这里Session…

2024几个测试接口的好工具,效率加倍~

作为一名后端程序员&#xff0c;一定要对自己写的接口负责&#xff0c;保证接口的正确和稳定性。因此&#xff0c;接口测试也是后端开发中的关键环节。 但我相信&#xff0c;很多朋友是懒得测试接口的&#xff0c;觉得这很麻烦。一般自己写的接口自己都不调用&#xff0c;而是…

RT-Thread入门笔记4-跑马灯线程实例

RT-Thread操作系统是基于线程调度的多任务系统。 线程状态切换 调度过程是一种完全抢占式的基于优先级的调度算法。 支持8/32/256优先级&#xff0c;其中0表示最高&#xff0c;7/31/255表示最低。最低优先级7/31/255优先级用于空闲线程。 支持以相同优先级运行的线程。 共享时…

如何理解线程池中的参数设计

如何理解线程池中的参数设计 你的线程池的参数怎么配置&#xff1f;线程数量设置多少合理&#xff1f;如何确定一个线程池中的人物已经完成了为什么不建议使用java自带的Executors创建线程池线程池里面的阻塞队列设置多少合理&#xff1f; 考察&#xff1a;了解你对技术的掌握…

springboot摄影跟拍预定管理系统源码和论文

首先,论文一开始便是清楚的论述了系统的研究内容。其次,剖析系统需求分析,弄明白“做什么”,分析包括业务分析和业务流程的分析以及用例分析,更进一步明确系统的需求。然后在明白了系统的需求基础上需要进一步地设计系统,主要包罗软件架构模式、整体功能模块、数据库设计。本项…

如何通过 3 个步骤,管理项目可交付成果?

没有可交付成果&#xff0c;就没有项目。无论是构建软件、公寓、汽车还是其他东西&#xff0c;项目工作都可以定义为实现项目可交付成果。 项目管理中的可交付成果 项目可交付成果是项目要实现的最终结果。“可交付成果 "的内容没有限制&#xff0c;可以是实体产品&…

石大版跳一跳(UPC)

题目描述 还记得微信上那个风靡全国的跳一跳小程序吧&#xff0c;估计曾经也受到不少我校同学的喜爱吧。话说唐克也在玩这款游戏&#xff0c;不过&#xff0c;与一般玩家的境界不一样&#xff0c;唐克并不沉迷于游戏&#xff0c;唐克玩游戏是为了开发游戏&#xff0c;作为中国石…

tiktok_浅谈hook ios之发包x-ss-stub

frida-trace ios手机一部&#xff0c;需要越狱的电脑一台idacrackerXI 目标app&#xff1a; ipa 包&#xff0c;点击前往 密码&#xff1a;8urs 协议分析起始从抓包开始&#xff0c;个人习惯 一般安卓逆向可以直接搜关键词&#xff0c;但是ios 都在 Mach-O binary (reverse…

[JAVA数据结构] 认识 Iterable、Collection、List 的常见方法签名以及含义

目录 (一)Iterable 1. 介绍 2. 常见方法 (二)Collection 1. 介绍 2. 常见方法 (三) List 1. 介绍 2. 常见方法 总结 (一) Iterable 1. 介绍 Iterable接口是Java中的一个接口&#xff0c;它是集合框架中的根接口之一。Iterable接口表示实现了迭代功能&#xff0c;即可以通过迭…

JetCache源码解析——缓存处理

在Java技术体系中&#xff0c;如果想要在不改变已有代码逻辑的情况下&#xff0c;对已有的函数进行功能增强&#xff0c;一般可以使用两种方式&#xff0c;如AOP&#xff08;Aspect Oriented Programming&#xff09;&#xff0c;即面向切面编程&#xff0c;以及代理模式&#…

最详细手把手教你安装 Vivado2019.2

Vivado 是 FPGA 厂商赛灵思公司&#xff08;Xilinx&#xff09;于 2012 年起发布的集成设计环境。 Vivado 2019.2 是 2019 年 Xilinx 推出的 Vivado 最后一个版本&#xff0c;相对稳定&#xff0c;并推出了新式的嵌入式开发平台 Vitis。 软件下载 官网可下载各个版本百度网盘…

Android 通知简介

Android 通知简介 1. 基本通知 图1: 基本通知详情 小图标 : 必须提供,通过 setSmallIcon( ) 进行设置.应用名称 : 由系统提供.时间戳 : 由系统提供,也可隐藏时间.大图标(可选) : 可选内容(通常仅用于联系人照片,请勿将其用于应用图标),通过setLargeIcon( ) 进行设置.标题 : 可选…

一日难再晨及时当勉励 date

文章目录 Linux shell 获取更改系统时间默认输入显示时区世界协调时格式化日期更多信息 Linux shell 获取更改系统时间 … note:: 时光只解催人老&#xff0c;不信多情&#xff0c;长恨离亭&#xff0c;泪滴春衫酒易醒。 - 晏殊《采桑子时光只解催人老》date命令可以用来打印…

12.扩展字典(ExtensionDictionary)

愿你出走半生,归来仍是少年! 环境:.NET FrameWork4.5、ObjectArx 2016 64bit、Entity Framework 6. 在10.扩展数据(XData)中我们讲到每个DbObject有一个XData对象可以存储数据,除此之外每个DbObject对象还有一个ExtensionDictionary(扩展字典)可以进行数据存储。…

P1328 [NOIP2014 提高组] 生活大爆炸版石头剪刀布————C++

目录 [NOIP2014 提高组] 生活大爆炸版石头剪刀布题目背景题目描述输入格式输出格式样例 #1样例输入 #1样例输出 #1 样例 #2样例输入 #2样例输出 #2 提示 解题思路Code调用函数的Code&#xff08;看起来简洁一点&#xff09;运行结果 [NOIP2014 提高组] 生活大爆炸版石头剪刀布 …

Tensorflow2.0笔记 - Tensor的数据索引和切片

主要涉及的了基础下标索引"[]",逗号",",冒号":",省略号"..."操作&#xff0c;以及gather,gather_nd和boolean_mask的相关使用方法。 import tensorflow as tf import numpy as nptf.__version__tensor tf.random.uniform([1,5,5,3],…

if单分支,二分支,多分支,语句嵌套,while语句,for语句(Python实现)

一、主要目的&#xff1a; 1&#xff0e;熟悉程序设计结构的三种方式 2.掌握if单分支语句、if二分支语句、if多分支语句及if语句嵌套的使用方法 3.掌握while语句的使用方法 4.掌握for语句的使用方法 5.掌握循环嵌套的使用方法 二、主要内容和结果展现&#xff1a; 1&…

国产ULN2803达林顿驱动芯片为什么可兼容TI ULN2803A的参数特性分享,且可用于红外摄像机等产品中

随着安防视频监控系统工程的需求越来越广&#xff0c;销量也与日俱增。在红外摄像机的红外LED驱动电路应用中&#xff0c;驱动大多数选用达林顿驱动芯片&#xff0c;行业上有用到TI 的ULN2803A&#xff0c;在目前行情&#xff0c;国外芯片紧缺的情况下&#xff0c;不少企业会多…