Mô-đun Bazel là một dự án Bazel có thể có nhiều phiên bản, mỗi phiên bản phát hành siêu dữ liệu về các mô-đun khác mà nó phụ thuộc vào. Điều này tương tự như các khái niệm quen thuộc trong các hệ thống quản lý phần phụ thuộc khác, chẳng hạn như artifact Maven, package npm, module Go hoặc crate Cargo.
Một mô-đun phải có tệp MODULE.bazel ở thư mục gốc của kho lưu trữ (bên cạnh tệp
WORKSPACE). Tệp này là tệp kê khai của mô-đun, khai báo tên,
phiên bản, danh sách các phần phụ thuộc trực tiếp và các thông tin khác. Ví dụ cơ bản
module(name = "my-module", version = "1.0")
bazel_dep(name = "rules_cc", version = "0.0.1")
bazel_dep(name = "protobuf", version = "3.19.0")
Để thực hiện phân giải mô-đun, Bazel bắt đầu bằng cách đọc tệp
MODULE.bazel của mô-đun gốc, sau đó liên tục yêu cầu tệp
MODULE.bazel của mọi phần phụ thuộc từ sổ đăng ký Bazel cho đến khi
phát hiện ra toàn bộ biểu đồ phần phụ thuộc.
Theo mặc định, Bazel sẽ chọn một phiên bản của mỗi mô-đun để sử dụng. Bazel biểu thị mỗi mô-đun bằng một kho lưu trữ và tham khảo lại sổ đăng ký để tìm hiểu cách xác định từng kho lưu trữ.
Định dạng phiên bản
Bazel có một hệ sinh thái đa dạng và các dự án sử dụng nhiều lược đồ kiểm soát phiên bản. Cho đến nay, lược đồ phổ biến nhất là SemVer, nhưng cũng có những dự án nổi bật sử dụng các lược đồ khác như Abseil, có các phiên bản dựa trên ngày, ví dụ: 20210324.2).
Vì lý do này, Bzlmod áp dụng một phiên bản thoải mái hơn của đặc tả SemVer. Các điểm khác biệt bao gồm:
- SemVer quy định rằng phần "phát hành" của phiên bản phải bao gồm 3
phân đoạn:
MAJOR.MINOR.PATCH. Trong Bazel, yêu cầu này được nới lỏng để cho phép bất kỳ số lượng phân đoạn nào. - Trong SemVer, mỗi phân đoạn trong phần "phát hành" phải chỉ là các chữ số. Trong Bazel, yêu cầu này được nới lỏng để cho phép cả chữ cái và ngữ nghĩa so sánh khớp với "mã nhận dạng" trong phần "bản phát hành trước".
- Ngoài ra, ngữ nghĩa của việc tăng phiên bản chính, phiên bản phụ và phiên bản bản vá không được thực thi. Tuy nhiên, hãy xem cấp độ tương thích để biết thông tin chi tiết về cách chúng tôi biểu thị khả năng tương thích ngược.
Mọi phiên bản SemVer hợp lệ đều là phiên bản mô-đun Bazel hợp lệ. Ngoài ra, hai
phiên bản SemVer a và b so sánh a < b nếu và chỉ khi điều tương tự xảy ra khi
chúng được so sánh dưới dạng phiên bản mô-đun Bazel.
Chọn phiên bản
Hãy xem xét vấn đề về phần phụ thuộc hình kim cương, một vấn đề chính trong không gian quản lý phần phụ thuộc có phiên bản. Giả sử bạn có biểu đồ phần phụ thuộc:
A 1.0
/ \
B 1.0 C 1.1
| |
D 1.0 D 1.1
Bạn nên sử dụng phiên bản D nào? Để giải quyết câu hỏi này, Bzlmod sử dụng thuật toán
Chọn phiên bản tối thiểu
(MVS) được giới thiệu trong hệ thống mô-đun Go. MVS giả định rằng tất cả các phiên bản mới
của một mô-đun đều tương thích ngược, do đó, sẽ chọn phiên bản cao nhất
do bất kỳ phần phụ thuộc nào chỉ định (D 1.1 trong ví dụ của chúng tôi). Thuật toán này được gọi là "tối thiểu"
vì D 1.1 là phiên bản sớm nhất có thể đáp ứng các yêu cầu của chúng tôi –
ngay cả khi D 1.2 hoặc phiên bản mới hơn tồn tại, chúng tôi cũng không chọn các phiên bản đó. Việc sử dụng MVS sẽ tạo ra một
quy trình chọn phiên bản có độ trung thực cao và có thể tái tạo.
Các phiên bản bị loại bỏ
Sổ đăng ký có thể khai báo một số phiên bản nhất định là bị loại bỏ nếu bạn nên tránh sử dụng các phiên bản đó
(chẳng hạn như đối với các lỗ hổng bảo mật). Bazel sẽ đưa ra lỗi khi chọn phiên bản bị loại bỏ của một mô-đun. Để khắc phục lỗi này, hãy nâng cấp lên phiên bản mới hơn,
không bị loại bỏ hoặc sử dụng cờ
--allow_yanked_versions
để cho phép phiên bản bị loại bỏ một cách rõ ràng.
Cấp độ tương thích
Trong Go, giả định của MVS về khả năng tương thích ngược hoạt động vì nó coi
các phiên bản không tương thích ngược của một mô-đun là một mô-đun riêng biệt. Về
SemVer, điều đó có nghĩa là A 1.x và A 2.x được coi là các mô-đun riêng biệt và có thể
cùng tồn tại trong biểu đồ phần phụ thuộc đã phân giải. Đổi lại, điều này được thực hiện bằng cách
mã hoá phiên bản chính trong đường dẫn gói trong Go, vì vậy, không có xung đột nào trong thời gian
biên dịch hoặc thời gian liên kết.
Tuy nhiên, Bazel không thể đưa ra những đảm bảo như vậy, vì vậy, nó cần số "phiên bản chính"
để phát hiện các phiên bản không tương thích ngược. Số này được gọi là
cấp độ tương thích và được chỉ định bởi mỗi phiên bản mô-đun trong chỉ thị
module(). Với thông tin này, Bazel có thể đưa ra lỗi khi nó
phát hiện thấy các phiên bản của cùng một mô-đun có các cấp độ tương thích khác nhau
tồn tại trong biểu đồ phần phụ thuộc đã phân giải.
Chế độ ghi đè
Chỉ định chế độ ghi đè trong tệp MODULE.bazel để thay đổi hành vi của quá trình phân giải mô-đun Bazel. Chỉ chế độ ghi đè của mô-đun gốc mới có hiệu lực – nếu một mô-đun được
dùng làm phần phụ thuộc, thì chế độ ghi đè của mô-đun đó sẽ bị bỏ qua.
Mỗi chế độ ghi đè được chỉ định cho một tên mô-đun nhất định, ảnh hưởng đến tất cả các phiên bản của mô-đun đó trong biểu đồ phần phụ thuộc. Mặc dù chỉ chế độ ghi đè của mô-đun gốc mới có hiệu lực, nhưng chúng có thể dành cho các phần phụ thuộc bắc cầu mà mô-đun gốc không phụ thuộc trực tiếp vào.
Ghi đè một phiên bản
The single_version_override
phục vụ nhiều mục đích:
- Với thuộc tính
version, bạn có thể ghim một phần phụ thuộc vào một phiên bản cụ thể, bất kể phiên bản nào của phần phụ thuộc được yêu cầu trong biểu đồ phần phụ thuộc. - Với thuộc tính
registry, bạn có thể buộc phần phụ thuộc này đến từ một sổ đăng ký cụ thể, thay vì tuân theo quy trình chọn sổ đăng ký thông thường registry. - Với các thuộc tính
patch*, bạn có thể chỉ định một tập hợp các bản vá để áp dụng cho mô-đun đã tải xuống.
Tất cả các thuộc tính này đều không bắt buộc và có thể kết hợp với nhau.
Ghi đè nhiều phiên bản
Bạn có thể chỉ định multiple_version_override
để cho phép nhiều phiên bản của cùng một mô-đun cùng tồn tại trong
biểu đồ phần phụ thuộc đã phân giải.
Bạn có thể chỉ định danh sách phiên bản được phép một cách rõ ràng cho mô-đun. Tất cả các phiên bản này phải có trong biểu đồ phần phụ thuộc trước khi phân giải – phải có một số phần phụ thuộc bắc cầu tuỳ thuộc vào từng phiên bản được phép. Sau khi phân giải, chỉ các phiên bản được phép của mô-đun mới còn lại, trong khi Bazel nâng cấp các phiên bản khác của mô-đun lên phiên bản được phép cao hơn gần nhất ở cùng cấp độ tương thích. Nếu không có phiên bản được phép cao hơn ở cùng cấp độ tương thích tồn tại, Bazel sẽ đưa ra lỗi.
Ví dụ: nếu các phiên bản 1.1, 1.3, 1.5, 1.7, và 2.0 tồn tại trong
biểu đồ phần phụ thuộc trước khi phân giải và phiên bản chính là cấp độ tương thích
:
- Chế độ ghi đè nhiều phiên bản cho phép
1.3,1.7và2.0dẫn đến việc1.1được nâng cấp lên1.3,1.5được nâng cấp lên1.7và các phiên bản khác vẫn giữ nguyên. - Chế độ ghi đè nhiều phiên bản cho phép
1.5và2.0dẫn đến lỗi, vì1.7không có phiên bản cao hơn ở cùng cấp độ tương thích để nâng cấp. - Chế độ ghi đè nhiều phiên bản cho phép
1.9và2.0dẫn đến lỗi, vì1.9không có trong biểu đồ phần phụ thuộc trước khi phân giải.
Ngoài ra, người dùng cũng có thể ghi đè sổ đăng ký bằng registry
thuộc tính, tương tự như chế độ ghi đè một phiên bản.
Ghi đè không phải sổ đăng ký
Chế độ ghi đè không phải sổ đăng ký sẽ xoá hoàn toàn một mô-đun khỏi quá trình phân giải phiên bản. Bazel
không yêu cầu các tệp MODULE.bazel này từ sổ đăng ký mà từ
chính kho lưu trữ.
Bazel hỗ trợ các chế độ ghi đè không phải sổ đăng ký sau:
Tên kho lưu trữ và phần phụ thuộc nghiêm ngặt
Tên chính tắc của một kho lưu trữ hỗ trợ
mô-đun là module_name~version (ví dụ: bazel_skylib~1.0.3). Đối với các mô-đun có
chế độ ghi đè không phải sổ đăng ký, hãy thay thế phần version
bằng chuỗi override. Xin lưu ý rằng định dạng tên chính tắc không phải là API
mà bạn nên phụ thuộc vào và có thể thay đổi bất kỳ lúc nào.
Tên hiển thị của một kho lưu trữ hỗ trợ mô-đun cho các phần phụ thuộc trực tiếp của mô-đun đó mặc định là tên mô-đun, trừ phi thuộc tính repo_name của chỉ thị bazel_dep cho biết khác. Xin lưu ý rằng điều này có nghĩa là một mô-đun chỉ có thể tìm thấy các phần phụ thuộc trực tiếp
của mô-đun đó. Điều này giúp ngăn chặn các lỗi vô tình do những thay đổi trong
các phần phụ thuộc bắc cầu.
Tiện ích mô-đun cũng có thể đưa thêm các kho lưu trữ vào phạm vi hiển thị của một mô-đun.