- Json5/Yaml support
- Convert images
- Manifest editing
- Validation
- Includes
- @oninit
- Bindings
- Async Task Generator
- Tracking transpilied files
- Logger
- Type Gen
- Web Server
- Locale validation
- Protobuf Generator
Playlet implements a few Brighterscript Plugins. The plugins inject themselves in the compilation process, allowing the modification of bs scripts, xml components, and even assets or the app manifest. Let's start with a simple one:
This plugin allows us to include jsonc
, json5
and yaml
files into the app as static assets. They are an alternative to plain json
, which for instance, does not support comments. Additionally, this plugin minifies the json
files.
Brightscript can only parse json natively using ParseJson. But it is nice to format config files as we please, include comments, and use the Json5 format in general. Additionally, we get smaller files than the original assets.
At the end of the build, the plugin scans for certain files (jsonc
, json
, json5
and yaml
), and converts them to plain json, while keeping the original file name. That way they are all parsable by the ParseJson function, even though their original format is not compatible.
This plugin allows us to convert svg files to png files.
This is used to convert splash screen, app poster, as well as icons in the app. Having svg images (or vector images in general) as the source of truth is better, in case images need to be modified or resized. Since Roku can't load them, we convert them to png.
Each .svg
file found will have an associated .svg.meta.json5
file, which will contain the hash (MD5) of the svg, conversion paramters, and the hash of the output files. This allows us to hash existing files and see if a conversion is needed.
Notice how each .svg
file can be configured to generate multiple png files. This is especially the case to generate the app artwork (splash screen/poster) with different sizes from the same logo .svg
file.
This plugin allows us to make modifications to the app manifest dynamically, mostly based on compiler flags.
A few reasons. For example:
- If
--debug
flag is passed to the compiler, we want theDEBUG
compilation constant to be true - If
--test-mode
flag is passed, Turnplaylet-lib
into a standalone app instead of a component library, so we can run it as a standalone app for testing - Allows us to
DEBUG_HOST_IP_ADDRESS
with the local ip address of the debugging machine, so thatplaylet-app
is able to find theplaylet-lib.zip
file that VSCode is serving during a debug session.
We simply read the content of the original manifest, and modify it before the compilation starts. When the build is finished, we restore the source file of the manifest.
This plugin allows us to add custom diagnostic errors messages to source code, to prevent using certain patterns of functions.
Let's see an example. consider this code:
node.ObserveField("someField", "OnSomeFieldChanged")
This code makes it possible to make a typo in "OnSomeFieldChanged", and this callback ends up never firing.
Let's suppose we want to eliminate this pattern and always use:
node.ObserveField("someField", FuncName(OnSomeFieldChanged))
Where FuncName
will determine the function name at runtime. This way, we will get a linter error if there's a typo, and adds a bit of safety.
It's usually possible to create all types of validation plugins using the brigherscript plugin APIs. But this particular plugin is generic enough to allow us to define rules based on Regex. To define the rule in the previous example, we simply add this to the bsconfig.json
file:
"validation": [
{
"code": "NO_OBSERVE_STRING_CALLBACK",
"regex": "\\.observeField(scoped)?\\s*\\(\\s*\"\\w+\"\\s*,\\s*\"\\w+\"\\s*\\)",
"regexFlags": "ig",
"message": "observeField(\"field\", \"callback\") is not allowed. Use observeField(\"field\", FuncName(callback)) instead."
}
],
The plugin will scan all the code for usage matching "regex"
with the "regexFlags"
. If found, it will error with the "message"
and the error code "code"
.
The code allows us to ignore this diagnostics message, using the following:
' bs:disable-next-line NO_OBSERVE_STRING_CALLBACK
node.ObserveField("someField", "OnSomeFieldChanged")
This plugin allows us "include" different pieces of a component, enabling Composition over inheritance
Consider a "behaviour" that we want in multiple components. For example:
<!-- Beep.xml -->
<component name="Beep" extends="Node">
<interface>
<field id="beepNow" type="boolean" alwaysNotify="true" onChange="OnBeepSet" />
</interface>
<script type="text/brightscript" uri="pkg:/path/to/Beep.bs" />
</component>
' Beep.bs
function OnBeepSet()
print("BEEP BEEP!")
end function
Imagine you want to enable many components to "Beep". You have a few options:
- Inheritance: you make a base component that can "Beep", and extend it in different components. This can seem ok at first, but as your app scales, it becomes a burden to manage these inheritance relationships.
- Copy the fields, and the
<script>
tags to components everywhere. Can lead to lots of duplication, is error prone, but it can work - Use this plugin, which will do all the copying for you!
The plugin looks for the includes
attribute in components:
<!-- MyComponent.xml -->
<component name="MyComponent" extends="Node" includes="Beep">
</component>
Once it finds it, it will try to look for Beep.part.xml
, and will do the following:
- Copy all
field
s fromBeep
toMyComponent
- Copy all
function
s fromBeep
toMyComponent
- Copy all
<script>
tags fromBeep
toMyComponent
- Copy all children nodes from
Beep
toMyComponent
- Remove the
includes
attribute, since it is technically not a valid attribute for Roku components
After it is transpiled, MyComponent.xml
should look like this
<component name="MyComponent" extends="Node">
<interface>
<field id="beepNow" type="boolean" alwaysNotify="true" onChange="OnBeepSet" />
</interface>
<script type="text/brightscript" uri="pkg:/path/to/Beep.bs" />
</component>
This way, we can share the functionality in a composable way, but still keep the code base sane enough.
A couple of more details:
- the "behaviours" are supposed to be pieces of functionality, and not to be used as components. This is why we explicitely look for files named
*.part.xml
, to be more intentional with what is a component, and what is a "part". - It's possible to include multiple "parts" in a component, separated by a comma:
<!-- MyComponent.xml -->
<component name="MyComponent" extends="Node" includes="Beep,FadeIn">
</component>
@oninit
is an annotation that can be added to functions, and then these functions will be called in the Init()
function of the component it is in.
Consider the components "parts" from Includes plugin. Sometimes these behaviours need to initialize. Without the @oninit
annotation, components including a part need to be aware of of any functions that we need to call in Init()
. But the reality is, this is an implementation detail that the including component should not be concerned with.
Simply this code:
function Init()
end function
@oninit
function SomeFunction()
end function
would transpile to:
function Init()
SomeFunction() ' auto-generated!
end function
function SomeFunction()
end function
Important implementation details:
- A component including a function annotated with
@oninit
must have anInit()
function.- If needed, add an empty
Init()
function where@oninit
functions will be called
- If needed, add an empty
- An
Init()
function that will be calling@oninit
functions must be included in only one component- Let's explain why with an example:
- We have two components:
ComponentA
andComponentB
- We have a script named
MyCompScript.bs
which has anInit()
function - We have a script named
MyBehaviour.bs
which has a function namedDoSomethingOnInit()
annotated with@oninit
ComponentA
includesMyCompScript.bs
andMyBehaviour.bs
ComponentB
includes onlyMyCompScript.bs
- With this setup, the compiler will call
DoSomethingOnInit()
in theInit()
function ofMyCompScript.bs
ComponentA
would work fineComponentB
will have an error, since it doesn't includeMyBehaviour.bs
, but callsDoSomethingOnInit()
- We have two components:
- For simplicity, the convention in this code base is to always keep the
Init()
function of a component in a bs file of the same name.- For example, the
Init()
function ofMyComponent.xml
would live inMyComponent.bs
- For example, the
- Let's explain why with an example:
- functions with
@oninit
must have a name. - functions with
@oninit
must have no arguments.
This plugin allows us to "bind" different node references together. It's a simple form of dependency injection to manage dependencies between nodes.
Consider a real (but simplified) example from Playlet Lib:
<ApplicationInfo id="ApplicationInfo" />
<Invidious id="Invidious" />
<PlayletWebServer
id="WebServer"
port="8888" />
The Invidious
node is able to provide a link that will be used to login. Let's assume it is something like:
http://192.168.1.2:8888/invidious/login
Invidious
needs to ask PlayletWebServer
for the server url, and append /invidious/login
.
To create the url, PlayletWebServer
needs to get the local ip address from ApplicationInfo
, to create a string in the form of http://IP_ADDRESS:PORT
.
These different dependencies need to be managed somehow. Without plugins, a combintion of the following would be necessary:
m.top.findNode
(see ifSGNodeDict.findNode)- This is useful to find child nodes, but it is tricky to find parent or sibling nodes
- using the attribute
role
(see Defining SceneGraph components)- This is also shortcut for assigning a child node to a field, and is pretty limited
- using the attribute value starting with
dictionary:
(see Defining SceneGraph components)- The
dictionary:
is close in functionality to what we want, but it is heavily dependent on the order of the defined nodes, and pretty limited in terms of what it can do.
- The
With this plugin, it is simple to assign the dependencies on each node by using special annotations:
<ApplicationInfo id="ApplicationInfo" />
<Invidious id="Invidious"
webServer="bind:../WebServer" />
<PlayletWebServer
id="WebServer"
port="8888"
applicationInfo="bind:../ApplicationInfo" />
And now Invidious
has a reference to the WebServer
, and the WebServer
has a reference to the ApplicationInfo
.
Nodes can reference other nodes they depend on by describing the path to reach said nodes.
An important thing to know about nodes lifecycle is that child node get created and initialized first. This means when Init()
gets called, the children of a node are ready, but the node itself is not attached to its parent yet.
This makes it difficult to prepare references between nodes at Init()
time, if the nodes are siblings for example. This is why we introduce a different lifecycle event to nodes: OnNodeReady()
.
With that in mind, let's dive into it.
In the Includes plugin, we talked about how to include different parts into a component. For binding, the part AutoBind
is required, since it contains the logic to do binding.
<component name="MyComponent" extends="Group" includes="AutoBind">
</component>
The plugin introduces two types of binds:
- In Component field references
This is done at the declaration level of a field in a node. For example:
<component name="MyComponent" extends="Group" includes="AutoBind">
<interface>
<!-- Tries to find a sibling to the current node, with a node ID "AppController" -->
<field id="appController" type="node" bind="../AppController" />
</interface>
</component>
- In Component child declarations. For example
<component name="MyComponent" extends="Group" includes="AutoBind">
<children>
<!-- Tries to find a sibling to Node1, with a node ID "Node2" -->
<Group id="Node1" field1="bind:../Node2" />
<!-- Tries to find a sibling to Node2, with a node ID "Node1" -->
<Group id="Node2" field1="bind:../Node1" />
</children>
</component>
Note that even though it looks like this is not valid xml declarations, the plugin will take care of transpiling to correct SceneGraph components.
Paths are used in both types of bindings are used to describe the relative (or absolute) of the nodes. They are very similar to unix file system paths.
.
indicates the current node..
indicates the parent node/
indicates the scene node when used at the start of the path- Anything else indicates the ID of the node we're looking for.
This means /Node1/../Node2/./Node3
would translate to:
node = m.top.getScene() ' "/"
node = node.findNode("Node1") ' "Node1"
node = node.getParent() ' ".."
node = node.findNode("Node2") ' "Node2"
node = node ' "."
node = node.findNode("Node3") ' "Node3"
Note that since we use findNode
, it means finding a node by ID can dig multiple levels into a node.
Let's go step by step on what happens at compile time, and at runtime.
Consider this component:
<component name="MyComponent" extends="Group" includes="AutoBind">
<interface>
<field id="appController" type="node" bind="../AppController" />
</interface>
<children>
<Group id="Node1" field1="bind:../Node2" />
<Group id="Node2" field1="bind:../Node1" />
</children>
</component>
First, the plugin generate a structure describing the bindings that need to be done, and generates this function
@oninit
function InitializeBindings()
m.top.bindings = {
fields: {
"appController": "../AppController"
},
childProps: {
"Node1": {
"field1": "../Node2"
},
"Node2": {
"field1": "../Node1"
}
}
}
end function
Notice how the function has @oninit
, which means it will be called in the Init()
function.
When the bindings
field is set, the node registers itself to a global array of nodes that need binding.
Next, once all nodes are registered, we need to call a function called AutoBindSceneGraph()
. This need to be called once from the root node's Init()
(concretely, here)
AutoBindSceneGraph()
will loop through all the nodes that require binding, and use the information we assigned to m.top.bindings
of each node to populate all the needed references.
Next, all fields of type node will be added to the component scope m
. This means if you have a field named field1
, you can access it with m.field1
in addition to m.top.field1
. This is just for conveninece.
Finally, the AutoBindSceneGraph()
will signal to all AutoBind
nodes that bindings are ready, by setting the field binding_done
, and calling the function OnNodeReady()
So in most cases, we use Init()
as a first step of initialization where all child nodes are ready, but not all dependencies, while OnNodeReady()
indicates that everything is ready for use, and that nodes can call dependency nodes.
We've talked about nodes that are mostly declared in the xml
files. What about dynamically created nodes with the CreateObject
function? There's a pattern for that as well:
' Create a component
myNode = CreateObject("roSGNode", "MyComponent")
' Add it to the right parent
myParentNode.appendChild(myNode)
' Trigger the binding manually
myNode@.BindNode()
calling BindNode
will trigger all the binding steps just for this node. But it is expected that all its dependencies are already in the scene and can be found.
This plugin generates Task
components a scripts that makes it simpler to call a function in a background thread, and get the result in a callback function.
To implement functionality that runs in background threads, Tasks must be used.
Setting up a task involves lots of boilerplate: setting up the task in an xml file, defining inputs and outputs, code to listen for a task to finish, and so on.
Additionally, there's no standard way to do error handling, cancellation, and so on.
Use a function with the attribute @asynctask
.
In a separate file, define the task function:
@asynctask
function MyBackgroundTask(input as object) as object
myVar = input.myVar
value = DoWork(myVar)
return {
value: value
}
end function
And use the function like so
import "pkg:/source/AsyncTask/Tasks.bs"
import "pkg:/source/AsyncTask/AsyncTask.bs"
AsyncTask.Start(Tasks.MyBackgroundTask, {myVar: "some input"}, function(output as object) as void
if output.success
print(output.result.value)
end if
end function)
The rest will be taken care of by the plugin, which will generate an xml
file containing the task component, and a script that handles calling the function.
Although very useful, this pattern might not be the best for long running tasks (like the Web Server) or other tasks that require task reuse. This is because AsyncTask.Start
create a new instance of the task every time.
By now, you realize we make a lot of transformations to the source code through many plugins. As a sanity check, this plugin allows us to keep track of the transpiled files, so we can be sure that the output is what we expect.
This is useful to catch errors with the plugin usage, of bugs with the plugins themselves.
When transpiling a file, say MyComponent.xml
, the plugin will see if there is a MyComponent.transpiled.xml
right next to it in the source.
If a file is found, then the transpiled file of MyComponent.xml
will be copied back from the staging folder to MyComponent.transpiled.xml
.
Additionally, if there's a folder MyFolder
and MyFolder.transpiled
in the source, then all transpiled files of MyFolder
will also be copied back to MyFolder.transpiled
. Keeping track of entire folders is primarily to track of transpiled files of test nodes and scripts.
To reduce the noise of this plugin, it only runs when we're making a test build.
Because we continiously make test build in Github actions, the tracked transpilied files will be automatically added to PRs, so that we notice if outputs do not look right.
The logger plugin does some code transformation to log calls, which can add caller path at compile time, or the filename, based on debug/release configuration.
The logger plugin searches for calls to the predefined functions LogError
, LogWarn
, LogInfo
and LogDebug
. Once found, it does the following (we take LogError
as an example):
- It replaces calls of
LogError
withLogErrorX
where X is the number of arguments.- For example, it replaces
LogError("Cannot complete operation:", error)
withLogError2("Cannot complete operation:", error)
- For example, it replaces
- Based on usage, it generates
LogErrorX
functions used. Example:
function LogError2(arg0, arg1) as void
logger = m.global.logger
if logger.logLevel < 0
return
end if
m.global.logger.logLine = "[ERROR]" + ToString(arg0) + " " + ToString(arg1)
end function
- Finally, it adds the caller file and line number to the args.
LogError("Cannot complete operation:", error)
is replaced withLogError(CALLER, "Cannot complete operation:", error)
CALLER
would be the full path to the source .bs file and line number, for easier debugging- In release mode,
CALLER
would be only the file name and line number, without the full path. This is so that dev directory does not get "leaked" in releases.
This is implemented to make it easy to identify where logs came from, and to generate log functions with the right number of parameters based on usage. This way we do not need all combination of functions (LogError
, LogWarn
, LogInfo
and LogDebug
multiplied by number of parameters, LogError1
, LogError2
, LogError3
, and so on). Since brightscript does not support function templates or function overloading, it is either this method, or use optional parameters (but then we would have to check which parameter was passed, and which was is using the default value.) and end up with a log function with logs of if's else'es.
To understand more about what's being generated, download playlet-lib.zip
from releases, extract, and inspect the file source/utils/Logging.brs
. You will see it contains log functions in the form LogVerbX
, that are generated based on usage.
To give credit where credit is due, some of the ideas here (like adding the source location to the log) came from roku-log
I found myself implementing type checking functions like IsBool
, IsInt
, and IsString
, and functions for getting a valid type, like ValidBool
, ValidInt
, and ValidString
. The implementation is the same, why not generate them?
Simply we declare them in an object describing the type, the interface, and the default value, and generate all corresponding functions.
Sample output:
' Returns true if the given object is of type Bool, false otherwise
function IsBool(obj as dynamic) as boolean
return obj <> invalid and GetInterface(obj, "ifBoolean") <> invalid
end function
' Returns the given object if it is of type Bool, otherwise returns the default value `false`
function ValidBool(obj as dynamic) as boolean
if obj <> invalid and GetInterface(obj, "ifBoolean") <> invalid
return obj
else
return false
end if
end function
Handles routing annotations (e.g. `@get("/api/something")) so that they are handled by the routers/middlewares. It's nicer than having to register things by hand.
Routers need to inherit from Http.HttpRouter
. Then, once a request handled is annotated, e.g.
@get("/")
function GoHome(context as object) as boolean
response = context.response
response.Redirect("/index.html")
return true
end function
The route and the handler are added to the router. In this example, the /
route redirects to index/html
.
There one special annotation, and one special route:
@all
matches all methods*
matches all paths
So @all(*)
would be a middleware that will be matched with any request.
Roku's built-in localization system has some undesired side effects: for example, it might translate node ids, which would break the relationship between the nodes. This plugin enforces some rules to ensure translation files are kept up to date with code implementation. See the next section for the validation rules.
The plugin checks for the following:
- Translated words and sentenses are defined in an enum annotated with
@locale
- All the values in the
@locale
enums must be provided in the English (en_US
) translation. This is becauseen_US
is the fallback translation when the current localte translations are not complete. en_US
must have matching source and translation values, so it can act as a fallback translation.- All
source
translations from all languages must be present in the@locale
enums. This is to prevent renaming the values in the code without updating translation files. - Translation
source
keys can't be used except for certain fields such as"text"
and"title"
. This is to prevent the accidental localization of non-display fields, such as node ids.
To generate BrighterScript encoding/decoding functions for Protobuf types. These are needed for some functionality (e.g. Search filters in Innertube backend)
The plugin scans for .proto
files, and parse them into plain objects using protocol-buffers-schema
Then the right functions that get generated to serialize/deserialize types.
Please note this generator is VERY bare bone: only features necessary for Playlet were implemented.
The generation can be improved as needed.