參考1:基于space command 從 git commit 自動化代碼風格 https://juejin.cn/post/6844903555866820622
參考2: git hook原理 https://c.isme.pub/2017/06/14/git-hooks/
背景
大多數 iOS 開發者應該都知道 Xcode 的插件 Clang Format,它是基于 clang-format 命令行工具的一個 Xcode 插件,但是這款插件在Xcode9上已經無法使用了,因為Xcode9的插件機制已經變了。
現在可以使用這一款XcodeClangFormat,具體的使用方式點擊鏈接,大家自行去看吧。這款有個缺點,就是不能像之前那款插件可以設置在保存時自動格式化(這其實也不能怪作者,Xcode新的機制不允許)。 不過使用這種插件還是不夠方便,你還得手動選中文件或者代碼再按快捷鍵來格式化,很容易忘,而導致把不規范的代碼直接提交到倉庫里了。
那么有沒有一種方式,可以讓我在敲代碼的時候隨心所欲,提交時又能提醒我然后自動幫我格式化嗎?
思路
這里我直接介紹一款神器Space Commander,它利用 Git Hooks ,在 commit 之前檢查代碼風格是否符合規范,只有符合規范的代碼才允許提交,列出不符合規范的文件,同時提供 Shell 腳本來自動格式化。接下來我介紹下如何使用。
目標
- 理解目前格式化的手段,基于.clang_format 文件做
- spacecommander這個開源組件可以內部做具體代碼格式合法檢測、它消---費.clang_format做自動化代碼
- 使用 xcode build hook 啟動 spacecommander 自動化下載組件
- 腳本啟動 spacecommander
跟著步驟來一遍
第一步:項目根目錄下新增一個下載并且啟動 spacecommander 腳本文件 :format-check.sh
#!/usr/bin/env bash
orgin=`git rev-parse --show-toplevel` #展示工作根目錄
if [ ! -d .git ];
then exit 0
fi
if [ -d .spacecommander ];
then exit 0
fi
mkdir .spacecommander
cd .spacecommander
git clone git@git.koolearn-inc.com:k12/iOS/SpaceCommand.git spacecommander
cd $orgin
bash "$(pwd)/.spacecommander/spacecommander/setup-repo.sh"
exit 0
第二步:利用編譯的插樁
方式1: clang 自定義插件
方式2: xcode -> build phase -> add run script (目前使用這種方式)增加一下腳本,啟動spacecommander下載啟動腳本
cd ${SRCROOT}
chmod u+x ./format-check.sh
sh format-check.sh
執行 command + b,結果如下:
- 工程根目錄下會生成一個.spacecommander隱藏文件,已多一分.clang_format副本,這個文件用于做自動化檢測代碼風格的基礎,我們可以修改內部的規則
- .git文件夾 目錄會自動生成一個pre-commit文件 ,用于hook我們的hook操作
第三步:我們在自己的工程里隨便增加一部分代碼,隨便風格編寫
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view.
//方式1
//self.threadAliveHelper = [[GGZResidentThread alloc] init];
//方式2
self.threadAliveHelper = [[GGZKeepAliveHelper alloc] init];
UIView *view1 = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 100, 100)];
NSLog(@"view1-frame:%@", NSStringFromCGRect(view1.frame));
view1.bounds = CGRectMake(0, 0, 50, 50);
NSLog(@"view2-frame:%@", NSStringFromCGRect(view1.frame));
view1.backgroundColor = [UIColor redColor];
[self.view addSubview:view1];
UIView *view2 = [[UIView alloc] initWithFrame:CGRectMake(50, 30, 300, 300)];
[self.view addSubview:view2];
view2.backgroundColor = [UIColor yellowColor];
[view2 addSubview:view1];}
- (IBAction)resumeResidentThread:(id)sender
{
[self.threadAliveHelper resume];
}
- (void)test
{
NSLog(@"aa");
}
- (void)test2
{
NSLog(@"aa");
}
cmd上執行 git commit -m '修改文件' 或者source tree 提交修改
這時會示我們代碼風格根clang_format 文件不一致的風格,提示如下?
根據提示:我們的 ViewController.m 風格不對 ,執行提示的命令去格式化
"/Users/gegaozhao/Desktop/Person/練習/IOS-learning/TEST12-Threadkeepalive/.spacecommander/spacecommander"/format-objc-files.sh
注意:引號不要漏掉
格式化后代碼:
@interface ViewController ()
{
BOOL _stopLoop;
}
@property (nonnull, strong) id<GGZKeepThreadProtocol> threadAliveHelper;
@end
@implementation ViewController
- (void)viewDidLoad
{
[super viewDidLoad];
// Do any additional setup after loading the view.
//方式1
//self.threadAliveHelper = [[GGZResidentThread alloc] init];
//方式2
self.threadAliveHelper = [[GGZKeepAliveHelper alloc] init];
UIView *view1 = [[UIView alloc] initWithFrame:CGRectMake(0, 0, 100, 100)];
NSLog(@"view1-frame:%@", NSStringFromCGRect(view1.frame));
view1.bounds = CGRectMake(0, 0, 50, 50);
NSLog(@"view2-frame:%@", NSStringFromCGRect(view1.frame));
view1.backgroundColor = [UIColor redColor];
[self.view addSubview:view1];
UIView *view2 = [[UIView alloc] initWithFrame:CGRectMake(50, 30, 300, 300)];
[self.view addSubview:view2];
view2.backgroundColor = [UIColor yellowColor];
[view2 addSubview:view1];
}
- (IBAction)resumeResidentThread:(id)sender
{
[self.threadAliveHelper resume];
}
- (void)test
{
NSLog(@"aa");
}
- (void)test2
{
NSLog(@"aa");
}
思考。。。
我們實現的是半自動化,我們執行commit 后還是需要我們執行提示的命令去修改 可不可以 我們直接提示 然后按照提示去修改呢
進一步繼續優化??
關于這里有點復雜,需要了解 git hook 原理 和 clang 插樁操作 ,參考下開頭我提供的文檔
這里執行給你們提供我完善的 Space Commander 組件庫 ,執行git commit 提示是否自動格式化代碼 ,可以將前面的第一步:format-check.sh腳本 git clone 地址換為我的 Space Commander 的 git@ 地址即可 ,你也可以down下來自己看下,是否存在安全因素再選擇是否用,具體修改如下:
常見問題
問題1: 如果你command + b 發現 跟目錄沒有 spacecommander
原因1:你的format-check.sh 的git 地址不對,很有可能,這個不可要 fan qiang .
原因2:你的github的受本地的nds影響,修改電腦dns即可
問題2: .git / hook/ 沒有 pre-commit 執行文件
原因:你的工程的format-check.sh 沒有執行 spacecommander 目錄下的setup-repo.sh的路徑不對
如果你想省事,可以直接下載我的 Space Commander 然后解壓到根目錄下即可,要保證和腳本format-check.sh路徑一致
其他
- 如果你想省事,可以直接down下我的 Space Commander 然后解壓到根目錄下即可,要保證和腳本format-check.sh路徑一致
- 如果你想要用自己的規則,可以去Space Commander倉庫中改真身,也可以用新的.clang-format文件替換掉這個替身。
-其實上面的方式,需要每一個開發者第一次command +b 下載對應的space command ,我們后面使用gitlab或者其他的代碼管理倉的web hook方式,首先在jenkins我們的flow 構建工程勾選代碼提交監聽,這時會生成對應的webhook地址,這時將地址放在gitlab/設置/webhook,這樣你每一次提交,會觸發這個地址,jenkins會對提交的commit代碼進行自動化格式化,這種方式方便了許多