feat: 完成微信小程序消息推送

This commit is contained in:
橙子
2024-11-09 19:05:42 +08:00
parent 3fbaffe9a2
commit 83fb93da11
10 changed files with 213 additions and 10 deletions

View File

@@ -0,0 +1,71 @@
using Yi.Framework.WeChat.MiniProgram.Abstract;
namespace Yi.Framework.WeChat.MiniProgram.HttpModels;
public class SubscribeNoticeRequest
{
/// <summary>
///用户openid可以是小程序的openid也可以是mp_template_msg.appid对应的公众号的openid
/// </summary>
public string touser { get; set; }
/// <summary>
/// 小程序模板id
/// </summary>
public string template_id { get; set; }
/// <summary>
/// 小程序模板消息的数据
/// </summary>
public Dictionary<string, keyValueItem> data { get; set; }
/// <summary>
/// 默认为正式版
/// </summary>
public string miniprogram_state { get; set; } = "formal";
/// <summary>
/// 默认为中文
/// </summary>
public string lang { get; set; } = "zh_CN";
}
public class SubscribeNoticeInput
{
/// <summary>
///用户openid可以是小程序的openid也可以是mp_template_msg.appid对应的公众号的openid
/// </summary>
public string touser { get; set; }
/// <summary>
/// 小程序模板id
/// </summary>
public string template_id { get; set; }
/// <summary>
/// 公众号模板消息的数据
/// </summary>
public Dictionary<string, keyValueItem> data { get; set; }
}
public class SubscribeNoticeResponse : IErrorObjct
{
public int errcode { get; set; }
public string errmsg { get; set; }
}
public class keyValueItem
{
public keyValueItem(string value)
{
this.value = value;
}
public string value { get; set; }
}

View File

@@ -10,4 +10,11 @@ public interface IWeChatMiniProgramManager
/// <param name="input"></param> /// <param name="input"></param>
/// <returns></returns> /// <returns></returns>
Task<Code2SessionResponse> Code2SessionAsync(Code2SessionInput input); Task<Code2SessionResponse> Code2SessionAsync(Code2SessionInput input);
/// <summary>
/// 向用户发送订阅消息要openid
/// </summary>
/// <param name="input"></param>
/// <returns></returns>
Task SendSubscribeNoticeAsync(SubscribeNoticeInput input);
} }

View File

@@ -1,5 +1,6 @@
using System.Net.Http.Json; using System.Net.Http.Json;
using Microsoft.Extensions.Options; using Microsoft.Extensions.Options;
using Newtonsoft.Json;
using Volo.Abp.DependencyInjection; using Volo.Abp.DependencyInjection;
using Yi.Framework.Core.Extensions; using Yi.Framework.Core.Extensions;
using Yi.Framework.WeChat.MiniProgram.HttpModels; using Yi.Framework.WeChat.MiniProgram.HttpModels;
@@ -44,4 +45,32 @@ public class WeChatMiniProgramManager : IWeChatMiniProgramManager, ISingletonDep
return responseBody; return responseBody;
} }
} }
/// <summary>
/// 发送模板订阅消息
/// </summary>
/// <param name="input"></param>
public async Task SendSubscribeNoticeAsync(SubscribeNoticeInput input)
{
var token = await _weChatToken.GetTokenAsync();
string url = $"https://api.weixin.qq.com/cgi-bin/message/subscribe/send?access_token={token}";
var req = new SubscribeNoticeRequest
{
touser = input.touser,
template_id = input.template_id,
data = input.data,
miniprogram_state = _options.Notice.State??"formal"
};
req.template_id=req.template_id?? _options.Notice.TemplateId;
using (HttpClient httpClient = new HttpClient())
{
var body =new StringContent(JsonConvert.SerializeObject(req));
HttpResponseMessage response = await httpClient.PostAsync(url, body);
var responseBody = await response.Content.ReadFromJsonAsync<SubscribeNoticeResponse>();
responseBody.ValidateSuccess();
}
}
} }

View File

@@ -12,4 +12,19 @@ public class WeChatMiniProgramOptions
/// </summary> /// </summary>
public string AppSecret { get; set; } public string AppSecret { get; set; }
/// <summary>
/// 消息
/// </summary>
public WeChatMiniProgramNoticeItem Notice { get; set; }
}
public class WeChatMiniProgramNoticeItem
{
/// <summary>
/// 模板id
/// </summary>
public string TemplateId { get; set; }
public string State { get; set; }
} }

View File

