springcloud阿里云OCR(个人证照)识别对接

由于项目要用到实名认证功能,因此OCR识别是前提,我们是用的阿里云的OCR,首先需要在阿里云上面开通OCR功能

开通地址:https://ocr.console.aliyun.com/overview

然后就是写后端接口:

首先是pom依赖:

java 复制代码
 <dependency>
            <groupId>com.aliyun</groupId>
            <artifactId>ocr_api20210707</artifactId>
            <version>3.1.3</version>
        </dependency>

接口层:

java 复制代码
    /**
     * ocr识别
     *
     * @param bytes
     * @param originalFilename
     * @return
     */
    AppMemberUserOcrRespVO ocrIdentify(byte[] bytes, String originalFilename);

接口实现:

java 复制代码
    @Value("${aliyun.appId}")
    private String accessKeyId;

    @Value("${aliyun.appSecret}")
    private String accessKeySecret;

    private static final String END_POINT = "ocr-api.cn-hangzhou.aliyuncs.com";
    private static final Integer SUCCESS_CODE = 200;
    // 正面
    private static final String FACE = "face";
    // 背面
    private static final String BACK = "back";


@Override
    public AppMemberUserOcrRespVO ocrIdentify(byte[] bytes, String originalFilename) {
        String url = fileApi.createFile(bytes, originalFilename);
        if (StringUtils.isBlank(url)) {
            throw exception(OCR_FILE_UPLOAD_FAIL);
        }
        com.aliyun.credentials.Client credential = new com.aliyun.credentials.Client();
        com.aliyun.teaopenapi.models.Config config = new com.aliyun.teaopenapi.models.Config()
                .setAccessKeyId(accessKeyId)
                .setAccessKeySecret(accessKeySecret)
                .setCredential(credential);
        config.endpoint = END_POINT;
        com.aliyun.ocr_api20210707.Client client = null;
        try {
            client = new com.aliyun.ocr_api20210707.Client(config);
        } catch (Exception e) {
            log.error("ocr client init fail", e);
            throw exception(OCR_CLIENT_INIT_FAIL);
        }
        com.aliyun.ocr_api20210707.models.RecognizeIdcardRequest recognizeIdcardRequest = new com.aliyun.ocr_api20210707.models.RecognizeIdcardRequest().setUrl(url);
        com.aliyun.teautil.models.RuntimeOptions runtime = new com.aliyun.teautil.models.RuntimeOptions();
        AppMemberUserOcrRespVO appMemberUserOcrRespVO = new AppMemberUserOcrRespVO();
        try {
            com.aliyun.ocr_api20210707.models.RecognizeIdcardResponse resp = client.recognizeIdcardWithOptions(recognizeIdcardRequest, runtime);
            if (resp == null || !SUCCESS_CODE.equals(resp.getStatusCode())) {
                throw exception(OCR_RESULT_FAIL);
            }
            RecognizeIdcardResponseBody body = resp.getBody();
            String data = body.getData();

            if (StringUtils.isBlank(data)) {
                log.error("OCR response data is empty");
                throw exception(OCR_RESULT_FAIL);
            }
            JSONObject jsonObject = JSON.parseObject(data);
            JSONObject dataObject = jsonObject.getJSONObject("data");
            if (dataObject == null) {
                log.error("OCR response data object is null");
                throw exception(OCR_RESULT_FAIL);
            }

            // 处理身份证正面信息
            processFrontSide(dataObject, appMemberUserOcrRespVO);
            // 处理身份证背面信息
            processBackSide(dataObject, appMemberUserOcrRespVO);
        } catch (TeaException error) {
            log.error("OCR recognition failed - Message: {}, Recommend: {}", error.getMessage(), error.getData().get("Recommend"));
            throw exception(OCR_RESULT_FAIL);
        } catch (Exception _error) {
            log.error("OCR recognition failed unexpectedly - Message: {}", _error.getMessage(), _error);
            throw exception(OCR_RESULT_FAIL);
        }
        return appMemberUserOcrRespVO;
    }

    /**
     * 提取处理身份证正面信息的方法
     *
     * @param dataObject
     * @param respVO
     */
    private void processFrontSide(JSONObject dataObject, AppMemberUserOcrRespVO respVO) {
        if (dataObject.containsKey(FACE)) {
            JSONObject faceData = dataObject.getJSONObject(FACE).getJSONObject("data");
            if (faceData != null) {
                respVO.setAddress(getStringValue(faceData, "address"));
                respVO.setBirthDate(getStringValue(faceData, "birthDate"));
                respVO.setEthnicity(getStringValue(faceData, "ethnicity"));
                respVO.setIdNumber(getStringValue(faceData, "idNumber"));
                respVO.setName(getStringValue(faceData, "name"));
                respVO.setSex(getStringValue(faceData, "sex"));
            }
        }
    }

    /**
     * 提取处理身份证背面信息的方法
     *
     * @param dataObject
     * @param respVO
     */
    private void processBackSide(JSONObject dataObject, AppMemberUserOcrRespVO respVO) {
        if (dataObject.containsKey(BACK)) {
            JSONObject backData = dataObject.getJSONObject(BACK).getJSONObject("data");
            if (backData != null) {
                respVO.setIssueAuthority(getStringValue(backData, "issueAuthority"));
                respVO.setValidPeriod(getStringValue(backData, "validPeriod"));
            }
        }
    }

    /**
     * 安全获取JSON字符串值的辅助方法
     *
     * @param jsonObject
     * @param key
     * @return
     */
    private String getStringValue(JSONObject jsonObject, String key) {
        Object value = jsonObject.get(key);
        return value != null ? value.toString() : null;
    }

