在目前的工作中,Bitbucket 同时被我们用作代码仓库和私有 NPM 仓库。通过 git+ssh
指定 tag
来安装依赖,对于纯 JavaScript 的项目十分方便。
随着部分依赖开始使用 TypeScript,基于现有使用方式,需要对这部分依赖采取下面的一种方式进行管理。
- 直接将编译后的
dist
文件提交到主分支上。 - 在
postinstall
中进行编译。当依赖被安装后,就会自动编译。
但无论哪种方式,都会有一些瑕疵。
- 直接将
dist
提交到主分支上,在做 Code review 时会额外多出许多文件干扰,项目结构也不优雅。 - 通过
postinstall
的方式对于纯 JavaScript 的项目就需要额外安装typescript
,同样不优雅。
出于人力以及迁移仓库的成本考虑(我们有近 100 个依赖,且每个依赖有许多版本并且被不同的服务引用),对此只能通过改进 Git flow 来解决上述瑕疵。
目标
重新设计的 Git flow 需要满足以下目标。
- 开发分支上不需要编译后
dist
文件。 - 不需要通过
postinstall
的方式来编译。 - 引用的方式不发生改变,还是通过
git+ssh
指定tag
的方式安装依赖。 - 需要考虑到
hotfix
的情况。
Git flow 改进
为了实现上述目标,下面是改进后的 Git flow。
对于正常的开发,当代码合并到 master
后,先推送 src
tag 并修改 .gitignore
临时创建 build
分支。随后在 build
分支编译并且打上正式 tag,正式 tag 包含 dist
文件,可以被其他服务引用。
当需要进行 hotfix
时,就可以从对应的 src
tag 上切出分支进行 fix,之后再以相同的步骤合入 master
分支。
Pipeline 脚本设置
上述的 Git flow 在代码合并之后有打 tag、修改 .gitignore
、编译再发布的步骤。即便有文档说明,在实际操作上也很容易出错。因此十分有必要将这些步骤自动化。利用 Bitbucket Pipeline 在代码合入 master
分支时自动执行。
下面是示例代码。
- Pipeline 部分
branches:
master: # 当合入主分支时才会触发
- step:
name: build
deployment: Build # 以 Deployment 的方式执行,名字可以随意指定
script:
- git remote set-url origin ${BITBUCKET_GIT_SSH_ORIGIN}
- ./scripts/build.sh # 实际执行的脚本
- 打 tag 以及编译脚本。
pipeline 中的 build.sh
,整个 Git flow 的核心。实现打 src tag、编译以及推送编译后的正式 tag。
tag 的版本遵循 semver 规则,当前 tag 为 package.json
中的 version
。下一个版本则来源于分支名的前缀,比如分支名为 major/xxxx
即下一个版本升级主版本号;如果是 feature/xxx
则为 minor
。利用 npm version
命令升级标签。
#!/usr/bin/env bash
set -e
# Create tags in the bitbucket pipline, env from bitbucket pipeline
source_branch=$(git log --format=%B -n 1 $BITBUCKET_COMMIT | awk '{print $3}')
if [[ -n $source_branch ]]; then
# PART 1: Get the next tag
echo "Triggered by pull request ${source_branch}"
if [[ ${source_branch} == feature/* ]]; then
level=minor
elif [[ ${source_branch} == bugfix/* ]]; then
level=patch
elif [[ ${source_branch} == hotfix/* ]]; then
level=prerelease
elif [[ ${source_branch} == major/* ]]; then
level=major
else
echo "Nothing happen on branch: ${BITBUCKET_BRANCH}"
echo "Source branch: ${source_branch}"
fi
if [[ ! -z ${level} ]]; then
# PART 2: Create src-tag and push back
current_version=$(node -e "console.log(require('./package.json').version)") && echo "Current version - ${current_version}"
echo "Crated version on branch: ${BITBUCKET_BRANCH}. Tag: ${current_version}"
next_version=$(npx semver ${current_version} -i ${level})
source_tag=v${next_version}-src
echo "Build source tag: ${source_tag} on branch: ${BITBUCKET_BRANCH}"
# Create the source tag and push back
npm version ${next_version} --no-git-tag-version # This will not create a git tag just update the version in the package.json
git add --all
git commit -m "[skip ci] ${current_version} --> ${next_version}" # [skip ci] Will not trigger this pipeline again
git tag -am "[skip ci] source tag: ${source_tag}" ${source_tag}
git push origin ${BITBUCKET_BRANCH} ${source_tag}
# PART 3: Add the dist folder and create release tag
# Add the dist folder
sed -i 's/dist//g' .gitignore
# build ts
npm run build
# push to tag
git add --all
git commit -m "[skip ci] Build release tag: ${next_version} on branch: ${BITBUCKET_BRANCH}"
git tag -am "[skip ci] release tag: ${next_version}" ${next_version} # only push the tag
git push origin refs/tags/${next_version}
fi
fi
Shell 脚本最好放在 Deployment 中执行。因为 Deployment 是阻塞队列,可以保证同一时间只有一个脚本在运行,从而避免 tag 重复冲突的问题。
同时对于开发分支的命名也需要做一定的限制。比如可以使用 husky 等工具。
拓展
这一套 Git flow 可以在任何的代码托管工具使用,即便没有类似 Pipeline/GitAction 的触发方式同样也可以通过跑脚本的方式实现。同时可以实现自动打 tag 的功能。
虽然没有完美的 Git flow,但一套合适的 Git flow 也可以让我们的工作更加规范,提高工作的效率。
本文由mdnice多平台发布