0
  • 聊天消息
  • 系统消息
  • 评论与回复
登录后你可以
  • 下载海量资料
  • 学习在线课程
  • 观看技术视频
  • 写文章/发帖/加入社区
会员中心
创作中心

完善资料让更多小伙伴认识你,还能领取20积分哦,立即完善>

3天内不再提示

浅析单测在商家前端业务中的实践

OSC开源社区 来源:得物技术 2023-01-12 10:28 次阅读

1

背景

商家系统是提供给得物商家在得物平台上可以稳定运营的服务抓手,前端代码也伴随着系统的发展而不断壮大。这样将导致文档却更新不及时,最后想再通过这些文档回溯业务逻辑也非常困难。

且若代码结构上没有关注,动辄就会产出一个大几千行的文件,人员交替维护的时候很难理清里面的逻辑,维护非常困难。

2

前端单测的难点

为解决上述痛点,早在单测之前,团队上已经做了一些其他事情来使文档更清晰、代码质量更高,如写需求系分文档、通过整洁架构(The clean architecture)对代码进行分层、code review等等。但这些其实都只是外在的约束,只有内在的代码能真正经得住单测的推敲,才能更好的保障我们的代码质量。

但目前现状是前端大部分情况下都没有接触到单测,仅在组件库或工具类的项目里有一些。这并不代表业务项目中前端就无法单测, 而是因为一些客观原因,导致前端在单测上的投入相对较少。

前端开发的内容比较杂,一个需求不仅仅是功能函数的编写,还有UI的展示、dom交互的绑定等等,且若想单测完全覆盖,将包含非常多的内容,对业务前端来说成本太高。

前端UI框架层出不穷,在业务开发的时候,依赖框架也很容易将代码逻辑和UI等完全耦合在一起,导致一个文件上千行,很难对这种代码找到单测的切入点。

单测上手本身就有一定的门槛,要写出可维护性高的单测更不简单,会让不熟悉的人望而却步。

3

单测即文档

鉴于上面的第一个难点,前端涉及的内容太杂,我们肯定无法给所有的代码覆盖单测,去测到代码的各个角落。再结合上我们自己本身的痛点(文档更新不及时,人员轮转成本高),因此以“单测即文档”为目标,我们只用覆盖业务逻辑上的单测即可,只关注业务流程的衔接,通过用例将业务流程讲清楚,对于单测的分支覆盖率也不做强硬的要求。

Use Cases

因此,要在团队落地单测的第一步即是识别出实现业务逻辑的代码模块。若在较早的时候,想找到这个切入点可能还真没有什么好的方法,因为全是几千行的大文件,且逻辑和UI都耦合在一起。 正如前面所说,在单测推行前,我们已经做了一些代码准备工作。得益于“整洁架构”的推行,在开发需求的同时,已逐渐在对代码进行解耦重构,其核心就是依据各部分代码作用的不同将其拆分成不同的层次,在各层次间制定了明确的依赖原则,达到与框架无关与外部服务无关并可测试的目的。

78a80d70-85ec-11ed-bfe3-dac502259ad0.png

经过分层后,我们将业务逻辑主要都落在了usecase这一层,在我们的代码结构上,它的作用是将业务流程串联起来,且它仅依赖entities(主要对服务端返回数据做适配和检查)层,逻辑独立不会因为依赖框架或UI的变化而无法运行。

相较于后端服务,前端应用通常并不会承载如计算、存储等实实在在的业务逻辑,同时由于现在微服务架构的流行,前端应用往往会承担很重的胶水逻辑,即将各个微服务的逻辑串联在一起,从而跑通业务流程。

因此,前端在编写usecase的时候,我们会更注重主子函数的拆分,让主usecase更纯粹的去描述业务流程,而将部分具体的实现拆分到子函数中去实现。

/*
    usecase聚焦流程的描述,诸如url链接拼接、活动期查询等具体逻辑都拆分到了其他的模块中
*/
async function exportActivityLog({count, formValues}: {count: number;formValues: LogData}) {
  if (count > 5000) {
    message.error('导出文件数量不得超过5000!')
    return
  }
  const res = await checkIsDuringTheEventApi()
  if (res.isDuring) {
    message.error('活动期间,功能暂不可用,如有疑问联系运营');
    return
  }
  const url = generateDownloadUrl({ formValues })
  downloadExcelFile(url)
}


function generateDownloadUrl() {
  // 省略
}
因此,对usecase层写单测,正是我们要找的最好切入点,其既能满足我们将业务文档进行补充,同时又能有单测模块的产出,保障我们的代码质量和程序的稳定性。

4

单测实践

在识别出要覆盖单测的代码模块之后,下一步自然就是落地单测用例。

