מושגי Android DSL

זוהי סקירה כללית של המושגים הבסיסיים של שפת התחום הספציפית (DSL) לאוטומציה ב-Android.

רכיבי אוטומציה

אוטומציה מורכבת מהרכיבים הבסיסיים הבאים, שבדרך כלל נבדקים בסדר הזה:

  1. סימן לתחילת הפעולה – מגדיר את התנאים הראשוניים שמפעילים את האוטומציה, כמו שינוי במאפיין. לכל פעולה אוטומטית צריך להיות סימן לתחילת הפעולה.
  2. תנאי – אילוצים נוספים להערכה אחרי הפעלת האוטומציה. הביטוי בתנאי צריך להיות שווה ל-true כדי שהפעולות של האוטומציה ימשיכו.
  3. פעולה – פקודות או עדכוני סטטוס שמתבצעים כשכל התנאים מתקיימים.

לדוגמה, יכול להיות שיש לכם פעולה אוטומטית שמחלישה את האורות בחדר כשמפעילים את הטלוויזיה בחדר בין השקיעה לזריחה. בדוגמה הזו:

  1. Starter – הטלוויזיה הופעלה, וזהו שינוי במצב של מאפיין הטלוויזיה.
  2. תנאי – המערכת מעריכה את השעה הנוכחית בבית שבו נמצאת הטלוויזיה.
  3. פעולה – עוצמת ההארה של הנורות באותו חדר שבו נמצא הטלוויזיה תפחת.

האוטומציה תופעל כשהטלוויזיה בחדר תופעל, אבל היא תתבצע רק אם התנאי 'השעה היא בין השקיעה לזריחה' יתקיים.

בנוסף למבנה הבסיסי, האוטומציות בממשקי ה-API של Home מכילות גם מטא-נתונים, כמו שם ותיאור, שאפשר להשתמש בהם כדי לזהות אותן עבור מפתחים ומשתמשים.

צמתים

בממשקי ה-API של Home, המבנה הלוגי של אוטומציה מורכב מצמתים. צמתים הם יחידות מופשטות לשימוש חוזר שמייצגות התנהגויות של ישויות או זרימות של ביצוע. לכל צומת יכולות להיות משתני קלט וגם משתני פלט, שיכולים לשמש צמתים אחרים.

טבלה: סוגים של צמתי אוטומציה
צומת סוג צומת הטמעה של Kotlin תיאור
Starter התנהגותי StarterNodeDsl הפעלה של אוטומציה כשמצב המאפיין (כל מאפיין) משתנה.
StateReader התנהגותי StateReaderNodeDsl קריאה של מאפיין של תכונה ומאפשרת לכם לתעד את הערך שלו לשימוש בצמתי תנאים.
פעולה התנהגותי ActionNodeDsl הפעלת פקודות של מאפיינים.
רציף תהליך הביצוע SequentialFlow מבצעת צמתי פעולה מקוננים ברצף. זו התנהגות ברירת המחדל של ההרצה.
מקביל תהליך הביצוע ParallelFlow מבצעת צמתים מקוננים של פעולות במקביל.
Condition תהליך הביצוע ConditionNodeDsl שינוי מותנה של תהליך הביצוע על סמך הערכות של ביטויים לוגיים. אפשר לשייך תנאים ל-starter (תנאים ספציפיים ל-starter) או להגדיר תנאים גלובליים (שחלים על כל ה-starters).
בחירה תהליך הביצוע SelectFlow מאפשר להפעיל פעולה אוטומטית באמצעות יותר מסימן אחד לתחילת פעולה.
ביטוי ערך Expression יכול להיות הערך של מאפיין של תכונה, קבוע או ערך מילולי, והוא חייב להיות רשימה, מספר, ערך בוליאני או מחרוזת.

צמתים התנהגותיים

צמתים כמו סימנים לתחילת פעולה ופעולות הם צמתים התנהגותיים. חבילות Starter מפעילות אוטומציה על סמך שינויים במאפייני המכשיר. פעולות ששולחות פקודות למכשיר או מעדכנות מאפיינים.

צמתי התנהגות קשורים בדרך כלל למאפייני המכשיר ולמצב מאפיין הפלט לשימוש כקלט בצמתים אחרים.

צמתים של זרימת ביצוע

חלק מהצמתים מייצגים זרימות ביצוע, כמו רציף ומקביל. כל אחד מהצמתים האלה מכיל את הצמתים ההתנהגותיים שמגדירים את האוטומציה.

לדוגמה, זרימה רציפה עשויה להכיל צמתים שמופעלים בסדר רציף. בדרך כלל אלה יהיו starter,‏ condition ו-action.

תהליכי עבודה לביצוע רציף
איור 1: תרשים זרימה של אוטומציה רציפה

בזרימה מקבילית יכולים להיות כמה צמתי פעולה שמופעלים בו-זמנית, למשל הדלקה של כמה אורות בו-זמנית. צמתים שמופיעים אחרי זרימה מקבילה לא יופעלו עד שכל הענפים של הזרימה המקבילה יסתיימו.

זרימות ביצוע מקבילות
איור 2: זרימת אוטומציה מקבילית

סוג נוסף של תהליך ביצוע הוא תהליך מותנה, שיכול לשנות את תהליך הביצוע על סמך הערכה של ביטוי.

