feat: 完成微信小程序消息推送
This commit is contained in:
@@ -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; }
|
||||||
|
}
|
||||||
@@ -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);
|
||||||
}
|
}
|
||||||
@@ -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();
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
@@ -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; }
|
||||||
}
|
}
|
||||||
@@ -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)
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -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("点击前往小程序,可在仓库或者记录中查看") },
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -0,0 +1,7 @@
|
|||||||
|
namespace Yi.Framework.DigitalCollectibles.Domain.Shared.Etos;
|
||||||
|
|
||||||
|
public class WeChatMiniProgramNoticeEto
|
||||||
|
{
|
||||||
|
public Guid UserId { get; set; }
|
||||||
|
public String Title { get; set; }
|
||||||
|
}
|
||||||
@@ -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);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -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);
|
||||||
}
|
}
|
||||||
@@ -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);
|
||||||
|
|||||||
Reference in New Issue
Block a user