一、提示词

问题描述

在使用了@badeball/cypress-cucumber-preprocessor插件后,存在不支持nodejs原生的promise和async/await语法问题

执行用例命令

npx cypress run --env configFile=mhesi-staging,TAGS="@API005" --spec "cypress/integration/AL/API/*.feature"

报错信息

Error: Cucumber preprocessor detected that you returned a native promise from a function handler, this is not supported. Using async / await is generally speaking not supported when using Cypress, period, preprocessor or not.
      at runStepWithLogGroup (https://thai-portal.staging.mhesi-skill.org/__cypress/tests?p=cypress\integration\AL\API\api.feature:17265:17)
      at Context.eval (https://thai-portal.staging.mhesi-skill.org/__cypress/tests?p=cypress\integration\AL\API\api.feature:17751:58)

约束条件 

1.在Cypress 10.0.0+框架下的自动化测试项目中,使用了大量的async/await语法,修复工作量比较大,需要维持这种使用

2.cypress-cucumber-preprocessor插件不支持Cypress 10.0.0+版本,因此替换成@badeball/cypress-cucumber-preprocessor插件,请不要建议我替换成cypress-cucumber-preprocessor插件

3.在./cypress/api和./cypress/integration/_utils目录下的js文件,需要兼容在nodejs和browser环境中运行,因此涉及fs,path和crypto等模块的调用,代码需要考虑兼容两个环境,最好在统一的文件中封装处理好后,便于项目全局可调用

任务

提供一个解决方案,在Cypress 10.0.0+版本中,能够支持async/await的原生promise语法,使用@badeball/cypress-cucumber-preprocessor插件的解决方案,或者其它cucumber插件的结局方案均可

方案验证方法

命令npx cypress run --env configFile=mhesi-staging,TAGS="@API005" --spec "cypress/integration/AL/API/*.feature"能够执行成功

二、AI给的解决方案

1.编写一个wrapAsync函数,兼容cucumber preprocessor

cypress/support/async-wrapper.js文件

/*** Utility to wrap async functions for Cypress cucumber compatibility* Converts async/await to Cypress chainable commands*//*** Wraps an async function to make it compatible with Cypress cucumber preprocessor* @param {Function} asyncFn - The async function to wrap* @returns {Function} - A function that returns a Cypress chainable*/
export function wrapAsync(asyncFn) {return function(...args) {return cy.wrap(null).then(() => {return asyncFn.apply(this, args);});};
}/*** Wraps an async function and handles any errors appropriately* @param {Function} asyncFn - The async function to wrap* @returns {Function} - A function that returns a Cypress chainable with error handling*/
export function wrapAsyncWithErrorHandling(asyncFn) {return function(...args) {return cy.wrap(null).then(async () => {try {const result = await asyncFn.apply(this, args);return result;} catch (error) {cy.log(`Error in async function: ${error.message}`);throw error;}});};
}/*** Helper to execute multiple async operations in sequence* @param {Array} asyncOperations - Array of async functions to execute* @returns {Function} - A function that returns a Cypress chainable*/
export function executeAsyncSequence(asyncOperations) {return cy.wrap(null).then(async () => {const results = [];for (const operation of asyncOperations) {const result = await operation();results.push(result);}return results;});
}/*** Helper to execute async operation with retry logic* @param {Function} asyncFn - The async function to retry* @param {number} maxRetries - Maximum number of retries* @param {number} delay - Delay between retries in ms* @returns {Function} - A function that returns a Cypress chainable*/
export function wrapAsyncWithRetry(asyncFn, maxRetries = 3, delay = 1000) {return function(...args) {return cy.wrap(null).then(async () => {let lastError;for (let i = 0; i <= maxRetries; i++) {try {return await asyncFn.apply(this, args);} catch (error) {lastError = error;if (i < maxRetries) {cy.log(`Retry ${i + 1}/${maxRetries} after error: ${error.message}`);await new Promise(resolve => setTimeout(resolve, delay));}}}throw lastError;});};
}

2.提供一个跨环境的适配层,以满足nodejs和browser两个运行环境

cypress/support/cross-env-utils.js

/*** Cross-environment compatibility layer for Node.js modules* Works in both Node.js and browser environments*/let fs, path, crypto;// Detect environment and import appropriate modules
if (typeof window === 'undefined') {// Node.js environmenttry {fs = require('fs-extra');path = require('path');crypto = require('crypto');} catch (e) {// Fallback if modules are not availableconsole.warn('Node.js modules not available:', e.message);}
} else {// Browser environment - use polyfills or mocksfs = require('../mocks/fs-mock');path = require('path-browserify');crypto = require('crypto-browserify');
}/*** File system operations compatible with both environments*/
export const FileSystem = {readFileSync: (filePath, encoding = 'utf8') => {if (typeof window === 'undefined' && fs) {return fs.readFileSync(filePath, encoding);}// Browser fallback - could use cy.readFile or fetchthrow new Error('File system operations not supported in browser environment');},writeFileSync: (filePath, data, encoding = 'utf8') => {if (typeof window === 'undefined' && fs) {return fs.writeFileSync(filePath, data, encoding);}throw new Error('File system operations not supported in browser environment');},existsSync: (filePath) => {if (typeof window === 'undefined' && fs) {return fs.existsSync(filePath);}return false;},readJsonSync: (filePath) => {if (typeof window === 'undefined' && fs && fs.readJSONSync) {return fs.readJSONSync(filePath);}// Fallback for browserconst content = FileSystem.readFileSync(filePath);return JSON.parse(content);}
};/*** Path operations compatible with both environments*/
export const PathUtils = {join: (...paths) => {if (path && path.join) {return path.join(...paths);}// Simple fallback for browserreturn paths.join('/').replace(/\/+/g, '/');},resolve: (...paths) => {if (path && path.resolve) {return path.resolve(...paths);}// Simple fallback for browserreturn PathUtils.join(...paths);},dirname: (filePath) => {if (path && path.dirname) {return path.dirname(filePath);}// Simple fallbackreturn filePath.split('/').slice(0, -1).join('/') || '/';},basename: (filePath, ext) => {if (path && path.basename) {return path.basename(filePath, ext);}// Simple fallbackconst name = filePath.split('/').pop() || '';return ext && name.endsWith(ext) ? name.slice(0, -ext.length) : name;}
};/*** Crypto operations compatible with both environments*/
export const CryptoUtils = {createHash: (algorithm) => {if (crypto && crypto.createHash) {return crypto.createHash(algorithm);}throw new Error('Crypto operations not available in this environment');},randomBytes: (size) => {if (crypto && crypto.randomBytes) {return crypto.randomBytes(size);}// Browser fallback using Web Crypto APIif (typeof window !== 'undefined' && window.crypto && window.crypto.getRandomValues) {const buffer = new Uint8Array(size);window.crypto.getRandomValues(buffer);return buffer;}throw new Error('Random bytes generation not available in this environment');},createUuid: () => {if (crypto && crypto.randomUUID) {return crypto.randomUUID();}// Fallback UUID v4 generatorreturn 'xxxxxxxx-xxxx-4xxx-yxxx-xxxxxxxxxxxx'.replace(/[xy]/g, function(c) {const r = Math.random() * 16 | 0;const v = c === 'x' ? r : (r & 0x3 | 0x8);return v.toString(16);});}
};/*** Environment detection utilities*/
export const Environment = {isNode: () => typeof window === 'undefined',isBrowser: () => typeof window !== 'undefined',isCypress: () => typeof window !== 'undefined' && typeof window.Cypress !== 'undefined'
};/*** Async operation wrapper for cross-environment compatibility*/
export const AsyncUtils = {delay: (ms) => new Promise(resolve => setTimeout(resolve, ms)),retryAsync: async (fn, maxRetries = 3, delay = 1000) => {let lastError;for (let i = 0; i <= maxRetries; i++) {try {return await fn();} catch (error) {lastError = error;if (i < maxRetries) {await AsyncUtils.delay(delay);}}}throw lastError;}
};// Export everything as default for easy importing
export default {FileSystem,PathUtils,CryptoUtils,Environment,AsyncUtils
};

.cypress/mocks/fs-mock.js

/*** Mock fs module for browser environment*/const fsMock = {readFileSync: (filePath, encoding = 'utf8') => {throw new Error('File system operations not supported in browser environment. Use cy.readFile() instead.');},writeFileSync: (filePath, data, encoding = 'utf8') => {throw new Error('File system operations not supported in browser environment.');},existsSync: (filePath) => {return false;},readJSONSync: (filePath) => {throw new Error('File system operations not supported in browser environment. Use cy.fixture() instead.');},ensureDirSync: (dirPath) => {// No-op in browser},copySync: (src, dest) => {throw new Error('File system operations not supported in browser environment.');}
};module.exports = fsMock;

3.更新Cucumber测试步骤

原步骤

//添加培训到频道
When(/^user blinds Train item "([^"]*)" in Library "([^"]*)" to Channel "([^"]*)"$/, (trainName, libraryName, channelName) => {libraryName = cy.data(libraryName);trainName = cy.data(trainName);channelName = cy.data(channelName);cy.wrap(Library.queryLibrary(libraryName)).then((libraryId) => {cy.wrap(Channel.queryChannel(channelName)).then((channelId) => {cy.wrap(Channel.addTrain(channelId, "Recommended by window", libraryId, trainName)).then(() => {cy.wrap(Channel.saveChannel(channelId));});});});
});

修改后的步骤(代码可读性和可维护性提升)

import { wrapAsync, wrapAsyncWithErrorHandling } from '../../../support/async-wrapper';//添加培训到频道
When(/^user blinds Train item "([^"]*)" in Library "([^"]*)" to Channel "([^"]*)"$/, wrapAsync(async (trainName, libraryName, channelName) => {libraryName = cy.data(libraryName);trainName = cy.data(trainName);channelName = cy.data(channelName);const libraryId = await Library.queryLibrary(libraryName);const channelId = await Channel.queryChannel(channelName);await Channel.addTrain(channelId, "Recommended by window", libraryId, trainName);await Channel.saveChannel(channelId);
}));

 4.更新Cucumber插件配置

