CheckStyle in Pre-Commit
养成编码习惯的第一步. checkstyle + precommit
简单说,我们在每次git commit
时,都进行一次checkstyle.
个人而言,每当你敲下任何代码的时候,都会受到许多制约,就像开车时候,你需要受到交规制约一样,一个简单的转向灯,能够让周围的人明白你的意图
团队而言,所有不符合代码规范的代码,都完全无法进入到git 仓库,他们被挡在了个人电脑这个层级上.
guideline
- 配置checkstyle 运行方式
- 配置git hooks
- 实现全团队checkstyle
- 配置checkstyle 规则
配置 checkstyle
选择自己希望运行的方式, checkstyle 支持命令行, gradle, maven, 或者是java -jar 的方式运行.
文章推荐使用gradle的方式, 不为什么, 就是因为配置简单.
关于gradle 配置的文档可以参考 checkstyle in gradle
1
2
3
plugins {
id 'checkstyle'
}
默认情况, checkstyle plugin 会从特定路径下寻找配置文件, 默认路径如下:
1
2
3
4
5
<root>
└── config
└── checkstyle
└── checkstyle.xml
└── suppressions.xml
- Checkstyle configuration files -> checkstyle.xml
- Primary Checkstyle configuration file -> suppressions.xml
他们的规则我还没想好呢, 待会我们会提到, 我们先用一个简单的吧~
1
2
3
4
5
6
7
8
9
10
<?xml version="1.0"?>
<!DOCTYPE module PUBLIC "-//Puppy Crawl//DTD Check Configuration 1.3//EN" "http://checkstyle.sourceforge.net/dtds/configuration_1_3.dtd">
<module name="Checker">
<property name="charset" value="UTF-8" />
<module name="LineLength">
<property name="max" value="120" />
<property name="ignorePattern" value="^package.*|^import.*|a href|href|http://|https://|ftp://" />
</module>
</property>
</module>
👆🏻️顾名思义, 这个文件是在检查编码是否属于UTF-8, 并且单行不应该超过120个字符~ 接下来, 我们就来到命令行里输入, 如果懒的自己搭, 可以用我现有的项目
1
2
git clone git@github.com:leweii/demo.git
./gradlew checkstyleMain
配置git hooks
git hooks 本质是一系列shell脚本, 他会在git 各个阶段自动调用, 我们要做的就是定义自己的pre-commit 脚本, 让checkstyle 成为里面的一个步骤.
最快的做法是把example mv 过来.
1
2
mv .git/hooks/pre-commit.sample .git/hooks/pre-commit
chmod +x .git/hooks/pre-commit
- 通过git status 拿到所有改动的文件
- 过滤出java 文件
- 对每个文件checkstyle
- 如果check 不通过, 则返回1, 通过返回0
这是我的pre-commit
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
#!/bin/sh
#
# An example hook script to verify what is about to be committed.
# Called by "git commit" with no arguments. The hook should
# exit with non-zero status after issuing an appropriate message if
# it wants to stop the commit.
#
# To enable this hook, rename this file to "pre-commit".
# 一个打印函数,前后加空行
function print(){
echo "===========$*============"
}
print 提高修养
wd=`pwd`
rm -f temp
is_err=0
re=`./gradlew checkstyleMain >> temp`
err=`cat temp | grep "ERROR"`
if [ $err = *"ERROR"* ];then
print $err
is_err=1
fi
print "检查完成"
rm -f temp
if [ $is_err -ne 0 ];
then
print "请先符合style才能提交!"
exit 1
fi
exit 0
单单这样配置, 还是不够哦~ 因为hooks的目录是不会被git 同步, 那么我们要怎样把自己的checkstyle 共享给全团队呢?👇🏻️
实现全团队checkstyle
通过core.hooksPath
来配置指定的hooks path.
simply->
1
2
git config core.hooksPath ./hooks
mv .git/hooks/pre-commit ./hooks/
这样团队里所有人都能通用一套checkstyle 辽~
配置checkstyle 规则
这个纯粹就是文档功夫, 请参阅 Checkers.
举例子说明吧, 我今天想检查无用的import 和不应该出现的空格.
依葫芦画瓢~, 在你的Checker 里加上.
1
2
3
4
5
6
7
8
9
10
<module name="TreeWalker">
<module name="UnusedImports"/>
<module name="WhitespaceAfter"/>
<module name="WhitespaceAround">
<property name="tokens"
value="ASSIGN,DIV_ASSIGN,PLUS_ASSIGN,MINUS_ASSIGN,STAR_ASSIGN,
MOD_ASSIGN,SR_ASSIGN,BSR_ASSIGN,SL_ASSIGN,BXOR_ASSIGN,
BOR_ASSIGN,BAND_ASSIGN"/>
</module>
</module>
效果如下:
最后, 一些问题总结
如果你看到这样的东东, 大概率是xml 文件定义错误了, 或者checkstyle 版本和xml 不对应.
要想debug 具体原因, 就可以打开 --stacktrace
1
gradle checkstyleMain --stacktrace
附录:
- download google checkstyle -> checkstyle.xml
- download sun checkstyle -> checkstyle.xml
- download -> suppressions.xml
因为以前cdci都是SED team小伙伴帮忙专门配置的, 自己胡乱按理论写了一通, 并没有实际验证过, 结果发现心里负担很重, 自己的博客还是需要负责任的验证再发布一遍, 之后推翻重写了一遍, 总算是心安了.