v0.1
This commit is contained in:
		| @@ -12,35 +12,35 @@ import ( | ||||
| // 封装电子合同相关的核心信息,包含合同状态和签署流程管理 | ||||
| type ContractInfo struct { | ||||
| 	// 合同基本信息 | ||||
| 	ContractFileID     string    `json:"contract_file_id"`     // 合同文件ID | ||||
| 	EsignFlowID        string    `json:"esign_flow_id"`        // e签宝签署流程ID | ||||
| 	ContractURL        string    `json:"contract_url"`         // 合同文件访问链接 | ||||
| 	ContractSignURL    string    `json:"contract_sign_url"`    // 合同签署链接 | ||||
| 	 | ||||
| 	ContractFileID  string `json:"contract_file_id"`  // 合同文件ID | ||||
| 	EsignFlowID     string `json:"esign_flow_id"`     // e签宝签署流程ID | ||||
| 	ContractURL     string `json:"contract_url"`      // 合同文件访问链接 | ||||
| 	ContractSignURL string `json:"contract_sign_url"` // 合同签署链接 | ||||
|  | ||||
| 	// 合同元数据 | ||||
| 	ContractTitle      string    `json:"contract_title"`       // 合同标题 | ||||
| 	ContractVersion    string    `json:"contract_version"`     // 合同版本 | ||||
| 	TemplateID         string    `json:"template_id"`          // 模板ID | ||||
| 	 | ||||
| 	ContractTitle   string `json:"contract_title"`   // 合同标题 | ||||
| 	ContractVersion string `json:"contract_version"` // 合同版本 | ||||
| 	TemplateID      string `json:"template_id"`      // 模板ID | ||||
|  | ||||
| 	// 签署相关信息 | ||||
| 	SignerAccount      string    `json:"signer_account"`       // 签署人账号 | ||||
| 	SignerName         string    `json:"signer_name"`          // 签署人姓名 | ||||
| 	TransactorPhone    string    `json:"transactor_phone"`     // 经办人手机号 | ||||
| 	TransactorName     string    `json:"transactor_name"`      // 经办人姓名 | ||||
| 	TransactorIDCardNum string   `json:"transactor_id_card_num"` // 经办人身份证号 | ||||
| 	 | ||||
| 	SignerAccount       string `json:"signer_account"`         // 签署人账号 | ||||
| 	SignerName          string `json:"signer_name"`            // 签署人姓名 | ||||
| 	TransactorPhone     string `json:"transactor_phone"`       // 经办人手机号 | ||||
| 	TransactorName      string `json:"transactor_name"`        // 经办人姓名 | ||||
| 	TransactorIDCardNum string `json:"transactor_id_card_num"` // 经办人身份证号 | ||||
|  | ||||
| 	// 时间信息 | ||||
| 	GeneratedAt        *time.Time `json:"generated_at,omitempty"`        // 合同生成时间 | ||||
| 	SignFlowCreatedAt  *time.Time `json:"sign_flow_created_at,omitempty"` // 签署流程创建时间 | ||||
| 	SignedAt           *time.Time `json:"signed_at,omitempty"`           // 签署完成时间 | ||||
| 	ExpiresAt          *time.Time `json:"expires_at,omitempty"`          // 签署链接过期时间 | ||||
| 	 | ||||
| 	GeneratedAt       *time.Time `json:"generated_at,omitempty"`         // 合同生成时间 | ||||
| 	SignFlowCreatedAt *time.Time `json:"sign_flow_created_at,omitempty"` // 签署流程创建时间 | ||||
| 	SignedAt          *time.Time `json:"signed_at,omitempty"`            // 签署完成时间 | ||||
| 	ExpiresAt         *time.Time `json:"expires_at,omitempty"`           // 签署链接过期时间 | ||||
|  | ||||
| 	// 状态信息 | ||||
| 	Status             string    `json:"status"`               // 合同状态 | ||||
| 	SignProgress       int       `json:"sign_progress"`        // 签署进度 | ||||
| 	 | ||||
| 	Status       string `json:"status"`        // 合同状态 | ||||
| 	SignProgress int    `json:"sign_progress"` // 签署进度 | ||||
|  | ||||
| 	// 附加信息 | ||||
| 	Metadata           map[string]interface{} `json:"metadata,omitempty"` // 元数据 | ||||
| 	Metadata map[string]interface{} `json:"metadata,omitempty"` // 元数据 | ||||
| } | ||||
|  | ||||
| // ContractStatus 合同状态常量 | ||||
| @@ -57,19 +57,19 @@ const ( | ||||
| // NewContractInfo 创建合同信息值对象 | ||||
| func NewContractInfo(contractFileID, esignFlowID, contractURL, contractSignURL string) (*ContractInfo, error) { | ||||
| 	info := &ContractInfo{ | ||||
| 		ContractFileID:   strings.TrimSpace(contractFileID), | ||||
| 		EsignFlowID:      strings.TrimSpace(esignFlowID), | ||||
| 		ContractURL:      strings.TrimSpace(contractURL), | ||||
| 		ContractSignURL:  strings.TrimSpace(contractSignURL), | ||||
| 		Status:           ContractStatusGenerated, | ||||
| 		SignProgress:     0, | ||||
| 		Metadata:         make(map[string]interface{}), | ||||
| 		ContractFileID:  strings.TrimSpace(contractFileID), | ||||
| 		EsignFlowID:     strings.TrimSpace(esignFlowID), | ||||
| 		ContractURL:     strings.TrimSpace(contractURL), | ||||
| 		ContractSignURL: strings.TrimSpace(contractSignURL), | ||||
| 		Status:          ContractStatusGenerated, | ||||
| 		SignProgress:    0, | ||||
| 		Metadata:        make(map[string]interface{}), | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := info.Validate(); err != nil { | ||||
| 		return nil, fmt.Errorf("合同信息验证失败: %w", err) | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return info, nil | ||||
| } | ||||
|  | ||||
| @@ -78,27 +78,27 @@ func (c *ContractInfo) Validate() error { | ||||
| 	if err := c.validateContractFileID(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateEsignFlowID(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateContractURL(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateContractSignURL(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateSignerInfo(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateStatus(); err != nil { | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -107,12 +107,12 @@ func (c *ContractInfo) validateContractFileID() error { | ||||
| 	if c.ContractFileID == "" { | ||||
| 		return errors.New("合同文件ID不能为空") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 简单的格式验证 | ||||
| 	if len(c.ContractFileID) < 10 { | ||||
| 		return errors.New("合同文件ID格式不正确") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -121,12 +121,12 @@ func (c *ContractInfo) validateEsignFlowID() error { | ||||
| 	if c.EsignFlowID == "" { | ||||
| 		return errors.New("e签宝流程ID不能为空") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 简单的格式验证 | ||||
| 	if len(c.EsignFlowID) < 10 { | ||||
| 		return errors.New("e签宝流程ID格式不正确") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -135,18 +135,18 @@ func (c *ContractInfo) validateContractURL() error { | ||||
| 	if c.ContractURL == "" { | ||||
| 		return errors.New("合同访问链接不能为空") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// URL格式验证 | ||||
| 	urlPattern := `^https?://.*` | ||||
| 	matched, err := regexp.MatchString(urlPattern, c.ContractURL) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("合同访问链接格式验证错误: %w", err) | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if !matched { | ||||
| 		return errors.New("合同访问链接格式不正确,必须以http://或https://开头") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -155,18 +155,18 @@ func (c *ContractInfo) validateContractSignURL() error { | ||||
| 	if c.ContractSignURL == "" { | ||||
| 		return errors.New("合同签署链接不能为空") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// URL格式验证 | ||||
| 	urlPattern := `^https?://.*` | ||||
| 	matched, err := regexp.MatchString(urlPattern, c.ContractSignURL) | ||||
| 	if err != nil { | ||||
| 		return fmt.Errorf("合同签署链接格式验证错误: %w", err) | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if !matched { | ||||
| 		return errors.New("合同签署链接格式不正确,必须以http://或https://开头") | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -177,11 +177,11 @@ func (c *ContractInfo) validateSignerInfo() error { | ||||
| 		if c.SignerAccount == "" { | ||||
| 			return errors.New("签署人账号不能为空") | ||||
| 		} | ||||
| 		 | ||||
|  | ||||
| 		if c.SignerName == "" { | ||||
| 			return errors.New("签署人姓名不能为空") | ||||
| 		} | ||||
| 		 | ||||
|  | ||||
| 		if c.TransactorPhone != "" { | ||||
| 			// 手机号格式验证 | ||||
| 			phonePattern := `^1[3-9]\d{9}$` | ||||
| @@ -189,12 +189,12 @@ func (c *ContractInfo) validateSignerInfo() error { | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("经办人手机号格式验证错误: %w", err) | ||||
| 			} | ||||
| 			 | ||||
|  | ||||
| 			if !matched { | ||||
| 				return errors.New("经办人手机号格式不正确") | ||||
| 			} | ||||
| 		} | ||||
| 		 | ||||
|  | ||||
| 		if c.TransactorIDCardNum != "" { | ||||
| 			// 身份证号格式验证 | ||||
| 			idPattern := `^[1-9]\d{5}(19|20)\d{2}(0[1-9]|1[0-2])(0[1-9]|[12]\d|3[01])\d{3}[\dXx]$` | ||||
| @@ -202,13 +202,13 @@ func (c *ContractInfo) validateSignerInfo() error { | ||||
| 			if err != nil { | ||||
| 				return fmt.Errorf("经办人身份证号格式验证错误: %w", err) | ||||
| 			} | ||||
| 			 | ||||
|  | ||||
| 			if !matched { | ||||
| 				return errors.New("经办人身份证号格式不正确") | ||||
| 			} | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -223,13 +223,13 @@ func (c *ContractInfo) validateStatus() error { | ||||
| 		ContractStatusRejected, | ||||
| 		ContractStatusCancelled, | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	for _, status := range validStatuses { | ||||
| 		if c.Status == status { | ||||
| 			return nil | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return fmt.Errorf("无效的合同状态: %s", c.Status) | ||||
| } | ||||
|  | ||||
| @@ -240,7 +240,7 @@ func (c *ContractInfo) SetSignerInfo(signerAccount, signerName, transactorPhone, | ||||
| 	c.TransactorPhone = strings.TrimSpace(transactorPhone) | ||||
| 	c.TransactorName = strings.TrimSpace(transactorName) | ||||
| 	c.TransactorIDCardNum = strings.TrimSpace(transactorIDCardNum) | ||||
| 	 | ||||
|  | ||||
| 	return c.validateSignerInfo() | ||||
| } | ||||
|  | ||||
| @@ -248,15 +248,15 @@ func (c *ContractInfo) SetSignerInfo(signerAccount, signerName, transactorPhone, | ||||
| func (c *ContractInfo) UpdateStatus(status string) error { | ||||
| 	oldStatus := c.Status | ||||
| 	c.Status = status | ||||
| 	 | ||||
|  | ||||
| 	if err := c.validateStatus(); err != nil { | ||||
| 		c.Status = oldStatus // 回滚 | ||||
| 		return err | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 根据状态更新进度 | ||||
| 	c.updateProgressByStatus() | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -271,7 +271,7 @@ func (c *ContractInfo) updateProgressByStatus() { | ||||
| 		ContractStatusRejected:  50, | ||||
| 		ContractStatusCancelled: 0, | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if progress, exists := progressMap[c.Status]; exists { | ||||
| 		c.SignProgress = progress | ||||
| 	} | ||||
| @@ -283,7 +283,7 @@ func (c *ContractInfo) MarkAsSigning() error { | ||||
| 	c.SignProgress = 50 | ||||
| 	now := time.Now() | ||||
| 	c.SignFlowCreatedAt = &now | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -293,7 +293,7 @@ func (c *ContractInfo) MarkAsSigned() error { | ||||
| 	c.SignProgress = 100 | ||||
| 	now := time.Now() | ||||
| 	c.SignedAt = &now | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -302,7 +302,7 @@ func (c *ContractInfo) MarkAsExpired() error { | ||||
| 	c.Status = ContractStatusExpired | ||||
| 	now := time.Now() | ||||
| 	c.ExpiresAt = &now | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -310,7 +310,7 @@ func (c *ContractInfo) MarkAsExpired() error { | ||||
| func (c *ContractInfo) MarkAsRejected() error { | ||||
| 	c.Status = ContractStatusRejected | ||||
| 	c.SignProgress = 50 | ||||
| 	 | ||||
|  | ||||
| 	return nil | ||||
| } | ||||
|  | ||||
| @@ -319,7 +319,7 @@ func (c *ContractInfo) IsExpired() bool { | ||||
| 	if c.ExpiresAt == nil { | ||||
| 		return false | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return time.Now().After(*c.ExpiresAt) | ||||
| } | ||||
|  | ||||
| @@ -344,7 +344,7 @@ func (c *ContractInfo) GetStatusName() string { | ||||
| 		ContractStatusRejected:  "被拒绝", | ||||
| 		ContractStatusCancelled: "已取消", | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	if name, exists := statusNames[c.Status]; exists { | ||||
| 		return name | ||||
| 	} | ||||
| @@ -364,7 +364,7 @@ func (c *ContractInfo) GetMaskedSignerAccount() string { | ||||
| 	if len(c.SignerAccount) <= 6 { | ||||
| 		return c.SignerAccount | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 保留前3位和后3位,中间用*替代 | ||||
| 	return c.SignerAccount[:3] + "***" + c.SignerAccount[len(c.SignerAccount)-3:] | ||||
| } | ||||
| @@ -374,7 +374,7 @@ func (c *ContractInfo) GetMaskedTransactorPhone() string { | ||||
| 	if len(c.TransactorPhone) != 11 { | ||||
| 		return c.TransactorPhone | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 保留前3位和后4位,中间用*替代 | ||||
| 	return c.TransactorPhone[:3] + "****" + c.TransactorPhone[7:] | ||||
| } | ||||
| @@ -384,7 +384,7 @@ func (c *ContractInfo) GetMaskedTransactorIDCardNum() string { | ||||
| 	if len(c.TransactorIDCardNum) != 18 { | ||||
| 		return c.TransactorIDCardNum | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 保留前6位和后4位,中间用*替代 | ||||
| 	return c.TransactorIDCardNum[:6] + "********" + c.TransactorIDCardNum[14:] | ||||
| } | ||||
| @@ -411,7 +411,7 @@ func (c *ContractInfo) Equals(other *ContractInfo) bool { | ||||
| 	if other == nil { | ||||
| 		return false | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return c.ContractFileID == other.ContractFileID && | ||||
| 		c.EsignFlowID == other.EsignFlowID && | ||||
| 		c.Status == other.Status | ||||
| @@ -435,7 +435,7 @@ func (c *ContractInfo) Clone() *ContractInfo { | ||||
| 		Status:              c.Status, | ||||
| 		SignProgress:        c.SignProgress, | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 复制时间字段 | ||||
| 	if c.GeneratedAt != nil { | ||||
| 		generatedAt := *c.GeneratedAt | ||||
| @@ -453,7 +453,7 @@ func (c *ContractInfo) Clone() *ContractInfo { | ||||
| 		expiresAt := *c.ExpiresAt | ||||
| 		cloned.ExpiresAt = &expiresAt | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 复制元数据 | ||||
| 	if c.Metadata != nil { | ||||
| 		cloned.Metadata = make(map[string]interface{}) | ||||
| @@ -461,38 +461,38 @@ func (c *ContractInfo) Clone() *ContractInfo { | ||||
| 			cloned.Metadata[k] = v | ||||
| 		} | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return cloned | ||||
| } | ||||
|  | ||||
| // String 返回合同信息的字符串表示 | ||||
| func (c *ContractInfo) String() string { | ||||
| 	return fmt.Sprintf("合同信息[文件ID:%s, 流程ID:%s, 状态:%s, 进度:%d%%]",  | ||||
| 		c.ContractFileID,  | ||||
| 		c.EsignFlowID,  | ||||
| 		c.GetStatusName(),  | ||||
| 	return fmt.Sprintf("合同信息[文件ID:%s, 流程ID:%s, 状态:%s, 进度:%d%%]", | ||||
| 		c.ContractFileID, | ||||
| 		c.EsignFlowID, | ||||
| 		c.GetStatusName(), | ||||
| 		c.SignProgress) | ||||
| } | ||||
|  | ||||
| // ToMap 转换为map格式(用于序列化) | ||||
| func (c *ContractInfo) ToMap() map[string]interface{} { | ||||
| 	result := map[string]interface{}{ | ||||
| 		"contract_file_id":        c.ContractFileID, | ||||
| 		"esign_flow_id":           c.EsignFlowID, | ||||
| 		"contract_url":            c.ContractURL, | ||||
| 		"contract_sign_url":       c.ContractSignURL, | ||||
| 		"contract_title":          c.ContractTitle, | ||||
| 		"contract_version":        c.ContractVersion, | ||||
| 		"template_id":             c.TemplateID, | ||||
| 		"signer_account":          c.SignerAccount, | ||||
| 		"signer_name":             c.SignerName, | ||||
| 		"transactor_phone":        c.TransactorPhone, | ||||
| 		"transactor_name":         c.TransactorName, | ||||
| 		"transactor_id_card_num":  c.TransactorIDCardNum, | ||||
| 		"status":                  c.Status, | ||||
| 		"sign_progress":           c.SignProgress, | ||||
| 		"contract_file_id":       c.ContractFileID, | ||||
| 		"esign_flow_id":          c.EsignFlowID, | ||||
| 		"contract_url":           c.ContractURL, | ||||
| 		"contract_sign_url":      c.ContractSignURL, | ||||
| 		"contract_title":         c.ContractTitle, | ||||
| 		"contract_version":       c.ContractVersion, | ||||
| 		"template_id":            c.TemplateID, | ||||
| 		"signer_account":         c.SignerAccount, | ||||
| 		"signer_name":            c.SignerName, | ||||
| 		"transactor_phone":       c.TransactorPhone, | ||||
| 		"transactor_name":        c.TransactorName, | ||||
| 		"transactor_id_card_num": c.TransactorIDCardNum, | ||||
| 		"status":                 c.Status, | ||||
| 		"sign_progress":          c.SignProgress, | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 添加时间字段 | ||||
| 	if c.GeneratedAt != nil { | ||||
| 		result["generated_at"] = c.GeneratedAt | ||||
| @@ -506,11 +506,11 @@ func (c *ContractInfo) ToMap() map[string]interface{} { | ||||
| 	if c.ExpiresAt != nil { | ||||
| 		result["expires_at"] = c.ExpiresAt | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	// 添加元数据 | ||||
| 	if c.Metadata != nil { | ||||
| 		result["metadata"] = c.Metadata | ||||
| 	} | ||||
| 	 | ||||
|  | ||||
| 	return result | ||||
| }  | ||||
| } | ||||
|   | ||||
		Reference in New Issue
	
	Block a user