移除package.json中的cucumber插件配置部分

  "cypress-cucumber-preprocessor": {"nonGlobalStepDefinitions": true,"stepDefinitions": ["cypress/integration/**/*.js","cypress/integration/**/**/*.js"]},

根目录下创建.cypress-cucumber-preprocessorrc文件

{"stepDefinitions": "cypress/integration/**/*.js","filterSpecs": true,"omitFiltered": true,"nonGlobalStepDefinitions": false
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如若转载,请注明出处:http://www.pswp.cn/web/86964.shtml
繁体地址,请注明出处:http://hk.pswp.cn/web/86964.shtml
英文地址,请注明出处:http://en.pswp.cn/web/86964.shtml

如若内容造成侵权/违法违规/事实不符,请联系英文站点网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++多线程【Linux】

Linux的多线程 Linux的子线程实际上也是个进程&#xff0c;但是比传统的进程轻量化。 pthread pthread是用于Linux系统下的线程库&#xff0c;头文件是<pthread.h>。C11 之前的多线程开发高度依赖平台原生 API&#xff0c;Windows 以 CreateThread 和内核对象为核心&am…

Windows 环境下 NVM 命令详解:多版本 Node.js 管理利器

“一个 Node.js 版本走天下&#xff1f;太局限了&#xff01;试试 nvm&#xff0c;版本切换如丝般顺滑。” 什么是 NVM NVM&#xff08;Node Version Manager&#xff09;是一个命令行工具&#xff0c;允许你安装并在多个 Node.js 版本之间自由切换。 在 Linux/macOS 下常用的…

一二级路由之间的传参方式以及高亮问题

实现如下图所示的一二级路由的高亮情况&#xff1a; 在一级路由APP.vue下设置&#xff1a; .head a.router-link-active {background-color: rgb(235, 221, 204); }在二级路由Mycenter.vue下设置&#xff1a; /* 要求在点击跳转到mycenter_lianxi页面时候父路由保持高亮…

前端JavaScript力扣HOT100刷题【51-100】

注&#xff1a;纯手打&#xff0c;如有错误欢迎评论区交流&#xff01; 转载请注明出处&#xff1a;https://blog.csdn.net/testleaf/article/details/148953015 编写此文是为了更好地学习前端知识&#xff0c;如果损害了有关人的利益&#xff0c;请联系删除&#xff01; 本文章…

智能制造数字孪生集成交付生态链:智慧产线极速克隆,孪生重构生产周期

在智能制造的浪潮中&#xff0c;数字孪生技术正以前所未有的速度重塑制造业的生产模式。从产品设计到生产制造&#xff0c;再到运维管理&#xff0c;数字孪生通过构建物理世界的虚拟镜像&#xff0c;实现了生产全流程的数字化映射与优化。 山东融谷信息以“智能制造数字孪生集成…

非常详细版: dd.device.geolocation 钉钉微应用获取定位,移动端 PC端都操作,Vue实现钉钉微应用获取精准定位并渲染在地图组件上

dd.device.geolocation 钉钉微应用获取定位,钉钉微应用获取精准定位并渲染在地图组件上 ,手机端 PC端要都可用 【dd.device.geolocation是需要鉴权的哦】 想要的数据和效果图 想要的数据格式 代码 <template><div class="dialogStyles"

鸿蒙5:组件状态共享

目录 1. 组件状态共享 1.1 状态共享-父子传值&#xff1a;Local、Param、Event 1.2 状态共享-父子双向绑定!! 1.3 跨代共享&#xff1a;Provider和Consumer 1.3.1 aliasName和属性名 1.3.2 实现跨代共享 1.3.3 装饰复杂类型&#xff0c;配合Trace一起使用 1.3.4 支持共…

【MySQL】12. C语言与数据库的连接

1. 下载MySQL的连接库 sudo apt install -y libmysqlclient-dev 2. MySQL连接库的常用接口介绍 通过下面的样例了解MYSQL的常用接口&#xff1a; #include <iostream> #include <mysql/mysql.h> using namespace std;const char *host "localhost";…

[springboot系列] 探秘JUnit 5: Java单元测试利器

介绍 JUnit 5 是一个用于 Java 编程语言的单元测试框架&#xff0c;它是 JUnit 框架的第五个版本&#xff0c;与 JUnit 4 相比&#xff0c;JUnit 5 提供了许多改进和新特性&#xff0c;包括更好的扩展性、灵活性和对现代 Java 特性的支持。 JUnit 5 由三个主要的子模块组成&a…

开源 java android app 开发(十三)绘图定义控件、摇杆控件的制作

文章的目的为了记录使用java 进行android app 开发学习的经历。本职为嵌入式软件开发&#xff0c;公司安排开发app&#xff0c;临时学习&#xff0c;完成app的开发。开发流程和要点有些记忆模糊&#xff0c;赶紧记录&#xff0c;防止忘记。 相关链接&#xff1a; 开源 java an…

Python 库 包 sentence-transformers

sentence-transformers 是一个非常流行的 Python 库&#xff0c;专门用于将文本&#xff08;句子、段落、文档&#xff09;转换为高质量的语义向量&#xff08;嵌入&#xff09;。它基于 Transformer 架构&#xff08;如 BERT、RoBERTa、DistilBERT 等&#xff09; 的预训练模型…

《聚类算法》入门--大白话篇:像整理房间一样给数据分类

一、什么是聚类算法&#xff1f; 想象一下你的衣柜里堆满了衣服&#xff0c;但你不想一件件整理。聚类算法就像一个聪明的助手&#xff0c;它能自动帮你把衣服分成几堆&#xff1a;T恤放一堆、裤子放一堆、外套放一堆。它通过观察衣服的颜色、大小、款式这些特征&#xff0c;把…

AutoGen(五) Human-in-the-Loop(人类在环)实战与进阶:多智能体协作与Web交互全流程(附代码)

AutoGen Human-in-the-Loop&#xff08;人类在环&#xff09;实战与进阶&#xff1a;多智能体协作与Web交互全流程&#xff08;附代码&#xff09; 引言&#xff1a;AI自动化的极限与人类参与的价值 在大模型&#xff08;LLM&#xff09;驱动的AI应用开发中&#xff0c;完全自…

并查集 Union-Find

目录 引言 简单介绍 浅浅总结 算法图解 初始化 根节点查找 集合合并 连通性检查 例题 大概思路 完整代码&#xff1a; 引言 一个小小的并查集让我们在ccpc卡了那么久(还有unordered_map,如果不是忘了map自动排序这么一回事也不至于试那么多发)&#xff0c;至今仍然心有…

书籍在行列都排好序的矩阵中找数(8)0626

题目&#xff1a; 给定一个有N*M的整型矩阵matrix和一个整数K&#xff0c;matrix的每一行和每一列都是排好序的。实现一个函数&#xff0c;判断K是否在matrix中。 0 1 2 5 2 3 4 7 4 4 4 8 5 …

深度学习04 卷积神经网络CNN

卷积神经网络与人工神经网络关系与区别 概念 卷积神经网络&#xff08;Convolutional Neural Network, CNN&#xff09;是人工神经网络&#xff08;Artificial Neural Network, ANN&#xff09;的一种特殊形式&#xff0c;两者在核心思想和基础结构上存在关联&#xff0c;但在…

vue基础之组件通信(VUE3)

文章目录 前言一、父子组件通信1.父组件向子组件通信2.子组件向父组件通信3.ref父组件直接操作子组件通信。 二、跨代通信1. 跨层级通信2.事件总线通信 总结 前言 vue3的组件通信和vue2相比在语法上会有些差距&#xff0c;且vue3有的通信方式也在功能上比vue2更加完善&#xf…

【RidgeUI AI+系列】中文重复统计器

中文重复统计器 文字重复统计是一个使用文本处理工具&#xff0c; 输入文本内容并指定最小词长度后&#xff0c; 就能自动高亮显示重复的词。 本教程将会借助AI实现这个应用的开发 页面脚本编写 该工具的基础流程较为清晰&#xff1a;用户输入一段文字后&#xff0c;调用提取…

代码随想录|图论|05岛屿数量(深搜DFS)

leetcode:99. 岛屿数量 题目 题目描述&#xff1a; 给定一个由 1&#xff08;陆地&#xff09;和 0&#xff08;水&#xff09;组成的矩阵&#xff0c;你需要计算岛屿的数量。岛屿由水平方向或垂直方向上相邻的陆地连接而成&#xff0c;并且四周都是水域。你可以假设矩阵外均…

数据结构-第二节-堆栈与队列

一、概念&#xff1a; 堆栈与队列也是线性表&#xff0c;但是&#xff1a; 堆栈&#xff1a;只能在一个端进行插入删除&#xff0c;此端称为栈顶。&#xff08;特点&#xff1a;后来居上&#xff09; 队列&#xff1a;在一端进行插入&#xff08;队尾&#xff09;&#xff0…