基于vite3的monorepo前端工程搭建

2023-06-08 0 776

原副标题:如前所述vite3的monorepo后端工程建设构筑

译者 | 天猫云合作开发人员-天猫信息控制技术 牛志伟

一、控制技术栈优先选择1. 标识符库管理组织工作形式 – Monorepo:将数个工程项目存放在同一个标识符复本

基于vite3的monorepo前端工程搭建

▪优先选择理据 1:数个应用领域(能按销售业务线商品发射率分割)在同一 repo 管理组织工作,易于标准化管理组织工作标识符规范化、共享资源组织工作流

▪优先选择理据 2:化解跨工程项目 / 应用领域间力学微观的标识符F83E43Se,不必透过正式发布 / 安装 npm 包化解共享资源难题

2. 倚赖管理组织工作 – PNPM:消解倚赖提高、规范化流形内部结构

▪优先选择理据 1:透过软 / 硬镜像形式,最小某种程度节约系统资源

▪优先选择理据 2:化解幻影倚赖难题,管理组织工作更明晰

3. 构筑辅助工具 – Vite:如前所述 ESM 和 Rollup 的构筑辅助工具

▪优先选择理据:省却邻近地区合作开发时的校对操作过程,提高邻近地区合作开发组织工作效率

4. 后端框架 – Vue3:Composition API

▪优先选择理据:除了组件F83E43Se之外,还能F83E43Se一些共同的逻辑状态,比如请求接口 loading 与结果的逻辑

5. 模拟接口返回数据 – Mockjs

▪优先选择理据:前后端标准化了数据内部结构后,即可分离合作开发,降低后端合作开发倚赖,缩短合作开发周期

二、目1. 常规 / 简单模式:根据文件功能类型集中管理组织工作“`

mesh-fe

├── .husky #git提交标识符触发

│ ├── commit-msg

│ └── pre- commit

├── mesh- server#倚赖的node服务

│ ├── mock

│ │ └── data-service #mock接口返回结果

│ └── package.json

├── README.md

├── package.json

├── pnpm-workspace.yaml #PNPM组织工作空间

├── .eslintignore #排除eslint检查

├── .eslintrc.js #eslint配置

├── .gitignore

├── .stylelintignore #排除stylelint检查

├── stylelint.config.js #style样式规范化

├── commitlint.config.js#git提交信息规范化

├── prettier.config.js #格式化配置

├── index.html #入口页面

└── mesh- client#不同的web应用领域package

├── vite-vue3

├── src

├── api #api调用接口层

├── assets #静态资源相关

├── components #公共组件

├── config #公共配置,如字典/枚举等

├── hooks #逻辑F83E43Se

├── layout #router中使用的父布局组件

├── router #路由配置

├── stores #pinia全局状态管理组织工作

├── types #ts类型声明

├── utils

│ ├── index.ts

│ └── request.js #Axios接口请求封装

├── views #主要页面

├── main.ts #js入口

└── App.vue

“`2. 如前所述 domain 领域模式:根据销售业务模块集中管理组织工作“`

mesh-fe

├── .husky#git提交标识符触发

│ ├── commit-msg

│ └── pre-commit

├── mesh-server#倚赖的 node服务

│ ├── mock

│ │ └──data-service#mock接口返回结果

│ └── package.json

├── README.md

├── package.json

├── pnpm-workspace.yaml#PNPM组织工作空间

├── .eslintignore#排除 eslint检查

├── .eslintrc.js#eslint配置

├── .gitignore

├── .stylelintignore#排除 stylelint检查

├──stylelint.config.js#style样式规范化

├── commitlint.config.js#git提交信息规范化

├── prettier.config.js#格式化配置

├── index.html#入口页面

└── mesh-client#不同的 web应用领域 package

├── vite-vue3

├── src#按销售业务领域分割

├── assets#静态资源相关

├── components#公共组件

├──domain#领域

│ ├── config.ts

│ ├── service.ts

│ ├── store.ts

│ ├── type.ts

├── hooks#逻辑F83E43Se

├── layout#router中使用的父布局组件

├── router#路由配置

├── utils

│ ├── index.ts

│ └── request.js#Axios接口请求封装

├── views#主要页面

├── main.ts#js入口

└── App.vue

“`

