Add service_kyc
All checks were successful
Client CMS Check Build (NixCN CMS) TeamCity build finished
Backend Check Build (NixCN CMS) TeamCity build finished

Signed-off-by: Asai Neko <sugar@sne.moe>
This commit is contained in:
2026-02-01 13:15:17 +08:00
parent a2eb882398
commit 0ac96ab3e6
23 changed files with 1831 additions and 89 deletions

View File

@@ -0,0 +1,251 @@
package service_kyc
import (
"context"
"nixcn-cms/data"
"nixcn-cms/internal/exception"
"nixcn-cms/internal/kyc"
"nixcn-cms/service/shared"
"github.com/google/uuid"
)
type KycQueryData struct {
KycId string `json:"kyc_id"`
}
type KycQueryPayload struct {
Context context.Context
Data *KycQueryData
}
type KycQueryResponse struct {
Status string `json:"status"` // success | pending | failed
}
type KycQueryResult struct {
Common shared.CommonResult
Data *KycQueryResponse
}
func (self *KycServiceImpl) QueryKyc(payload *KycQueryPayload) (result *KycQueryResult) {
var err error
sessionState, err := kyc.GetSessionState(payload.Context, self.PassportReaderSessionId)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInvalidInput).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
if sessionState == kyc.StateApproved {
sessionDetails, err := kyc.GetSessionDetails(payload.Context, self.PassportReaderSessionId)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInvalidInput).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
var kycInfo = kyc.PassportResp{
GivenNames: sessionDetails.GivenNames,
Surname: sessionDetails.Surname,
Nationality: sessionDetails.Nationality,
DateOfBirth: sessionDetails.DateOfBirth,
DocumentType: sessionDetails.DocumentType,
DocumentNumber: sessionDetails.DocumentNumber,
ExpiryDate: sessionDetails.ExpiryDate,
}
if kycInfo.DocumentType != "PASSPORT" || kycInfo.DocumentNumber != self.PassportId {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInvalidInput).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
encodedKycInfo, err := kyc.EncodeAES(kycInfo)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeSpecific).
SetOriginal(exception.KycQueryFailed).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
kycId, err := uuid.Parse(payload.Data.KycId)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceEvent).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorUuidParseFailed).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
var updates = map[string]any{
"kyc_info": kycInfo,
}
err = new(data.Kyc).
SetKycInfo(*encodedKycInfo).
UpdateByKycID(payload.Context, &kycId, updates)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorDatabase).
SetError(err).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonSuccess).
SetError(nil).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 200,
Exception: exception,
},
Data: &KycQueryResponse{
Status: "success",
},
}
return
}
if sessionState == kyc.StateCreated || sessionState == kyc.StateInitiated || sessionState == kyc.StateCompleted {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonSuccess).
SetError(nil).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 200,
Exception: exception,
},
Data: &KycQueryResponse{
Status: "pending",
},
}
return
}
if sessionState == kyc.StateFailed || sessionState == kyc.StateAborted || sessionState == kyc.StateRejected {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceQuery).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonSuccess).
SetError(nil).
Throw(payload.Context)
result = &KycQueryResult{
Common: shared.CommonResult{
HttpCode: 200,
Exception: exception,
},
Data: &KycQueryResponse{
Status: "failed",
},
}
return
}
return
}

View File

@@ -0,0 +1,15 @@
package service_kyc
type KycService interface {
SessionKyc(*KycSessionPayload) *KycSessionResult
QueryKyc(*KycQueryPayload) *KycQueryResult
}
type KycServiceImpl struct {
PassportId string `json:"passport_id"`
PassportReaderSessionId int `json:"passport_reader_session_id"`
}
func NewKycService() KycService {
return &KycServiceImpl{}
}

View File