接口出参:

java 复制代码
import io.swagger.v3.oas.annotations.media.Schema;
import lombok.AllArgsConstructor;
import lombok.Data;
import lombok.NoArgsConstructor;

@Schema(description = "用户 APP - OCR信息 Response VO")
@Data
@NoArgsConstructor
@AllArgsConstructor
public class AppMemberUserOcrRespVO {
    /**
     * 正面信息
     */
    @Schema(description = "姓名", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "章三")
    private String name;

    @Schema(description = "性别", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "男")
    private String sex;

    @Schema(description = "民主", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "汉族")
    private String ethnicity;

    @Schema(description = "出生日期", requiredMode = Schema.RequiredMode.NOT_REQUIRED)
    private String birthDate;

    @Schema(description = "住址", requiredMode = Schema.RequiredMode.NOT_REQUIRED)
    private String address;

    @Schema(description = "身份证好吗", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "330301200001090002")
    private String idNumber;

    /**
     * 反面信息
     */
    @Schema(description = "签发机关", requiredMode = Schema.RequiredMode.NOT_REQUIRED, example = "北京市公安局")
    private String issueAuthority;

    @Schema(description = "有效期限", requiredMode = Schema.RequiredMode.NOT_REQUIRED)
    private String validPeriod;
}

阿里云开发文档和调试地址:

https://api.aliyun.com/api/ocr-api/2021-07-07/RecognizeIdcard?spm=5176.23043878_1479102120.console-base_help.53.182e1cdcLtXfrz&sdkStyle=dara&RegionId=cn-hangzhou&params={%22body%22:%228ac2446b-7197-4470-95f6-cfd12172f4ad%22}&tab=DOC&lang=JAVA

入参和出参的字段都是有说明的:

然后我用apiFox测试:

相关推荐
Inhand陈工1 天前
基于台达PLC与映翰通IG502的智慧水产养殖精准投喂与远程运维解决方案
运维·人工智能·物联网·阿里云·信息与通信
Database_Cool_1 天前
什么是数据仓库物化视图?AnalyticDB MySQL 实时物化视图能力解析
人工智能·mysql·阿里云
Database_Cool_1 天前
大规模数据分析降本指南:AnalyticDB Serverless 弹性架构实战
数据仓库·阿里云·架构·数据分析·serverless
Sour1 天前
PDF翻译卡住不动怎么办?扫描件、OCR 和大文件排查清单
前端·pdf·ocr
tiancaijiben1 天前
阿里云Kubernetes集群托管完全指南:从创建到生产级运维
云计算
我是小bā吖1 天前
Claude Code 模型接入阿里云 AI 网关并统计不同使用者的模型用量
网络·人工智能·阿里云
翼龙云_cloud1 天前
阿里云国际代理商:如何使用RDS MySQL 构建网站数据库?
数据库·mysql·阿里云
旗讯数字1 天前
旗讯 OCR 工业手写识别解决方案|破解车间纸质表单录入难题,加速生产数字化转型
大数据·ocr
互联网推荐官1 天前
上海软件定制开发公司推荐:从PaaS工程化路径看D-coding的技术取舍
云原生·云计算·paas·软件开发·开发经验·上海
sbjdhjd1 天前
从零搭建企业级 CI/CD(下):Jenkins+GitLab+Harbor 全链路实战指南
git·servlet·ci/cd·云原生·云计算·gitlab·jenkins