能根据具体销售业务场景,优先选择以上 2 种形式其中之一。

三、构筑部分细节1.Monorepo+PNPM 集中管理组织工作数个应用领域(workspace)

▪根目录创建 pnpm-workspace.yaml,mesh-client 文件夹下每个应用领域都是一个 package,间能相互添加邻近地区倚赖:pnpm install <name>

packages:

# all packages in direct subdirs of packages/

mesh-client/*

# exclude packages that are inside test directories

!**/test/**

pnpm install #安装所有package中的倚赖

pnpm install -w axios #将axios库安装到根目录

pnpm –filter | -F <name> <command> #执行某个package下的命令

▪与 NPM 安装的一些区别:

▪所有倚赖都会安装到根目录 node_modules/.pnpm 下;

▪package 中 packages.json 中下不会显示幻影倚赖(比如 tslib@types/webpack-dev),需要显式安装,否则报错

▪安装的包首先会从当前 workspace 中查找,如果有存在则 node_modules 创建软连接指向邻近地区 workspace

▪”mock”: “workspace:^1.0.0”

2.Vue3 请求接口相关封装

▪request.ts 封装:主要是对接口请求和返回做拦截处理,重写 get/post 方法支持泛型

importaxios, { AxiosError } fromaxios

importtype{ AxiosRequestConfig, AxiosResponse } fromaxios

// 创建 axios 实例

constservice = axios.create({

baseURL: import.meta.env.VITE_APP_BASE_URL,

timeout: 1000* 60* 5, // 请求超时时间

headers: { Content-Type: application/json;charset=UTF-8},

})

consttoLogin = ( sso: string) => {

constcur =window.location.href

consturl = ` ${sso}${ encodeURIComponent(cur)} `

window.location.href = url

}

// 服务器状态码错误处理

consthandleError = ( error: AxiosError) => {

if(error.response) {

switch(error.response.status) {

case401:

// todo

toLogin( import.meta.env.VITE_APP_SSO)

break

// case 404:

// router.push(/404)

// break

// case 500:

// router.push(/500)

// break

default:

break

}

}

returnPromise.reject(error)

}

// request interceptor

service.interceptors.request.use(( config) => {

consttoken =

if(token) {

config.headers![ Access-Token] = token// 让每个请求携带自定义 token 请根据实际情况自行修改

}

returnconfig

}, handleError)

// response interceptor

service.interceptors.response.use(( response: AxiosResponse<ResponseData>) => {

const{ code } = response.data

if(code ===10000) {

toLogin( import.meta.env.VITE_APP_SSO)

} elseif(code !== 00000) {

// 抛出错误信息,页面处理

returnPromise.reject(response.data)

}

// 返回正确数据

returnPromise.resolve(response)

// return response

}, handleError)

// 后端返回数据内部结构泛型,根据实际工程项目调整

interfaceResponseData<T = unknown> {

code: string

message: string

result: T

}

exportconsthttpGet =async<T, D = any> ( url: string, config?: AxiosRequestConfig<D> ) => {

returnservice.get<ResponseData<T>>( url, config). then( ( res) => res.data )

}

exportconsthttpPost= async< T, D= any>(

url: string,

data?: D,

config?: AxiosRequestConfig<D>,

) => {

returnservice.post<ResponseData<T>> ( url, data, config). then( ( res) => res.data)

}

export{ serviceasaxios}

exporttype{ ResponseData}

▪useRequest.ts 封装:如前所述 vue3 Composition API,将请求参数、状态以及结果等逻辑封装F83E43Se

import{ ref } fromvue

importtype{ Ref } fromvue

import{ ElMessage } fromelement-plus

importtype{ ResponseData } from@/utils/request

exportconstuseRequest = <T, P = any>(

api: ( …args: P[]) =>Promise<ResponseData<T>>,

defaultParams?: P,

) => {

constparams = ref<P> asRef<P>

if(defaultParams) {

params.value = {

…defaultParams,

}

}

constloading = ref( false)

constresult = ref<T>

constfetchResource = async(…args: P[]) => {

loading.value = true

returnapi(…args)

.then( ( res) => {

if(!res?.result) return

result.value = res.result

})

.catch( ( err) => {

result.value = undefined

ElMessage({

message: typeoferr === string? err : err?.message || error,

type: error,

offset: 80,

})

})

.finally( => {

loading.value = false

})

}

return{

params,

loading,

result,

fetchResource,

}

}

▪API 接口层

import{ httpGet } from@/utils/request

constAPI = {

getLoginUserInfo:/userInfo/getLoginUserInfo,

}

typeUserInfo = {

userName: string

realName: string

}

exportconstgetLoginUserInfoAPI ==> httpGet<UserInfo>(API.getLoginUserInfo)

▪页面使用:接口返回结果 userInfo,能自动推断出 UserInfo 类型,

//方式一:推荐

const{

loading,

result: userInfo,

fetchResource: getLoginUserInfo,

}= useRequest(getLoginUserInfoAPI)

//形式二:不推荐,每次使用接口时都需要重复定义type

typeUserInfo = {

userName: string

realName: string

}

const{

loading,

result: userInfo,

fetchResource: getLoginUserInfo,

}= useRequest<UserInfo>(getLoginUserInfoAPI)

onMounted(async=> {

awaitgetLoginUserInfo

if(!userInfo.value) return

constuser = useUserStore

user.$patch({

userName: userInfo.value.userName,

realName: userInfo.value.realName,

})

})

3.Mockjs 模拟后端接口返回数据

importMock frommockjs

constBASE_URL =/api

Mock.mock( ` ${BASE_URL}/user/list` , {

code: 00000,

message: 成功,

result|10-20: [

{

uuid: @guid,

name: @name,

tag: @title,

age: @integer(18, 35),

modifiedTime: @datetime,

status: @cword(“01”),

},

],

})

四、标准化规范化

1.ESLint

注意:不同框架下,所需要的 preset 或 plugin 不同,建议将公共部分提取并配置在根目录中,package 中的 eslint 配置设置 extends。

/* eslint-env node */

