Observable Notebooks: Promise syntax vs await syntax

Observable Notebooks* support Promise syntax, and the newer ECMAScript async/await syntax. In this post, I explore more comparisons of their syntax, and some pros/cons of each.

Observable supports Promise syntax as shown in their introduction to Promises.

Syntax Examples

await syntax and Promise syntax are not new to Observable notebooks*; the pros and cons apply in modern Javascript too; check out these other resources that compare the two in Javascript. Mozilla’s await documentation also gives an example of equivalent await vs Promise syntax.

TODO: Footnote Don’t confuse Observable Notebooks with “Observables” (Reactive) (RxJS); my blog is referring to Observable Notebooks. If you’re interested in “reactive” Observable *objects vs Promise objects, check out a different article.

However, some nuances are specific to Observable notebooks; The Observable Introduction to Promises gives an example of Promise syntax and an example of await syntax; the examples produce the same result (so they’re equivalent).

Note that both examples begin with an asyc operation using Promise.delay(...); the string "Hello, " is returned after 3 seconds.

Promise syntax example

Promises.delay(3000, "Hello, ").then(greeting => greeting + "world!")

await syntax example

{
  let greeting = await Promises.delay(3000, "Hello, ");
  return greeting + "world!";
}

Promise syntax notes

Implicitly await promises across cell boundaries

Because Observable implicitly awaits promises across cell boundaries, and the return value of the cell is a Promise (Promises.delay(...).then(greeting => greeting + "world!") returns a Promise); Observable gets the awaited result: "Hello, world!":

Observable implicitly awaits promises across cell boundaries, so you often don’t need to deal with a promise directly.

Implicit vs explicit await

Observable implicitly awaits promises, but we could also make it explicit by adding the await keyword in the cell’s expression:

await Promises.delay(3000, "Hello, ").then(greeting => greeting + "world!")

Promise.all

Promise.all is supports parallelism better than await. Note the Mozilla documentation) (emphasis added)

If you wish to safely perform two or more jobs in parallel, you must await a call to Promise.all, or Promise.allSettled.

So parallelism is best supposed by Promise.all, as repeated here

To run promises in parallel, create an array of promises and then use Promise.all(promisesArray).

Error Handling

Note the Mozilla documentation also compares concurrentStart (“CONCURRENT START with await”) vs concurrentPromise (“CONCURRENT START with Promise.all”):

Note that functions concurrentStart and concurrentPromise are not functionally equivalent.

await syntax notes

Using with a code block

The await syntax example uses a code block.

While await does not require a code block. This specific await example does require a code block because it defines local variables using the let keyword. (Observable’s Introduction to Code explains the difference between two types of cell values: expressions and blocks (aka code blocks))

To demonstrate that the await syntax does not require a code block; this cell uses await, but doesn’t assign the result to let greeting; just evaluates with parentheses – no code block required:

(await Promises.delay(3000, "Hello, ")) + "world"

await makes your code “easier to follow”

TODO: Find where this argument.

async keyword is implied

await syntax is designed to work with async functions, according to the ECMAScript specification, and Mozilla:

The await operator… can only be used inside an async function.

So why don’t we use the word async , anywhere in our Observable notebook code, above? It seems like await is allowed at the top-level of Observable and certain Javascript code: an Observable cell expression an Observable cell block Chrome console

But notice when you try to use await inside an explicit function the function must be marked async, or your code will fail

This code fails in Chrome console: new_func = function(){ await new Promise((resolve)=>setTimeout( ()=>resolve('hi'), 3000)) }

Uncaught SyntaxError: await is only valid in async function

This code fails in Observable:

{
  let new_func = function(){ await Promise((resolve)=>setTimeout( ()=>resolve('hi'), 3000)) }
}

SyntaxError: Unexpected token (pointing after the await; before the Promise)

To fix it, add the async keyword before the function. (Remember, Observable’s not Javascript)

Conclusion

Here are some takeaways: