Add new Hcs.Broker to communicate with ГИС ЖКХ via CryptoPro LibCore

This commit is contained in:
2025-09-28 15:45:15 +09:00
parent 904988780a
commit 2b49320014
171 changed files with 185618 additions and 0 deletions

View File

@ -0,0 +1,107 @@
using Hcs.Broker.Api.Payload.Payments;
using Hcs.Broker.Api.Request.Exception;
using Hcs.Broker.Internal;
using Hcs.Service.Async.Payments;
namespace Hcs.Broker.Api.Request.Payments
{
internal class ImportNotificationsOfOrderExecutionRequest(Client client) : PaymentsRequestBase(client)
{
protected override bool CanBeRestarted => false;
internal async Task<bool> ExecuteAsync(ImportNotificationsOfOrderExecutionPayload payload, CancellationToken token)
{
ThrowIfPayloadIncorrect(payload);
// http://open-gkh.ru/Payment/importNotificationsOfOrderExecutionRequest.html
var request = new importNotificationsOfOrderExecutionRequest
{
Id = Constants.SIGNED_XML_ELEMENT_ID,
version = "10.0.1.1",
Items = [GetNotificationFromPayload(payload)]
};
var result = await SendAndWaitResultAsync(request, async asyncClient =>
{
var response = await asyncClient.importNotificationsOfOrderExecutionAsync(CreateRequestHeader(), request);
return response.AckRequest.Ack;
}, token);
result.Items.OfType<ErrorMessageType>().ToList().ForEach(error =>
{
throw RemoteException.CreateNew(error.ErrorCode, error.Description);
});
result.Items.OfType<CommonResultType>().ToList().ForEach(commonResult =>
{
commonResult.Items.OfType<ErrorMessageType>().ToList().ForEach(error =>
{
throw RemoteException.CreateNew(error.ErrorCode, error.Description);
});
});
return true;
}
private void ThrowIfPayloadIncorrect(ImportNotificationsOfOrderExecutionPayload payload)
{
if (string.IsNullOrEmpty(payload.orderId))
{
throw new ArgumentException($"{nameof(payload.orderId)} is empty");
}
if (payload.month.HasValue && !payload.year.HasValue)
{
throw new ArgumentException($"{nameof(payload.month)} has value but {nameof(payload.year)} has not");
}
if (!payload.month.HasValue && payload.year.HasValue)
{
throw new ArgumentException($"{nameof(payload.year)} has value but {nameof(payload.month)} has not");
}
if (string.IsNullOrEmpty(payload.paymentDocumentId))
{
throw new ArgumentException($"{nameof(payload.paymentDocumentId)} is empty");
}
if (string.IsNullOrEmpty(payload.paymentDocumentGUID))
{
throw new ArgumentException($"{nameof(payload.paymentDocumentGUID)} is empty");
}
}
private importNotificationsOfOrderExecutionRequestNotificationOfOrderExecution139Type GetNotificationFromPayload(ImportNotificationsOfOrderExecutionPayload payload)
{
var notification = new importNotificationsOfOrderExecutionRequestNotificationOfOrderExecution139Type()
{
TransportGUID = Guid.NewGuid().ToString(),
OrderInfo = new NotificationOfOrderExecution139TypeOrderInfo()
{
OrderID = payload.orderId,
OrderDate = payload.orderDate,
Amount = payload.amount,
Items = [payload.paymentDocumentId, payload.paymentDocumentGUID],
ItemsElementName = [ItemsChoiceType4.PaymentDocumentID, ItemsChoiceType4.PaymentDocumentGUID]
}
};
if (payload.onlinePayment.HasValue && payload.onlinePayment.Value)
{
notification.OrderInfo.OnlinePayment = true;
notification.OrderInfo.OnlinePaymentSpecified = true;
}
if (payload.month.HasValue)
{
notification.OrderInfo.MonthAndYear = new NotificationOfOrderExecution139TypeOrderInfoMonthAndYear()
{
Year = payload.year.Value,
Month = payload.month.Value
};
}
return notification;
}
}
}

View File

