Skip to content

Commit

Permalink
Browse files Browse the repository at this point in the history
 into release-1.0.0
  • Loading branch information
TwitchBronBron committed Nov 25, 2024
2 parents 2a6afd9 + f6dedf7 commit 57fa2ad
Show file tree
Hide file tree
Showing 14 changed files with 1,013 additions and 153 deletions.
4 changes: 2 additions & 2 deletions .github/workflows/build.yml
Original file line number Diff line number Diff line change
Expand Up @@ -27,8 +27,8 @@ jobs:
- run: npm run lint
- run: npm run test
- run: npm run publish-coverage
# disable this task until we have v1-suported versions of the projects to test
# - run: npm run test-related-projects
# this is stalling out for some reason, so disable it for now
#- run: npm run test-related-projects
npm-release:
#only run this task if a tag starting with 'v' was used to trigger this (i.e. a tagged release)
if: startsWith(github.ref, 'refs/tags/v')
Expand Down
10 changes: 10 additions & 0 deletions CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -358,6 +358,16 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
## [0.66.0-alpha.0](https://github.com/rokucommunity/brighterscript/compare/v0.65.1...v0.66.0-alpha.0) - 2023-06-09
### Changed
- all the type tracking stuff!



## [0.68.0](https://github.com/rokucommunity/brighterscript/compare/v0.67.8...v0.68.0) - 2024-11-21
### Changed
- Fix issues with the ast walkArray function ([#1347](https://github.com/rokucommunity/brighterscript/pull/1347))
- Optimize ternary transpilation for assignments ([#1341](https://github.com/rokucommunity/brighterscript/pull/1341))



## [0.67.8](https://github.com/rokucommunity/brighterscript/compare/v0.67.7...v0.67.8) - 2024-10-18
### Changed
- upgrade to [roku-deploy@3.12.2](https://github.com/rokucommunity/roku-deploy/blob/master/CHANGELOG.md#3122---2024-10-18). Notable changes since 3.12.1:
Expand Down
32 changes: 32 additions & 0 deletions docs/plugins.md
Original file line number Diff line number Diff line change
Expand Up @@ -273,6 +273,38 @@ export interface CompilerPlugin {
afterProvideReferences?(event: AfterProvideReferencesEvent): any;


/**
* Called before the `provideDocumentSymbols` hook
*/
beforeProvideDocumentSymbols?(event: BeforeProvideDocumentSymbolsEvent): any;
/**
* Provide all of the `DocumentSymbol`s for the given file
* @param event
*/
provideDocumentSymbols?(event: ProvideDocumentSymbolsEvent): any;
/**
* Called after `provideDocumentSymbols`. Use this if you want to intercept or sanitize the document symbols data provided by bsc or other plugins
* @param event
*/
afterProvideDocumentSymbols?(event: AfterProvideDocumentSymbolsEvent): any;


/**
* Called before the `provideWorkspaceSymbols` hook
*/
beforeProvideWorkspaceSymbols?(event: BeforeProvideWorkspaceSymbolsEvent): any;
/**
* Provide all of the workspace symbols for the entire project
* @param event
*/
provideWorkspaceSymbols?(event: ProvideWorkspaceSymbolsEvent): any;
/**
* Called after `provideWorkspaceSymbols`. Use this if you want to intercept or sanitize the workspace symbols data provided by bsc or other plugins
* @param event
*/
afterProvideWorkspaceSymbols?(event: AfterProvideWorkspaceSymbolsEvent): any;


onGetSemanticTokens?: PluginHandler<OnGetSemanticTokensEvent>;
//scope events
onScopeValidate?(event: OnScopeValidateEvent): any;
Expand Down
109 changes: 80 additions & 29 deletions docs/ternary-operator.md
Original file line number Diff line number Diff line change
@@ -1,20 +1,61 @@
# Ternary (Conditional) Operator: ?
The ternary (conditional) operator is the only BrighterScript operator that takes three operands: a condition followed by a question mark (?), then an expression to execute (consequent) if the condition is true followed by a colon (:), and finally the expression to execute (alternate) if the condition is false. This operator is frequently used as a shortcut for the if statement. It can be used in assignments, and in any other place where an expression is valid. Due to ambiguity in the brightscript syntax, ternary operators cannot be used as standalone statements. See the [No standalone statements](#no-standalone-statements) section for more information.
The ternary (conditional) operator takes three operands: a condition followed by a question mark (?), then an expression to execute (consequent) if the condition is true followed by a colon (:), and finally the expression to execute (alternate) if the condition is false. This operator is frequently used as a shorthand version of an if statement. It can be used in assignments or any place where an expression is valid. Due to ambiguity in the brightscript syntax, ternary operators cannot be used as standalone statements. See the [No standalone statements](#no-standalone-statements) section for more information.

## Warning
<p style="background-color: #fdf8e3; color: #333; padding: 20px">The <a href="https://developer.roku.com/docs/references/brightscript/language/expressions-variables-types.md#optional-chaining-operators">optional chaining operator</a> was added to the BrightScript runtime in <a href="https://developer.roku.com/docs/developer-program/release-notes/roku-os-release-notes.md#roku-os-110">Roku OS 11</a>, which introduced a slight limitation to the BrighterScript ternary operator. As such, all ternary expressions must have a space to the right of the question mark when followed by <b>[</b> or <b>(</b>. See the <a href="#">optional chaning</a> section for more information.
</p>

## Basic usage
The basic syntax is: `<condition> ? <value if true> : <value if false>`.

Here's an example:
```BrighterScript
authStatus = user <> invalid ? "logged in" : "not logged in"
authStatus = user2 <> invalid ? "logged in" : "not logged in"
```

transpiles to:

```BrightScript
authStatus = bslib_ternary(user <> invalid, "logged in", "not logged in")
if user2 <> invalid then
authStatus = "logged in"
else
authStatus = "not logged in"
end if
```

As you can see, when used in the right-hand-side of an assignment, brighterscript transpiles the ternary expression into a simple `if else` block.
The `bslib_ternary` function checks the condition, and returns either the consequent or alternate. In these optimal situations, brighterscript can generate the most efficient code possible which is equivalent to what you'd write yourself without access to the ternary expression.

This also works for nested ternary expressions:
```BrighterScript
result = true ? (true ? "one" : "two") : "three"
```

transpiles to:

```BrightScript
if true then
if true then
result = "one"
else
result = "two"
end if
else
result = "three"
end if
```

## Use in complex expressions
Ternary can also be used in any location where expressions are supported, such as function calls or within array or associative array declarations. In some situations it's much more difficult to convert the ternary expression into an `if else` statement, so we need to leverage the brighterscript `bslib_ternary` library function instead. Consider the following example:

```BrighterScript
printAuthStatus(user2 <> invalid ? "logged in" : "not logged in")
```

transpiles to:

```BrightScript
printAuthStatus(bslib_ternary(user2 <> invalid, "logged in", "not logged in"))
```

The `bslib_ternary` function checks the condition, and returns either the consequent or alternate.
Expand All @@ -24,86 +65,94 @@ There are some important implications to consider for code execution order and s
Consider:

```BrighterScript
a = user = invalid ? "no name" : user.name
printUsername(user = invalid ? "no name" : user.name)
```

transpiles to:
<details>
<summary>View the transpiled BrightScript code</summary>

```BrightScript
a = (function(__bsCondition, user)
printUsername((function(__bsCondition, user)
if __bsCondition then
return "no name"
else
return user.name
end if
end function)(user = invalid, user)
end function)(user = invalid, user))
```
</details>

This code will crash because `user.invalid` will be evaluated.

To avoid this problem the transpiler provides conditional scope protection, as discussed in the following section.
If we were to transpile this to leverage the `bslib_ternary` function, it would crash at runtime because `user.name` will be evaluated even when `user` is `invalid`. To avoid this problem the transpiler provides conditional scope protection, as discussed in the following section.

## Scope protection

For conditional language features such as the _conditional (ternary) operator_, BrighterScript sometimes needs to protect against unintended performance hits.

There are 2 possible ways that your code can be transpiled:
Sometimes BrighterScript needs to protect against unintended performance hits. There are 3 possible ways that your code can be transpiled:

### Simple
In this situation, BrighterScript has determined that both the consequent and the alternate are side-effect free and will not cause rendezvous. This means BrighterScript can use a simpler and more performant transpile target.
### Optimized `if else` block
In this situation, brighterscript can safely convert the ternary expression into an `if else` block. This is typically available when ternary is used in assignments.

```BrighterScript
a = user = invalid ? "not logged in" : "logged in"
m.username = m.user <> invalid ? m.user.name : invalid
```

transpiles to:

```BrightScript
a = bslib_ternary(user = invalid, "not logged in", "logged in")
if m.user <> invalid then
m.username = m.user.name
else
m.username = invalid
end if
```

### BrighterScript `bslib_ternary` library function
In this situation, BrighterScript has determined that both the consequent and the alternate are side-effect free and will not cause rendezvous or cloning, but the code was too complex to safely transpile to an `if else` block so we will leverage the `bslib_ternary` function instead.

```BrighterScript
a = user = invalid ? defaultUser : user
printAuthStatus(user = invalid ? "not logged in" : "logged in")
```

transpiles to:

```BrightScript
a = bslib_ternary(user = invalid, defaultUser, user)
printAuthStatus(bslib_ternary(user = invalid, "not logged in", "logged in"))
```


### Scope capturing
In this situation, BrighterScript has detected that your ternary operation will have side-effects or could possibly result in a rendezvous. BrighterScript will create an immediately-invoked-function-expression to capture all of the referenced local variables. This is in order to only execute the consequent if the condition is true, and only execute the alternate if the condition is false.
In this situation, BrighterScript has detected that your ternary operation will have side-effects or could possibly result in a rendezvous, and could not be transpiled to an `if else` block. BrighterScript will create an [immediately-invoked-function-expression](https://en.wikipedia.org/wiki/Immediately_invoked_function_expression) to capture all of the referenced local variables. This is in order to only execute the consequent if the condition is true, and only execute the alternate if the condition is false.

```BrighterScript
a = user = invalid ? "no name" : user.name
printUsername(user = invalid ? "no name" : user.name)
```

transpiles to:

```BrightScript
a = (function(__bsCondition, user)
printUsername((function(__bsCondition, user)
if __bsCondition then
return "no name"
else
return user.name
end if
end function)(user = invalid, user)
end function)(user = invalid, user))
```

```BrighterScript
a = user = invalid ? getNoNameMessage(m.config) : user.name + m.accountType
printMessage(user = invalid ? getNoNameMessage(m.config) : user.name + m.accountType)
```

transpiles to:

```BrightScript
a = (function(__bsCondition, getNoNameMessage, m, user)
printMessage((function(__bsCondition, getNoNameMessage, m, user)
if __bsCondition then
return getNoNameMessage(m.config)
else
return user.name + m.accountType
end if
end function)(user = invalid, getNoNameMessage, m, user)
end function)(user = invalid, getNoNameMessage, m, user))
```

### Nested Scope Protection
Expand All @@ -114,17 +163,18 @@ m.increment = function ()
m.count++
return m.count
end function
result = (m.increment() = 1 ? m.increment() : -1) = -1 ? m.increment(): -1
printResult((m.increment() = 1 ? m.increment() : -1) = -1 ? m.increment(): -1)
```

transpiles to:

```BrightScript
m.count = 1
m.increment = function()
m.count++
return m.count
end function
result = (function(__bsCondition, m)
printResult((function(__bsCondition, m)
if __bsCondition then
return m.increment()
else
Expand All @@ -136,7 +186,7 @@ result = (function(__bsCondition, m)
else
return -1
end if
end function)(m.increment() = 1, m)) = -1, m)
end function)(m.increment() = 1, m)) = -1, m))
```


Expand All @@ -155,7 +205,7 @@ end function
```

## No standalone statements
The ternary operator may only be used in expressions and may not be used in standalone statements because the BrightScript grammer uses `=` for both assignments (`a = b`) and conditions (`if a = b`)
The ternary operator may only be used in expressions, and may not be used in standalone statements because the BrightScript grammer uses `=` for both assignments (`a = b`) and conditions (`if a = b`)

```brightscript
' this is generally not valid
Expand All @@ -173,6 +223,7 @@ This expression can be interpreted in two completely separate ways:
```brightscript
'assignment
a = (myValue ? "a" : "b'")
'ternary
(a = myValue) ? "a" : "b"
```
Expand Down
2 changes: 1 addition & 1 deletion scripts/compile-doc-examples.ts
Original file line number Diff line number Diff line change
Expand Up @@ -197,7 +197,7 @@ class DocCompiler {
const program = new Program({
rootDir: `${__dirname}/rootDir`,
files: [
'source/main.brs'
'source/main.bs'
],
//use the current bsconfig
...(this.bsconfig ?? {})
Expand Down
78 changes: 75 additions & 3 deletions src/astUtils/creators.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,9 +3,9 @@ import type { Identifier, Token } from '../lexer/Token';
import type { SGToken } from '../parser/SGTypes';
import { SGAttribute, SGComponent, SGInterface, SGInterfaceField, SGInterfaceFunction, SGScript } from '../parser/SGTypes';
import { TokenKind } from '../lexer/TokenKind';
import type { Expression } from '../parser/AstNode';
import { CallExpression, DottedGetExpression, FunctionExpression, LiteralExpression, VariableExpression } from '../parser/Expression';
import { Block, MethodStatement } from '../parser/Statement';
import type { Expression, Statement } from '../parser/AstNode';
import { LiteralExpression, CallExpression, DottedGetExpression, VariableExpression, FunctionExpression } from '../parser/Expression';
import { AssignmentStatement, Block, DottedSetStatement, IfStatement, IndexedSetStatement, MethodStatement } from '../parser/Statement';

const tokenDefaults = {
[TokenKind.BackTick]: '`',
Expand Down Expand Up @@ -263,3 +263,75 @@ export function createSGScript(attributes: { type?: string; uri?: string }) {
startTagClose: { text: '/>' }
});
}

export function createIfStatement(options: {
if?: Token;
condition: Expression;
then?: Token;
thenBranch: Block;
else?: Token;
elseBranch?: IfStatement | Block;
endIf?: Token;
}) {
return new IfStatement(
{
if: options.if ?? createToken(TokenKind.If),
condition: options.condition,
then: options.then ?? createToken(TokenKind.Then),
thenBranch: options.thenBranch,
else: options.else ?? createToken(TokenKind.Else),
elseBranch: options.elseBranch,
endIf: options.endIf ?? createToken(TokenKind.EndIf)
}
);
}

export function createBlock(options: { statements: Statement[] }) {
return new Block(options);
}

export function createAssignmentStatement(options: {
name: Identifier | string;
equals?: Token;
value: Expression;
}) {
return new AssignmentStatement({
equals: options.equals ?? createToken(TokenKind.Equal),
name: typeof options.name === 'string' ? createIdentifier(options.name) : options.name,
value: options.value
});
}

export function createDottedSetStatement(options: {
obj: Expression;
dot?: Token;
name: Identifier | string;
equals?: Token;
value: Expression;
}) {
return new DottedSetStatement({
obj: options.obj,
name: typeof options.name === 'string' ? createIdentifier(options.name) : options.name,
value: options.value,
dot: options.dot,
equals: options.equals ?? createToken(TokenKind.Equal)
});
}

export function createIndexedSetStatement(options: {
obj: Expression;
openingSquare?: Token;
indexes: Expression[];
closingSquare?: Token;
equals?: Token;
value: Expression;
}) {
return new IndexedSetStatement({
obj: options.obj,
indexes: options.indexes,
value: options.value,
openingSquare: options.openingSquare ?? createToken(TokenKind.LeftSquareBracket),
closingSquare: options.closingSquare ?? createToken(TokenKind.RightSquareBracket),
equals: options.equals ?? createToken(TokenKind.Equal)
});
}
Loading

0 comments on commit 57fa2ad

Please sign in to comment.