诺诺开放平台电子发票对接

需求

公司因自有订单业务规模不断扩大,产生了线上电子发票开具的需求,对接的是诺诺开发平台。

开发指南

申请诺诺资质:申请成为诺诺平台资质,提交资料,一次性费用2w+,每年一定维护费用。

创建应用类型

诺诺开放平台支持如下两种应用类型,企业可根据业务需求选择。

  1. 自用型:接入诺诺开放平台业务能力,为自己公司开发应用。自助接入建议使用自用型应用


    自用型流程图
  2. 第三方应用:第三方接入(帮助其他企业开发)建议使用第三方应用。系统服务商可创建第三方应用,开发应用服务于商户,可代商户发起调用。进行第三方调用前,需在应用中添加对应功能并获得商户授权。


    第三方应用流程图
自用型对接

自用型可以理解为公司给客户开票


应用列表
  1. 创建自用型应用获取到APPKey和APPSecret
  2. 获取access_token
    access_token是开发者调用开放平台接口的调用凭据,开发者通过应用参数向诺诺开放平台调用令牌接口地址获取access_token。令牌有效期默认24h(也可在创建应用时设置token永不过期,我们创建的是默认24h),且令牌30天内的调用上限为50次 ,请开发者做好令牌的管理。
    private String getNNToken(String redisKey) {
        // 获取token
        Object token = redisUtil.get(redisKey);
        if (ObjectUtils.isNotEmpty(token)) {
            return (String) token;
        }
        String result = NNOpenSDK.getIntance().getMerchantToken(nnAppkey, nnAppSecret, nn_accessToken_url);
        HashMap tokenMap = JSON.parseObject(result, HashMap.class);
        token = tokenMap.get("access_token”);
        if (ObjectUtils.isEmpty(token)) {
            String msg = "获取token出错:" + tokenMap.get("error") + " " + tokenMap.get("error_description”);
            throw new MsgException(msg);
        }
        // 缓存token 比诺诺先过期
        long expires_in = (Integer) tokenMap.get("expires_in") - 60 * 60;
        redisUtil.set(redisKey, token, expires_in);
        return (String) token;
    }

获取到的access_token,存入到redis,就可以使用access_token,调用开票、重开、查询、发送email等接口

@Override
    public AjaxResponse<Object> ybInvoice(InvoiceOrderDTO orderDTO) throws Exception {
        // 校验order信息
        String content = generateOrder(orderDTO);
        
        NNOpenSDK sdk = NNOpenSDK.getIntance();
        String method = InvoiceMethodEnum.INVOICE_METHOD_NEW.getKey();
        String senId = IdUtils.simpleUUID();
        // 获取token
        String token = getNNToken(yuanben_redisKey);
        
        //调用诺诺接口的时间
        long reqApiTimes = System.currentTimeMillis();
        String result = sdk.sendPostSyncRequest(nnUrl, senId, nnAppkey, nnAppSecret, token, StaticValue.SHUNDIAN_TAXNUM, method, content);
        log.info("ybInvoice result = " + result);
        AjaxResponse<Object> ajaxResponse = generateAjaxResponse(result, reqApiTimes);
        return ajaxResponse;
    }

    @Override
    public AjaxResponse<Object> ybReInvoice(String serialNo) {
        if (StringUtils.isBlank(serialNo)) {
            throw new MsgException("发票流水号不能空!”);
        }
        NNOpenSDK sdk = NNOpenSDK.getIntance();
        String method = InvoiceMethodEnum.INVOICE_METHOD_REINVOICE.getKey();
        String senId = IdUtils.simpleUUID();
        // 获取token
        String token = getNNToken(yuanben_redisKey);
        Map<String, Object> contentMap = new HashMap<>();
        contentMap.put("fpqqlsh", serialNo);
        String content = JSON.toJSONString(contentMap);
        //调用诺诺接口的时间
        long reqApiTimes = System.currentTimeMillis();
        String result = sdk.sendPostSyncRequest(nnUrl, senId, nnAppkey, nnAppSecret, token, StaticValue.SHUNDIAN_TAXNUM, method, content);
        log.info("result = " + result);

        AjaxResponse<Object> ajaxResponse = generateAjaxResponse(result, reqApiTimes);
        if (ajaxResponse.isState()) {
            saveInvoiceRecord(StaticValue.SHUNDIAN_TAXNUM, serialNo, System.currentTimeMillis(), 0);
        }
        return ajaxResponse;
    }

    @Override
    public AjaxResponse<Object> ybQueryInvoiceResult(String serialNos, String orderNos, String isOfferInvoiceDetail) {

        if (StringUtils.isBlank(serialNos) && StringUtils.isBlank(orderNos)) {
            throw new MsgException("发票流水号或订单编号,两字段二选一”);
        }
        if (StringUtils.isBlank(isOfferInvoiceDetail)) {
            isOfferInvoiceDetail = “0”;
        }
        String[] serialNosArray = null;
        String[] orderNosArray = null;
        if (StringUtils.isNotBlank(serialNos)) {
            serialNosArray = serialNos.split(",”);
        }
        if (StringUtils.isNotBlank(orderNos)) {
            orderNosArray = orderNos.split(",”);
        }

        NNOpenSDK sdk = NNOpenSDK.getIntance();
        String method = InvoiceMethodEnum.INVOICE_METHOD_RESULT.getKey();
        String senId = IdUtils.simpleUUID();
        String token = getNNToken(yuanben_redisKey);

        Map<String, Object> contentMap = new HashMap<>();
        if (StringUtils.isNotBlank(serialNos)) {
            contentMap.put("serialNos", serialNosArray);
        } else {
            contentMap.put("orderNos", orderNosArray);
        }
        contentMap.put("isOfferInvoiceDetail", isOfferInvoiceDetail);
        String content = JSON.toJSONString(contentMap);
        //调用诺诺接口的时间
        long reqApiTimes = System.currentTimeMillis();
        String result = sdk.sendPostSyncRequest(nnUrl, senId, nnAppkey, nnAppSecret, token, StaticValue.SHUNDIAN_TAXNUM, method, content);
        log.info("result = " + result);
        
        return generateAjaxResponse(result, reqApiTimes);
    }

    @Override
    public AjaxResponse<Object> ybDeliveryInvoice(String invoiceCode, String invoiceNum, String phone, String mail) {
        NNOpenSDK sdk = NNOpenSDK.getIntance();
        String method = InvoiceMethodEnum.INVOICE_METHOD_DELIVERY.getKey();
        String senId = IdUtils.simpleUUID();
        String token = getNNToken(yuanben_redisKey);

        AssertUtil.notBlank(invoiceCode, "发票代码不能为空”);
        AssertUtil.notBlank(invoiceNum, "发票号码不能为空”);

        if (StringUtils.isBlank(phone) && StringUtils.isBlank(mail)) {
            throw new MsgException("交付手机号,和交付邮箱至少有一个不为空”);
        }

        Map<String, Object> contentMap = new HashMap<>();
        if (StringUtils.isNotBlank(phone)) {
            contentMap.put("phone", phone);
        } else {
            contentMap.put("mail", mail);
        }
        contentMap.put("taxnum", StaticValue.SHUNDIAN_TAXNUM);
        contentMap.put("invoiceCode", invoiceCode);
        contentMap.put("invoiceNum", invoiceNum);
        String content = JSON.toJSONString(contentMap);
        //调用诺诺接口的时间
        long reqApiTimes = System.currentTimeMillis();
        String result = sdk.sendPostSyncRequest(nnUrl, senId, nnAppkey, nnAppSecret, token, StaticValue.SHUNDIAN_TAXNUM, method, content);
        log.info("result = " + result);

        return generateAjaxResponse(result, reqApiTimes);
    }

    @Override
    public AjaxResponse<Object> ybCancelInvoice(String invoiceId, String invoiceCode, String invoiceNo) {
        NNOpenSDK sdk = NNOpenSDK.getIntance();
        String method = InvoiceMethodEnum.INVOICE_METHOD_CANCEL.getKey();
        String senId = IdUtils.simpleUUID();
        String token = getNNToken(yuanben_redisKey);

        AssertUtil.notBlank(invoiceId, "发票流水号不能为空”);
        AssertUtil.notBlank(invoiceCode, "发票代码不能为空”);
        AssertUtil.notBlank(invoiceNo, "发票号码不能为空”);

        Map<String, Object> contentMap = new HashMap<>();
        contentMap.put("invoiceId", invoiceId);
        contentMap.put("invoiceCode", invoiceCode);
        contentMap.put("invoiceNo", invoiceNo);
        String content = JSON.toJSONString(contentMap);
        //调用诺诺接口的时间
        long reqApiTimes = System.currentTimeMillis();
        String result = sdk.sendPostSyncRequest(nnUrl, senId, nnAppkey, nnAppSecret, token, StaticValue.SHUNDIAN_TAXNUM, method, content);
        log.info("result = " + result);

        return generateAjaxResponse(result, reqApiTimes);
    }

校验订单必传参数(客户税务信息,订单信息)后,组装订单默认参数,商户信息从数据库查询。

/**
     * 组装订单必填信息
     * @Params: 
     * @DateTime: 2022/7/15 下午4:22
     * @Author: zenghao
     */
    private String generateOrder(InvoiceOrderDTO orderDTO) throws Exception {
        ClientCacheModel clientCacheModel = ThreadLocalClientCache.get();
        // 查询商户信息
        TinSalerPO salerPO = salerDAO.findByPrimaryKey(clientCacheModel.getGuid());
        // 生成订单号、默认蓝票、默认发邮件
        orderDTO.setOrderNo(getOrderNo());
        orderDTO.setInvoiceDate(DateTime.getCurrentDate_YYYYMMDDHHMMSS());
        orderDTO.setInvoiceType(InvoiceTypeEnum.BLUE_TICKET.getKey());
        // 0 代表开票成功后发送电子票到指定邮件
        orderDTO.setPushMode("0”);
        orderDTO.setBuyerPhone(orderDTO.getBuyerTel());
        // 商户信息
        orderDTO.setSalerAccount(salerPO.getAccountBank());
        orderDTO.setSalerAddress(salerPO.getAddress());
        orderDTO.setSalerTel(salerPO.getTel());
        orderDTO.setSalerTaxNum(salerPO.getTaxNum());
        // 设置回调地址,开票成功后,回调接口处理后续业务
        orderDTO.setCallBackUrl(callBackUrl);
        
        Map<String, Object> map = new HashMap<>();
        map.put("order", orderDTO);
        String content = JSON.toJSONString(map);
        log.info(orderDTO.toString());
        return content;
    }

重点提一点,提交成功后,返回是开票提交成功,会立即返回开票流水号,这里电子票的状态不是最终状态,如果使用流水号查询开票信息,有可能是开票中状态。
这里我们需要利用比较关键的参数,callBackUrl,写入订单成功后的回调地址,利用回调接口,读取返回的订单流水号和订单json信息,然后进行一些数据存储的业务处理。

@RequestMapping("/callback”)
    @ResponseBody
    public AjaxResponse invoiceCallback(HttpServletRequest request) throws Exception {
        //返回的内容
        String content = request.getParameter("content”);
        Map map = JSON.parseObject(content, Map.class);
        //发票流水号
        String serialNum = (String) map.get("c_fpqqlsh”);
        //商户税号
        String saleTaxNum = (String) map.get("c_saletaxnum”);
        invoiceRecordService.invoiceCallback(serialNum, saleTaxNum);
        return AjaxResponse.ok();
    }
第三方应用对接

第三方应用对接可以理解为公司的客户给它的客户开票

  1. 公司的客户也需要提交资料到诺诺平台,获取资质;
  2. 注册诺诺平台账号;
  3. 授权成为创建应用下的商户;

利用回调地址可以接受商户授权成功后code,利用code调用生成access_token接口,有了这个access_token,就可以进行开票等接口的正常调用了。创建应用时选择的是access_token永久有效,商户的access_token存入数据库已备用。

private String getNNToken(String code, String taxNum) {
        Object token = null;
        String result = NNOpenSDK.getIntance().getISVToken(tpa_nnAppkey, tpa_nnAppSecret, code, taxNum, auth_redirect_uri, nn_accessToken_url);
        HashMap tokenMap = JSON.parseObject(result, HashMap.class);
        token = tokenMap.get("access_token");
        if (ObjectUtils.isEmpty(token)) {
            String msg = "获取token出错:" + tokenMap.get("error") + " " + tokenMap.get("error_description");
            throw new MsgException(msg);
        }
        log.info("taxNum = " + taxNum + "access_token = " + tokenMap.toString());
        return (String) token;
    }

创建第三方应用时,设置的授权回调地址设置很重要。

    @GetMapping("/tpa/merchant/code")
    @ResponseBody
    public AjaxResponse tpAppMerchant(@RequestParam String code, @RequestParam String taxnum) throws Exception {
        invoiceService.tpAppGetMerchantToken(code, taxnum);
        return AjaxResponse.ok();
    }

授权后可以再商户管理查看商户授权信息


截屏2022-07-17 下午1.58.40.png

第三方应用类型在流程上稍微绕一些,商户获取资质有一定的费用。

对接过程中也遇到了不少问题,需要与诺诺人员沟通。

需要注意的点:
  1. 调用开票等接口前,需要插入税盘,安装诺诺的软件,启动助手。
  1. 对接没有使用沙盒环境(用不了,喊用正式环境),使用的正式环境开的测试票,最后红冲了。

  2. 调用平台C回调地址设置
    诺诺平台-A,开票中间平台-B,调用平台-C
    如果做的是开票中间平台B,假如C调用B接口后,B调用A,A回调B,B回调C,所以需要B在调用接口前需要提供回调地址给B。我们做法是,公司信息注册在B平台时需要提供回调地址存储在数据库,开票成功后回调C使用。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
禁止转载,如需转载请通过简信或评论联系作者。
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 160,227评论 4 364
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 67,755评论 1 298
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 109,899评论 0 244
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 44,257评论 0 213
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 52,617评论 3 288
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 40,757评论 1 221
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 31,982评论 2 315
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 30,715评论 0 204
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 34,454评论 1 246
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 30,666评论 2 249
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 32,148评论 1 261
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 28,512评论 3 258
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 33,156评论 3 238
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 26,112评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 26,896评论 0 198
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 35,809评论 2 279
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 35,691评论 2 272

推荐阅读更多精彩内容