หน้านี้ครอบคลุมพื้นฐานของการใช้มาโคร รวมถึงกรณีการใช้งานทั่วไป การแก้ไขข้อบกพร่อง และการตั้งค่า
มาโครคือฟังก์ชันที่เรียกจากไฟล์ 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()) สามารถสร้างอินสแตนซ์จากโมดูล native ได้ดังนี้
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(แอตทริบิวต์ชื่อของมาโคร) ดังนี้- 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อาร์กิวเมนต์นี้ไม่ควรเป็น อาร์กิวเมนต์ที่ไม่บังคับ (อย่าระบุค่าเริ่มต้น)
- ฟังก์ชันสาธารณะควรใช้สตริงเอกสารตามรูปแบบของ Python 
- ในไฟล์ - BUILDอาร์กิวเมนต์- nameของมาโครต้องเป็นอาร์กิวเมนต์คีย์เวิร์ด (ไม่ใช่อาร์กิวเมนต์ตำแหน่ง)
- แอตทริบิวต์ - nameของกฎที่สร้างโดยมาโครควรมีอาร์กิวเมนต์ชื่อเป็นคำนำหน้า เช่น- macro(name = "foo")สามารถสร้าง- cc_library- fooและ genrule- foo_gen
- ในกรณีส่วนใหญ่ พารามิเตอร์ที่ไม่บังคับควรมีค่าเริ่มต้นเป็น - None- Noneสามารถส่งไปยังกฎเนทีฟได้โดยตรง ซึ่งจะถือว่าเหมือนกับว่าคุณไม่ได้ส่งอาร์กิวเมนต์ใดๆ ดังนั้นจึงไม่จำเป็นต้องแทนที่ด้วย- 0,- Falseหรือ- []เพื่อจุดประสงค์นี้ แต่แมโครควรเลื่อน ไปใช้กฎที่สร้างขึ้น เนื่องจากค่าเริ่มต้นของกฎอาจซับซ้อนหรืออาจเปลี่ยนแปลงไป ตามเวลา นอกจากนี้ พารามิเตอร์ที่ตั้งค่าเริ่มต้นอย่างชัดเจน จะดูแตกต่างจากพารามิเตอร์ที่ไม่เคยตั้งค่า (หรือตั้งค่าเป็น- None) เมื่อเข้าถึง ผ่านภาษาการค้นหาหรือส่วนประกอบภายในของระบบบิลด์
- มาโครควรมีอาร์กิวเมนต์ - visibilityที่ไม่บังคับ