| 纷享销客 | 帮助中心 - 北京易动纷享科技有限责任公司

Groovy代码示例


package com.facishare.function.core.template.groovy

import com.fxiaoke.functions.Fx
import com.fxiaoke.functions.http.HttpResult
import com.fxiaoke.functions.service.esig.ElectionSignPlugin
import com.fxiaoke.functions.service.esig.model.AddEmployees
import com.fxiaoke.functions.service.esig.model.CancelSign
import com.fxiaoke.functions.service.esig.model.DownloadSignFile
import com.fxiaoke.functions.service.esig.model.FileAttachment
import com.fxiaoke.functions.service.esig.model.GetSignUrl
import com.fxiaoke.functions.service.esig.model.InitiateSign
import com.fxiaoke.functions.service.esig.model.InnerAppSignerData
import com.fxiaoke.functions.service.esig.model.OuterAppSignerData
import com.fxiaoke.functions.service.esig.model.RemoveEmployees
import com.fxiaoke.functions.service.esig.model.SignCallback
import com.fxiaoke.functions.service.esig.model.SignLocation
import com.fxiaoke.functions.service.esig.model.TaskStatus
import com.fxiaoke.functions.service.esig.model.UrgeSign
import com.fxiaoke.functions.time.DateTime
import com.fxiaoke.functions.utils.Strings

import static com.fxiaoke.functions.Fx.log
/**
 * @type classes
 * @returntype
 * @namespace electronic_sign
 */
class Esign implements ElectionSignPlugin {

    //电子厂商url, 此处要替换
    private static final String BASE_URL = "https://smlopenapi.esign.cn"
    private static final String CONTENT_TYPE = "application/json; charset=UTF-8"
    private static final String CONTENT_TYPE_UPLOAD = "application/octet-stream"
    private static final String ACCEPT = "*/*"

    //应用id,此处要替换
    private static final String APP_ID = "7438894881";
    //应用key,此处要替换
    private static final String APP_KEY = "3155ca27042ed83469cf5baef3de84ef";
    //先用该方法获取租户账号id,写到上边, createCompanyAccount("xx公司"), 此处要替换
    private static final String AUTHORIZED_ACCOUNT_ID = "d8e826359a8c4eecbac9295f58b49603"
    //电子厂商通知纷享的接口url, 此处要替换
    private static final String CALLBACK_URL = "//m.sxgtbz.com/customerprovider/callback/signCallback/xxx企业id加密值"

    /**
     * 拼接http请求头
     */
    private static Map getHeaders(String accept, String contentType, String signOpenCaSignature, String contentMD5) {
        DateTime d = DateTime.now();
        Map headerMap = [:];
        headerMap.put("X-Tsign-Open-App-Id", APP_ID);
        headerMap.put("X-Tsign-Open-Auth-Mode", "Signature");
        headerMap.put("X-Tsign-Open-Ca-Timestamp", String.valueOf(d.toTimestamp()));
        headerMap.put("Accept", accept);
        headerMap.put("Content-Type", contentType);
        headerMap.put("X-Tsign-Open-Ca-Signature", signOpenCaSignature);
        headerMap.put("Content-MD5", contentMD5);
        return headerMap;
    }
    /**
     * 拼接纯文本
     */
    private static String getPlaintext(String method, String accept, String contentMD5, String contentType, String uri) {
        String date = "";
        String plaintext = method + "\n" + accept + "\n" + contentMD5 + "\n" + contentType + "\n" + date + "\n" + "" + uri;
        return plaintext;
    }
    /**
     * 编码纯文本
     */
    private static String encodePlainText(String plaintext) {
        def signatureBytes = Fx.crypto.hmac.encrypt("HmacSHA256", APP_KEY, plaintext);
        return Fx.crypto.base64.encode(signatureBytes);
    }
    /**
     * 取md5值
     */
    private static String getContentMD5(Map map) {
        //body转成String
        def requestBodyJson = Fx.json.toJson(map)
        //String转成bytes
        def bytes = Strings.toUTF8Bytes(requestBodyJson)
        //进行MD5加密
        def message = Fx.crypto.MD5.encode2Bytes(bytes);
        //进行Base64加密,返回最终需要的MD5密文
        return Fx.crypto.base64.encode(message);
    }
    /**
     * 取md5值
     */
    private static String getContentMD5(byte[] bytes) {
        //进行MD5加密
        def message = Fx.crypto.MD5.encode2Bytes(bytes);
        //进行Base64加密,返回最终需要的MD5密文
        return Fx.crypto.base64.encode(message);
    }

    //增加员工
    public AddEmployees.Result addEmployees(AddEmployees.Arg arg) {
        //标准log,请保留
        log.info("addEmployees arg:" + arg)
        List<AddEmployees.EmployeeData> employeeDataList = arg.getEmployeeDataList()

        employeeDataList.each { employeeData ->
            String accountId = createPersonalAccount(employeeData.getUserName(), employeeData.getMobile())
            log.info("添加人员 accountId:" + accountId)
            //此处人员必须标记为已认证
            Map objectData = [:]
            objectData.put("esign_provider_user_id", accountId);
            objectData.put("status", "verified");
            Fx.object.update("ExternalPlatPerCompObj", employeeData.getObjectId(), objectData);
        }
        AddEmployees.Result result = AddEmployees.Result.builder().success(true).build();
        //标准log,请保留
        log.info("addEmployees result:" + result)
        return result;
    }

