package contract
import (
"encoding/json"
"errors"
"fmt"
"github.com/beego/beego/v2/client/orm"
"github.com/shopspring/decimal"
"hongze/hz_crm_api/models"
"hongze/hz_crm_api/models/company"
"hongze/hz_crm_api/models/contract"
"hongze/hz_crm_api/models/contract/request"
"hongze/hz_crm_api/models/system"
"hongze/hz_crm_api/services"
"hongze/hz_crm_api/utils"
"mime/multipart"
"os"
"reflect"
"regexp"
"sort"
"strconv"
"strings"
"time"
)
// AddContract 新增合同
func AddContract(req request.AddContractReq, sellerId, productId int, sellerName string) (newContract *contract.Contract, err error, errMsg string) {
originalPrice := req.OriginalPrice
price := req.Price
if originalPrice < 0 {
err = errors.New("合同金额不能小于0")
return
}
//如果优惠后金额小于等于0,那么就是等于优惠前的金额
if price <= 0 {
price = originalPrice
}
//合同日期
var startDate, endDate time.Time
if req.ContractBusinessType == "代付合同" {
businessContractInfo, tmpErr := contract.GetContractById(req.RelationContractId)
if tmpErr != nil {
err = tmpErr
return
}
startDate = businessContractInfo.StartDate
endDate = businessContractInfo.EndDate
ok, tmpErr := CheckContractPrice(businessContractInfo, price)
if tmpErr != nil {
err = tmpErr
return
}
if !ok {
err = errors.New("合同金额异常,不能高于业务合同金额")
errMsg = "合同金额异常,不能高于业务合同金额"
return
}
} else {
tmpStartDate, tmpErr := time.ParseInLocation(utils.FormatDate, req.StartDate, time.Local)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同开始日期转换失败:", tmpErr))
return
}
startDate = tmpStartDate
tmpEndDate, tmpErr := time.ParseInLocation(utils.FormatDate, req.EndDate, time.Local)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同结束日期转换失败:", tmpErr))
return
}
endDate = tmpEndDate
}
companyName := req.CompanyName
if companyName == "" {
err = errors.New("客户名称必填")
return
}
creditCode := req.CreditCode
if companyName == "" {
err = errors.New("社会统一信用代码必填")
return
}
contractCode, err := contract.GetCompanyContractCode(productId, req.ContractBusinessType)
if err != nil {
err = errors.New(fmt.Sprint("合同编号生成异常,Err:", err))
return
}
//权益的新签合同重新定义,如果合同起始时间在第一份新签合同起始日期 1年之内的,仍为新签合同。
if productId == utils.COMPANY_PRODUCT_RAI_ID && req.ContractType == "" {
//根据社会信用码获取客户信息
companyInfo, tmpErr := company.GetCompanyByCreditCode(creditCode)
//如果查询异常,且并不是在系统中找不到该社会信用码,那么就异常返回
if tmpErr != nil {
err = errors.New(fmt.Sprint("根据社会信用码获取客户信息,Err:", tmpErr))
return
}
totalLastYear, tmpErr := GetCompanyContractCountRaiByLastYear(companyInfo.CompanyId, req.StartDate)
if tmpErr != nil {
err = errors.New(fmt.Sprint("获取第一份新签合同起始日期是否1年之内失败,Err:", tmpErr))
return
}
if totalLastYear > 0 {
req.ContractType = "新签合同"
} else {
req.ContractType = "续约合同"
}
}
contractInfo := &contract.Contract{
ContractCode: contractCode,
SellerId: sellerId,
SellerName: sellerName,
ProductId: productId,
ContractType: req.ContractType,
ContractBusinessType: req.ContractBusinessType,
Status: "待提交",
StartDate: startDate,
EndDate: endDate,
OriginalPrice: originalPrice,
Price: price,
PayRemark: utils.TrimStr(req.PayRemark), //付款方式说明,长度255位
PayChannel: utils.TrimStr(req.PayChannel), //付款渠道,长度255位
CompanyName: utils.TrimStr(companyName),
CreditCode: utils.TrimStr(creditCode),
ProvinceId: req.ProvinceId,
Province: utils.TrimStr(req.Province),
CityId: req.CityId,
City: utils.TrimStr(req.City),
Address: utils.TrimStr(req.Address),
Fax: utils.TrimStr(req.Fax),
Phone: utils.TrimStr(req.Phone),
Postcode: utils.TrimStr(req.Postcode),
Remark: req.Remark,
SellerRemark: req.SellerRemark, //销售备注
TemplateId: req.TemplateId,
SourceId: req.SourceId, //来源合同id
ModifyTime: time.Now(),
CreateTime: time.Now(),
}
contractServiceSlice, err := getContractServiceAndDetail(contractInfo.ProductId, req.Service, true)
if err != nil {
return
}
newContract, err = contract.AddContract(contractInfo, contractServiceSlice, req.RelationContractId)
if err != nil {
return
}
//添加操作日志
remark := "新增合同模板"
if newContract.SourceId > 0 {
remark = "复制模板"
}
_ = contract.AddContractOperationRecord(newContract.ContractId, newContract.SellerId, 0, "add", newContract.SellerName, remark)
return
}
// EditContract 编辑合同
func EditContract(req request.EditContractReq, opUser *system.Admin) (newContract *contract.Contract, err error, errMsg string) {
//查询当前合同信息
nowContract, err := contract.GetContractById(req.ContractId)
if err != nil {
if err.Error() == utils.ErrNoRow() {
err = errors.New(fmt.Sprint("根据合同编号:", req.ContractId, " 找不到初始合同"))
}
return
}
if nowContract.SellerId != opUser.AdminId {
err = errors.New("当前账号无操作权限")
return
}
ignoreStatus := []string{"待提交", "已撤回", "已驳回", "已审批"}
if !strings.Contains(strings.Join(ignoreStatus, ","), nowContract.Status) {
err = errors.New("合同状态异常,不允许修改,当前合同状态:" + nowContract.Status)
return
}
originalPrice := req.OriginalPrice
price := req.Price
if originalPrice < 0 {
err = errors.New("合同金额不能小于0")
return
}
//如果优惠后金额小于等于0,那么就是等于优惠前的金额
if price <= 0 {
price = originalPrice
}
checkPrice := price //代付合同,需要检查的需要支付的余额
// 是否更新业务合同金额
updateBusinessContractPrice := 0.0
if nowContract.Status == "已审批" && nowContract.ContractBusinessType == "代付合同" {
updateBusinessContractPrice = nowContract.Price
//因为当前代付合同是已审批的,所以修改重审的时候:
//需要检查的需要支付的余额 = 当前提交的金额 - 之前提交的金额
checkPrice = price - nowContract.Price
}
//合同日期
var startDate, endDate time.Time
if nowContract.ContractBusinessType == "代付合同" {
businessContractInfo, tmpErr := contract.GetContractById(req.RelationContractId)
if tmpErr != nil {
err = tmpErr
return
}
startDate = businessContractInfo.StartDate
endDate = businessContractInfo.EndDate
ok, tmpErr := CheckContractPrice(businessContractInfo, checkPrice)
if tmpErr != nil {
err = tmpErr
return
}
if !ok {
err = errors.New("合同金额异常,不能高于业务合同金额")
errMsg = "合同金额异常,不能高于业务合同金额"
return
}
} else {
tmpStartDate, tmpErr := time.ParseInLocation(utils.FormatDate, req.StartDate, time.Local)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同开始日期转换失败:", tmpErr))
return
}
startDate = tmpStartDate
tmpEndDate, tmpErr := time.ParseInLocation(utils.FormatDate, req.EndDate, time.Local)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同结束日期转换失败:", tmpErr))
return
}
endDate = tmpEndDate
}
companyName := utils.TrimStr(req.CompanyName)
if companyName == "" {
err = errors.New("客户名称必填")
return
}
creditCode := utils.TrimStr(req.CreditCode)
if companyName == "" {
err = errors.New("社会统一信用代码必填")
return
}
modifyContent := "" //变更记录
nowContract.ContractType = req.ContractType //合同类型
nowContract.StartDate = startDate //合同开始日期
nowContract.EndDate = endDate //合同结束日期
nowContract.OriginalPrice = originalPrice //原始金额
nowContract.Price = price //优惠后的金额
nowContract.PayRemark = utils.TrimStr(req.PayRemark) //付款方式说明,长度255位
nowContract.PayChannel = utils.TrimStr(req.PayChannel) //付款渠道,长度255位
nowContract.CompanyName = companyName //企业客户名称
nowContract.CreditCode = creditCode //组织社会信用代码
nowContract.ProvinceId = req.ProvinceId //省级id
nowContract.Province = utils.TrimStr(req.Province) //省
nowContract.CityId = req.CityId //市级id
nowContract.City = utils.TrimStr(req.City) //市
nowContract.Address = utils.TrimStr(req.Address) //详细地址
nowContract.Fax = utils.TrimStr(req.Fax) //传真
nowContract.Phone = utils.TrimStr(req.Phone) //电话
nowContract.FileUrl = "" //修改合同后,需要移除原有的合同附件
nowContract.Postcode = utils.TrimStr(req.Postcode) //邮编
nowContract.Remark = req.Remark //补充说明
nowContract.SellerRemark = req.SellerRemark //销售备注
nowContract.ModifyTime = time.Now() //更新时间
nowContract.Status = "待提交" //合同状态
//获取服务信息
contractServiceSlice, err := getContractServiceAndDetail(nowContract.ProductId, req.Service, true)
if err != nil {
return
}
//最近一次审批通过或者驳回的申请单
lastContractApproval, err := contract.GetLastContractApprovalByContractIdAndStatus(nowContract.ContractId, "contract")
if err != nil && err.Error() != utils.ErrNoRow() {
return
}
//如果有申请记录,且申请单的状态是已审批,那么审核内容是:修改重审
if lastContractApproval != nil {
//旧的合同信息
var snapshotContract contract.ContractDetail
jsonErr := json.Unmarshal([]byte(lastContractApproval.ContractDetail), &snapshotContract)
if jsonErr != nil {
err = jsonErr
return
}
//applyContent = "修改重审"
modifyContentStrList := make([]string, 0)
//客户信息
updateCompanyInfoStrList := contrastCompanyInfo(snapshotContract, *nowContract)
if len(updateCompanyInfoStrList) > 0 {
updateCompanyInfoStr := ``
if len(updateCompanyInfoStrList) == 1 {
updateCompanyInfoStr = fmt.Sprint("客户信息发生了变化:" + updateCompanyInfoStrList[0])
} else {
updateCompanyInfoStr = fmt.Sprint("客户信息发生了变化:", strings.Join(updateCompanyInfoStrList, ""))
}
modifyContentStrList = append(modifyContentStrList, updateCompanyInfoStr)
}
//金额、付款、合同日期方式
updatePayInfoStrList := contrastPayInfo(snapshotContract, *nowContract)
modifyContentStrList = append(modifyContentStrList, updatePayInfoStrList...)
if len(modifyContentStrList) > 0 {
modifyContent = strings.Join(updatePayInfoStrList, "
")
}
//服务内容
updateServiceStr, tmpErr := contrastServiceInfo(snapshotContract, contractServiceSlice)
if tmpErr != nil {
err = tmpErr
return
}
if updateServiceStr != "" {
modifyContentStrList = append(modifyContentStrList, updateServiceStr)
}
//补充内容
if nowContract.Remark != snapshotContract.Remark {
snapshotContractRemark := snapshotContract.Remark
if snapshotContractRemark == "" {
snapshotContractRemark = "无"
}
nowContractRemark := nowContract.Remark
if nowContractRemark == "" {
nowContractRemark = "无"
}
updateStr := fmt.Sprint("原补充内容:", snapshotContractRemark, "现补充内容:", nowContractRemark)
modifyContentStrList = append(modifyContentStrList, updateStr)
}
if len(modifyContentStrList) > 0 {
modifyContent = strings.Join(modifyContentStrList, "
")
}
}
nowContract.ModifyContent = modifyContent //修改信息
updateDetailCol := []string{"ContractType", "StartDate", "EndDate", "OriginalPrice", "Price", "PayRemark", "PayChannel", "CompanyName", "CreditCode", "ProvinceId", "Province", "CityId", "City", "Address", "Fax", "Phone", "FileUrl", "Postcode", "Remark", "SellerRemark", "ModifyContent", "ModifyTime", "Status"}
//修改合同
err = contract.EditContract(nowContract, updateDetailCol, contractServiceSlice, req.RelationContractId)
//添加操作日志
remark := "编辑合同模板"
_ = contract.AddContractOperationRecord(nowContract.ContractId, nowContract.SellerId, 0, "edit", nowContract.SellerName, remark)
//修改业务合同的已支付金额
if updateBusinessContractPrice > 0 {
go updateContractPaidPriceByPaymentOnBehalfContractId(nowContract.ContractId, -nowContract.Price)
}
//赋值返回
newContract = nowContract
return
}
// contrastCompanyInfo 比对客户信息是否变更
func contrastCompanyInfo(snapshotContract contract.ContractDetail, nowContract contract.Contract) (updateStrList []string) {
//客户信息比对
if nowContract.CompanyName != snapshotContract.CompanyName {
updateStrList = append(updateStrList, fmt.Sprint("原客户名称:", snapshotContract.CompanyName, "现客户名称:", nowContract.CompanyName))
}
if nowContract.CreditCode != snapshotContract.CreditCode {
updateStrList = append(updateStrList, fmt.Sprint("原社会信用码:", snapshotContract.CreditCode, "现社会信用码:", nowContract.CreditCode))
}
if nowContract.Province != snapshotContract.Province || nowContract.City != snapshotContract.City {
updateStrList = append(updateStrList, fmt.Sprint("原公司地址:", snapshotContract.Province, snapshotContract.City, "现公司地址:", nowContract.Province, nowContract.City))
}
if nowContract.Address != snapshotContract.Address {
snapshotContractAddress := snapshotContract.Address
if snapshotContractAddress == "" {
snapshotContractAddress = `无`
}
nowContractAddress := nowContract.Address
if nowContractAddress == "" {
nowContractAddress = `无`
}
updateStrList = append(updateStrList, fmt.Sprint("原详细地址:", snapshotContractAddress, "现详细地址:", nowContractAddress))
}
if nowContract.Fax != snapshotContract.Fax {
snapshotContractFax := snapshotContract.Fax
if snapshotContractFax == "" {
snapshotContractFax = `无`
}
nowContractFax := nowContract.Fax
if nowContractFax == "" {
nowContractFax = `无`
}
updateStrList = append(updateStrList, fmt.Sprint("原传真:", snapshotContractFax, "现传真:", nowContractFax))
}
if nowContract.Phone != snapshotContract.Phone {
snapshotContractPhone := snapshotContract.Phone
if snapshotContractPhone == "" {
snapshotContractPhone = `无`
}
nowContractPhone := nowContract.Phone
if nowContractPhone == "" {
nowContractPhone = `无`
}
updateStrList = append(updateStrList, fmt.Sprint("原电话:", snapshotContractPhone, "现电话:", nowContractPhone))
}
if nowContract.Postcode != snapshotContract.Postcode {
snapshotContractPostcode := snapshotContract.Postcode
if snapshotContractPostcode == "" {
snapshotContractPostcode = `无`
}
nowContractPostcode := nowContract.Postcode
if nowContractPostcode == "" {
nowContractPostcode = `无`
}
updateStrList = append(updateStrList, fmt.Sprint("原邮编:", snapshotContractPostcode, "现邮编:", nowContractPostcode))
}
return
}
// contrastPayInfo 比对合同金额、付款方式是否变更
func contrastPayInfo(snapshotContract contract.ContractDetail, nowContract contract.Contract) (updateStrList []string) {
//合同信息比对
if nowContract.StartDate != snapshotContract.StartDate || nowContract.EndDate != snapshotContract.EndDate {
updateStrList = append(updateStrList, fmt.Sprint("原合同期限:", snapshotContract.StartDate.Format(utils.FormatDate), " ~ ", snapshotContract.EndDate.Format(utils.FormatDate), "现合同期限:", nowContract.StartDate.Format(utils.FormatDate), " ~ ", nowContract.EndDate.Format(utils.FormatDate)))
}
if nowContract.ContractBusinessType == "代付合同" {
if nowContract.Price != snapshotContract.Price {
updateStrList = append(updateStrList, fmt.Sprint("原代付金额:", decimal.NewFromFloat(snapshotContract.Price).String(), "现代付金额:", decimal.NewFromFloat(nowContract.Price).String()))
}
} else {
if nowContract.OriginalPrice != snapshotContract.OriginalPrice {
updateStrList = append(updateStrList, fmt.Sprint("原合同金额:", decimal.NewFromFloat(snapshotContract.OriginalPrice).String(), "现合同金额:", decimal.NewFromFloat(nowContract.OriginalPrice).String()))
}
if nowContract.Price != snapshotContract.Price {
updateStrList = append(updateStrList, fmt.Sprint("原优惠后金额:", decimal.NewFromFloat(snapshotContract.Price).String(), "现优惠后金额:", decimal.NewFromFloat(nowContract.Price).String()))
}
}
if nowContract.PayRemark != snapshotContract.PayRemark {
snapshotContractInfo := snapshotContract.PayRemark
if snapshotContractInfo == "" {
snapshotContractInfo = "无"
}
nowContractInfo := nowContract.PayRemark
if nowContractInfo == "" {
nowContractInfo = "无"
}
updateStrList = append(updateStrList, fmt.Sprint("原付款方式:", snapshotContractInfo, "现付款方式:", nowContractInfo))
}
if nowContract.PayChannel != snapshotContract.PayChannel {
snapshotContractInfo := snapshotContract.PayChannel
if snapshotContractInfo == "" {
snapshotContractInfo = "无"
}
nowContractInfo := nowContract.PayChannel
if nowContractInfo == "" {
nowContractInfo = "无"
}
updateStrList = append(updateStrList, fmt.Sprint("原付款方:", snapshotContractInfo, "现付款方:", nowContractInfo))
}
return
}
// contrastServiceInfo 比对合同服务内容是否变更
func contrastServiceInfo(snapshotContract contract.ContractDetail, contractServiceSlice []*contract.ContractServiceAndDetail) (updateStr string, err error) {
serviceList := snapshotContract.Service
nowServiceList := contractServiceSlice
// CRM8.8-权益的合同服务主客观合并
if snapshotContract.ProductId == 2 {
updateStr = handleRaiContractModifyContentByServices(serviceList, nowServiceList)
return
}
snapshotPermissionList, err := GetPermissionByContractService(snapshotContract.ProductId, serviceList)
if err != nil {
return
}
allPermissionMap := make(map[int]string)
snapshotPermissionIdList := make([]int, 0)
for _, permissionItem := range snapshotPermissionList {
for _, permission := range permissionItem.Items {
allPermissionMap[permission.ChartPermissionId] = permission.PermissionName
if snapshotContract.ProductId == 2 {
allPermissionMap[permission.ChartPermissionId] = permission.Remark
}
}
snapshotPermissionIdList = append(snapshotPermissionIdList, permissionItem.CheckList...)
}
if len(snapshotPermissionIdList) > 0 {
sort.Ints(snapshotPermissionIdList)
}
nowPermissionList, err := GetPermissionByContractService(snapshotContract.ProductId, nowServiceList)
if err != nil {
return
}
nowPermissionIdList := make([]int, 0)
for _, permissionItem := range nowPermissionList {
nowPermissionIdList = append(nowPermissionIdList, permissionItem.CheckList...)
}
if len(nowPermissionIdList) > 0 {
sort.Ints(nowPermissionIdList)
}
//长度不等
if len(snapshotPermissionIdList) != len(nowPermissionIdList) {
goto FalseContrast
}
for i := 0; i < len(snapshotPermissionIdList); i++ {
if snapshotPermissionIdList[i] != nowPermissionIdList[i] {
//权限id不匹配
goto FalseContrast
}
}
return
FalseContrast:
snapshotContractServiceStrList := make([]string, 0)
for _, id := range snapshotPermissionIdList {
if permissionName, ok := allPermissionMap[id]; ok {
snapshotContractServiceStrList = append(snapshotContractServiceStrList, permissionName)
}
}
nowContractServiceStrList := make([]string, 0)
for _, id := range nowPermissionIdList {
if permissionName, ok := allPermissionMap[id]; ok {
nowContractServiceStrList = append(nowContractServiceStrList, permissionName)
}
}
snapshotContractServiceStr := `无`
if len(snapshotContractServiceStrList) > 0 {
snapshotContractServiceStr = strings.Join(snapshotContractServiceStrList, ",")
}
nowContractServiceStr := `无`
if len(nowContractServiceStrList) > 0 {
nowContractServiceStr = strings.Join(nowContractServiceStrList, ",")
}
updateStr = fmt.Sprint("原服务内容:", snapshotContractServiceStr, "现服务内容:", nowContractServiceStr)
return
}
// DeleteContract 删除合同
func DeleteContract(contractId int, opUser *system.Admin) (err error) {
//查询当前合同信息
nowContract, err := contract.GetContractById(contractId)
if err != nil {
if err.Error() == utils.ErrNoRow() {
err = errors.New(fmt.Sprint("根据合同编号:", contractId, " 找不到初始合同"))
}
return
}
if nowContract.SellerId != opUser.AdminId {
err = errors.New("当前账号无操作权限")
return
}
ignoreStatus := []string{"已审批", "待审批"}
if strings.Contains(strings.Join(ignoreStatus, ","), nowContract.Status) {
err = errors.New("合同状态异常,不允许删除,当前合同状态:" + nowContract.Status)
return
}
//如果删除状态 >0,那么代表已经被删除了
if nowContract.IsDelete > 0 {
err = errors.New("该合同已删除")
return
}
//假删除
err = contract.DeleteContract(nowContract)
if err != nil {
return
}
//添加操作日志
//remark := "编辑合同模板"
//_ = contract.AddContractOperationRecord(nowContract.ContractId, nowContract.SellerId, 0, "edit", nowContract.SellerName, remark)
return
}
// InvalidContract 作废合同
func InvalidContract(contractId int, opUser *system.Admin) (err error) {
//查询当前合同信息
nowContract, err := contract.GetContractById(contractId)
if err != nil {
if err.Error() == utils.ErrNoRow() {
err = errors.New(fmt.Sprint("根据合同编号:", contractId, " 找不到初始合同"))
}
return
}
if nowContract.SellerId != opUser.AdminId && opUser.RoleTypeCode != utils.ROLE_TYPE_CODE_COMPLIANCE {
err = errors.New("当前账号无操作权限")
return
}
if nowContract.Status != "已审批" {
err = errors.New("合同状态异常,不允许作废,当前合同状态:" + nowContract.Status)
return
}
//如果删除状态 >0,那么代表已经被删除了
if nowContract.IsDelete > 0 {
err = errors.New("该合同已删除")
return
}
//合同作废
err = contract.InvalidContract(nowContract)
if err != nil {
return
}
//添加操作日志
remark := "作废合同模板"
_ = contract.AddContractOperationRecord(nowContract.ContractId, opUser.AdminId, 0, "invalid", opUser.AdminName, remark)
//修改业务合同的已支付金额
if nowContract.ContractBusinessType == "代付合同" {
go updateContractPaidPriceByPaymentOnBehalfContractId(nowContract.ContractId, -nowContract.Price)
}
return
}
// UploadFile 上传更新合同附件
func UploadFile(contractId int, ext string, fileMulti multipart.File, opUser *system.Admin) (err error) {
//获取合同信息
contractInfo, err := contract.GetContractById(contractId)
if err != nil {
return
}
//合同状态判断
if contractInfo.Status != "已审批" {
err = errors.New("合同状态异常,不允许更新合同附件,当前合同状态:" + contractInfo.Status)
return
}
if contractInfo.SellerId != opUser.AdminId {
err = errors.New("合同异常,只允许本人更新合同附件")
return
}
//保存的文件名
fileName := contractInfo.CompanyName + "_" + contractInfo.ContractCode
//非正式环境下,文件名上面还是加上随机数
if utils.RunMode != "release" {
fileName += "_" + utils.GetRandStringNoSpecialChar(10)
}
fileName += ext
//上传到阿里云
resourceUrl, err := services.UploadToOssAndFileName(fileMulti, fileName)
if err != nil {
err = errors.New("文件保存失败,Err:" + err.Error())
return
}
contractInfo.FileUrl = resourceUrl
contractInfo.ModifyTime = time.Now()
err = contractInfo.Update([]string{"FileUrl", "ModifyTime"})
if err != nil {
return
}
//添加操作日志
remark := "更新合同附件"
_ = contract.AddContractOperationRecord(contractInfo.ContractId, opUser.AdminId, 0, "update_file", opUser.RealName, remark)
return
}
// UploadCheckBackFile 上传签回合同附件
func UploadCheckBackFile(contractId int, ext string, fileMulti multipart.File, opUser *system.Admin) (resourceUrl string, err error) {
//获取合同信息
contractInfo, err := contract.GetContractById(contractId)
if err != nil {
return
}
//合同状态判断
if contractInfo.Status != "已审批" && contractInfo.Status != "已签回" {
err = errors.New("合同状态异常,不允许上传签回合同附件,当前合同状态:" + contractInfo.Status)
return
}
//保存的文件名
fileName := contractInfo.CompanyName + "_" + contractInfo.ContractCode + "(签回)"
//非正式环境下,文件名上面还是加上随机数
if utils.RunMode != "release" {
fileName += "_" + utils.GetRandStringNoSpecialChar(10)
}
fileName += ext
//上传到阿里云
resourceUrl, err = services.UploadToOssAndFileName(fileMulti, fileName)
if err != nil {
err = errors.New("文件保存失败,Err:" + err.Error())
return
}
contractInfo.CheckBackFileUrl = resourceUrl
contractInfo.Status = "已签回"
contractInfo.ModifyTime = time.Now()
contractInfo.CheckBackFileTime = time.Now()
err = contractInfo.Update([]string{"CheckBackFileUrl", "Status", "ModifyTime", "CheckBackFileTime"})
if err != nil {
return
}
//同步更新客户那边提交审批的合同
o := orm.NewOrm()
sql := `update company_contract set img_url = ? where contract_code=? and source="系统合同" `
_, err = o.Raw(sql, contractInfo.CheckBackFileUrl, contractInfo.ContractCode).Exec()
//添加操作日志
remark := "上传签回合同附件"
_ = contract.AddContractOperationRecord(contractInfo.ContractId, opUser.AdminId, 0, "upload", opUser.RealName, remark)
return
}
// UploadRescindFile 上传解约合同附件
func UploadRescindFile(contractId int, ext string, fileMulti multipart.File, opUser *system.Admin) (err error) {
//获取合同信息
contractInfo, err := contract.GetContractById(contractId)
if err != nil {
return
}
//合同状态判断
if contractInfo.Status != "已签回" {
err = errors.New("合同状态异常,不允许上传解约合同附件,当前合同状态:" + contractInfo.Status)
return
}
//保存的文件名
fileName := contractInfo.CompanyName + "_" + contractInfo.ContractCode + "(解约)"
//非正式环境下,文件名上面还是加上随机数
if utils.RunMode != "release" {
fileName += "_" + utils.GetRandStringNoSpecialChar(10)
}
fileName += ext
//上传到阿里云
resourceUrl, err := services.UploadToOssAndFileName(fileMulti, fileName)
if err != nil {
err = errors.New("文件保存失败,Err:" + err.Error())
return
}
contractInfo.RescindFileUrl = resourceUrl
contractInfo.Status = "已解约"
contractInfo.ModifyTime = time.Now()
contractInfo.RescindTime = time.Now()
err = contractInfo.Update([]string{"RescindFileUrl", "Status", "ModifyTime", "RescindTime"})
if err != nil {
return
}
//添加操作日志
remark := "上传解约合同附件"
_ = contract.AddContractOperationRecord(contractInfo.ContractId, opUser.AdminId, 0, "rescind", opUser.RealName, remark)
//修改业务合同的已支付金额
if contractInfo.ContractBusinessType == "代付合同" {
go updateContractPaidPriceByPaymentOnBehalfContractId(contractInfo.ContractId, -contractInfo.Price)
}
return
}
// 根据id获取合同详情(包含服务)
func GetContractDetail(contractId int) (contractDetail *contract.ContractDetail, err error) {
contractDetail, err = getContractDetail(contractId)
if err != nil {
if err.Error() == utils.ErrNoRow() {
err = errors.New("找不到该合同")
}
return
}
relationContractDetailList := make([]*contract.ContractDetail, 0)
switch contractDetail.ContractBusinessType {
case "业务合同":
tmpList, tmpErr := contract.GetContractRelationListByRelationContractId(contractDetail.ContractId)
if tmpErr != nil {
err = errors.New("查询业务关联合同异常:" + tmpErr.Error())
return
}
//获取所有关联合同详情
for _, relationContract := range tmpList {
tmpContractDetail, tmpErr := getContractDetail(relationContract.PaymentOnBehalfContractId)
if tmpErr != nil {
err = errors.New("查询业务关联合同详情异常:" + tmpErr.Error())
return
}
relationContractDetailList = append(relationContractDetailList, tmpContractDetail)
}
case "代付合同":
tmpList, tmpErr := contract.GetContractRelationListByPaymentOnBehalfContractId(contractDetail.ContractId)
if tmpErr != nil {
err = errors.New("查询代付关联合同异常:" + tmpErr.Error())
return
}
//获取所有关联合同详情
for _, relationContract := range tmpList {
tmpContractDetail, tmpErr := getContractDetail(relationContract.ContractId)
if tmpErr != nil {
err = errors.New("查询业务关联合同详情异常:" + tmpErr.Error())
return
}
relationContractDetailList = append(relationContractDetailList, tmpContractDetail)
}
}
contractDetail.RelationContractDetailList = relationContractDetailList
return
}
// getContractDetail 根据id获取合同详情(包含服务)
func getContractDetail(contractId int) (contractDetail *contract.ContractDetail, err error) {
//获取合同基础信息详情
contractDetail, err = contract.GetContractDetailById(contractId)
if err != nil {
if err.Error() == utils.ErrNoRow() {
err = errors.New("找不到该合同")
}
return
}
//获取合同服务内容详情
serviceList, err := getContractServiceAndDetailList(contractId)
if err != nil {
err = errors.New(fmt.Sprint("查找合同服务异常", err))
return
}
contractDetail.Service = serviceList
return
}
// getContractServiceAndDetailList 根据合同Id获取服务详情列表
func getContractServiceAndDetailList(contractId int) (serviceList []*contract.ContractServiceAndDetail, err error) {
serviceList, err = contract.GetContractServiceAndDetailList(contractId)
if err != nil {
err = errors.New(fmt.Sprint("查找合同服务异常", err))
return
}
// 权益合同主客观合并
//if mergeEquity && len(serviceList) > 0 && serviceList[0].ProductId == 2 {
// mergeList, e := EquityMergeSubjectAndObject(serviceList)
// if e != nil {
// err = errors.New(fmt.Sprint("合并权益合同服务异常", err))
// return
// }
// serviceList = mergeList
//}
newLen := len(serviceList)
for i := 0; newLen > i; i++ {
if serviceList[i].HasDetail == "是" {
list, detailErr := contract.GetContractServiceDetailListByServiceId(serviceList[i].ContractServiceId)
if detailErr != nil {
err = errors.New(fmt.Sprint("查找合同服务详情异常", detailErr))
return
}
serviceList[i].DetailList = list
}
}
return
}
// GetContractDetailByAddContractReq 根据请求参数获取合同详情结构体数据
func GetContractDetailByAddContractReq(req request.AddContractReq, sellerId, productId int, sellerName string) (contractDetail *contract.ContractDetail, err error) {
originalPrice := req.OriginalPrice
price := req.Price
if originalPrice < 0 {
err = errors.New("合同金额不能小于0")
return
}
//如果优惠后金额小于等于0,那么就是等于优惠前的金额
if price <= 0 {
price = originalPrice
}
companyName := req.CompanyName
if companyName == "" {
err = errors.New("客户名称必填")
return
}
creditCode := req.CreditCode
if companyName == "" {
err = errors.New("社会统一信用代码必填")
return
}
contractCode, err := contract.GetCompanyContractCode(productId, req.ContractBusinessType)
if err != nil {
err = errors.New(fmt.Sprint("合同编号生成异常,Err:", err))
return
}
//校验合同模板类型
contractTemplate, err := contract.GetContractTemplateByTemplateId(req.TemplateId)
if err != nil {
return
}
if contractTemplate.ProductId != productId {
err = errors.New("模板类型与当前账户类型不符")
return
}
contractDetail = &contract.ContractDetail{
ContractId: int(time.Now().Unix()),
ContractCode: contractCode,
SellerId: sellerId,
SellerName: sellerName,
ProductId: productId,
ContractBusinessType: req.ContractBusinessType,
ContractType: req.ContractType,
Status: "待提交",
OriginalPrice: originalPrice,
Price: price,
PayRemark: req.PayRemark, //付款方式说明,长度255位
PayChannel: req.PayChannel, //付款渠道,长度255位
CompanyName: companyName,
CreditCode: creditCode,
ProvinceId: req.ProvinceId,
Province: req.Province,
CityId: req.CityId,
City: req.City,
Address: req.Address,
Fax: req.Fax,
Phone: req.Phone,
Postcode: req.Postcode,
Remark: req.Remark,
TemplateId: req.TemplateId,
SourceId: req.SourceId, //来源合同id
ModifyTime: time.Now(),
CreateTime: time.Now(),
}
//合同日期
if req.ContractBusinessType == "代付合同" {
//关联合同
relationContractDetailList := make([]*contract.ContractDetail, 0)
tmpContractDetail, tmpErr := getContractDetail(req.RelationContractId)
if tmpErr != nil {
err = errors.New("查询业务关联合同详情异常:" + tmpErr.Error())
return
}
relationContractDetailList = append(relationContractDetailList, tmpContractDetail)
contractDetail.RelationContractDetailList = relationContractDetailList
} else {
tmpStartDate, tmpErr := time.Parse(utils.FormatDate, req.StartDate)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同开始日期转换失败:", tmpErr))
return
}
contractDetail.StartDate = tmpStartDate
tmpEndDate, tmpErr := time.Parse(utils.FormatDate, req.EndDate)
if tmpErr != nil {
err = errors.New(fmt.Sprint("合同结束日期转换失败:", tmpErr))
return
}
contractDetail.EndDate = tmpEndDate
contractServiceSlice, tmpErr := getContractServiceAndDetail(productId, req.Service, false)
if tmpErr != nil {
err = tmpErr
return
}
contractDetail.Service = contractServiceSlice
}
return
}
// 根据请求数据获取套餐服务详情信息
func getContractServiceAndDetail(productId int, reqService []request.AddContractServiceReq, base642Image bool) (contractServiceSlice []*contract.ContractServiceAndDetail, err error) {
contractServiceSlice = make([]*contract.ContractServiceAndDetail, 0)
if len(reqService) < 0 {
err = errors.New("请至少选择一种套餐")
return
}
for i := 0; i < len(reqService); i++ {
item := reqService[i]
detail := item.Detail
contractServiceDetailSlice := make([]*contract.ContractServiceDetail, 0)
if len(detail) > 0 {
for j := 0; j < len(detail); j++ {
detailItem := detail[j]
cellMap := make(map[string]string)
for k := 0; k < len(detailItem); k++ {
key := fmt.Sprint("cell_", k+1)
tmpVal, jsonErr := json.Marshal(detailItem[k])
if jsonErr != nil {
err = errors.New(fmt.Sprint("json转换失败:", jsonErr))
return
}
cellMap[key] = string(tmpVal)
}
contractServiceDetail := &contract.ContractServiceDetail{
Col1: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_1"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col2: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_2"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col3: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_3"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col4: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_4"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col5: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_5"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col6: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_6"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
Col7: func(cellMap map[string]string) string {
tmpVal, ok := cellMap["cell_7"]
if ok {
return tmpVal
} else {
return ""
}
}(cellMap),
ServiceTemplateId: item.ServiceTemplateId,
CreateTime: time.Now(),
}
contractServiceDetailSlice = append(contractServiceDetailSlice, contractServiceDetail)
}
}
hasDetail := "否"
if len(contractServiceDetailSlice) > 0 {
hasDetail = "是"
}
//报价单图片地址
valueStr := item.Value
if base642Image {
b, _ := regexp.MatchString(`^data:\s*image\/(\w+);base64,`, valueStr)
if b {
imageUrl, uploadErr := UploadImageBase64(valueStr)
if uploadErr != nil {
err = errors.New(fmt.Sprint("base64图片上传失败:", uploadErr))
return
}
valueStr = imageUrl
}
}
//寻找该系统模板
systemContractServiceTem, tmpErr := contract.GetContractServiceTemplateById(item.ServiceTemplateId)
if tmpErr != nil {
err = errors.New("找不到该模板")
return
}
tmpContractService := &contract.ContractServiceAndDetail{
ServiceTemplateId: item.ServiceTemplateId,
Title: item.Title,
Value: valueStr,
ProductId: productId,
//CreateTime: time.Now(),
HasDetail: hasDetail,
TableValue: systemContractServiceTem.TableValue,
DetailList: contractServiceDetailSlice,
ChartPermissionId: systemContractServiceTem.ChartPermissionId,
}
contractServiceSlice = append(contractServiceSlice, tmpContractService)
}
return
}
// 获取品种列数据(因为前端可以随意插入列数据,所以需要根据列名来搜索对应的品种列)
func GetProductCelData(detail contract.ContractServiceDetail, tableCel *request.AddContractServiceDetailReq) (err error) {
t := reflect.TypeOf(detail)
v := reflect.ValueOf(detail)
for k := 0; k < t.NumField(); k++ {
//获取结构体的参数名
tmpName := t.Field(k).Name
if strings.Contains(tmpName, "Col") {
//获取结构体该参数名的值
tmpValue := v.Field(k).String()
//如果值不为空的话,那么做下json转换
if tmpValue != "" {
tmpErr := json.Unmarshal([]byte(tmpValue), tableCel)
if tmpErr != nil {
err = tmpErr
continue
}
if tableCel.HeadName == "品种" {
err = nil
return
}
}
}
}
return
}
// 将base64的图片上传至oss
func UploadImageBase64(base64Str string) (resourceUrl string, err error) {
ext := ".png"
uploadDir := "./static"
randStr := utils.GetRandStringNoSpecialChar(28)
fileName := randStr + ext
fpath := uploadDir + "/" + fileName
b, _ := regexp.MatchString(`^data:\s*image\/(\w+);base64,`, base64Str)
if !b {
err = errors.New("图片格式不正确")
return
}
re, _ := regexp.Compile(`^data:\s*image\/(\w+);base64,`)
base64Str = re.ReplaceAllString(base64Str, "")
base64Str = strings.Replace(base64Str, " ", "", -1)
base64Str = strings.Replace(base64Str, " ", "", -1)
//imgByte, err := base64.StdEncoding.DecodeString(base64Str)
//err=ioutil.WriteFile(fpath,imgByte,os.ModeAppend)
//if err!=nil {
// br.Msg = "图片保存失败"
// br.ErrMsg = "图片保存失败,Err:"+err.Error()
// return
//}
err = utils.SaveBase64ToFile(base64Str, fpath)
if err != nil {
return
}
fmt.Println("end")
defer os.Remove(fpath)
savePath := utils.Upload_Audio_Dir + time.Now().Format("200601/20060102/")
savePath += fileName
//上传到阿里云
err = services.UploadVideoAliyun(fileName, fpath, savePath)
if err != nil {
return
}
utils.FileLog.Info("%s:", time.Now().Format(utils.FormatDateTime))
utils.FileLog.Info("end update oss ")
resourceUrl = utils.Imghost + savePath
item := new(models.Resource)
item.ResourceUrl = resourceUrl
item.ResourceType = 1
item.CreateTime = time.Now()
_, err = models.AddResource(item)
if err != nil {
return
}
return
}
// GetPermissionByContractService 通过合同服务,获取权限id列表
func GetPermissionByContractService(productId int, serviceList []*contract.ContractServiceAndDetail) (permissionList []*company.PermissionLookList, err error) {
//获取已经勾选的的权限
checkPermissionIdMap, err := GetServicePermissionMap(serviceList)
if err != nil {
return
}
permissionList, err = GetPermissionByPermissionIdMap(productId, checkPermissionIdMap)
return
}
// GetPermissionByPermissionIdMap 通过权限id的map,获取权限列表
func GetPermissionByPermissionIdMap(productId int, checkPermissionIdMap map[int]int) (permissionList []*company.PermissionLookList, err error) {
//子权限切片集合
allPermissions, err := company.GetPermissionLookItemsByProductId(productId)
if err != nil {
err = fmt.Errorf("获取权限失败,Err:%s", err.Error())
return
}
permissionMap := make(map[int][]*company.PermissionLookItem, 0)
for _, v := range allPermissions {
if v.ParentId > 0 {
permissionMap[v.ParentId] = append(permissionMap[v.ParentId], v)
}
}
//遍历获取
for _, v := range allPermissions {
if v.ParentId == 0 {
checkList := make([]int, 0)
plist := new(company.PermissionLookList)
plist.Items = make([]*company.PermissionLookItem, 0)
items, ok1 := permissionMap[v.ChartPermissionId]
if ok1 {
for _, n := range items {
if _, ok := checkPermissionIdMap[n.ChartPermissionId]; ok {
checkList = append(checkList, n.ChartPermissionId)
} else if _, ok2 := checkPermissionIdMap[n.ChartPermissionId+utils.PERMISSION_ID_UPGRADE]; ok2 {
checkList = append(checkList, n.ChartPermissionId)
n.IsUpgrade = 1
}
}
plist.Items = items
}
plist.ClassifyName = v.PermissionName
plist.CheckList = checkList
permissionList = append(permissionList, plist)
}
}
return
}
// GetServicePermissionMap 获取合同服务中的权限id map
func GetServicePermissionMap(serviceList []*contract.ContractServiceAndDetail) (checkPermissionIdMap map[int]int, err error) {
checkPermissionIdMap = make(map[int]int)
//获取已经勾选的的权限
for _, contractService := range serviceList {
//如果是权益,那么研选必要(2022-07-06 10:29:11移除)
//if contractService.ProductId == 2 {
// chartPermissionId := 31 //研选服务
// checkPermissionIdMap[chartPermissionId] = chartPermissionId
//}
//看看系统中有没有映射对应的权限,如果有的话,那么直接返回,没有那么可能是特殊服务,需要去遍历
if contractService.ChartPermissionId > 0 {
checkPermissionIdMap[contractService.ChartPermissionId] = contractService.ChartPermissionId
continue
}
switch contractService.ServiceTemplateId {
case 1: //ficc 大套餐
allFiccPermissions, e := company.GetPermissionLookItemsByProductId(utils.COMPANY_PRODUCT_FICC_ID)
if e != nil {
err = fmt.Errorf("获取权限失败 Err:%v", e)
return
}
permissionMap := make(map[int][]*company.PermissionLookItem, 0)
for _, v := range allFiccPermissions {
if v.ParentId > 0 {
permissionMap[v.ParentId] = append(permissionMap[v.ParentId], v)
}
}
for _, v := range allFiccPermissions {
if v.ParentId == 0 {
//合同这边市场策略不需要体现出来,所以调整返回
if v.PermissionName == "市场策略" {
continue
}
items, ok := permissionMap[v.ChartPermissionId]
if ok {
for _, n := range items {
checkPermissionIdMap[n.ChartPermissionId] = n.ChartPermissionId
}
}
}
}
case 2: //ficc小套餐
permissionValues := []string{
"FICC周报", "商品双周报+线上电话会讨论会
(由弘则的研究员主持线上讨论)", "数据点评",
}
for _, detail := range contractService.DetailList {
var tableCel request.AddContractServiceDetailReq
tmpErr := json.Unmarshal([]byte(detail.Col1), &tableCel)
if tmpErr != nil {
err = tmpErr
return
}
if strings.Contains(strings.Join(permissionValues, ","), tableCel.Value) {
//获取品种列数据
var tableCel2 request.AddContractServiceDetailReq
tmpErr := GetProductCelData(*detail, &tableCel2)
if tmpErr != nil {
err = tmpErr
return
}
for _, serviceId := range tableCel2.ValueId {
checkPermissionIdMap[serviceId] = serviceId
}
}
}
case 4, 5, 6: //权益大套餐
permissionFiccClassifyArr := []string{"权益"}
for _, v := range permissionFiccClassifyArr {
items, tmpErr := company.GetPermissionLookItems(2, v)
if tmpErr != nil {
err = tmpErr
return
}
for _, n := range items {
//如果是专家行业,那么 结束当前循环,进入下一循环(产品需求:专家行业不在里面)
if n.ChartPermissionId == 29 || n.ChartPermissionId == 31 {
continue
}
checkPermissionIdMap[n.ChartPermissionId] = n.ChartPermissionId
}
}
}
}
return
}
// GetServicePackageType 获取合同服务中的 大小套餐类型
func GetServicePackageType(serviceList []*contract.ContractServiceAndDetail) (packageType int, err error) {
//获取已经勾选的的权限
for _, contractService := range serviceList {
switch contractService.ServiceTemplateId {
case 1: //ficc 大套餐
packageType = 1
case 2: //ficc小套餐
packageType = 2
case 4, 5, 6: //权益大套餐
//packageType = 1
}
}
return
}
// GetRaiServicePackageType 获取合同服务中的 权益的大小套餐类型
func GetRaiServicePackageType(serviceList []*contract.ContractServiceAndDetail) (raiPackageType int) {
for _, contractService := range serviceList {
switch contractService.ServiceTemplateId {
case 5: // 45w大套餐
raiPackageType = 2
case 6: // 70w大套餐
raiPackageType = 1
}
if raiPackageType > 0 {
break
}
}
return
}
// CheckContractPrice 校验代付合同金额是否 大于 业务合同金额(可能存在多份合同,需要遍历,如果大于的话,校验不通过,需要驳回)
func CheckContractPrice(contractInfo *contract.Contract, price float64) (ok bool, err error) {
list, err := contract.GetContractRelationListByContractId(contractInfo.ContractId)
if err != nil {
return
}
payedPrice := 0.0
for _, v := range list {
payedPrice += v.Price
}
if (contractInfo.Price - payedPrice) < price {
return
}
ok = true
return
}
// UploadCheckBackFileFromSeal 更新用印上传签回合同附件
func UploadCheckBackFileFromSeal(contractId int, fileUrl string, opUser *system.Admin) (err error) {
// 获取合同信息
contractInfo, err := contract.GetContractById(contractId)
if err != nil {
return
}
// 合同状态判断
if contractInfo.Status != "已审批" && contractInfo.Status != "已签回" {
err = errors.New("合同状态异常,不允许上传签回合同附件,当前合同状态:" + contractInfo.Status)
return
}
contractInfo.CheckBackFileUrl = fileUrl
contractInfo.Status = "已签回"
contractInfo.ModifyTime = time.Now()
contractInfo.CheckBackFileTime = time.Now()
err = contractInfo.Update([]string{"CheckBackFileUrl", "Status", "ModifyTime", "CheckBackFileTime"})
if err != nil {
return
}
// 同步更新客户那边提交审批的合同
o := orm.NewOrm()
sql := `update company_contract set img_url = ? where contract_code=? and source = "系统合同" `
_, err = o.Raw(sql, contractInfo.CheckBackFileUrl, contractInfo.ContractCode).Exec()
// 添加操作日志
remark := "上传签回合同附件"
_ = contract.AddContractOperationRecord(contractInfo.ContractId, opUser.AdminId, 0, "upload", opUser.RealName, remark)
return
}
//func init() {
// fmt.Println("contract init start")
// go FixEquityContractService()
// fmt.Println("contract init end")
//}
// FixEquityContractService 修复权益合同服务内容-CRM8.8
func FixEquityContractService() (err error) {
defer func() {
if err != nil {
fmt.Println("FixEquityContractService Err: " + err.Error())
}
}()
// 获取所有权益的合同(不考虑状态)
productId := 2
contractList, e := contract.GetContractListByProductId(productId)
if e != nil {
err = errors.New("获取权益合同列表失败")
return
}
contractLen := len(contractList)
if contractLen == 0 {
fmt.Println("无合同需要修复")
return
}
fmt.Println("待修复权益合同数", contractLen)
//return
// 合同IDs
contractIdArr := make([]string, 0)
contractIdServices := make(map[int][]*contract.ContractService, 0)
for i := 0; i < contractLen; i++ {
contractIdArr = append(contractIdArr, strconv.Itoa(contractList[i].ContractId))
newServices := make([]*contract.ContractService, 0)
contractIdServices[contractList[i].ContractId] = newServices
}
contractIds := strings.Join(contractIdArr, ",")
// 合同服务内容
serviceList, e := contract.GetContractServiceListByIds(contractIds)
if e != nil {
err = errors.New("获取权益合同服务内容列表失败")
return
}
servicesLen := len(serviceList)
if servicesLen == 0 {
fmt.Println("无合同服务需要修复")
return
}
for i := 0; i < servicesLen; i++ {
contractIdServices[serviceList[i].ContractId] = append(contractIdServices[serviceList[i].ContractId], serviceList[i])
}
// 服务套餐模板
templateList, e := contract.GetAllContractServiceTemplateList()
if e != nil {
err = errors.New("获取服务套餐模板列表失败")
return
}
// 主客观服务套餐
titleArr := []string{"医药", "消费", "科技", "智造"}
subTempMap := make(map[string]*contract.ContractServiceTemplate, 0)
objTempMap := make(map[string]*contract.ContractServiceTemplate, 0)
templateIdMap := make(map[int]*contract.ContractServiceTemplate, 0)
templateLen := len(templateList)
for i := 0; i < templateLen; i++ {
templateIdMap[templateList[i].ServiceTemplateId] = templateList[i]
}
for i := 0; i < templateLen; i++ {
if templateList[i].Title == "主观" {
parentTemp := templateIdMap[templateList[i].Pid]
subTempMap[parentTemp.Title] = templateList[i]
continue
}
if templateList[i].Title == "客观" {
parentTemp := templateIdMap[templateList[i].Pid]
objTempMap[parentTemp.Title] = templateList[i]
continue
}
}
// 遍历合同
for i := 0; i < contractLen; i++ {
contractId := contractList[i].ContractId
fmt.Printf("正在修复第%d个, 合同ID: %d\n", i+1, contractId)
// 合同服务内容
contractServices := contractIdServices[contractId]
insertArr := make([]*contract.ContractService, 0)
serviceLen := len(contractServices)
// 是否含有医药、消费、科技、智造
for _, v := range titleArr {
for ii := 0; ii < serviceLen; ii++ {
createTime := contractServices[ii].CreateTime
// 含有四种套餐之一
if contractServices[ii].Title == v {
template := templateIdMap[contractServices[ii].ServiceTemplateId]
pid := template.ServiceTemplateId
insertSubject := true
insertObject := true
// 去重, 校验是否已存在主客观
for iii := 0; iii < serviceLen; iii++ {
t := templateIdMap[contractServices[iii].ServiceTemplateId]
if t.Pid == pid && t.Title == "主观" {
insertSubject = false
continue
}
if t.Pid == pid && t.Title == "客观" {
insertObject = false
continue
}
}
if insertSubject {
subTemp := subTempMap[v]
title := v + "(" + subTemp.Title + ")"
insertArr = append(insertArr, &contract.ContractService{
ContractId: contractId,
ProductId: productId,
ServiceTemplateId: subTemp.ServiceTemplateId,
Title: title,
Value: subTemp.Value,
TableValue: subTemp.TableValue,
HasDetail: "否",
ChartPermissionId: subTemp.ChartPermissionId,
CreateTime: createTime,
})
}
if insertObject {
objTemp := objTempMap[v]
title := v + "(" + objTemp.Title + ")"
insertArr = append(insertArr, &contract.ContractService{
ContractId: contractId,
ProductId: productId,
ServiceTemplateId: objTemp.ServiceTemplateId,
Title: title,
Value: objTemp.Value,
TableValue: objTemp.TableValue,
HasDetail: "否",
ChartPermissionId: objTemp.ChartPermissionId,
CreateTime: createTime,
})
}
break
}
}
}
// 新增主客观
if len(insertArr) > 0 {
if e := contract.MultiCreateContractService(insertArr); e != nil {
fmt.Printf("第%d个修复失败, 合同ID: %d, Err: %s\n", i+1, contractId, e.Error())
return
}
fmt.Printf("第%d个修复成功, 合同ID: %d\n", i+1, contractId)
} else {
fmt.Printf("第%d个已修复过, 忽略合同ID: %d\n", i+1, contractId)
}
}
return
}
// CRM8.8-权益合同主客观套餐合并
func EquityMergeSubjectAndObject(serviceList []*contract.ContractServiceAndDetail) (mergeList []*contract.ContractServiceAndDetail) {
serviceLen := len(serviceList)
if serviceLen == 0 {
return serviceList
}
mergeArr := []string{"医药", "消费", "科技", "智造"}
// 获取模板列表
templateList, e := contract.GetAllContractServiceTemplateList()
if e != nil {
return serviceList
}
parentTempMap := make(map[string]int, 0)
templateMap := make(map[int]*contract.ContractServiceTemplate, 0)
for _, v := range templateList {
if utils.InArrayByStr(mergeArr, v.Title) {
parentTempMap[v.Title] = v.ServiceTemplateId
}
templateMap[v.ServiceTemplateId] = v
}
// 计算每个行业的子套餐数, 并判断所有套餐所属行业
countIndustryMap := make(map[string]int, 0)
serviceIndustryMap := make(map[int]string, 0)
industryServiceMap := make(map[string]*contract.ContractServiceAndDetail, 0)
childrenServiceMap := make(map[string]*contract.ContractServiceAndDetail, 0)
for i := 0; i < serviceLen; i++ {
temp := templateMap[serviceList[i].ServiceTemplateId]
if temp == nil {
continue
}
// 行业套餐
if utils.InArrayByStr(mergeArr, temp.Title) {
serviceIndustryMap[i] = temp.Title
industryServiceMap[temp.Title] = serviceList[i]
continue
}
// 主/客观
parentTemp := templateMap[temp.Pid]
if parentTemp != nil && utils.InArrayByStr(mergeArr, parentTemp.Title) {
countIndustryMap[parentTemp.Title] += 1
serviceIndustryMap[i] = parentTemp.Title
childrenServiceMap[parentTemp.Title] = serviceList[i]
}
}
mergeList = make([]*contract.ContractServiceAndDetail, 0)
// 遍历每一个套餐, 取出所属的行业, 不属于合并的行业则直接加入数组
// 属于合并的行业, 则校验行业子套餐数, 若子套餐数为1, 则取取主/客观套餐, 否则取行业套餐
mergeIndustryMap := make(map[string]int, 0)
for i := 0; i < serviceLen; i++ {
industryName := serviceIndustryMap[i]
// 不需要合并的行业
if industryName == "" {
mergeList = append(mergeList, serviceList[i])
continue
}
// 未合并则进行合并
if mergeIndustryMap[industryName] > 0 {
continue
}
count := countIndustryMap[industryName]
if count == 1 {
// 取主/客观套餐
mergeList = append(mergeList, childrenServiceMap[industryName])
} else {
// 0或者2都取行业套餐
mergeList = append(mergeList, industryServiceMap[industryName])
}
mergeIndustryMap[industryName] += 1
}
return
}
// CRM8.8-权益合同主客观权限的合并
func EquityMergeSubjectAndObjectPermission(list []*company.CompanyContractPermissionName) (newList []*company.CompanyContractPermissionName) {
if len(list) == 0 {
return list
}
// contractPermissionMap示例:
// map[
// 665:map[
// 专家:[0xc0002e24d0]
// 买方研选:[0xc0002e2630]
// 医药:[0xc0002e2370 0xc0002e28f0]
// 智造:[0xc0002e2160 0xc0002e26e0]
// 消费:[0xc0002e22c0 0xc0002e2840]
// 科技:[0xc0002e2210 0xc0002e2790]
// 策略:[0xc0002e2420]
// 路演服务:[0xc0002e2580]
// ]
// 666:map[
// 智造:[0xc0002e29a0 0xc0002e2b00]
// 科技:[0xc0002e2a50 0xc0002e2bb0]
// ]
// ]
contractPermissionMap := make(map[int]map[string][]*company.CompanyContractPermissionName)
for _, v := range list {
permissionMap, ok := contractPermissionMap[v.CompanyContractId]
if ok {
permissionItemMap, ok2 := permissionMap[v.ChartPermissionName]
if !ok2 {
permissionItemMap = make([]*company.CompanyContractPermissionName, 0)
}
permissionItemMap = append(permissionItemMap, v)
permissionMap[v.ChartPermissionName] = permissionItemMap
} else {
permissionMap = make(map[string][]*company.CompanyContractPermissionName)
permissionMap[v.ChartPermissionName] = append(permissionMap[v.ChartPermissionName], v)
}
contractPermissionMap[v.CompanyContractId] = permissionMap
}
newList = make([]*company.CompanyContractPermissionName, 0)
for _, v := range contractPermissionMap {
for _, p := range v {
if len(p) == 1 {
p[0].ChartPermissionName = p[0].PermissionRemark
}
newList = append(newList, p[0])
}
}
return
}
// HandleEquityPermissionLookList 权益-PermissionLookList主客观合并
func HandleEquityPermissionLookList(permissionLookList []*company.PermissionLookList) []*company.PermissionLookList {
if len(permissionLookList) == 0 {
return permissionLookList
}
checkList := permissionLookList[0].CheckList
permissionList := permissionLookList[0].Items
newPermissionList := HandleEquityPermissionLookItem(checkList, permissionList)
permissionLookList[0].Items = newPermissionList
return permissionLookList
}
// HandleEquityContractPermissionList 权益-ContractPermissionList主客观合并
func HandleEquityContractPermissionList(contractPermissionList []*company.ContractPermissionList) []*company.ContractPermissionList {
if len(contractPermissionList) == 0 {
return contractPermissionList
}
checkList := contractPermissionList[0].CheckList
permissionList := contractPermissionList[0].Items
newPermissionList := HandleEquityPermissionLookItem(checkList, permissionList)
contractPermissionList[0].Items = newPermissionList
return contractPermissionList
}
// HandleEquityPermissionLookItem 权益-权限展示的主客观合并处理
func HandleEquityPermissionLookItem(checkList []int, permissionList []*company.PermissionLookItem) (newPermissionList []*company.PermissionLookItem) {
permissionLen := len(permissionList)
if len(checkList) <= 0 || permissionLen <= 0 {
return permissionList
}
// 若某个行业的主客观权限均被满足,则只留其中一个权限
// 1.取出每个行业对应的其中一个权限
industryName := []string{"医药", "消费", "科技", "智造"}
namePermissionMap := make(map[string]*company.PermissionLookItem)
for i := 0; i < permissionLen; i++ {
pName := permissionList[i].PermissionName
if utils.InArrayByStr(industryName, pName) && namePermissionMap[pName] != nil {
namePermissionMap[pName] = permissionList[i]
}
}
// 2.计算哪些行业主客观权限都有
countIndustryMap := make(map[string]int, 0)
for i := 0; i < permissionLen; i++ {
p := permissionList[i]
if utils.InArrayByStr(industryName, p.PermissionName) && utils.InArrayByInt(checkList, p.ChartPermissionId) {
countIndustryMap[p.PermissionName] += 1
}
}
// 3.重新组成一个权限列表
mergeIndustryMap := make(map[string]int, 0)
newPermissionList = make([]*company.PermissionLookItem, 0)
for i := 0; i < permissionLen; i++ {
item := permissionList[i]
if utils.InArrayByStr(industryName, item.PermissionName) {
// 只有主客观之一, permissionName取remark
if countIndustryMap[item.PermissionName] == 1 {
// CRM13.2 标准合同新增升级套餐
if item.IsUpgrade == 1 {
item.PermissionName += "(升级)"
newPermissionList = append(newPermissionList, item)
} else {
item.PermissionName = item.Remark
newPermissionList = append(newPermissionList, item)
}
continue
}
// 主客观都有
if countIndustryMap[item.PermissionName] > 1 && mergeIndustryMap[item.PermissionName] == 0 {
mergeIndustryMap[item.PermissionName] += 1
// CRM13.2 标准合同新增升级套餐
if item.IsUpgrade == 1 {
item.PermissionName += "(升级)"
}
newPermissionList = append(newPermissionList, item)
}
} else {
newPermissionList = append(newPermissionList, item)
}
}
return
}
// handleRaiContractModifyContentByServices 权益合同-服务内容变更内容
func handleRaiContractModifyContentByServices(originService []*contract.ContractServiceAndDetail, newService []*contract.ContractServiceAndDetail) (tips string) {
if originService == nil || newService == nil {
return
}
oldArr := make([]string, 0)
newArr := make([]string, 0)
oldStr := "无"
newStr := "无"
o := EquityMergeSubjectAndObject(originService)
n := EquityMergeSubjectAndObject(newService)
for i := range o {
oldArr = append(oldArr, o[i].Title)
}
if len(oldArr) > 0 {
oldStr = strings.Join(oldArr, ",")
}
for i := range n {
newArr = append(newArr, n[i].Title)
}
if len(newArr) > 0 {
newStr = strings.Join(newArr, ",")
}
tips = fmt.Sprint("原服务内容:", oldStr, "现服务内容:", newStr)
return
}
// 获取权益近一年是否有新签合同
func GetCompanyContractCountRaiByLastYear(companyId int, startDate string) (total int, err error) {
firstContractRai, err := company.GetFirstContractRai(companyId)
if err != nil && err.Error() != utils.ErrNoRow() {
return
}
//没有合同这第一份合同就属于新签合同
if firstContractRai == nil {
err = nil
total = 1
return
}
//判断是否是在一年以内
startDateTime, _ := time.Parse(utils.FormatDate, startDate)
startDateTime = startDateTime.AddDate(-1, 0, 0)
startDateTimeFirst, _ := time.Parse(utils.FormatDate, firstContractRai.StartDate)
if startDateTime.Before(startDateTimeFirst) {
total = 1
}
return
}
func initCRM15_5() {
var condition string
var pars []interface{}
condition = " AND rai_contract_type = '新签合同' AND product_id = 2 "
companyContractList, e := company.GetCompanyContractList(condition, pars)
if e != nil && e.Error() != utils.ErrNoRow() {
fmt.Println(e)
return
}
var CompanyContractIdS []int
for k, v := range companyContractList {
fmt.Println(k)
firstContractRai, err := company.GetFirstContractRai(v.CompanyId)
if err != nil {
fmt.Println(err)
continue
}
if firstContractRai.CompanyContractId == v.CompanyContractId {
continue
}
//判断是否是在一年以内
startDateTime, _ := time.Parse(utils.FormatDate, v.StartDate)
startDateTime = startDateTime.AddDate(-1, 0, 0)
startDateTimeFirst, _ := time.Parse(utils.FormatDate, firstContractRai.StartDate)
if !startDateTime.Before(startDateTimeFirst) {
CompanyContractIdS = append(CompanyContractIdS, v.CompanyContractId)
}
}
fmt.Println("companyContractIds", len(CompanyContractIdS))
fmt.Println("companyContractIds", (CompanyContractIdS))
//
//fmt.Println(strings.Join(companyContractIds, ","))
//err := company.UpdateCompanyContractRaiContractTypeInit(strings.Join(companyContractIds, ","))
//fmt.Println(err)
return
}