前面已说过,写单测本身就有一定的门槛,但既然要写就应写可维护性和稳定性高的单测。否则代码稍微一重构,单测崩了;或代码真崩了的时候,单测却没又通过了。

根据前面的描述可以看出,我们对于用例的可读性(文档性)和稳定性有极高的诉求,对于用例所测试的逻辑范围要求不高,这个准则对于后续的单测用例的设计取舍会有很大的影响。

4.1 用例设计

首先我们需要确定设计用例的切入点,目前单测社区内比较流行的模式无非TDD和BDD两种: TDD:测试驱动开发,偏向于去测到函数的各个功能运行的结果是否符合预期,由于是通过先写用例去驱动业务逻辑的实现,因此用例的设计往往更偏技术实现。 BDD:行为驱动开发,流程上是TDD模式的一种分支,区别在于在构思用例的时候更多的是以用户行为(user story)的角度去考虑。

7928282a-85ec-11ed-bfe3-dac502259ad0.png

关于两者更多的区别,大家可以网上查阅到更多的资料,这里就不再赘述。为了我们单测的稳定可维护性,且以文档为导向的我们,自然是选用了BDD的模式,只测业务行为逻辑,不关注功能函数的输出正确与否(这块目前可在自测和测试兄弟团队那边帮忙保障)。这样除非业务流程发生变更,否则代码一般的重构或调整都不会影响到单测的运行,不会造成单测的雪崩。

4.2 用例结构

在用例结构上,为了配合“单测即文档”的初衷并更好的配合BDD,我们在社区常见的AAA(Arrange-Act-Assert)和GWT(Given-When-Then)两种结构之间选择了后者。

无论AAA还是GWT最终都会形成一个三段式的用例结构,其区别仍然在于AAA的构思更倾向于技术实现,GWT更倾向于业务流程。虽然结构一样,但设计出来的用例内容会有很大区别。

Given-When-Then

Given:一个上下文,指定和准备测试的预设

When:进行一系列操作,即所要执行的操作

Then:得到可观察的结果,即需要检测的断言

我们根据GWT的提供了单测的基本模板,供组内同学写单测时直接使用。

function init() {
  const checkIsDuringTheEventApi = jest.fn();
  const downloadExcelFile = jest.fn();
  const exportActivityLog = buildMakeExportActivityLog({checkIsDuringTheEventApi, downloadExcelFile})


  return {
    checkIsDuringTheEventApi,
    downloadExcelFile,
    exportActivityLog
  }
}


