在單一儲存庫中工作最強大的方面之一是能夠在套件/團隊/層次結構之間共用程式碼。在這篇文章中,我將嘗試解釋一個非常簡單的現實世界場景
想像一下,您想要開發一個函式庫來顯示以兆位元組為單位的檔案大小,您認為這可能對您的 monorepo 的其他部分有用。該函式庫接受整數大小(例如:2048 位元組),並且可以傳回人性化字串(例如:2 MB)。為了增加一些品質保證,我們還將為其編寫一個測試。
從上面的場景我們知道,我們需要將此函數開發為共享庫,然後由另一個套件導入以供使用。 Bazel 允許我們在庫中定義函數並將其匯出到需要它的其他服務,從而使這變得非常簡單。正如我之前在本文底部連結的文章中所解釋的,我們還可以控制哪些其他函式庫也可以匯入它以供使用。
出於程式碼組織的目的,我們將在工作區的根目錄下有一個庫目錄,其中有一個名為 humanize_filesize 的子目錄,我們將在其中編寫庫代碼。
讓我們在 humanize_filesize.go 中寫一些非常基本的 Go 程式碼
package humanize_filesize import "fmt" // GetHumanizedFilesize takes size_in_bytes as an int32 pointer and returns the size in megabytes. func GetHumanizedFilesize(size_in_bytes *int32) string { if size_in_bytes != nil { size_in_megabytes := float64(*size_in_bytes) / (1024 * 1024) return fmt.Sprintf("%.4f MB", size_in_megabytes) } return "0 MB" }
此程式碼僅將 int32 作為輸入,並傳回計算出的 4 位十進位精度的可讀兆位元組字串
這個功能絕對不全面,絕對可以改進,但這不是這次練習的重點。
還要斷言我們的邏輯按預期工作,我們將在名為 humanize_filesize_test.go 的文件中添加一個非常基本的測試以及 go 代碼
package humanize_filesize import ( "testing" ) func TestHumanizeFilesize(t *testing.T) { tests := []struct { name string size_in_bytes *int32 expected string }{ { name: "nil bytes", size_in_bytes: nil, expected: "0 MB", }, { name: "2048 bytes", size_in_bytes: int32Ptr(2048), expected: "0.0020 MB", }, { name: "0 bytes", size_in_bytes: int32Ptr(0), expected: "0.0000 MB", }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { result := GetHumanizedFilesize(tt.size_in_bytes) if result != tt.expected { t.Errorf("expected %s, got %s", tt.expected, result) } }) } } func int32Ptr(n int32) *int32 { return &n }
一個非常簡單的測試,以 nil、int32 和 0 作為輸入進行基本測試
現在是如何匯出此函數的有趣部分,以便可以將其匯入到其他套件或服務中。這是我們必須定義 BUILD.bazel 檔案的地方。
load("@rules_go//go:def.bzl", "go_library", "go_test") go_library( name = "humanize_filesize", srcs = ["humanize_filesize.go"], importpath = "basil/libraries/humanize_filesize", visibility = ["//visibility:public"], ) go_test( name = "humanize_filesize_test", srcs = ["humanize_filesize_test.go"], embed = [":humanize_filesize"], )
在這裡我們定義了兩個主要規則。一個用於實際庫,一個用於我們編寫的測試文件。
go_library 定義目標 humanize_filesize 使用 humanize_filesize.go 作為其來源之一,可以透過 importpath 中指定的路徑導入,並且它在工作區中公開可見,可供其他套件導入。我們將在以後的帖子中學習如何控制可見性。
go_test 定義了一個測試目標,其中嵌入了 go_library 輸出中的程式碼。
此時我們應該能夠透過運行我們的測試套件來測試庫,如下所示
bazel build //... && bazel run //libraries/ humanize_filesize: humanize_filesize_test
您應該可以看到如下所示的測試輸出,表示所有測試都已通過。
package humanize_filesize import "fmt" // GetHumanizedFilesize takes size_in_bytes as an int32 pointer and returns the size in megabytes. func GetHumanizedFilesize(size_in_bytes *int32) string { if size_in_bytes != nil { size_in_megabytes := float64(*size_in_bytes) / (1024 * 1024) return fmt.Sprintf("%.4f MB", size_in_megabytes) } return "0 MB" }
?哇喔! ! ! ?現在我們知道我們的圖書館正在按預期運作。
現在讓我們在 services 目錄中的服務 service1 中使用這個庫,我們將使用以下 go 程式碼和 BUILD.bazel 檔案在工作區的根目錄中建立該目錄。
service1.go
package humanize_filesize import ( "testing" ) func TestHumanizeFilesize(t *testing.T) { tests := []struct { name string size_in_bytes *int32 expected string }{ { name: "nil bytes", size_in_bytes: nil, expected: "0 MB", }, { name: "2048 bytes", size_in_bytes: int32Ptr(2048), expected: "0.0020 MB", }, { name: "0 bytes", size_in_bytes: int32Ptr(0), expected: "0.0000 MB", }, } for _, tt := range tests { t.Run(tt.name, func(t *testing.T) { result := GetHumanizedFilesize(tt.size_in_bytes) if result != tt.expected { t.Errorf("expected %s, got %s", tt.expected, result) } }) } } func int32Ptr(n int32) *int32 { return &n }
BUILD.bazel
load("@rules_go//go:def.bzl", "go_library", "go_test") go_library( name = "humanize_filesize", srcs = ["humanize_filesize.go"], importpath = "basil/libraries/humanize_filesize", visibility = ["//visibility:public"], ) go_test( name = "humanize_filesize_test", srcs = ["humanize_filesize_test.go"], embed = [":humanize_filesize"], )
Go 程式碼非常簡單,它會導入我們先前聲明的函式庫,並使用我們庫中的 GetHumanizedFilesize 函數,傳遞一個隨機整數值並列印輸出。
現在,當執行 bazel build //services/service1 時,bazel 將解析我們目標的所有依賴項,包括我們開發和建置它們的程式庫。
service1 現在可以使用 bazel run //services/service1 執行,因為我們只定義了一個二進位目標。如果您有多個二進位目標,例如:serviceX,您可以使用 bazel run //services/service1:serviceX 來執行目標。預設情況下,當不指定目標時,bazel 將始終嘗試尋找與目錄同名的二進位目標並執行它。
那麼...就這樣吧。我們已經製作了您的第一個共享庫,可供我們 monorepo 的其他部分使用。
此範例的所有程式碼都可以在 https://github.com/nixclix/basil/pull/3/commits/61c673b8757860bd5e60eb2ab6c35f3f4da78c87
如果您喜歡這篇文章的內容,請隨時分享。另外,請訂閱並留下評論,表達您對這篇文章的看法,以及您是否希望看到我改進的地方。
以上是monorepo 中共享庫的實際範例的詳細內容。更多資訊請關注PHP中文網其他相關文章!