หน้านี้ครอบคลุมข้อมูลเบื้องต้นเกี่ยวกับการใช้มาโคร รวมถึงกรณีการใช้งานทั่วไป การแก้ไขข้อบกพร่อง และข้อกำหนด
มาโครคือฟังก์ชันที่เรียกจากไฟล์ BUILD ซึ่งสามารถสร้างอินสแตนซ์ของกฎได้
โดยหลักๆ แล้วจะใช้มาโครเพื่อการห่อหุ้มและการนำโค้ดของกฎที่มีอยู่และมาโครอื่นๆ กลับมาใช้ซ้ำ เมื่อสิ้นสุดระยะการโหลด
มาโครจะไม่มีอยู่อีกต่อไป
และ Bazel จะเห็นเฉพาะชุดกฎที่เป็นรูปธรรมซึ่งสร้างอินสแตนซ์แล้ว
การใช้งาน
กรณีการใช้งานทั่วไปสำหรับมาโครคือเมื่อคุณต้องการนำกฎกลับมาใช้ซ้ำ
ตัวอย่างเช่น genrule ในไฟล์ BUILD จะสร้างไฟล์โดยใช้ //:generator ที่มีอาร์กิวเมนต์ some_arg ซึ่งฮาร์ดโค้ดไว้ในคำสั่ง
genrule(
name = "file",
outs = ["file.txt"],
cmd = "$(location //:generator) some_arg > $@",
tools = ["//:generator"],
)
หากต้องการสร้างไฟล์เพิ่มเติมด้วยอาร์กิวเมนต์ที่แตกต่างกัน คุณอาจต้องการแยกโค้ดนี้ไปยังฟังก์ชันมาโคร สมมติว่าเราเรียกมาโครว่า file_generator ซึ่งมีพารามิเตอร์ name และ arg ให้แทนที่ genrule ด้วยโค้ดต่อไปนี้
load("//path:generator.bzl", "file_generator")
file_generator(
name = "file",
arg = "some_arg",
)
file_generator(
name = "file-two",
arg = "some_arg_two",
)
file_generator(
name = "file-three",
arg = "some_arg_three",
)
ในที่นี้ คุณจะโหลดสัญลักษณ์ file_generator จากไฟล์ .bzl ที่อยู่ในแพ็กเกจ //path การใส่คำจำกัดความของฟังก์ชันมาโครในไฟล์ .bzl แยกต่างหากจะช่วยให้ไฟล์ BUILD ของคุณสะอาดและประกาศได้ ไฟล์ .bzl สามารถโหลดได้จากแพ็กเกจใดก็ได้ในเวิร์กสเปซ
สุดท้าย ให้เขียนคำจำกัดความของมาโครใน path/generator.bzl เพื่อห่อหุ้มและกำหนดพารามิเตอร์คำจำกัดความ genrule เดิม
def file_generator(name, arg, visibility=None):
native.genrule(
name = name,
outs = [name + ".txt"],
cmd = "$(location //:generator) %s > $@" % arg,
tools = ["//:generator"],
visibility = visibility,
)
นอกจากนี้ คุณยังใช้มาโครเพื่อเชื่อมโยงกฎต่างๆ เข้าด้วยกันได้ด้วย ตัวอย่างนี้แสดง genrule ที่เชื่อมโยงกัน ซึ่ง genrule หนึ่งใช้เอาต์พุตของ genrule ก่อนหน้าเป็นอินพุต
def chained_genrules(name, visibility=None):
native.genrule(
name = name + "-one",
outs = [name + ".one"],
cmd = "$(location :tool-one) $@",
tools = [":tool-one"],
visibility = ["//visibility:private"],
)
native.genrule(
name = name + "-two",
srcs = [name + ".one"],
outs = [name + ".two"],
cmd = "$(location :tool-two) $< $@",
tools = [":tool-two"],
visibility = visibility,
)
ตัวอย่างจะกำหนดค่าการมองเห็นให้กับ genrule ที่ 2 เท่านั้น ซึ่งช่วยให้ผู้เขียนมาโครซ่อนเอาต์พุตของกฎระดับกลางไม่ให้เป้าหมายอื่นๆ ในเวิร์กสเปซต้องพึ่งพา
การขยายมาโคร
เมื่อต้องการตรวจสอบสิ่งที่มาโครทำ ให้ใช้คำสั่ง query กับ --output=build เพื่อดูรูปแบบที่ขยาย
$ bazel query --output=build :file
# /absolute/path/test/ext.bzl:42:3
genrule(
name = "file",
tools = ["//:generator"],
outs = ["//test:file.txt"],
cmd = "$(location //:generator) some_arg > $@",
)
การสร้างอินสแตนซ์ของกฎดั้งเดิม
กฎดั้งเดิม (กฎที่ไม่ต้องมีคำสั่ง load() ) สามารถ
สร้างอินสแตนซ์ได้จากโมดูล ดั้งเดิม:
def my_macro(name, visibility=None):
native.cc_library(
name = name,
srcs = ["main.cc"],
visibility = visibility,
)
หากต้องการทราบชื่อแพ็กเกจ (เช่น ไฟล์ BUILD ใดที่เรียกใช้
มาโคร) ให้ใช้ฟังก์ชัน native.package_name()
โปรดทราบว่า native ใช้ได้เฉพาะในไฟล์ .bzl เท่านั้น และใช้ในไฟล์ BUILD ไม่ได้
การแก้ปัญหาป้ายกำกับในมาโคร
เนื่องจากมาโครได้รับการประเมินในระยะการโหลด สตริงป้ายกำกับ เช่น "//foo:bar" ที่เกิดขึ้นในมาโครจะได้รับการตีความแบบสัมพันธ์กับไฟล์ BUILD ที่ใช้มาโครนั้นๆ แทนที่จะสัมพันธ์กับไฟล์ .bzl ที่กำหนดไว้ โดยทั่วไปแล้วลักษณะการทำงานนี้ไม่เป็นที่ต้องการสำหรับมาโครที่ตั้งใจไว้ให้ใช้ในที่เก็บข้อมูลอื่นๆ เช่น เนื่องจากเป็นส่วนหนึ่งของชุดกฎ Starlark ที่เผยแพร่
หากต้องการให้มีลักษณะการทำงานเหมือนกับกฎ Starlark ให้รวมสตริงป้ายกำกับด้วยเครื่องมือสร้าง
Label
# @my_ruleset//rules:defs.bzl
def my_cc_wrapper(name, deps = [], **kwargs):
native.cc_library(
name = name,
deps = deps + select({
# Due to the use of Label, this label is resolved within @my_ruleset,
# regardless of its site of use.
Label("//config:needs_foo"): [
# Due to the use of Label, this label will resolve to the correct target
# even if the canonical name of @dep_of_my_ruleset should be different
# in the main repo, such as due to repo mappings.
Label("@dep_of_my_ruleset//tools:foo"),
],
"//conditions:default": [],
}),
**kwargs,
)
การแก้ไขข้อบกพร่อง
bazel query --output=build //my/path:allจะแสดงลักษณะของไฟล์BUILDหลังจากประเมิน มาโคร, globs, ลูปทั้งหมดจะขยาย ข้อจำกัดที่ทราบ: ขณะนี้ระบบยังไม่แสดงนิพจน์selectในเอาต์พุตคุณสามารถกรองเอาต์พุตตาม
generator_function(ฟังก์ชันที่สร้างกฎ) หรือgenerator_name(แอตทริบิวต์ name ของมาโคร) ได้โดยใช้คำสั่งbash $ bazel query --output=build 'attr(generator_function, my_macro, //my/path:all)'หากต้องการทราบตำแหน่งที่แน่นอนที่สร้างกฎ
fooในไฟล์BUILDคุณสามารถลองใช้เคล็ดลับต่อไปนี้ แทรกบรรทัดนี้ไว้ใกล้ด้านบนของBUILDไฟล์:cc_library(name = "foo")เรียกใช้ Bazel คุณจะได้รับข้อยกเว้นเมื่อสร้างกฎfoo(เนื่องจากชื่อซ้ำกัน) ซึ่งจะแสดงสแต็กเทรซแบบเต็มนอกจากนี้ คุณยังใช้คำสั่ง print เพื่อแก้ไขข้อบกพร่องได้ด้วย โดยคำสั่งนี้จะแสดงข้อความเป็นบรรทัดบันทึก
DEBUGในระยะการโหลด ยกเว้นในกรณีที่พบได้ยาก ให้นำการเรียกprintออก หรือทำให้การเรียกดังกล่าวเป็นแบบมีเงื่อนไขภายใต้พารามิเตอร์debuggingซึ่งมีค่าเริ่มต้นเป็นFalseก่อนที่จะส่งโค้ดไปยังที่เก็บ
ข้อผิดพลาด
หากต้องการแสดงข้อผิดพลาด ให้ใช้ฟังก์ชัน fail
อธิบายให้ผู้ใช้ทราบอย่างชัดเจนว่าเกิดอะไรขึ้นและวิธีแก้ไขไฟล์ BUILD
คุณไม่สามารถดักจับข้อผิดพลาดได้
def my_macro(name, deps, visibility=None):
if len(deps) < 2:
fail("Expected at least two values in deps")
# ...
ข้อกำหนด
ฟังก์ชันสาธารณะทั้งหมด (ฟังก์ชันที่ไม่ขึ้นต้นด้วยขีดล่าง) ที่สร้างอินสแตนซ์ของกฎต้องมีอาร์กิวเมนต์
nameอาร์กิวเมนต์นี้ไม่ควรเป็นอาร์กิวเมนต์ที่ไม่บังคับ (อย่ากำหนดค่าเริ่มต้น)ฟังก์ชันสาธารณะควรใช้ docstring ตามข้อกำหนดของ Python conventions
ในไฟล์
BUILDอาร์กิวเมนต์nameของมาโครต้องเป็นอาร์กิวเมนต์คีย์เวิร์ด (ไม่ใช่ positional argument)แอตทริบิวต์
nameของกฎที่สร้างโดยมาโครควรมีอาร์กิวเมนต์ name เป็นคำนำหน้า ตัวอย่างเช่นmacro(name = "foo")สามารถสร้างcc_libraryfooและ genrulefoo_genในกรณีส่วนใหญ่ พารามิเตอร์ที่ไม่บังคับควรมีค่าเริ่มต้นเป็น
Noneคุณสามารถส่งNoneไปยังกฎดั้งเดิมได้โดยตรง ซึ่งกฎดั้งเดิมจะถือว่าเป็นการไม่ส่งอาร์กิวเมนต์ใดๆ ดังนั้นจึงไม่จำเป็นต้องแทนที่ด้วย0,Falseหรือ[]เพื่อจุดประสงค์นี้ แต่มาโครควรเลื่อนการดำเนินการไปยังกฎที่สร้างขึ้น เนื่องจากค่าเริ่มต้นของกฎอาจซับซ้อนหรือเปลี่ยนแปลงไปตามกาลเวลา นอกจากนี้ พารามิเตอร์ที่ตั้งค่าอย่างชัดเจนเป็นค่าเริ่มต้นจะมีลักษณะแตกต่างจากพารามิเตอร์ที่ไม่เคยตั้งค่า (หรือตั้งค่าเป็นNone) เมื่อเข้าถึงผ่านภาษาการค้นหาหรือส่วนภายในของระบบบิลด์มาโครควรมีอาร์กิวเมนต์
visibilityที่ไม่บังคับ