require( @rushstack/eslint-patch/modern-module-resolution)

module.exports = {

root: true,

extends: [

plugin:vue/vue3-essential,

eslint:recommended,

@vue/eslint-config-type,

@vue/eslint-config-prettier,

],

overrides: [

{

files: [ cypress/e2e/**.{cy,spec}.{js,ts,jsx,tsx}],

extends: [ plugin:cypress/recommended],

},

],

parserOptions: {

ecmaVersion: latest,

},

rules: {

vue/no-deprecated-slot-attribute: off,

},

}

2.StyleLint

module. exports= {

extends: [ stylelint-config-standard, stylelint-config-prettier],

plugins: [ stylelint-order],

customSyntax: postcss-html,

rules: {

indentation: 2, //4空格

selector-class-pattern:

^(?:(?:o|c|u|t|s|is|has|_|js|qa)-)?[a-zA-Z0-9]+(?:-[a-zA-Z0-9]+)*(?:__[a-zA-Z0-9]+(?:-[a-zA-Z0-9]+)*)?(?:–[a-zA-Z0-9]+(?:-[a-zA-Z0-9]+)*)?(?:\[.+\])?$,

// at-rule-no-unknown: 屏蔽一些scss等语法检查

at-rule-no-unknown: [ true, { ignoreAtRules: [mixin, extend, content, export] }],

// css-next :global

selector-pseudo-class-no-unknown: [

true,

{

ignorePseudoClasses: [ global, deep],

},

],

order/order: [ custom-properties, declarations],

order/properties-alphabetical-order: true,

},

}

3.Prettier

module.exports={

printWidth:100,

singleQuote:true,

trailingComma:all,

bracketSpacing:true,

jsxBracketSameLine:false,

tabWidth:2,

semi:false,

}

4.CommitLint

module. exports= {

extends: [ @commitlint/config-conventional],

rules: {

type-enum: [

2,

always,

[ build, feat, fix, docs, style, refactor, test, chore, revert],

],

subject-full-stop: [ 0, never],

subject-case: [ 0, never],

},

}

五、附录:控制技术栈图谱

END

十年磨一剑,开源中国新使命

这里有最新开源资讯、软件更新、控制技术干货等内容

相关文章

发表评论
暂无评论
官方客服团队

为您解决烦忧 - 24小时在线 专业服务