לדוגמה, יכול להיות שיש לכם אוטומציה שמבצעת פעולה על סמך השאלה אם הלילה ירד. צומת של תנאי בודק את השעה ביום, ואז פועל לפי נתיב הביצוע המתאים על סמך הבדיקה הזו.

זרימת התנאים
איור 3: תהליך מותנה

זרימת בחירה שימושית כשרוצים להגדיר יותר מפעיל אחד שיכול להפעיל את האוטומציה. כשמקיפים שני סימנים או יותר לתחילת פעולה בזרימת select, כל אחד מהסימנים האלה יכול להפעיל את האוטומציה.

לדוגמה, אפשר לכתוב אוטומציה שמורידה את התריסים בשקיעה, אם הטמפרטורה עולה מעל סף מסוים או אם הבהירות עולה מעל סף מסוים. יש שלושה תרחישים נפרדים, ולכל אחד מהם יש תבנית התחלה משלו. כל שלושת התרחישים האלה נכללים בתהליך select.

בחירת תהליך
איור 4: בחירת זרימה

תהליכים מוטמעים

באוטומציות מורכבות, אפשר גם להשתמש בצמתים מוטמעים של זרימת הביצוע. לדוגמה, יכול להיות שיש לכם זרימה עוקבת שמבצעת זרימה מקבילה.

תהליכי ביצוע מוטמעים
איור 5: תהליכי ביצוע מוטמעים

אפשר להשתמש בצמתי DSL בתוך צמתי DSL אחרים ולשלב אותם בדרכים שונות כדי להתאים אותם לצרכים הספציפיים שלכם, בהתאם למגבלות שמפורטות בטבלה הבאה. הקישור בעמודה Builder מוביל לתיעוד של ה-builder ב-Kotlin, שבו מפורט מה מותר לשימוש בכל סוג של צומת.

טבלה: איך אפשר לשלב צמתים
צומת יכול להכיל את סוג הצומת והנתונים הבאים חייב להיות באחד מסוגי הצמתים הבאים
Starter ביטוי בחירה באפשרות 'רציף'
ManualStarter בחירה באפשרות 'רציף'
StateReader ביטוי (בדרך כלל כולל ערך של מאפיין מאפיין) פעולה, תנאי
פעולה פקודה, ישות, ביטוי מקביל, בחירה, רציף
רציף מקביל, בחירה, רציף
מקביל פעולה לפי הסדר
Condition ביטוי מקביל, רציף
בחירה תנאי, רציף, סימן לתחילת פעולה, סימן ידני לתחילת פעולה רציף, וחייב להיות הצומת הראשון בתהליך

Automation DSL

בממשקי ה-API של Home, פעולות אוטומטיות מוגדרות באמצעות Automation DSL (שפה ספציפית לדומיין). ה-DSL לאוטומציה מיושם כ-Kotlin DSL (שפה ספציפית לדומיין), באמצעות כלי בנייה בטוחים מסוג Kotlin, והוא מיועד במיוחד להגדרת תבניות אוטומציה.

כשמהדרים אוטומציה, כלי בנייה בטוחים לטיפוס ב-Kotlin יוצרים מחלקות נתונים ב-Kotlin, שמומרות לסדרת נתונים ב-JSON של מאגר פרוטוקולים. הנתונים האלה משמשים ליצירת קריאות לשירותי האוטומציה של Google.

ה-DSL של האוטומציה מפשט ומייעל את תהליך בניית האוטומציות. הוא משתמש באופן מובנה באותו מודל נתונים של Matter מאפיינים רגילים וsmart home מאפיינים שמופיעים ב-Device API.

בנוסף, שפת ה-DSL לאוטומציה מגדירה את הלוגיקה של אוטומציה במונחים של סוגי מכשירים מופשטים, ולא במונחים של מופעים ספציפיים של מכשירים שנמצאים בבית של משתמש. הוא מאפשר למפתח לספק פרמטרים של קלט שאפשר להשתמש בהם בזמן הריצה כדי לציין מופעים בפועל של מכשירים, וגם ערכים חשובים אחרים של פרמטרים.

תחביר ה-DSL דומה לזה של Kotlin, והוא בטוח באותה מידה מבחינת סוגים, אבל אוטומציה שנכתבת ב-DSL של Automation היא פשוטה ותמציתית יותר מאותה אוטומציה שנכתבת ב-Kotlin טהור.

דוגמה

הדוגמה הבאה היא אוטומציה שמפעילה מכשיר, שנכתבה באמצעות Automation DSL:

val automation = automation {
  name = "MyFirstAutomation"
  description = "If light1 is on, turn on light2."
  isActive = true
  sequential {
    val onOffTrait = starter<_>(device1, OnOffLightDevice, OnOff)
    condition() { expression = onOffTrait.onOff equals true }
    action(device2, OnOffLightDevice) { command(OnOff.on()) }
  }
}

האוטומציה הזו היא בסיסית מאוד: כשdevice1, מנורה, נדלקת (המאפיין onOff משתנה ל-true), נשלחת הפקודה on() להדלקת device2.

האוטומציה משתמשת בצומת sequential, שמציין שהצמתים שלו יפעלו בסדר עוקב.

בצומת sequential יש צמתים התנהגותיים כמו starter,‏ condition ו-action. הפלט של הצומת starter מוקצה למשתנה לשימוש בצומת condition.