@ -0,0 +1,93 @@
using Hcs.Broker.Api.Payload.Payments;
using Hcs.Broker.Api.Request.Exception;
using Hcs.Broker.Internal;
using Hcs.Service.Async.Payments;
namespace Hcs.Broker.Api.Request.Payments
{
internal class ImportSupplierNotificationsOfOrderExecutionRequest(Client client) : PaymentsRequestBase(client)
{
protected override bool CanBeRestarted => false;
internal async Task<bool> ExecuteAsync(ImportSupplierNotificationsOfOrderExecutionPayload payload, CancellationToken token)
{
ThrowIfPayloadIncorrect(payload);
// http://open-gkh.ru/Payment/importSupplierNotificationsOfOrderExecutionRequest.html
var request = new importSupplierNotificationsOfOrderExecutionRequest
{
Id = Constants.SIGNED_XML_ELEMENT_ID,
version = "10.0.1.1",
SupplierNotificationOfOrderExecution = [GetNotificationFromPayload(payload)]
};
var result = await SendAndWaitResultAsync(request, async asyncClient =>
{
var response = await asyncClient.importSupplierNotificationsOfOrderExecutionAsync(CreateRequestHeader(), request);
return response.AckRequest.Ack;
}, token);
result.Items.OfType<ErrorMessageType>().ToList().ForEach(error =>
{
throw RemoteException.CreateNew(error.ErrorCode, error.Description);
});
result.Items.OfType<CommonResultType>().ToList().ForEach(commonResult =>
{
commonResult.Items.OfType<ErrorMessageType>().ToList().ForEach(error =>
{
throw RemoteException.CreateNew(error.ErrorCode, error.Description);
});
});
return true;
}
private void ThrowIfPayloadIncorrect(ImportSupplierNotificationsOfOrderExecutionPayload payload)
{
if (payload.month.HasValue && !payload.year.HasValue)
{
throw new ArgumentException($"{nameof(payload.month)} has value but {nameof(payload.year)} has not");
}
if (!payload.month.HasValue && payload.year.HasValue)
{
throw new ArgumentException($"{nameof(payload.year)} has value but {nameof(payload.month)} has not");
}
if (string.IsNullOrEmpty(payload.paymentDocumentId))
{
throw new ArgumentException($"{nameof(payload.paymentDocumentId)} is empty");
}
}
private importSupplierNotificationsOfOrderExecutionRequestSupplierNotificationOfOrderExecution GetNotificationFromPayload(ImportSupplierNotificationsOfOrderExecutionPayload payload)
{
var notification = new importSupplierNotificationsOfOrderExecutionRequestSupplierNotificationOfOrderExecution()
{
TransportGUID = Guid.NewGuid().ToString(),
OrderDate = payload.orderDate,
Item = payload.paymentDocumentId,
ItemElementName = ItemChoiceType1.PaymentDocumentID,
Amount = payload.amount
};
if (payload.month.HasValue)
{
notification.OrderPeriod = new SupplierNotificationOfOrderExecutionTypeOrderPeriod()
{
Month = payload.month.Value,
Year = payload.year.Value
};
}
if (payload.onlinePayment.HasValue && payload.onlinePayment.Value)
{
notification.OnlinePayment = true;
notification.OnlinePaymentSpecified = true;
}
return notification;
}
}
}

View File

@ -0,0 +1,55 @@
using Hcs.Broker.Api.Request;
using Hcs.Broker.Api.Request.Adapter;
using Hcs.Service.Async.Payments;
using System.Threading.Tasks;
namespace Hcs.Service.Async.Payments
{
#pragma warning disable IDE1006
public partial class getStateResult : IGetStateResultMany { }
#pragma warning restore IDE1006
public partial class PaymentPortsTypeAsyncClient : IAsyncClient<RequestHeader>
{
public async Task<IGetStateResponse> GetStateAsync(RequestHeader header, IGetStateRequest request)
{
return await getStateAsync(header, (getStateRequest)request);
}
}
#pragma warning disable IDE1006
public partial class getStateResponse : IGetStateResponse
#pragma warning restore IDE1006
{
public IGetStateResult GetStateResult => getStateResult;
}
public partial class AckRequestAck : IAck { }
public partial class ErrorMessageType : IErrorMessage { }
#pragma warning disable IDE1006
public partial class getStateRequest : IGetStateRequest { }
#pragma warning restore IDE1006
}
namespace Hcs.Broker.Api.Request.Payments
{
internal class PaymentsRequestBase(Client client) :
RequestBase<getStateResult,
PaymentPortsTypeAsyncClient,
PaymentPortsTypeAsync,
RequestHeader,
AckRequestAck,
ErrorMessageType,
getStateRequest>(client)
{
protected override EndPoint EndPoint => EndPoint.PaymentsAsync;
protected override bool EnableMinimalResponseWaitDelay => true;
protected override bool CanBeRestarted => true;
protected override int RestartTimeoutMinutes => 20;
}
}