@@ -0,0 +1,400 @@
package service_kyc
import (
"context"
"encoding/base64"
"encoding/json"
"net/url"
"nixcn-cms/data"
"nixcn-cms/internal/exception"
"nixcn-cms/internal/kyc"
"nixcn-cms/service/shared"
"github.com/google/uuid"
)
// cnrid: {"legal_name":"", "resident_id":""}
// passport: {"id": ""}
type KycSessionData struct {
Type string `json:"type"` // cnrid | passport
Identity string `json:"identity"` // base64 json
UserId string `json:"user_id" swaggerignore:"true"`
}
type KycSessionPayload struct {
Context context.Context
Data *KycSessionData
}
type KycSessionResponse struct {
Status string `json:"status"` // success | processing
KycId *string `json:"kyc_id"`
RedirectUri *string `json:"redirect_uri"`
}
type KycSessionResult struct {
Common shared.CommonResult
Data *KycSessionResponse
}
func (self *KycServiceImpl) SessionKyc(payload *KycSessionPayload) (result *KycSessionResult) {
var err error
decodedIdentityByte, err := base64.StdEncoding.DecodeString(payload.Data.Identity)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorBase64DecodeFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
switch payload.Data.Type {
case "cnrid":
var info kyc.CNRidInfo
if err := json.Unmarshal(decodedIdentityByte, &info); err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorJsonDecodeFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
kycInfo := kyc.CNRidInfo{
LegalName: info.LegalName,
ResidentId: info.ResidentId,
}
aliCloudAuth, err := kyc.CNRidMD5AliEnc(&kycInfo)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInternal).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
kycResult, err := kyc.AliId2MetaVerify(aliCloudAuth)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInternal).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
if *kycResult != "1" {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeSpecific).
SetOriginal(exception.KycSessionFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
userId, err := uuid.Parse(payload.Data.UserId)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceEvent).
SetEndpoint(exception.EndpointEventServiceJoin).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorUuidParseFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
encodedKycInfo, err := kyc.EncodeAES(kycInfo)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeSpecific).
SetOriginal(exception.KycSessionFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
kycIdOrig, err := new(data.Kyc).
SetType("cnrid").
SetUserId(userId).
SetKycInfo(*encodedKycInfo).
Create(payload.Context)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorDatabase).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
kycId := kycIdOrig.String()
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonSuccess).
SetError(nil).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 200,
Exception: exception,
},
Data: &KycSessionResponse{
Status: "success",
KycId: &kycId,
RedirectUri: nil,
},
}
case "passport":
var info kyc.PassportInfo
if err := json.Unmarshal(decodedIdentityByte, &info); err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorJsonDecodeFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
self.PassportId = info.ID
sessionResponse, err := kyc.CreateSession(payload.Context)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInternal).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
self.PassportReaderSessionId = sessionResponse.ID
userId, err := uuid.Parse(payload.Data.UserId)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusServer).
SetService(exception.ServiceEvent).
SetEndpoint(exception.EndpointEventServiceJoin).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorUuidParseFailed).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
kycIdOrig, err := new(data.Kyc).
SetType("passport").
SetUserId(userId).
Create(payload.Context)
if err != nil {
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorDatabase).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 500,
Exception: exception,
},
Data: nil,
}
return
}
kycId := kycIdOrig.String()
kycBaseURL := "https://passportreader.app/open"
kycUrl, _ := url.Parse(kycBaseURL)
kycQuery := kycUrl.Query()
kycQuery.Set("token", sessionResponse.Token)
kycUrl.RawQuery = kycQuery.Encode()
redirectUri := kycUrl.String()
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonSuccess).
SetError(nil).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 200,
Exception: exception,
},
Data: &KycSessionResponse{
Status: "processing",
KycId: &kycId,
RedirectUri: &redirectUri,
},
}
default:
exception := new(exception.Builder).
SetStatus(exception.StatusUser).
SetService(exception.ServiceKyc).
SetEndpoint(exception.EndpointKycServiceSession).
SetType(exception.TypeCommon).
SetOriginal(exception.CommonErrorInvalidInput).
SetError(err).
Throw(payload.Context)
result = &KycSessionResult{
Common: shared.CommonResult{
HttpCode: 400,
Exception: exception,
},
Data: nil,
}
return
}
return
}