@@ -49,7 +49,7 @@ public class WeChatMiniProgramAccountService : ApplicationService
var openId = (await _weChatMiniProgramManager.Code2SessionAsync(new Code2SessionInput(intput.JsCode))) var openId = (await _weChatMiniProgramManager.Code2SessionAsync(new Code2SessionInput(intput.JsCode)))
.openid; .openid;
var authInfo = await _authService.TryGetByOpenIdAsync(openId, AuthTypeConst.WeChatMiniProgram); var authInfo = await _authService.TryGetAuthInfoAsync(openId, AuthTypeConst.WeChatMiniProgram);
if (authInfo is null) if (authInfo is null)
{ {
throw new UserFriendlyException("该小程序没有绑定任何账号", "2000", "Auth未找到对应关系"); throw new UserFriendlyException("该小程序没有绑定任何账号", "2000", "Auth未找到对应关系");
@@ -90,7 +90,7 @@ public class WeChatMiniProgramAccountService : ApplicationService
//是否已经授权过绑定过auth //是否已经授权过绑定过auth
bool isAuthed =true; bool isAuthed =true;
//如果openId没有绑定过代表第一次进入否则就是临时账号进行绑定 //如果openId没有绑定过代表第一次进入否则就是临时账号进行绑定
var authInfo= await _authService.TryGetByOpenIdAsync(openId,AuthTypeConst.WeChatMiniProgram); var authInfo= await _authService.TryGetAuthInfoAsync(openId,AuthTypeConst.WeChatMiniProgram);
//从来没绑定过 //从来没绑定过
if (authInfo is null) if (authInfo is null)
{ {

View File

@@ -0,0 +1,56 @@
using Volo.Abp.DependencyInjection;
using Volo.Abp.Domain.Services;
using Volo.Abp.EventBus;
using Yi.Framework.DigitalCollectibles.Domain.Shared.Consts;
using Yi.Framework.DigitalCollectibles.Domain.Shared.Etos;
using Yi.Framework.Rbac.Application.Contracts.IServices;
using Yi.Framework.WeChat.MiniProgram;
using Yi.Framework.WeChat.MiniProgram.HttpModels;
namespace Yi.Framework.DigitalCollectibles.Domain.Managers;
public class WeChatMiniProgramNoticeEventHandler : ILocalEventHandler<WeChatMiniProgramNoticeEto>, ITransientDependency
{
private readonly IWeChatMiniProgramManager _weChatMiniProgramManager;
private readonly IAuthService _authService;
public WeChatMiniProgramNoticeEventHandler(IWeChatMiniProgramManager weChatMiniProgramManager,
IAuthService authService)
{
_weChatMiniProgramManager = weChatMiniProgramManager;
_authService = authService;
}
public async Task HandleEventAsync(WeChatMiniProgramNoticeEto eventData)
{
var authInfo = await _authService.TryGetAuthInfoAsync(null, AuthTypeConst.WeChatMiniProgram, eventData.UserId);
await SendAsync(authInfo.OpenId, eventData.Title);
}
/// <summary>
/// 像用户发送微信消息
/// </summary>
/// <param name="userId"></param>
public async Task SendAsync(string openId, string title)
{
//成功挖到矿,可以发消息给用户了
await _weChatMiniProgramManager.SendSubscribeNoticeAsync(new SubscribeNoticeInput
{
touser = openId,
data = new Dictionary<string, keyValueItem>()
{
//活动名称
{ "thing9", new keyValueItem("恭喜挖到新的数字藏品") },
//奖品名称
{ "thing1", new keyValueItem(title) },
//中奖时间
{ "date5", new keyValueItem(DateTime.Now.ToString("yyyy/MM/dd HH:mm:ss")) },
//温馨提醒
{ "thing4", new keyValueItem("点击前往小程序,可在仓库或者记录中查看") },
}
});
}
}

View File

@@ -0,0 +1,7 @@
namespace Yi.Framework.DigitalCollectibles.Domain.Shared.Etos;
public class WeChatMiniProgramNoticeEto
{
public Guid UserId { get; set; }
public String Title { get; set; }
}

View File

@@ -1,11 +1,14 @@
using Volo.Abp.DependencyInjection; using Volo.Abp.DependencyInjection;
using Volo.Abp.EventBus; using Volo.Abp.EventBus;
using Volo.Abp.EventBus.Local;
using Yi.Framework.DigitalCollectibles.Domain.Entities; using Yi.Framework.DigitalCollectibles.Domain.Entities;
using Yi.Framework.DigitalCollectibles.Domain.Entities.Record; using Yi.Framework.DigitalCollectibles.Domain.Entities.Record;
using Yi.Framework.DigitalCollectibles.Domain.Managers; using Yi.Framework.DigitalCollectibles.Domain.Managers;
using Yi.Framework.DigitalCollectibles.Domain.Shared.Consts;
using Yi.Framework.DigitalCollectibles.Domain.Shared.Etos; using Yi.Framework.DigitalCollectibles.Domain.Shared.Etos;
using Yi.Framework.SqlSugarCore.Abstractions; using Yi.Framework.SqlSugarCore.Abstractions;
using Yi.Framework.WeChat.MiniProgram;
using Yi.Framework.WeChat.MiniProgram.HttpModels;
namespace Yi.Framework.DigitalCollectibles.Domain.EventHandlers; namespace Yi.Framework.DigitalCollectibles.Domain.EventHandlers;
@@ -18,13 +21,18 @@ public class SuccessMiningEventHandler : ILocalEventHandler<SuccessMiningEto>, I
private ISqlSugarRepository<CollectiblesAggregateRoot> _repository; private ISqlSugarRepository<CollectiblesAggregateRoot> _repository;
private readonly ISqlSugarRepository<CollectiblesUserStoreAggregateRoot> _userStoreRepository; private readonly ISqlSugarRepository<CollectiblesUserStoreAggregateRoot> _userStoreRepository;
private readonly ISqlSugarRepository<MiningPoolRecordAggregateRoot> _miningPoolRecordRepository; private readonly ISqlSugarRepository<MiningPoolRecordAggregateRoot> _miningPoolRecordRepository;
private readonly ILocalEventBus _localEvent;
public SuccessMiningEventHandler(MiningPoolManager miningPoolManager, public SuccessMiningEventHandler(MiningPoolManager miningPoolManager,
ISqlSugarRepository<CollectiblesAggregateRoot> repository, ISqlSugarRepository<CollectiblesUserStoreAggregateRoot> userStoreRepository, ISqlSugarRepository<MiningPoolRecordAggregateRoot> miningPoolRecordRepository) ISqlSugarRepository<CollectiblesAggregateRoot> repository,
ISqlSugarRepository<CollectiblesUserStoreAggregateRoot> userStoreRepository,
ISqlSugarRepository<MiningPoolRecordAggregateRoot> miningPoolRecordRepository, ILocalEventBus localEvent)
{ {
_miningPoolManager = miningPoolManager; _miningPoolManager = miningPoolManager;
_repository = repository; _repository = repository;
_userStoreRepository = userStoreRepository; _userStoreRepository = userStoreRepository;
_miningPoolRecordRepository = miningPoolRecordRepository; _miningPoolRecordRepository = miningPoolRecordRepository;
_localEvent = localEvent;
} }
public async Task HandleEventAsync(SuccessMiningEto eventData) public async Task HandleEventAsync(SuccessMiningEto eventData)
@@ -36,7 +44,7 @@ public class SuccessMiningEventHandler : ILocalEventHandler<SuccessMiningEto>, I
//新增全世界发现 //新增全世界发现
currentCollectibles.FindTotal += 1; currentCollectibles.FindTotal += 1;
await _repository.UpdateAsync(currentCollectibles); await _repository.UpdateAsync(currentCollectibles);
//使用结果新增给对应的用户 //使用结果新增给对应的用户
await _userStoreRepository.InsertAsync(new CollectiblesUserStoreAggregateRoot await _userStoreRepository.InsertAsync(new CollectiblesUserStoreAggregateRoot
{ {
@@ -44,8 +52,16 @@ public class SuccessMiningEventHandler : ILocalEventHandler<SuccessMiningEto>, I
CollectiblesId = eventData.CollectiblesId, CollectiblesId = eventData.CollectiblesId,
IsRead = false IsRead = false
}); });
//新增一条挖矿记录 //新增一条挖矿记录
await _miningPoolRecordRepository.InsertAsync(new MiningPoolRecordAggregateRoot(eventData.UserId,eventData.CollectiblesId)); await _miningPoolRecordRepository.InsertAsync(
new MiningPoolRecordAggregateRoot(eventData.UserId, eventData.CollectiblesId));
//给挖到矿的用户,发送微信小程序通知
await _localEvent.PublishAsync(new WeChatMiniProgramNoticeEto
{
UserId = eventData.UserId,
Title = $"{currentCollectibles.Rarity.GetRarityName()}-{currentCollectibles.Name}"
},false);
} }
} }

