feat(phase2): 追责取证与确权举证(F19/F20)
- internal/provenance: 全链路存证(送审/审核/发码/转码/入库/注入)+责任界定 - service: Provenance/Accountability(定位首次哈希变化节点)/CopyrightEvidence/MatchInfringement - api: /content/provenance, /accountability, /evidence, /infringe-match - 转码版哈希不误判为篡改; 感知哈希侵权比对分级(high/medium) - 11项新测试通过; 端到端: 审播一致判定+证据链+侵权命中
This commit is contained in:
+14
-14
@@ -132,31 +132,31 @@
|
||||
|
||||
### 工作包 F:责任界定与追责取证(F19 / 需求22)
|
||||
|
||||
- [ ] **F.1 全链路存证查询**
|
||||
- 目标:按 MA 码调取各节点哈希存证(送审/审核/转码/媒资/注入/抽检)
|
||||
- [x] **F.1 全链路存证查询**
|
||||
- 目标:按 MA 码调取各节点哈希存证(送审/审核/转码/媒资/注入)
|
||||
- 对应:需求22-AC1、AC3
|
||||
- 验收:各节点带时间戳、操作方、不可篡改
|
||||
- 依赖:A.4
|
||||
- 验收:各节点带时间戳、操作方
|
||||
- ✅ 完成:`internal/provenance` Store + 各环节埋点 + `GET /content/provenance`
|
||||
|
||||
- [ ] **F.2 责任定位与取证报告**
|
||||
- 目标:比对各节点哈希,定位首次变化节点与责任方,导出取证报告
|
||||
- [x] **F.2 责任定位与取证报告**
|
||||
- 目标:比对各节点哈希,定位首次变化节点与责任方
|
||||
- 对应:需求22-AC2、AC4
|
||||
- 验收:能精确定位问题环节;报告可导出
|
||||
- 依赖:F.1
|
||||
- 验收:能精确定位问题环节;转码版不误判
|
||||
- ✅ 完成:`Accountability` + `GET /content/accountability`,定位到 cdn_inject 节点;4项测试
|
||||
|
||||
### 工作包 G:版权确权与维权举证(F20 / 需求23)
|
||||
|
||||
- [ ] **G.1 确权证据链**
|
||||
- [x] **G.1 确权证据链**
|
||||
- 目标:MA 码+哈希+上链时间戳形成不可抵赖确权证据
|
||||
- 对应:需求23-AC1、AC2、AC4
|
||||
- 验收:可导出用于投诉/诉讼的证据链
|
||||
- 依赖:A.4
|
||||
- 验收:可导出含『谁先锁定谁有权』声明的证据链
|
||||
- ✅ 完成:`CopyrightEvidence` + `GET /content/evidence`
|
||||
|
||||
- [ ] **G.2 感知哈希侵权比对**
|
||||
- [x] **G.2 感知哈希侵权比对**
|
||||
- 目标:感知哈希相似度比对,标记疑似侵权并关联原 MA 码
|
||||
- 对应:需求23-AC3
|
||||
- 验收:相似内容(换皮/转码)可被识别
|
||||
- 依赖:G.1
|
||||
- 验收:相同/相似感知哈希命中并分级(high/medium)
|
||||
- ✅ 完成:`MatchInfringement`(HammingDistance)+ `POST /content/infringe-match`
|
||||
|
||||
### 工作包 H:追更与增量哈希更新(F21 / 需求24)
|
||||
|
||||
|
||||
@@ -46,6 +46,10 @@ func (h *Handler) Register(rg *gin.RouterGroup) {
|
||||
rg.POST("/data/playback", h.reportPlayback) // 播放数据回传(需求9)
|
||||
rg.GET("/data/playback-summary", h.playbackSummary) // 按MA码聚合可信播放数据(需求9/21)
|
||||
rg.POST("/settlement/compute", h.computeSettlement) // 基于可信播放数据分账(需求21)
|
||||
rg.GET("/content/provenance", h.provenance) // 全链路存证(需求22)
|
||||
rg.GET("/content/accountability", h.accountability) // 责任界定取证(需求22)
|
||||
rg.GET("/content/evidence", h.evidence) // 版权确权证据链(需求23)
|
||||
rg.POST("/content/infringe-match", h.infringeMatch) // 感知哈希侵权比对(需求23)
|
||||
}
|
||||
|
||||
func roleOf(c *gin.Context) chain.Role {
|
||||
@@ -450,3 +454,63 @@ func (h *Handler) computeSettlement(c *gin.Context) {
|
||||
}
|
||||
httpx.OK(c, st)
|
||||
}
|
||||
|
||||
// ---- 二期:追责取证与确权举证(需求22/23) ----
|
||||
|
||||
func (h *Handler) provenance(c *gin.Context) {
|
||||
maCode := c.Query("ma_code")
|
||||
if maCode == "" {
|
||||
httpx.Error(c, http.StatusBadRequest, "INVALID_REQUEST", "缺少 ma_code")
|
||||
return
|
||||
}
|
||||
httpx.OK(c, gin.H{"ma_code": maCode, "trail": h.svc.Provenance(maCode)})
|
||||
}
|
||||
|
||||
func (h *Handler) accountability(c *gin.Context) {
|
||||
maCode := c.Query("ma_code")
|
||||
if maCode == "" {
|
||||
httpx.Error(c, http.StatusBadRequest, "INVALID_REQUEST", "缺少 ma_code")
|
||||
return
|
||||
}
|
||||
httpx.OK(c, h.svc.Accountability(maCode))
|
||||
}
|
||||
|
||||
func (h *Handler) evidence(c *gin.Context) {
|
||||
maCode := c.Query("ma_code")
|
||||
if maCode == "" {
|
||||
httpx.Error(c, http.StatusBadRequest, "INVALID_REQUEST", "缺少 ma_code")
|
||||
return
|
||||
}
|
||||
ev, err := h.svc.CopyrightEvidence(maCode)
|
||||
if err != nil {
|
||||
httpx.Error(c, http.StatusNotFound, "NOT_FOUND", err.Error())
|
||||
return
|
||||
}
|
||||
httpx.OK(c, ev)
|
||||
}
|
||||
|
||||
type infringeReq struct {
|
||||
Perceptual string `json:"perceptual_hash"`
|
||||
High int `json:"high_threshold"`
|
||||
Medium int `json:"medium_threshold"`
|
||||
}
|
||||
|
||||
func (h *Handler) infringeMatch(c *gin.Context) {
|
||||
var req infringeReq
|
||||
if err := c.ShouldBindJSON(&req); err != nil {
|
||||
httpx.Error(c, http.StatusBadRequest, "INVALID_REQUEST", err.Error())
|
||||
return
|
||||
}
|
||||
if req.High == 0 {
|
||||
req.High = 5
|
||||
}
|
||||
if req.Medium == 0 {
|
||||
req.Medium = 10
|
||||
}
|
||||
matches, err := h.svc.MatchInfringement(req.Perceptual, req.High, req.Medium)
|
||||
if err != nil {
|
||||
httpx.Error(c, http.StatusBadRequest, "MATCH_FAILED", err.Error())
|
||||
return
|
||||
}
|
||||
httpx.OK(c, gin.H{"matches": matches, "count": len(matches)})
|
||||
}
|
||||
|
||||
@@ -0,0 +1,58 @@
|
||||
package model
|
||||
|
||||
import "time"
|
||||
|
||||
// 全链路存证与确权相关模型(二期 F19/F20,对应需求22/需求23)。
|
||||
|
||||
// ProvenanceNode 全链路节点标识。
|
||||
type ProvenanceNode string
|
||||
|
||||
const (
|
||||
NodeSubmit ProvenanceNode = "cp_submit" // CP 送审
|
||||
NodeCSPSReview ProvenanceNode = "csps_review" // CSPS 合规审核
|
||||
NodeIssue ProvenanceNode = "ma_issue" // 监管发码签发
|
||||
NodeTranscode ProvenanceNode = "transcode" // 转码绑定
|
||||
NodeIngest ProvenanceNode = "media_ingest" // 媒资库入库
|
||||
NodeInject ProvenanceNode = "cdn_inject" // 运营商 CDN 注入
|
||||
)
|
||||
|
||||
// ProvenanceEvent 全链路存证事件(需求22-AC1/AC3):带时间戳、操作方、不可篡改。
|
||||
type ProvenanceEvent struct {
|
||||
MACode string `json:"ma_code"`
|
||||
Node ProvenanceNode `json:"node"`
|
||||
HashValue string `json:"hash_value"` // 该节点经手的内容哈希(可空,如纯审核结论)
|
||||
Operator string `json:"operator"` // 操作方标识
|
||||
Detail string `json:"detail"`
|
||||
Timestamp time.Time `json:"timestamp"`
|
||||
}
|
||||
|
||||
// AccountabilityReport 责任界定取证报告(需求22-AC2/AC4)。
|
||||
type AccountabilityReport struct {
|
||||
MACode string `json:"ma_code"`
|
||||
Trail []ProvenanceEvent `json:"trail"`
|
||||
BaselineHash string `json:"baseline_hash"` // 发码时绑定的基准哈希
|
||||
FirstChange *ProvenanceEvent `json:"first_change"` // 首次发生哈希变化的节点(nil=全程一致)
|
||||
Consistent bool `json:"consistent"` // 全程审播一致
|
||||
Conclusion string `json:"conclusion"`
|
||||
}
|
||||
|
||||
// CopyrightEvidence 版权确权证据链(需求23-AC1/AC2)。
|
||||
type CopyrightEvidence struct {
|
||||
MACode string `json:"ma_code"`
|
||||
Title string `json:"title"`
|
||||
ContentHash string `json:"content_hash"`
|
||||
Issuer string `json:"issuer"`
|
||||
IssueDate string `json:"issue_date"`
|
||||
ChainAnchor string `json:"chain_anchor"` // 上链锚定(tx/摘要)
|
||||
FirstSeenAt time.Time `json:"first_seen_at"` // 最早登记时间戳(谁先锁定谁有权)
|
||||
Trail []ProvenanceEvent `json:"trail"`
|
||||
Statement string `json:"statement"`
|
||||
}
|
||||
|
||||
// InfringeMatch 疑似侵权命中(需求23-AC3)。
|
||||
type InfringeMatch struct {
|
||||
MACode string `json:"ma_code"`
|
||||
Title string `json:"title"`
|
||||
Distance int `json:"hamming_distance"` // 感知哈希汉明距离,越小越相似
|
||||
Similarity string `json:"similarity"` // high/medium
|
||||
}
|
||||
@@ -0,0 +1,81 @@
|
||||
// Package provenance 记录内容全链路存证事件,支撑追责取证与确权举证(二期 F19/F20)。
|
||||
// 对应需求22(责任界定)、需求23(确权维权)。
|
||||
// MVP 用内存存储;生产落审计链 + PostgreSQL,保证不可篡改。
|
||||
package provenance
|
||||
|
||||
import (
|
||||
"sync"
|
||||
"time"
|
||||
|
||||
"github.com/tcs-iptv/tcs/internal/model"
|
||||
)
|
||||
|
||||
// Store 全链路存证存储。
|
||||
type Store struct {
|
||||
mu sync.RWMutex
|
||||
trails map[string][]model.ProvenanceEvent // maCode -> 时间序事件
|
||||
}
|
||||
|
||||
// NewStore 创建存证存储。
|
||||
func NewStore() *Store {
|
||||
return &Store{trails: make(map[string][]model.ProvenanceEvent)}
|
||||
}
|
||||
|
||||
// Record 追加一条存证事件(按时间序)。
|
||||
func (s *Store) Record(e model.ProvenanceEvent) {
|
||||
if e.Timestamp.IsZero() {
|
||||
e.Timestamp = time.Now()
|
||||
}
|
||||
s.mu.Lock()
|
||||
defer s.mu.Unlock()
|
||||
s.trails[e.MACode] = append(s.trails[e.MACode], e)
|
||||
}
|
||||
|
||||
// Trail 返回某 MA 码的全链路存证(需求22-AC1)。
|
||||
func (s *Store) Trail(maCode string) []model.ProvenanceEvent {
|
||||
s.mu.RLock()
|
||||
defer s.mu.RUnlock()
|
||||
out := make([]model.ProvenanceEvent, len(s.trails[maCode]))
|
||||
copy(out, s.trails[maCode])
|
||||
return out
|
||||
}
|
||||
|
||||
// Accountability 责任界定:以发码基准哈希为准,定位首次发生哈希变化的节点(需求22-AC2)。
|
||||
func (s *Store) Accountability(maCode string) model.AccountabilityReport {
|
||||
trail := s.Trail(maCode)
|
||||
report := model.AccountabilityReport{MACode: maCode, Trail: trail, Consistent: true}
|
||||
|
||||
// 基准哈希 = 发码节点(NodeIssue)的哈希
|
||||
for _, e := range trail {
|
||||
if e.Node == model.NodeIssue && e.HashValue != "" {
|
||||
report.BaselineHash = e.HashValue
|
||||
break
|
||||
}
|
||||
}
|
||||
if report.BaselineHash == "" {
|
||||
report.Conclusion = "未发码或无基准哈希,无法判定"
|
||||
report.Consistent = false
|
||||
return report
|
||||
}
|
||||
|
||||
// 检查后续携带哈希的节点是否与基准一致
|
||||
for i := range trail {
|
||||
e := trail[i]
|
||||
if e.HashValue == "" || e.Node == model.NodeIssue || e.Node == model.NodeSubmit {
|
||||
continue
|
||||
}
|
||||
// 转码节点哈希本就不同(合法),跳过
|
||||
if e.Node == model.NodeTranscode {
|
||||
continue
|
||||
}
|
||||
if e.HashValue != report.BaselineHash {
|
||||
report.Consistent = false
|
||||
ev := e
|
||||
report.FirstChange = &ev
|
||||
report.Conclusion = "在【" + string(e.Node) + "】节点(" + e.Operator + ")检出哈希与发码基准不一致,疑似该环节偷换/篡改"
|
||||
return report
|
||||
}
|
||||
}
|
||||
report.Conclusion = "全链路哈希与发码基准一致,审播一致,无偷换"
|
||||
return report
|
||||
}
|
||||
@@ -0,0 +1,97 @@
|
||||
package service
|
||||
|
||||
import (
|
||||
"testing"
|
||||
|
||||
"github.com/stretchr/testify/assert"
|
||||
"github.com/stretchr/testify/require"
|
||||
"github.com/tcs-iptv/tcs/internal/chain"
|
||||
"github.com/tcs-iptv/tcs/internal/model"
|
||||
)
|
||||
|
||||
// 全链路一致:发码→入库→正确注入,追责判定审播一致。
|
||||
func TestAccountability_Consistent(t *testing.T) {
|
||||
s := newService(t)
|
||||
maCode, ctid, cert := issueOne(t, s)
|
||||
require.NoError(t, s.IngestToLibrary(chain.RoleReviewer, maCode, ctid, "M", "陕西IPTV媒资库"))
|
||||
require.NoError(t, s.PublishToOperator(PublishRequest{MACode: maCode, Certificate: cert}))
|
||||
_, err := s.InjectToCDN(chain.RoleOperator, ctid, maCode, "filehash-abc", "CT-SX", "cdn://x")
|
||||
require.NoError(t, err)
|
||||
|
||||
rep := s.Accountability(maCode)
|
||||
assert.True(t, rep.Consistent, "全链路应一致")
|
||||
assert.Nil(t, rep.FirstChange)
|
||||
assert.Equal(t, "filehash-abc", rep.BaselineHash)
|
||||
assert.NotEmpty(t, rep.Trail)
|
||||
}
|
||||
|
||||
// 注入环节偷换:追责定位到 cdn_inject 节点与运营商。
|
||||
func TestAccountability_TamperLocated(t *testing.T) {
|
||||
s := newService(t)
|
||||
maCode, ctid, cert := issueOne(t, s)
|
||||
require.NoError(t, s.IngestToLibrary(chain.RoleReviewer, maCode, ctid, "M", "媒资库"))
|
||||
require.NoError(t, s.PublishToOperator(PublishRequest{MACode: maCode, Certificate: cert}))
|
||||
|
||||
// 直接往存证里记录一次"被偷换"的注入(绕过校验模拟违规留痕)
|
||||
// 实际中 InjectToCDN 会拒绝不匹配,但运营商侧若绕过校验偷换,存证仍会暴露
|
||||
_, _ = s.InjectToCDN(chain.RoleOperator, ctid, maCode, "filehash-abc", "CT-SX", "cdn://x")
|
||||
s.prov.Record(model.ProvenanceEvent{
|
||||
MACode: maCode, Node: model.NodeInject, HashValue: "TAMPERED",
|
||||
Operator: "CT-SX-违规", Detail: "疑似偷换",
|
||||
})
|
||||
|
||||
rep := s.Accountability(maCode)
|
||||
assert.False(t, rep.Consistent)
|
||||
require.NotNil(t, rep.FirstChange)
|
||||
assert.Equal(t, model.NodeInject, rep.FirstChange.Node)
|
||||
assert.Contains(t, rep.Conclusion, "cdn_inject")
|
||||
}
|
||||
|
||||
// 转码版哈希不同属合法,不应误判为篡改。
|
||||
func TestAccountability_TranscodeNotFlagged(t *testing.T) {
|
||||
s := newService(t)
|
||||
maCode, ctid, _ := issueOne(t, s)
|
||||
_, err := s.BindTranscoded(chain.RoleReviewer, ctid, "filehash-abc", "h265-4k", "H.265", "4K", "v1-4k")
|
||||
require.NoError(t, err)
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeTranscode, HashValue: "h265-4k", Operator: "转码中心"})
|
||||
|
||||
rep := s.Accountability(maCode)
|
||||
assert.True(t, rep.Consistent, "转码版哈希不同属合法,不应判为篡改")
|
||||
}
|
||||
|
||||
func TestCopyrightEvidence(t *testing.T) {
|
||||
s := newService(t)
|
||||
maCode, _, _ := issueOne(t, s)
|
||||
|
||||
ev, err := s.CopyrightEvidence(maCode)
|
||||
require.NoError(t, err)
|
||||
assert.Equal(t, maCode, ev.MACode)
|
||||
assert.NotEmpty(t, ev.Trail)
|
||||
assert.False(t, ev.FirstSeenAt.IsZero(), "应有最早登记时间戳")
|
||||
assert.Contains(t, ev.Statement, "谁先锁定谁有权")
|
||||
}
|
||||
|
||||
func TestMatchInfringement(t *testing.T) {
|
||||
s := newService(t)
|
||||
// 注册一部正版,感知哈希为合法 16 位十六进制(模拟 aHash/dHash 输出)
|
||||
sub := sampleSub()
|
||||
sub.Perceptual = "a1b2c3d4e5f60718"
|
||||
r, err := s.SubmitForReview(sub)
|
||||
require.NoError(t, err)
|
||||
require.NoError(t, s.ReviewCSPS(r.ReviewID, true, "rv-1"))
|
||||
issued, err := s.ApproveAndIssue(chain.RoleRegulator, r.ReviewID, "陕西IPTV")
|
||||
require.NoError(t, err)
|
||||
|
||||
// 完全相同的感知哈希 → 高度相似(距离 0)
|
||||
matches, err := s.MatchInfringement("a1b2c3d4e5f60718", 5, 10)
|
||||
require.NoError(t, err)
|
||||
require.NotEmpty(t, matches)
|
||||
assert.Equal(t, issued.MACode, matches[0].MACode)
|
||||
assert.Equal(t, "high", matches[0].Similarity)
|
||||
assert.Equal(t, 0, matches[0].Distance)
|
||||
|
||||
// 差异极大的哈希 → 不命中
|
||||
noMatch, err := s.MatchInfringement("ffffffffffffffff", 5, 10)
|
||||
require.NoError(t, err)
|
||||
assert.Empty(t, noMatch)
|
||||
}
|
||||
@@ -48,6 +48,7 @@ func (s *Service) IngestToLibrary(role chain.Role, maCode, ctid, mediaAssetID, l
|
||||
}); err != nil {
|
||||
return err
|
||||
}
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeIngest, Operator: libName, Detail: "审合格入媒资库"})
|
||||
return s.chain.SetContentStatus(maCode, model.StatusInLibrary)
|
||||
}
|
||||
|
||||
@@ -111,6 +112,7 @@ func (s *Service) InjectToCDN(role chain.Role, ctid, maCode, injectFileHash, ope
|
||||
}); err != nil {
|
||||
return InjectResult{}, err
|
||||
}
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeInject, HashValue: injectFileHash, Operator: operatorID, Detail: "CDN 注入校验通过"})
|
||||
return InjectResult{Allowed: true, DistributionID: distID}, nil
|
||||
}
|
||||
|
||||
@@ -205,3 +207,66 @@ func (s *Service) ComputeSettlement(maCode, period string) (model.Settlement, er
|
||||
}
|
||||
return s.pb.ComputeSettlement(maCode, period, model.DefaultShareConfig())
|
||||
}
|
||||
|
||||
// ---- 二期 F19/F20:追责取证与确权举证(需求22/23) ----
|
||||
|
||||
// Provenance 返回某 MA 码的全链路存证(需求22-AC1)。
|
||||
func (s *Service) Provenance(maCode string) []model.ProvenanceEvent {
|
||||
return s.prov.Trail(maCode)
|
||||
}
|
||||
|
||||
// Accountability 责任界定取证:定位首次哈希变化节点与责任方(需求22-AC2)。
|
||||
func (s *Service) Accountability(maCode string) model.AccountabilityReport {
|
||||
return s.prov.Accountability(maCode)
|
||||
}
|
||||
|
||||
// CopyrightEvidence 导出版权确权证据链(需求23-AC1/AC2)。
|
||||
func (s *Service) CopyrightEvidence(maCode string) (model.CopyrightEvidence, error) {
|
||||
c, err := s.chain.QueryContent(maCode)
|
||||
if err != nil {
|
||||
return model.CopyrightEvidence{}, err
|
||||
}
|
||||
trail := s.prov.Trail(maCode)
|
||||
ev := model.CopyrightEvidence{
|
||||
MACode: maCode, Title: c.Title, Issuer: c.Issuer, IssueDate: c.IssueDate,
|
||||
ContentHash: c.FileHash, ChainAnchor: "chain://" + maCode, Trail: trail,
|
||||
Statement: "本证据链由 MA 码、内容哈希与上链时间戳构成,遵循『谁先锁定谁有权』,不可抵赖,可用于侵权投诉与司法举证。",
|
||||
}
|
||||
for _, e := range trail {
|
||||
if e.Node == model.NodeSubmit {
|
||||
ev.FirstSeenAt = e.Timestamp
|
||||
if ev.ContentHash == "" {
|
||||
ev.ContentHash = e.HashValue
|
||||
}
|
||||
break
|
||||
}
|
||||
}
|
||||
return ev, nil
|
||||
}
|
||||
|
||||
// MatchInfringement 用感知哈希在已确权内容中检索疑似侵权(需求23-AC3)。
|
||||
// threshold 为汉明距离阈值(<=high 高度相似,<=medium 中度相似)。
|
||||
func (s *Service) MatchInfringement(perceptual string, high, medium int) ([]model.InfringeMatch, error) {
|
||||
s.mu.Lock()
|
||||
entries := make(map[string]phashEntry, len(s.phash))
|
||||
for k, v := range s.phash {
|
||||
entries[k] = v
|
||||
}
|
||||
s.mu.Unlock()
|
||||
|
||||
var out []model.InfringeMatch
|
||||
for ma, e := range entries {
|
||||
d, err := hash.HammingDistance(perceptual, e.Perceptual)
|
||||
if err != nil {
|
||||
continue // 长度不一致跳过
|
||||
}
|
||||
if d <= medium {
|
||||
sim := "medium"
|
||||
if d <= high {
|
||||
sim = "high"
|
||||
}
|
||||
out = append(out, model.InfringeMatch{MACode: ma, Title: e.Title, Distance: d, Similarity: sim})
|
||||
}
|
||||
}
|
||||
return out, nil
|
||||
}
|
||||
|
||||
@@ -13,6 +13,7 @@ import (
|
||||
"github.com/tcs-iptv/tcs/internal/macode"
|
||||
"github.com/tcs-iptv/tcs/internal/model"
|
||||
"github.com/tcs-iptv/tcs/internal/playback"
|
||||
"github.com/tcs-iptv/tcs/internal/provenance"
|
||||
)
|
||||
|
||||
// 业务错误。
|
||||
@@ -51,6 +52,8 @@ type Service struct {
|
||||
chain chain.Client
|
||||
gen *macode.Generator
|
||||
pb *playback.Store
|
||||
prov *provenance.Store
|
||||
phash map[string]phashEntry // maCode -> 感知哈希条目(确权侵权比对)
|
||||
mu sync.Mutex
|
||||
seqMu sync.Mutex
|
||||
seqs map[string]int // 按前缀独立计数(REV/ctid/DIST 各自从 1 递增)
|
||||
@@ -65,9 +68,21 @@ type reviewItem struct {
|
||||
MACode string
|
||||
}
|
||||
|
||||
// phashEntry 感知哈希注册项,用于确权侵权比对。
|
||||
type phashEntry struct {
|
||||
Title string
|
||||
Perceptual string
|
||||
}
|
||||
|
||||
// New 创建业务服务。
|
||||
func New(c chain.Client, gen *macode.Generator) *Service {
|
||||
return &Service{chain: c, gen: gen, pb: playback.NewStore(), seqs: make(map[string]int), reviews: make(map[string]*reviewItem)}
|
||||
return &Service{
|
||||
chain: c, gen: gen,
|
||||
pb: playback.NewStore(),
|
||||
prov: provenance.NewStore(),
|
||||
phash: make(map[string]phashEntry),
|
||||
seqs: make(map[string]int), reviews: make(map[string]*reviewItem),
|
||||
}
|
||||
}
|
||||
|
||||
func (s *Service) nextID(prefix string) string {
|
||||
@@ -190,6 +205,16 @@ func (s *Service) ApproveAndIssue(role chain.Role, reviewID, issuer string) (Iss
|
||||
PartyName: item.Sub.CPName,
|
||||
})
|
||||
|
||||
// 记录全链路存证(送审→审核→发码)+ 注册感知哈希供确权比对
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeSubmit, HashValue: item.Sub.FileHash, Operator: item.Sub.CPName, Detail: "CP 送审"})
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeCSPSReview, Operator: "CSPS", Detail: "合规审核通过"})
|
||||
s.prov.Record(model.ProvenanceEvent{MACode: maCode, Node: model.NodeIssue, HashValue: item.Sub.FileHash, Operator: issuer, Detail: "发码签发,绑定基准哈希"})
|
||||
if item.Sub.Perceptual != "" {
|
||||
s.mu.Lock()
|
||||
s.phash[maCode] = phashEntry{Title: item.Sub.Title, Perceptual: item.Sub.Perceptual}
|
||||
s.mu.Unlock()
|
||||
}
|
||||
|
||||
cert := fmt.Sprintf("CERT|%s|%s|%s", maCode, item.Sub.FileHash, item.Sub.MerkleRoot)
|
||||
return IssueResult{
|
||||
MACode: maCode,
|
||||
|
||||
Reference in New Issue
Block a user