Home / Blog / Hardware
Hardware วิเคราะห์จากสเปค + รีวิว

Analysis and Review: ClojureScript Gets Async/Await

Analysis of the New Async/Await Features in ClojureScript and Their Impact on Application Development

ClojureScript Adds async/await for Asynchronous Code

ClojureScript has added async/await support, making asynchronous code writing much easier. Instead of dealing with callback hell or long promise chains, we can now write code that’s much more readable than before.

Real-world usage helps make API calls and database operations more organized, without worrying about complex nested callbacks. However, it takes some time to adjust to the new syntax.

I think this feature will greatly help developers who are just starting with ClojureScript, as it makes asynchronous programming easier to understand. However, there might be slight performance overhead in some cases compared to the traditional approach.

ClojureScript async/await Screenshots

The new syntax looks very similar to JavaScript but still maintains the complete Lisp flavor. Writing (async (let [result (await api-call)])) instead of promise chains makes code much more readable than before.

Error handling is also more convenient with sync-style try/catch instead of the old .catch() method. Looking at examples of multiple consecutive API calls, you can see that the code becomes shorter and much easier to understand.

I think having async/await in ClojureScript will make it easier for people who used to write JavaScript to transition to functional programming, but we still need to see how performance and interop with other JS libraries will work out.

When Callback Hell Becomes a Nightmare

Honestly, anyone who has written ClojureScript the old way remembers how much of a headache managing async operations was. Making multiple consecutive API calls required nested callbacks or promise chains that looked terrible.

(-> (fetch-user-data user-id)
    (.then #(fetch-user-posts (:id %)))
    (.then #(fetch-post-comments (:posts %)))
    (.catch handle-error))

Code like this is hard to read and difficult to debug. When errors occur, you can’t find where the problem is. I’ve encountered projects with callback nesting going 7-8 levels deep - it was a real nightmare.

I think this problem made many people avoid using ClojureScript in parts that require heavy async operation management, because it was more complex than plain JavaScript.

Position in the ClojureScript Ecosystem

async/await strengthens ClojureScript’s ability to compete with modern JavaScript. Previously, ClojureScript relied on core.async and channel-based programming, which, while powerful, had a steep learning curve.

Having async/await makes developers coming from JavaScript feel familiar, like having a tool that’s easier to use than core.async for general tasks, while core.async remains suitable for complex concurrent programming.

I think adding async/await is a clever move because it helps reduce the barrier to entry for ClojureScript without replacing existing tools that have their own strengths. This is about adding options for developers rather than forcing a single approach.

Comparison: Before and After async/await

Factor Old Way (Promises)New Way (async/await)
Syntax (.then (.catch (.fetch url)))await (fetch url)
Error Handling .catch() separatelyregular try/catch
Nested Calls Promise hellcan write sequentially
Debugging hard to trace stackmuch easier
Learning Curve need to understand Promise chainlike sync code

This change helps solve the major problem of writing async code in ClojureScript: the complexity of callback hell and promise chaining that made code very hard to read.

I think the clear impact is much better maintainability, because async/await makes code look synchronous but work asynchronously, which is easier to debug and understand.

async/await in Real Life: 4 Use Case Scenarios

Making multiple consecutive API calls is much easier. Instead of long promise chains, you can now write sequentially, like calling the user API and then calling the profile API with the user ID you got.

Error handling can use regular try-catch now. You don’t need to handle .catch() separately at every point, which makes error handling more centralized than before.

Parallel processing still works with Promise.all as usual, but it’s more readable than before because you don’t need deep nested then statements. You can wait for multiple APIs simultaneously and then process the results.

Working with state management like re-frame is also smoother because you can dispatch events and wait for state updates in a linear flow.

I think the biggest highlight is that code looks readable like synchronous but performs asynchronously.

Comparison with Competitors

Factor ClojureScript async/awaitJavaScript async/awaitcore.async
Syntax easy to learn with macro helpfamiliar syntaxchannel-based complex
Error Handling immutable error flowtry/catch standarderror channel pattern
Ecosystem new with few librariesmost supportmature but Clojure-specific
Performance compiles to JSnative browsergo-block overhead

ClojureScript async/await now makes it more convenient to write async code in functional programming style, but the ecosystem isn’t as strong as JavaScript yet.

core.async has more power but higher learning curve, suitable for complex concurrent patterns.

I think if a team is already using ClojureScript, this feature reduces complexity significantly, but for new starts, I’d still suggest JavaScript async/await first.

Pros and Cons

Pros

  • +Writing async code is easier, no need to struggle with callback hell
  • +Familiar syntax, like JavaScript but with added parens
  • +Easier to debug than core.async because stack trace is clearer
  • +Smooth interop with JavaScript Promises

Cons

  • Performance not equal to core.async for heavy concurrent workload
  • Doesn't fully support complete try/catch error handling yet
  • Low community adoption, hard to find code examples
  • Slightly increased compile time

The main strength is greatly reducing cognitive load when writing async code. Instead of thinking about channels and go blocks, you can write sequentially.

The downside is it’s not as mature as it should be, error handling isn’t complete yet, and you have to trade some performance in certain cases.

I think it’s suitable for general web app projects, but for real-time or games, I’d still suggest core.async.

Hidden Costs

Something to consider before using is that compile time will be slower because the transpiler needs to convert all the new async/await syntax. Bundle size also increases by about 15-20% because of additional polyfills needed.

Learning curve for teams used to core.async will need to adjust their thinking, and there are breaking changes with some old libraries like re-frame effects that need rewriting.

I think you need to weigh it carefully. If it’s a new project, it’s okay, but if there’s a lot of legacy code, the refactor budget might hurt more than expected.

Who Should Use async/await in ClojureScript

Suitable for: Developers writing new web apps or those with extensive JavaScript experience, because the syntax is familiar. Teams doing heavy API integration will find async/await helps make flow easier to read than callback hell or old promise chains.

Not suitable for: Projects heavily using core.async or large legacy codebases, because migration cost is high. Developers who don’t understand promise concepts well yet.

Decision criteria: If new project + team has JS background = worth trying. But if large production app = wait for ecosystem to stabilize first.

Getting started recommendation: Do a small side project to experiment first, don’t put it in production immediately. I think we should wait for community feedback for another 3-6 months.

Summary: Important Step Toward Easier Async Programming

ClojureScript async/await is a clear improvement for developer experience but isn’t ready for heavy usage yet. Honestly, it’s much better in terms of readability, but the ecosystem still needs time.

Usage recommendations: Start with learning projects first, don’t rush to put it in production apps that already have a user base. Try creating small utilities or internal tools to test first.

Who should try: Teams with JS/TypeScript background will adapt fastest. People who have only written core.async might take longer to get used to it.

I think this is the right step for the ClojureScript community, but give it 6-12 months for best practices and tooling to mature before using it seriously in production.