View File

@@ -5,6 +5,6 @@ namespace Yi.Framework.Rbac.Application.Contracts.IServices;
public interface IAuthService public interface IAuthService
{ {
Task<AuthOutputDto?> TryGetByOpenIdAsync(string openId, string authType); Task<AuthOutputDto?> TryGetAuthInfoAsync(string? openId, string authType, Guid? userId = null);
Task<AuthOutputDto> CreateAsync(AuthCreateOrUpdateInputDto input); Task<AuthOutputDto> CreateAsync(AuthCreateOrUpdateInputDto input);
} }

View File

@@ -113,9 +113,11 @@ namespace Yi.Framework.Rbac.Application.Services.Authentication
return (await GetListAsync(input)).Items; return (await GetListAsync(input)).Items;
} }
public async Task<AuthOutputDto?> TryGetByOpenIdAsync(string openId, string authType) public async Task<AuthOutputDto?> TryGetAuthInfoAsync(string? openId, string authType,Guid? userId=null)
{ {
var entity = await _repository._DbQueryable.Where(x => x.OpenId == openId) var entity = await _repository._DbQueryable
.WhereIF(openId is not null, x => x.OpenId == openId)
.WhereIF(userId is not null,x => x.UserId == userId)
.Where(x => x.AuthType == authType) .Where(x => x.AuthType == authType)
.FirstAsync(); .FirstAsync();
var output = await MapToGetOutputDtoAsync(entity); var output = await MapToGetOutputDtoAsync(entity);