    //创建个人账号
    private String createPersonalAccount(String userName, String mobile) {
        //标准log,请保留
        log.info("createPersonalAccount userName:" + userName + "mobile:" + mobile)
        Map requestBody = [:]
        requestBody.put("thirdPartyUserId", mobile)
        requestBody.put("name", userName)
        requestBody.put("mobile", mobile)
        //生成md5
        String contentMD5 = Esign.getContentMD5(requestBody)
        //创建人员接口
        String uri = "/v1/accounts/createByThirdPartyUserId"
        //获取签名字段
        String plaintext = Esign.getPlaintext("POST", ACCEPT, contentMD5, CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plaintext)
        //构建header
        Map headerMap = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, contentMD5);
        //构建url
        String url = BASE_URL + uri
        //标准log,请保留
        log.info("createPersonalAccount http requestBody:" + requestBody)
        def result = Fx.http.post(url, headerMap, requestBody)
        //标准log,请保留
        log.info("createPersonalAccount http response:" + result)
        String accountId = result[1]["content"]["data"]["accountId"] as String
        //标准log,请保留
        log.info("createPersonalAccount accountId:" + accountId)
        return accountId;
    }

    //移除员工
    public RemoveEmployees.Result removeEmployees(RemoveEmployees.Arg arg) {
        //标准log,请保留
        log.info("removeEmployees arg:" + arg)
        //标准log,请保留
        log.info("removeEmployees result:ll")
        return null;
    }

    //查询e签宝的文件转换后的状态
    private String queryFileStatus(String fileId) {
        //标准log,请保留
        log.info("queryFileStatus fileId:" + fileId)
        String uri = "/v1/files/{fileId}/status";
        uri = uri.replaceAll("\\{[fileId^}]*\\}", "" + fileId + "");
        String statusCode;
        //构建签名字段
        String plainText = Esign.getPlaintext("GET", ACCEPT, "", CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, "");
        //调用put接口
        def result = Fx.http.get(BASE_URL + uri, headers, 10000, false, 0);
        log.info(result)
        if (result[1]["statusCode"] != 200) {
            log.info("转pdf失败,再次查询")
            statusCode = "error"
        } else {
            Map contentss = result[1]["content"] as Map;
            log.info("检查文件转换状态:" + contentss)
            if (contentss['code'] as String != "0") {
                statusCode = contentss['message'];
            } else {
                statusCode = contentss['data']['status'];
            }
        }
        //标准log,请保留
        log.info("queryFileStatus statusCode:" + statusCode)
        return statusCode;
    }

    //发起签署任务
    public InitiateSign.Result initiateSign(InitiateSign.Arg arg) {
        //标准log,请保留
        log.info("initiateSign arg:" + arg)
        //e签宝返回的失败提示语?
        //Fx.message.throwErrorMessage("测试提示语")
        //创建签署的入参
        Map createFlowArg = [:]
        List<FileAttachment> fileAttachments = arg.getFileAttachments()
        //上传文件
        Map<String, String> fileMap = uploadFile(fileAttachments)
        //构建签署人信息
        List signers = covertSigners(arg.getInnerAppSignerDataList(), arg.getOuterAppSignerDataList(), fileMap)
        log.info("构建签署人信息:" + signers)
        //设置签署人数据
        createFlowArg.put("signers", signers)

        Map flowConfigInfo = [:]
        //签署回调地址
        flowConfigInfo.put("noticeDeveloperUrl", CALLBACK_URL)
        //签署通知方式
        flowConfigInfo.put("noticeType", "1")
        Map flowInfo = [:]
        //签署主题
        flowInfo.put("businessScene", arg.getTaskSubject())
        //设置自动归档(归档才能进行下载)
        flowInfo.put("autoArchive", true)
        //签署过期时间
        flowInfo.put("contractValidity", arg.getExpireTime())
        //签署配置
        flowInfo.put("flowConfigInfo", flowConfigInfo)
        //设置签署流程信息
        createFlowArg.put("flowInfo", flowInfo)

        //构建签署文件信息
        List docs = covertDocs(fileAttachments, fileMap)
        createFlowArg.put("docs", docs)
        String flowId = startSignFlow(createFlowArg)
        if (flowId.isEmpty()) {
            Fx.message.throwErrorMessage("发起签署流程失败")
        }
        InitiateSign.Result result = InitiateSign.Result.builder().taskId(flowId).build()
        //标准log,请保留
        log.info("initiateSign result:" + result)
        return result;
    }

    //发起签署任务
    private String startSignFlow(Map createFlowArg) {
        //标准log,请勿删除
        log.info("startSignFlow arg:" + createFlowArg)
        String createFlowUri = "/api/v2/signflows/createFlowOneStep"
        String createFlowUrl = BASE_URL + createFlowUri;
        //转为json串
        def requestBodyJson = Fx.json.toJson(createFlowArg)
        String createFlowContentMD5 = Esign.getContentMD5(createFlowArg)
        //获取签名字段
        String createFlowPlaintext = Esign.getPlaintext("POST", ACCEPT, createFlowContentMD5, CONTENT_TYPE, createFlowUri);
        //签名认证
        def createFlowSignOpenCaSignature = Esign.encodePlainText(createFlowPlaintext)
        //构建header
        Map createFlowHeader = Esign.getHeaders(ACCEPT, CONTENT_TYPE, createFlowSignOpenCaSignature, createFlowContentMD5);
        //标准log,请勿删除
        log.info("startSignFlow http arg:" + requestBodyJson)
        def createFlowResult = Fx.http.post(createFlowUrl, createFlowHeader, requestBodyJson, 60000, true, 3)
        //标准log,请勿删除
        log.info("startSignFlow http response:" + createFlowResult)

        if (createFlowResult[1]["statusCode"] != 200) {
            String contentJson = createFlowResult[1]["content"] as String
            Map contentMap = Fx.json.parse(contentJson)
            Fx.message.throwErrorMessage(contentMap["message"] as String)
        }
        Map contentMap = createFlowResult[1]["content"] as Map;
        if (contentMap["code"] != 0) {
            Fx.message.throwErrorMessage(contentMap["message"] as String)
        }
        String flowId = createFlowResult[1]["content"]["data"]["flowId"] as String

        //--------------------------------------发起签署流程--------------------------------------
        String startFlowUri = "/v1/signflows/{flowId}/start"
        startFlowUri = startFlowUri.replaceAll("\\{[flowId^}]*\\}", "" + flowId + "");
        String startFlowUrl = BASE_URL + startFlowUri
        //构建签名字段
        String startFlowPlainText = Esign.getPlaintext("PUT", ACCEPT, "", CONTENT_TYPE, startFlowUri);
        //签名认证
        def startFlowSignOpenCaSignature = Esign.encodePlainText(startFlowPlainText)
        //构建header
        Map startFlowHeader = Esign.getHeaders(ACCEPT, CONTENT_TYPE, startFlowSignOpenCaSignature, "");
        //调用put接口
        //标准log,请勿删除
        log.info("startSignFlow start http arg:" + "")
        def startFlowResult = Fx.http.put(startFlowUrl, startFlowHeader, "");
        log.info("startSignFlow start http result:" + startFlowResult)
        //异常检查
        if (startFlowResult[1]["statusCode"] != 200) {
            flowId = null;
        }
        //标准log,请勿删除
        log.info("startSignFlow result:" + flowId)
        return flowId
    }

    //上传文件
    private Map<String, String> uploadFile(List<FileAttachment> fileAttachments) {
        //标准log,请保留
        log.info("uploadFile arg:" + fileAttachments)
        Map<String, String> fileMap = [:]
        List<String> need2PdfFileIds = []
        fileAttachments.each { item ->
            String nPath = item.getPath() as String
            String ext = item.getExt() as String
            //下载文件,并获取byte流
            def ret = Fx.file.downloadFile(nPath)
            def fileDowloadData = ret[1]
            if (fileDowloadData == null) {
                log.info("downloadFile:" + ret[0] + "/" + ret[2])
                Fx.message.throwErrorMessage("下载文件失败:" + ret[2])
            }
            def fileData = fileDowloadData['fileData'] as byte[]

            //--------------------------------------上传文件--------------------------------------
            //进行MD5加密
            String getUploadUrlContentMd5 = Esign.getContentMD5(fileData)
            //入参body
            Map getUploadUrlArg = [:]
            if (ext != 'pdf') {
                getUploadUrlArg.put("convert2Pdf", "true")
            }
            getUploadUrlArg.put("contentMd5", getUploadUrlContentMd5)
            getUploadUrlArg.put("contentType", CONTENT_TYPE_UPLOAD)
            getUploadUrlArg.put("fileName", item.getFilename())
            getUploadUrlArg.put("fileSize", fileDowloadData["fileSize"] as Long);
            //上传接口
            String uri = "/v1/files/getUploadUrl"
            String url = BASE_URL + uri
            //签名字段
            String plaintext = Esign.getPlaintext("POST", ACCEPT, "", CONTENT_TYPE, uri);
            //签名认证
            String signOpenCaSignature = Esign.encodePlainText(plaintext)
            //构建header
            Map getUploadUrlHeader = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, "")
            //获取签署url
            def result = Fx.http.post(url, getUploadUrlHeader, getUploadUrlArg)
            log.info("获取签署url:" + result)
            def uploadUrl = result[1]["content"]["data"]["uploadUrl"]
            def fileId = result[1]["content"]["data"]["fileId"] as String
            log.info("fileId:" + fileId)
            Map putHeaderMap = [:]
            putHeaderMap.put("Content-MD5", getUploadUrlContentMd5)
            putHeaderMap.put("Content-Type", "application/octet-stream")
            //将文件put上去
            def putResult = Fx.http.put(uploadUrl as String, putHeaderMap, fileData, 50000, false, 0)
            log.info("putResult :" + putResult)
            if (putResult[1]['statusCode'] == 200) {
                fileMap.put(nPath, fileId)
            }
            if (ext != 'pdf') {
                need2PdfFileIds.add(fileId)
            }
        }
        //尝试睡眠
        trySleep(need2PdfFileIds)
        //标准log,请保留
        log.info("uploadFile result:" + fileMap)
        return fileMap
    }

    //尝试睡眠,等待文件转pdf
    private void trySleep(List<String> need2PdfFileIds) {
        //标准log,请保留
        log.info("trySleep arg:" + need2PdfFileIds)
        Boolean toPdfOk = false
        Range range = Ranges.of(1, 40)
        range.each {
            if (toPdfOk) {
                return
            }
            Boolean needWaiting2Pdf = false;
            need2PdfFileIds.each { fileId ->
                String statusfileId = queryFileStatus(fileId);
                log.info("状态值:" + statusfileId + "," + fileId)
                if (statusfileId != "5") {
                    needWaiting2Pdf = true
                }
            }
            //查询文件的上传状态,如果不是成功,则继续睡眠
            if (needWaiting2Pdf) {
                sleep(1000)
            } else {
                toPdfOk = true
            }
        }
        //标准log,请保留
        log.info("trySleep result")
    }

    //构建签署人信息
    private List covertSigners(List<InnerAppSignerData> innerAppSignerDataList,
                               List<OuterAppSignerData> outerAppSignerDataList,
                               Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertSigners arg innerAppSignerDataList:" + innerAppSignerDataList)
        log.info("covertSigners arg outerAppSignerDataList:" + outerAppSignerDataList)
        log.info("covertSigners arg fileMap:" + fileMap)
        //签署人信息
        List signers = []
        if (!innerAppSignerDataList.isEmpty()) {
            List innerSigners = covertInnerSigner(innerAppSignerDataList, fileMap)
            signers.addAll(innerSigners)
        }
        if (!outerAppSignerDataList.isEmpty()) {
            List outerSigners = covertOuterSigner(outerAppSignerDataList, fileMap)
            signers.addAll(outerSigners)
        }
        //标准log,请保留
        log.info("covertSigners result:" + signers)
        return signers;
    }

    //构建内部签署人信息
    private List covertInnerSigner(List<InnerAppSignerData> innerAppSignerDataList,
                                   Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertInnerSigner arg innerAppSignerDataList:" + innerAppSignerDataList)
        log.info("covertInnerSigner argfileMap:" + fileMap)
        //签署人信息
        List signers = []
        innerAppSignerDataList.each { item ->
            Map signer = [:]
            //签署人账号
            Map signerAccount = [:]
            signerAccount.put("signerAccountId", item.getSignerId())
            signerAccount.put("noticeType", "1")

            Integer signType = item.getSignType() as Integer

            if (signType == 2) {
                signerAccount.put("authorizedAccountId", AUTHORIZED_ACCOUNT_ID)
            }

            signer.put("signerAccount", signerAccount)

            //签署文件相关信息
            List signfields = []

            item.getSignLocationList().each { signLocation ->
                List covertSignfields = covertSignfields(signType, signLocation, fileMap)
                signfields.addAll(covertSignfields)
            }

            signer.put("signfields", signfields)
            signer.put("signOrder", item.getSignOrder())
            signers.add(signer)
        }
        //标准log,请保留
        log.info("covertInnerSigner result:" + signers)
        return signers
    }

    private List queryFieldKeyWords(SignLocation signLocation) {
        List keyWords = []
        keyWords.add(signLocation.getKeyWord())
        keyWords.add(signLocation.getTimeKeyWord())
        keyWords.add(signLocation.getCrossPageKeyWord())

        return keyWords
    }

    //构建外部签署人信息
    private List covertOuterSigner(List<OuterAppSignerData> outerAppSignerDataList,
                                   Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertOuterSigner arg outerAppSignerDataList:" + outerAppSignerDataList)
        log.info("covertOuterSigner arg fileMap:" + fileMap)
        //签署人信息
        List signers = []
        outerAppSignerDataList.each { item ->
            Integer signType = item.getSignType() as Integer
            //个人签署
            if (signType == 1) {
                Map outerPersonal = covertOuterPersonal(item, fileMap)
                signers.add(outerPersonal)
            }

            //企业签署
            if (signType == 2) {
                Map companySigner = covertOuterCompany(item, fileMap)
                signers.add(companySigner)
            }
        }
        //标准log,请保留
        log.info("covertOuterSigner result:" + signers)
        return signers;
    }

    private List covertKeyWordFields(String field,
                                     Integer signType,
                                     String keyWord,
                                     String timeKeyWord,
                                     Map keyWordMap) {
        //标准log,请保留
        log.info("covertKeyWordFields arg field:" + field)
        log.info("covertKeyWordFields arg signType:" + signType)
        log.info("covertKeyWordFields arg keyWord:" + keyWord)
        log.info("covertKeyWordFields arg timeKeyWord:" + timeKeyWord)
        log.info("covertKeyWordFields arg keyWordMap:" + keyWordMap)
        List signFields = []
        if (keyWord.isEmpty()) {
            Map signField = covertCommonSignField(field, signType)
            signFields.add(signField)
            return signFields
        }
        keyWord = keyWord.replaceAll("#", "") as String
        timeKeyWord = timeKeyWord.replaceAll("#", "") as String

        List positionList = keyWordMap[(keyWord)] as List
        positionList.each { position ->
            List coordinateList = position[("coordinateList")] as List
            if (!coordinateList.isEmpty()) {
                coordinateList.each { coordinate ->
                    Map signField = covertCommonSignField(field, signType)
                    signField.put("assignedPosbean", true)
                    signField.put("signType", 1)
                    Map posBean = [:]
                    posBean.put("posPage", position[("pageIndex")] as Integer)
                    posBean.put("posX", coordinate[("posx")])
                    posBean.put("posY", coordinate[("posy")])
                    signField.put("posBean", posBean)

                    Map timeKeyWordPosition = queryTimeKeywordPositionMap(timeKeyWord, keyWordMap)
                    if (timeKeyWordPosition != null) {
                        Map signDateBean = [:]
                        signDateBean.put("posPage", timeKeyWordPosition[("pageIndex")] as Integer)
                        signDateBean.put("posX", timeKeyWordPosition[("posx")])
                        signDateBean.put("posY", timeKeyWordPosition[("posy")])
                        signField.put("signDateBean", posBean)
                        signField.put("signDateBeanType", 1)
                    }
                    signFields.add(signField)
                }
            }
        }

        if (signFields.isEmpty()) {
            Map signField = covertCommonSignField(field, signType)
            signFields.add(signField)
            return signFields
        }
        //标准log,请保留
        log.info("covertKeyWordFields result:" + signFields)
        return signFields
    }

    private static Map queryTimeKeywordPositionMap(String timeKeyword, Map keywordMap) {
        //标准log,请保留
        log.info("queryTimeKeywordPositionMap arg timeKeyword:" + timeKeyword)
        log.info("queryTimeKeywordPositionMap arg keywordMap:" + keywordMap)
        if (timeKeyword.isEmpty()) {
            return [:]
        }
        log.info("queryTimeKeywordPositionMap timeKeyword:{}, keywordMap:{}" + timeKeyword + keywordMap)
        Map resultMap = [:]
        List keyword = keywordMap[(timeKeyword)] as List
        if (keyword.isEmpty()) {
            return [:]
        }
        resultMap.put("pageIndex", keyword[0]["pageIndex"] as Integer)
        List coordinateList = keyword[0]["coordinateList"] as List
        if (coordinateList.isEmpty()) {
            return [:]
        }
        Map xy = coordinateList[0] as Map
        resultMap.put("posx", xy["posx"] as String)
        resultMap.put("posy", xy["posy"] as String)
        log.info("queryTimeKeywordPositionMap result:{}" + resultMap)
        return resultMap
    }


    private List covertCrossKeyWordSignField(String field,
                                             Integer signType,
                                             String crossKeyWord,
                                             Map keyWordMap) {
        //标准log,请保留
        log.info("covertCrossKeyWordSignField arg field:" + field)
        log.info("covertCrossKeyWordSignField arg signType:" + signType)
        log.info("covertCrossKeyWordSignField arg crossKeyWord:" + crossKeyWord)
        log.info("covertCrossKeyWordSignField arg keyWordMap:" + keyWordMap)
        if (crossKeyWord.isEmpty()) {
            return null
        }
        crossKeyWord = crossKeyWord.replaceAll("#", "") as String
        log.info("crossKeyWord" + crossKeyWord)
        List signFields = []
        List positionList = keyWordMap[(crossKeyWord)] as List
        log.info("positionList" + positionList)
        positionList.each { position ->
            List coordinateList = position[("coordinateList")] as List
            if (!coordinateList.isEmpty()) {
                def coordinate = coordinateList[0]
                Map signField = covertCommonSignField(field, signType)
                signField.put("assignedPosbean", true)
                signField.put("signType", 2)
                Map posBean = [:]
                posBean.put("posPage", "all")
                posBean.put("posY", coordinate[("posy")])
                signField.put("posBean", posBean)
                signFields.add(signField)
                return signFields
            }
        }
        //标准log,请保留
        log.info("covertCrossKeyWordSignField result:" + signFields)
        return signFields
    }


    /**
     * 通用signField的构建
     * @param field
     * @param signType
     * @return
     */
    private Map covertCommonSignField(String field, Integer signType) {
        //标准log,请保留
        log.info("covertCommonSignField arg field:" + field)
        log.info("covertCommonSignField arg signType:" + signType)
        Map signField = [:]
        signField.put("signType", 0)
        signField.put("fileId", field)
        signField.put("assignedPosbean", false)
        //个人签署
        if (signType == 1) {
            signField.put("sealType", 0)
        }
        //企业签署
        if (signType == 2) {
            signField.put("sealType", 1)
            signField.put("actorIndentityType", 2)
        }
        //标准log,请保留
        log.info("covertCommonSignField result:" + signField)
        return signField;
    }

    private static Map covertPositionMap(List positionList) {
        Map positionMap = [:]
        positionList.each { item ->
            positionMap.put(item[("keyword")] as String, item[("positionList")])
        }
        return positionMap
    }

    //创建企业账号
    public String createCompanyAccount(String companyName) {
        //标准log,请保留
        log.info("createCompanyAccount arg:" + companyName)
        Map requestBody = [:]
        requestBody.put("thirdPartyUserId", companyName)
        requestBody.put("name", companyName)

        String contentMD5 = Esign.getContentMD5(requestBody)

        String uri = "/v1/organizations/createByThirdPartyUserId"
        //获取签名字段
        String plaintext = Esign.getPlaintext("POST", ACCEPT, contentMD5, CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plaintext)
        //构建header
        Map headerMap = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, contentMD5);
        //构建url
        String url = BASE_URL + uri
        def result = Fx.http.post(url, headerMap, requestBody)
        if (result[1]["statusCode"] != 200) {
            Fx.message.throwErrorMessage("创建企业失败")
        }
        String enterpriseAccount = result[1]["content"]["data"]["orgId"];
        //标准log,请保留
        log.info("createCompanyAccount result:" + enterpriseAccount)
        return enterpriseAccount
    }

    //构建外部签署人个人的入参
    private Map covertOuterPersonal(OuterAppSignerData outerAppSignerData, Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertOuterPersonal arg outerAppSignerData:" + outerAppSignerData)
        log.info("covertOuterPersonal arg fileMap:" + fileMap)
        String accountId = createPersonalAccount(outerAppSignerData.getSignerName(), outerAppSignerData.getNotifyAddress())
        Map signer = [:]
        //签署人账号
        Map signerAccount = [:]
        signerAccount.put("signerAccountId", accountId)
        signerAccount.put("noticeType", "1")
        signerAccount.put("actorIndentityType", 0)

        signer.put("signerAccount", signerAccount)

        //签署文件相关信息
        List signfields = []
        outerAppSignerData.getSignLocationList().each { signLocation ->
            List covertSignfields = covertSignfields(outerAppSignerData.getSignType(), signLocation, fileMap)
            signfields.addAll(covertSignfields)
        }

        signer.put("signfields", signfields)
        signer.put("signOrder", outerAppSignerData.getSignOrder())
        //标准log,请保留
        log.info("covertOuterPersonal result:" + signer)
        return signer
    }

    private List covertSignfields(Integer signType, SignLocation signLocation, Map fileMap) {
        //标准log,请保留
        log.info("covertSignfields arg signType:" + signType)
        log.info("covertSignfields arg signLocation:" + signLocation)
        log.info("covertSignfields arg fileMap:" + fileMap)
        List signfields = []
        String fileId = fileMap[(signLocation.getFilePath())]
        List positionList = searchWordsPosition(fileId, queryFieldKeyWords(signLocation))
        Map positionMap = covertPositionMap(positionList)
        List keyWordFields = covertKeyWordFields(fileId, signType, signLocation.getKeyWord(), signLocation.getTimeKeyWord(), positionMap)
        if (!keyWordFields.isEmpty()) {
            signfields.addAll(keyWordFields)
        }
        List crossKeyWords = covertCrossKeyWordSignField(fileId, signType, signLocation.getCrossPageKeyWord(), positionMap)
        if (!crossKeyWords.isEmpty()) {
            signfields.addAll(crossKeyWords)
        }
        //标准log,请保留
        log.info("covertSignfields result:" + signfields)
        return signfields
    }

    //构建外部签署人个人的入参
    private Map covertOuterCompany(OuterAppSignerData outerAppSignerData, Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertOuterCompany arg outerAppSignerData:" + outerAppSignerData)
        log.info("covertOuterCompany arg fileMap:" + fileMap)
        String companyAccountId = createCompanyAccount(outerAppSignerData.getCompanyName())
        String accountId = createPersonalAccount(outerAppSignerData.getSignerName(), outerAppSignerData.getNotifyAddress())
        Map signer = [:]
        //签署企业账号
        Map signerAccount = [:]
        signerAccount.put("signerAccountId", accountId)
        signerAccount.put("authorizedAccountId", companyAccountId)
        signerAccount.put("noticeType", "1")
        signerAccount.put("actorIndentityType", 2)
        signer.put("signerAccount", signerAccount)

        //签署文件相关信息
        List signfields = []
        outerAppSignerData.getSignLocationList().each { signLocation ->
            List covertSignfields = covertSignfields(outerAppSignerData.getSignType(), signLocation, fileMap)
            signfields.addAll(covertSignfields)
        }
        signer.put("signfields", signfields)
        signer.put("signOrder", outerAppSignerData.getSignOrder())
        //标准log,请保留
        log.info("covertOuterCompany result:" + signer)
        return signer
    }

    private List covertDocs(List<FileAttachment> fileAttachments, Map<String, String> fileMap) {
        //标准log,请保留
        log.info("covertDocs arg fileAttachments:" + fileAttachments)
        log.info("covertDocs arg fileMap:" + fileMap)
        //签署文件
        List docs = []
        if (fileAttachments.isEmpty()) {
            return docs
        }
        fileAttachments.each { item ->
            Map doc = [:]
            doc.put("fileId", fileMap[(item.getPath())])
            docs.add(doc)
        }
        //标准log,请保留
        log.info("covertDocs result:" + docs)
        return docs
    }

    //搜索关键字坐标
    public List searchWordsPosition(String fileId, List<String> keyWords) {
        //标准log,请保留
        log.info("searchWordsPosition arg fileId:" + fileId)
        log.info("searchWordsPosition arg keyWords:" + keyWords)
        String uri = "/v1/documents/{fileId}/searchWordsPosition";
        uri = uri.replaceAll("\\{[fileId^}]*\\}", "" + fileId + "");

        String query = "?keywords="
        keyWords.each { item ->
            query = query + item.replaceAll("#", "") as String
            query = query + ","
        }

        uri = uri + query;
        String url = BASE_URL + uri
        //构建签名字段
        String plainText = Esign.getPlaintext("GET", ACCEPT, "", CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, "");
        //调用get接口
        //标准log,请保留
        log.info("searchWordsPosition http request")
        def result = Fx.http.get(url, headers, 10000, false, 0);
        //标准log,请保留
        log.info("searchWordsPosition http response:" + result)
        def content = result[1]["content"]
        List wordsPositionList = []
        if (content["code"] == 0) {
            wordsPositionList = Fx.json.parseList(Fx.json.toJson(content["data"]))
        }
        //标准log,请保留
        log.info("searchWordsPosition result:" + wordsPositionList)
        return wordsPositionList
    }

    //撤销电子签
    public CancelSign.Result cancelSign(CancelSign.Arg arg) {
        //标准log,请保留
        log.info("cancelSign arg:" + arg)
        String flowId = arg.getTaskId();
        String uri = "/v1/signflows/{flowId}/revoke";
        uri = uri.replaceAll("\\{[flowId^}]*\\}", "" + flowId + "");
        String url = BASE_URL + uri;
        Map requestBody = [:]
        //生成md5
        String contentMD5 = Esign.getContentMD5(requestBody)
        //构建签名字段
        String plainText = Esign.getPlaintext("PUT", ACCEPT, contentMD5, CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, contentMD5);
        //调用put接口
        //标准log,请保留
        log.info("cancelSign http request:" + requestBody)
        def response = Fx.http.put(url, headers, requestBody);
        //标准log,请保留
        log.info("cancelSign http response:" + response)
        if (response[1]["statusCode"] != 200) {
            Fx.message.throwErrorMessage("撤销失败")
        }
        CancelSign.Result result = CancelSign.Result.builder().success(true).build();
        //标准log,请保留
        log.info("cancelSign result:" + result)
        return result
    }

    //下载签署文件
    public DownloadSignFile.Result downloadSignFile(DownloadSignFile.Arg arg) {
        //标准log,请保留
        log.info("downloadSignFile arg:" + arg)
        String flowId = arg.getTaskId();
        String uri = "/v1/signflows/{flowId}/documents";
        uri = uri.replaceAll("\\{[flowId^}]*\\}", "" + flowId + "");
        //构建签名字段
        String plainText = Esign.getPlaintext("GET", ACCEPT, "", CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, "");
        //调用put接口
        //标准log,请保留
        log.info("downloadSignFile http request")
        def response = Fx.http.get(BASE_URL + uri, headers, 10000, false, 0);
        //标准log,请保留
        log.info("downloadSignFile http response:" + response)
        if (response[1]["statusCode"] != 200) {
            Fx.message.throwErrorMessage("下载失败")
        }
        log.info("下载文件:" + response)
        List files = response[1]["content"]["data"]["docs"] as List
        //返回值
        List<FileAttachment> fileAttachments = []
        files.each() { x ->
            String fileUrl = x["fileUrl"] as String
            def (Boolean error, HttpResult reuslt, String errorMessage) = Fx.http.get(fileUrl, [:])
            if (error) {
                Fx.log.info("http 下载文件请求出错 : " + errorMessage)
                return null;
            }
            if (reuslt.statusCode != 200) {
                Fx.log.info("http 下载文件响应错误 :" + reuslt.content)
            }
            //上传图片到文件服务器
            byte[] bytes = reuslt.bytes
            String fileName = x["fileName"] as String
            String ext = getExt(fileName)
            def uploadReusult = Fx.file.uploadFile(ext, bytes.length, bytes)
            if (uploadReusult[0]) {
                Fx.log.info("上传图片错误 :" + uploadReusult[2])
                return null;
            }
            //将返回的路径更新到对象字段下
            String path = uploadReusult[1]['path'] as String

            FileAttachment fileAttachment = FileAttachment.builder().
                    filename(fileName).
                    ext(ext).
                    path(path).
                    size(uploadReusult[1]['size'] as Long).
                    build()

            fileAttachments.add(fileAttachment)
        }
        DownloadSignFile.Result result = DownloadSignFile.Result.builder().fileAttachments(fileAttachments).build();
        //标准log,请保留
        log.info("downloadSignFile result:" + result)
        return result;
    }

    //获取文件的后缀
    private static String getExt(String fileName) {
        String[] arr = fileName.split("\\.")
        return arr[-1]
    }

    //获取签署链接
    public GetSignUrl.Result getSignUrl(GetSignUrl.Arg arg) {
        //标准log,请保留
        log.info("getSignUrl arg:" + arg)
        String flowId = arg.getTaskId();
        String uri = "/v1/signflows/{flowId}/executeUrl";
        uri = uri.replaceAll("\\{[flowId^}]*\\}", "" + flowId + "");
        String accountId = arg.getSignerId()
        uri = uri + "?accountId=" + accountId
        uri = uri + "&amp;organizeId=" + "0"
        //构建签名字段
        String plainText = Esign.getPlaintext("GET", ACCEPT, "", CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, "");
        //调用put接口
        def response = Fx.http.get(BASE_URL + uri, headers, 10000, false, 0);
        log.info("getSignUrl http response:" + response)
        if (response[1]["statusCode"] != 200) {
            Fx.message.throwErrorMessage("获取签署url失败")
        }
        if (response[1]["content"]["code"] == 1437114) {
            Fx.message.throwErrorMessage(response[1]["content"]["message"] as String)
        }
        GetSignUrl.Result result = GetSignUrl.Result.builder().url(response[1]["content"]["data"]["shortUrl"] as String).build();
        //标准log,请保留
        log.info("getSignUrl result:" + result)
        return result;
    }

    //签署回调解析参数
    public SignCallback.Result signCallback(SignCallback.Arg arg) {
        //标准log,请保留
        log.info("signCallback arg:" + arg)
        String requestBody = arg.getRequestBodyByContent() as String;
        Map requestBodyMap = Fx.json.parse(requestBody)
        switch (requestBodyMap[("action")] as String) {
        //签署人签署完成
            case "SIGN_FLOW_UPDATE":
                SignCallback.Result result = buildFlowUpdateResult(requestBodyMap);
                //标准log,请保留
                log.info("signCallback update result:" + result)
                return result;
                //签署流程结束
            case "SIGN_FLOW_FINISH":
                SignCallback.Result result = buildFlowFinish(requestBodyMap);
                //标准log,请保留
                log.info("signCallback finish result:" + result)
                return result;
            default:
                //标准log,请保留
                log.info("signCallback error")
                return null;
        }
    }

    private SignCallback.Result buildFlowUpdateResult(Map requestBodyMap) {
        //标准log,请保留
        log.info("buildFlowUpdateResult arg:" + requestBodyMap)
        TaskStatus taskStatus;
        switch (requestBodyMap[("signResult")] as String) {
            case "2":
                taskStatus = TaskStatus.Signed;
                break;
            case "4":
                taskStatus = TaskStatus.VisaDenied;
                break;
            default:
                taskStatus = null;
                break;
        }
        SignCallback.Result result = SignCallback.Result.builder().
                taskId(requestBodyMap[("flowId")] as String).
                taskStatus(taskStatus).
                refusalReason(requestBodyMap["resultDescription"] as String).
                signerId(requestBodyMap[("accountId")] as String).
                callBackEvent(CallBackEvent.SignEvent).
                successMsg("success").
                successCode(200).
                build();
        //标准log,请保留
        log.info("buildFlowUpdateResult result:" + result)
        return result;
    }

    private SignCallback.Result buildFlowFinish(Map requestBodyMap) {
        //标准log,请保留
        log.info("buildFlowFinish arg:" + requestBodyMap)
        def flowStatus = requestBodyMap[("signResult")] as String
        TaskStatus taskStatus;
        if (flowStatus != null && !flowStatus.isEmpty() && flowStatus == '3' ) {
            taskStatus = TaskStatus.Revoked;
        } else {
            taskStatus = TaskStatus.Completed;
        }
        SignCallback.Result result = SignCallback.Result.builder().
                taskId(requestBodyMap[("flowId")] as String).
                taskStatus(taskStatus).
                refusalReason(requestBodyMap["statusDescription"] as String).
                callBackEvent(CallBackEvent.SignEvent).
                successMsg("success").
                successCode(200).
                build();
        //标准log,请保留
        log.info("buildFlowFinish result:" + result)
        return result;
    }

    //催签
    public UrgeSign.Result urgeSign(UrgeSign.Arg arg) {
        //标准log,请保留
        log.info("urgeSign arg:" + arg)
        String flowId = arg.getTaskId();
        String uri = "/v1/signflows/{flowId}/signers/rushsign";
        uri = uri.replaceAll("\\{[flowId^}]*\\}", "" + flowId + "");

        Map requestBody = [:]
        //生成md5
        String contentMD5 = Esign.getContentMD5(requestBody)
        //构建签名字段
        String plainText = Esign.getPlaintext("PUT", ACCEPT, contentMD5, CONTENT_TYPE, uri);
        //签名认证
        def signOpenCaSignature = Esign.encodePlainText(plainText)
        //构建header
        Map headers = Esign.getHeaders(ACCEPT, CONTENT_TYPE, signOpenCaSignature, contentMD5);
        //调用put接口
        def response = Fx.http.put(BASE_URL + uri, headers, requestBody);
        if (response[1]["statusCode"] != 200) {
            Fx.message.throwErrorMessage("催签失败")
        }
        UrgeSign.Result result = UrgeSign.Result.builder().success(true).build();
        //标准log,请保留
        log.info("urgeSign result:" + result)
        return result;
    }


    public static void main(String[] args) {
        // addEmployeesTest()
        // String ff = queryFileStatus("6c49acee65884f67ad7bbf2cff19ac00");
        // log.info(ff)
        // initiateSignTest()
        // cancelSignTest()
        // urgeSignTest()
        // downloadSignFileTest()
        // Sign sign = Fx.klass.newInstance('Sign') as Sign
        // List result = sign.searchWordsPosition("7fd63a8808214f06a2bf105a332af1fe", ["#{signer_1}", "#{signer_time_1}", "#{crosspage_signer_1}"])
        // log.info(result)
        // getSignUrlTest()
    }

    private static void initiateSignTest() {
        String arg = "{\"externalType\": 1, \"fileAttachments\": [{\"create_time\": 1641350684526, \"ext\": \"pdf\", \"filename\": \"电子签发起打印模板.pdf\", \"path\": \"N_202201_05_e7881c36d8e247c98eb8af61b6e8cc08\", \"size\": 5834 } ], \"innerAppSignerDataList\": [{\"notifyAddress\": \"17745113979\", \"signLocationList\": [{\"crossPageKeyWord\": \"#{crosspage_signer_1}\", \"filePath\": \"N_202112_24_5dd31bf0ddb1415ba587ec338c3af0d5.pdf\", \"keyWord\": \"#{signer_1}\", \"timeKeyWord\": \"#{signer_time_1}\"} ], \"signOrder\": 1, \"signType\": 1, \"signWay\": 1, \"signerId\": \"f0947f852b41404098dfdfab9139ee79\", \"signerName\": \"张宇\"} ], \"innerSignType\": 1, \"outerAppSignerDataList\": [], \"signWay\": 1, \"taskSubject\": \"测试签署电子合同\"}"
        InitiateSign.Arg initiateSignArg = Fx.json.parseObject(arg, InitiateSign.Arg.class)
        Esign sign = Fx.klass.newInstance('Esign') as Esign
        sign.initiateSign(initiateSignArg)
    }

    private static void addEmployeesTest() {
        Esign sign = Fx.klass.newInstance('Esign') as Esign

        AddEmployees.EmployeeData employeeData = AddEmployees.EmployeeData.builder().
                mobile("17745113979").
                objectId("61c56342a89376000199dfb1").
                userName("张宇").
                build()
        List employeeDataList = []
        employeeDataList.add(employeeData)

        AddEmployees.Arg arg = AddEmployees.Arg.builder().
                employeeDataList(employeeDataList).
                build()

        sign.addEmployees(arg)
    }

    private static void cancelSignTest() {
        Esign sign = Fx.klass.newInstance('Esign') as Esign
        CancelSign.Arg arg = CancelSign.Arg.builder().taskId("3c9a2be6db774c28bde575680d7cc1b7").build()
        sign.cancelSign(arg)
    }

    private static void urgeSignTest() {
        Esign sign = Fx.klass.newInstance('Esign') as Esign
        UrgeSign.Arg arg = UrgeSign.Arg.builder().taskId("3c9a2be6db774c28bde575680d7cc1b7").build()
        sign.urgeSign(arg)
    }

    private static void downloadSignFileTest() {
        Esign sign = Fx.klass.newInstance('Esign') as Esign
        DownloadSignFile.Arg arg = DownloadSignFile.Arg.builder().taskId("710d20bd61f44345b871e8ffa5a3c06f").build();
        sign.downloadSignFile(arg)
    }

    private static void getSignUrlTest() {
        Esign sign = Fx.klass.newInstance('Esign') as Esign
        GetSignUrl.Arg arg = GetSignUrl.Arg.builder().taskId("dcd85e52b07d4f3d92038047f7e856a2").signerId("addb6aceb3fb4d0ea9dc9ef4280e1a83").build()
        def result = sign.getSignUrl(arg)
        log.info(result)
    }
}
2024-10-22
0 0
Baidu
map