a8cdeeb838
- internal/nsdadapter/igw/client.go: REST-клиент ИШ (SendPackage, GetStatus, ListIncoming) с base64-JSON, ретраями на 5xx, 4xx без ретраев - internal/nsdadapter/router.go: маршрутизация MessageKind -> PackageType ЭДО (#M2MTR, #M2MTD, #M2MER, SUBBR/SUBER/SUB16, Assets_investment) - internal/nsdadapter/sender.go: реализация m2mcore.NSDSender (Send/SendDecision) через REST ИШ, сериализация Request/Decision в windows-1251 - internal/nsdadapter/config.go: профили guest/test3/prod × gost/rsa (URL ИШ, канал, контейнер ключа, retry) - internal/nsdadapter/onyx/onyx.go: скелет резервного канала WS ONYX (ждёт PR-6 crypto-service для подписи) - cmd/nsd-adapter/main.go: HTTP /healthz + фоновый поллер входящих по типам ЭДО; idle-режим без BJ_NSD_PROFILE make ci зелёный. Без внешних Go-зависимостей. Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
83 lines
3.1 KiB
Go
83 lines
3.1 KiB
Go
package nsdadapter
|
|
|
|
import (
|
|
"context"
|
|
"errors"
|
|
"fmt"
|
|
"time"
|
|
|
|
"git.zetit.ru/zuevav/Bridge-and-Join-s/internal/m2m"
|
|
"git.zetit.ru/zuevav/Bridge-and-Join-s/internal/nsdadapter/igw"
|
|
"git.zetit.ru/zuevav/Bridge-and-Join-s/internal/nsdxml"
|
|
)
|
|
|
|
// IGWClient — узкий интерфейс из igw.Client, нужный Sender'у. Введён
|
|
// ради тестируемости (mock в тестах) и независимости от ИШ-реализации.
|
|
type IGWClient interface {
|
|
SendPackage(ctx context.Context, channel, packageType string, body []byte) (string, error)
|
|
ListIncoming(ctx context.Context, channel string, since time.Time, packageType string) ([]igw.Package, error)
|
|
}
|
|
|
|
// Sender — реализация m2mcore.NSDSender поверх REST ИШ НРД.
|
|
type Sender struct {
|
|
profile Profile
|
|
client IGWClient
|
|
}
|
|
|
|
// NewSender собирает Sender для указанного профиля.
|
|
func NewSender(profile Profile, client IGWClient) *Sender {
|
|
return &Sender{profile: profile, client: client}
|
|
}
|
|
|
|
// Send сериализует M2MTransferRequest в windows-1251, маршрутизирует к
|
|
// типу пакета #M2MTR и отправляет в ИШ. M2MTransferResponse в этом
|
|
// канале возвращается асинхронно через ListIncoming, поэтому Send
|
|
// возвращает nil-response — реальный ответ забирает поллер.
|
|
func (s *Sender) Send(ctx context.Context, req *m2m.M2MTransferRequest) (*m2m.M2MTransferResponse, error) {
|
|
if req == nil {
|
|
return nil, errors.New("nsdadapter: Send: req=nil")
|
|
}
|
|
if err := req.Validate(); err != nil {
|
|
return nil, fmt.Errorf("nsdadapter: req.Validate: %w", err)
|
|
}
|
|
body, err := nsdxml.Marshal(req)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("nsdadapter: marshal Request: %w", err)
|
|
}
|
|
pkgType, err := RouteToPackageType(KindTransferRequest)
|
|
if err != nil {
|
|
return nil, err
|
|
}
|
|
pkgID, err := s.client.SendPackage(ctx, s.profile.Channel, string(pkgType), body)
|
|
if err != nil {
|
|
return nil, fmt.Errorf("nsdadapter: SendPackage: %w", err)
|
|
}
|
|
// Возвращаем псевдо-Response с GUID-ом для трассировки. Реальный
|
|
// M2MTransferResponse от НРД придёт через входящие пакеты, его
|
|
// обработает поллер cmd/nsd-adapter.
|
|
_ = pkgID
|
|
return nil, nil
|
|
}
|
|
|
|
// SendDecision сериализует и отправляет M2MTransferDecision.
|
|
func (s *Sender) SendDecision(ctx context.Context, decision *m2m.M2MTransferDecision) error {
|
|
if decision == nil {
|
|
return errors.New("nsdadapter: SendDecision: decision=nil")
|
|
}
|
|
if err := decision.Validate(); err != nil {
|
|
return fmt.Errorf("nsdadapter: decision.Validate: %w", err)
|
|
}
|
|
body, err := nsdxml.Marshal(decision)
|
|
if err != nil {
|
|
return fmt.Errorf("nsdadapter: marshal Decision: %w", err)
|
|
}
|
|
pkgType, err := RouteToPackageType(KindTransferDecision)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
if _, err := s.client.SendPackage(ctx, s.profile.Channel, string(pkgType), body); err != nil {
|
|
return fmt.Errorf("nsdadapter: SendPackage: %w", err)
|
|
}
|
|
return nil
|
|
}
|