describe('spec', () => {
  it('test', () => {
    // Given  准备用例所需的上下文
    const { checkIsDuringTheEventApi, downloadExcelFile, exportActivityLog } = init();


    // When 调用待测的函数
    exportActivityLog()


    // Then  断言
    expect('expect')
  })
})
对于一些校验简单模型的用例,通过init函数做一层封装就够用了。但对于业务逻辑比较复杂,字段比较多的模型,直接利用原生数据进行初始化对用例的可读性并不友好。
describe('spec', () => {
  it('个人卖家未发货的订单,允许进行取消操作', () => {
    // Bad case: 依赖字段较多,这样手动去创造字段数据可读性并不友好
    // 若case较多,这些字段要手动构建多次
    action({
      status: Status.待发货,
      merchantType: MerchantType.个人卖家,
      // ...还有一些其他必传字段
    })
  })
}
对于这种复杂场景,我们倾向于使用builder模式来构造数据,在较小的开发成本下保障用例的可读性和可维护性。

describe('spec', () => {
    it('个人卖家未发货的订单,允许进行取消操作', () => {
        // Good case:通过builder实现逻辑的复用和信息的聚焦
        const order = new OrderBuilder()
          .status("待发货")
          .merchantType("个人卖家")
          .build()


        action(order)


    })
})

4.3 用例描述

既然是要作为文档使用,那用例描述上也显得至关重要了。相比TDD对功能函数的单测,我们描述完全于GWT的用例结构对应(When时常会被省略掉),我们并不关心具体的技术实现细节,更多的是描述的这个业务的行为流程,思考函数最终想做什么,达到什么目的。基于意图,把被测函数当做黑盒,不用关注其中间的实现细节,究竟生成了什么临时变量、循环了几次、有什么判断等,而是通过用例描述将业务流程讲清楚。

describe('导出活动日志', () => {
  it('导出时,先查询当前活动状态,若状态是未在进行中,则执行导出操作', () => {
    // 省略...
  })
  it('导出时,若导出数量大于5000条,将不允许导出', () => {
    // 省略...
  })
})

上面是导出活动日志的一个操作,可以看出,用例的描述不会像测功能函数那样精简(入参是a,调用了啥函数必须返回b之类),但是将导出活动时,相应的调用流程和条件描述了出来,这样其他人在接手这块业务时,通过这个用例就能清楚知道在导出活动日志时需求上有些什么限制以及要做的操作。

4.4 用例断言

在确定好用例的设计思路和结构之后,我们在用例的校验内容上也做了一些取舍。针对社区上主导的经典测试(Classical)和模拟测试(Mockist)两大阵营,结合“单测即文档“的理念,我们对于业务流程的验证诉求非常强烈,因此选择了后者。

Classical风格是尽可能的使用真实对象和函数,让函数以及依赖都真实的执行;相对的,Mockist是想尽办法去mock,主张将所调用的被测函数全部mock。存在即合理,两个派各有利弊,并不存在一定谁好谁差。

要对用到的函数进行mock,在保证用例可维护性的前提下(比如不mock文件路径),我们需要对函数的依赖关系进行整理。得益于团队整洁架构的落地,目前应用的usecase层都已经通过依赖倒置对依赖关系做了很好的管理(usecase只依赖entity)。

export default function buildMakeExportActivityLog({checkIsDuringTheEventApi,downloadExcelFile}) {
  async function exportActivityLog({count,formValues}) {
    if (count > 5000) {
      message.error('导出文件数量不得超过5000!')
      return
    }
    const res = await checkIsDuringTheEventApi()
    if (res.isDuring) {
      message.error('活动期间,功能暂不可用,如有疑问联系运营');
      return
    }
    const url = generateDownloadUrl({ formValues })
    downloadExcelFile(url)
  }
}


// index.ts
import {checkIsDuringTheEventApi} from '@/services/activity'
import {downloadExcelFile} from '@/utils'
import buildMakeExportActivityLog from './makeExportActivityLog'


export const exportActivityLog = buildMakeExportActivityLog({cancel,printSaleTicket})
可以看到checkIsDuringTheEventApi以及downloadExcelFile这两个函数最终作为参数传入到实际的函数中,他们一个将会去发起请求,一个是会调用window的方法进行下载,通过依赖倒置就能方便我们对其进行模拟,在单测时就不会去真实执行这两个函数。
function init() {
  const checkIsDuringTheEventApi = jest.fn();
  const downloadExcelFile = jest.fn();
  const exportActivityLog = buildMakeExportActivityLog({checkIsDuringTheEventApi, downloadExcelFile})
  return {
    checkIsDuringTheEventApi,
    downloadExcelFile,
    exportActivityLog
  }
}
usecase中时常会有依赖的函数要去发起请求,在单测时我们不会去真实去发起这个请求,因此对于这类函数,我们都应mock掉,这样可保障我们用例的速度和稳定性。当然实际在写单测中,我们也不应该成为一个完全的mockist,无休止的进行mock,更好的方式是两者结合,否则滥用mock反而会导致单测写起来会更繁琐(因为要去mock所有调用的函数实现或场景),而且真实代码写起来也会很别扭(所有外部函数都依赖倒置)。

一个用例正确与否,最终依赖的是最后的断言,那对我们来说该怎样进行断言呢,如前面一直强调的一样,我们测的是逻辑行为,因此需断言的是某个行为的是否执行或者是否达到了什么目的。结合前面的mock,我们可对函数的调用情况进行捕获,针对上面发起取消退款的函数,断言的例子如下:

describe('导出活动日志', () => {
  it('导出时,先查询当前活动状态,若状态是未在进行中,则执行导出操作', () => {
    // 省略...
    expect(downloadExcelFile).toBeCalled()
  })


  it('导出时,若导出数量大于5000条,将不允许导出', () => {
    // 省略...
    expect(downloadExcelFile).not.toBeCalled();
  })
})

如上,断言的内容不是函数的实现细节,如参数是否正确,而是只断言行为是否执行,它能尽量保证做到若代码重构后,单测用例在不修改的情况下依然能健壮的运行,其只依赖需求的变更而做更改。同时为了维护用例的稳定性,单个用例我们通常仅执行一次断言(单一职责),断言的内容严格和描述的“Then”部分对应。

5

结语

商家以“单测即文档”的理念为落地方向,在代码设计以及用例的构思、结构、断言、描述等环节都做了一定取舍,最终在用例的书写成本、稳定性、可读性等各个方面取得了相对较好的平衡。

目前组内各个项目已逐渐沉淀了几百个用例,团队内相互支援或自己回顾时,通过这些用例就能知道这块逻辑在做什么事,在修改这些需求时通过测试用例也能尽快知道基本的业务逻辑,有了单测的保障,改起代码来更有底气,代码结构上,也更加的合理。在大家逐渐熟悉单测后,后续更会慢慢做到功能函数、UI等的单测覆盖,大家一起来保障商家前端业务的稳定发展。





审核编辑:刘清

声明:本文内容及配图由入驻作者撰写或者入驻合作网站授权转载。文章观点仅代表作者本人,不代表电子发烧友网立场。文章及其配图仅供工程师学习之用,如有内容侵权或者其他违规问题,请联系本站处理。 举报投诉
  • 逻辑电路
    +关注

    关注

    13

    文章

    494

    浏览量

    42555
  • TDD
    TDD
    +关注

    关注

    1

    文章

    121

    浏览量

    38149
  • BDD
    BDD
    +关注

    关注

    0

    文章

    6

    浏览量

    7518

原文标题:单测在商家前端业务中的实践

文章出处:【微信号:OSC开源社区,微信公众号:OSC开源社区】欢迎添加关注!文章转载请注明出处。

收藏 人收藏

    评论

    相关推荐

    基于电信业务的程序设计课程改革探索与实践

    【摘要】:本文针对我国大学程序设计课程教学存在的问题,以通信工程专业为背景,提出将专业知识与程序设计课程相结合的改革思路,指出根据专业特点优化课程教案、丰富课堂案例的方法,介绍了电信业务硬件平台
    发表于 04-24 10:06

    GMTC 大前端时代前端监控的最佳实践

    系统。(这时候你就会想)要是能知道用户报了什么错就好了。别怕,打开阿里云前端监控的【访问明细】搜索用户ID,直接可以看到该用户访问过程,到底报了什么错。有时候拿到了用户报错时的基本信息,也知道用户
    发表于 07-04 16:12

    高线性度芯片射频前端解决方案

    终端设备(如手机和PMP)越来越讲究轻薄小巧的今天,芯片RF(射频)前端终端应用市场上的前景越来越受到欢迎。可是,与数字电路不同,
    发表于 06-25 07:26

    OpenHarmony例模式实践

    概念在软件世界里面,实例是一个非常重要的概念。比如一个国家只有一个主席/总统/...一支军队只有一个最高统帅一个班级只有一个班主任...OpenHarmony实践OpenHarmony是如何实现
    发表于 09-15 09:27

    国产射频前端芯片

    AT2402E 是一款应用于无线通信的集成收发功能的射频前端芯片,芯片 内部集成了所需要的射频电路模块,集成度非常高,主要包括功率放大器(PA), 低噪声放大器(LNA),收发模式切换的开关
    发表于 02-02 15:16

    业务规则管理电信网管系统的应用

    主要介绍业务规则及业务规则管理的概念,讨论了业务规则管理系统的基本原理,实践一个简单的业务规则管理系统并应用于电信网络管理系统
    发表于 06-19 11:57 17次下载

    Ku波段接收前端抗干扰方法浅析

    Ku波段接收前端抗干扰方法浅析,下来看看。
    发表于 07-29 19:05 6次下载

    高通宣布与谷歌、hTC、三星和索尼移动射频前端业务方面展开合作

    射频前端对终端用户对其设备的移动体验至关重要,并能够促进移动行业的未来发展。过去的一年,高通射频前端领域可谓动作频频,经过一系列的合作,如今终于迎来一波重要的合作伙伴。今天CES的
    的头像 发表于 01-10 16:22 4486次阅读

    web前端开发实践的目录推荐

    本文档的主要内容详细介绍的是web前端开发实践的目录推荐。
    发表于 01-31 08:00 0次下载

    浅析风速传感器气象监测的重要作用

    浅析风速传感器气象监测的重要作用
    发表于 11-03 17:25 12次下载

    浅析高压放大器绝缘电阻电痕腐蚀程度分析的应用

    浅析高压放大器绝缘电阻电痕腐蚀程度分析的应用
    发表于 01-14 10:00 5次下载

    浅析质构仪还原剂对熟化陈米品质影响研究的应用

    浅析质构仪还原剂对熟化陈米品质影响研究的应用
    发表于 01-18 09:20 4次下载

    浅析低功耗应用克服低IQ挑战

    浅析低功耗应用克服低IQ挑战
    发表于 02-10 09:56 2次下载

    加快部署 5G 基站的最佳实践:RF 前端大规模 MIMO 入门

    加快部署 5G 基站的最佳实践:RF 前端大规模 MIMO 入门
    的头像 发表于 12-26 10:16 1733次阅读
    加快部署 5G 基站的最佳<b class='flag-5'>实践</b>:RF <b class='flag-5'>前端</b>大规模 MIMO 入门

    前端大数据产品的应用背景和应用原理

      导读: 本文由梯度科技前端研发部高级开发工程师贺信撰写,主要介绍如何基于前沿开源的前端技术方案实现微前端大数据平台中的应用落地,并对所取得的应用效果进行剖析。主要包括以下几个方面
    的头像 发表于 08-14 15:18 1277次阅读
    微<b class='flag-5'>前端</b><b class='flag-5'>在</b>大数据产品<b class='flag-5'>中</b>的应用背景和应用原理