Welcome to ballerina, the effortlessly elegant functional programming framework for frontend web development, with a particular but non-exclusive preference for React.
With special thanks to BLP Digital, where Ballerina has been built:
Ballerina is an advanced framework that really shines when building larger applications with lots of business logic. She is your best ally when you need to build something big and complex, and thanks to her grace and ability she will help you turn this complexity into elegant code.
Ballerina integrates the state of the art in functional programming and even category theory. She is great at these things, so that you don't necessarily need to be. You can just dance with her, and her skills will augment yours and make your work elegant and effective. And if you are a master of functional programming, together you will write code that can be, well, transcendental.
Here is my favorite snippet of code showcasing a timed animation that showcases a timed process with real-clock waiting operations:
export const Child2Animation =
Co.Repeat(
Co.Seq([
Co.SetState(Child2.Updaters.Core.a(replaceWith(1)).then(
Child2.Updaters.Core.b(replaceWith("")))),
Co.Wait(250),
Co.For(Range(0, 3))(
_ =>
Co.Seq([
Co.SetState(Child2.Updaters.Core.a(_ => _ * 2)),
Co.Wait(250),
Co.SetState(Child2.Updaters.Core.b(_ => _ + ".")),
Co.Wait(250),
])
)
])
);
Not only is the code pretty self explanatory, but the alternative implementation of state machines in any frontend framework would look terrible, easily be unmaintainable, and lose all the advantages of scoping, nesting, clear sequencing and parallelism as offered by coroutines, and much more.
And this doesn't even showcase the advanced operators Co.Any
, Co.All
, Co.Await
, or Co.Embed
...
If you want an extensive introduction to everything you can do with Ballerina, take a look at the guidelines.
You might have heard about the hello world syndrome, the terrible sickness that tool makers exploit to show off how their simple libraries make simple tasks simpler, without saying anything about what they do to complex tasks - because the truth is that simple libraries are irrelevant in the grand scheme of things.
Ballerina features tools that help you split up your code into domains, both horizontally and hierarchically. This way you can split your application into epics (horizontal split) which are neatly decomposed into smaller pieces (hierarchical split).
For example:
domains
│
└─parent
│ │
│ └─child1
│ │
│ └─child2
│
└─uncle
Ballerina then helps you partition each subdomain into state, state updaters, asynchronous background operations, APIs, and views with a series of idioms and supporting libraries based on the state of the art in category theory and functional programming.
Ballerina views are currently based on React, but the library is completely framework (and meta-framework) agnostic. Ballerina is light and flexible, and she can work well in any stage: Vue, Angular, jQuery, you name it, she will bring her grace anywhere you want.
Head over to the Playground and start checking out the extensive code of the sample application.
Launch the whole thing with a single command and head over to localhost:8081.
Check out the main.tsx and App.tsx entry points, and then take the plunge into the various domains and also the subdomains.
Create a new domain by running:
yarn create-domain --name child3 --path ./src/domains/parent/domains
and check out the structure it creates for you. Try adding the child3
subdomain to the state of the parent and to render it from the view of the parent. Be curious, experiment, and most of all: enjoy!
Coming soon: npm packages.
You are free to adapt the core for whatever purpose you have in mind. Just head over to the core folder and don't forget to run the core-startup.sh script. Remember that the playground uses npm link
to use the core as a dependency, so make sure to regularly re-run npm link ../core
from the playground folder so that any changes you make to the core are immediately visible from the playground.
You might want to check out some resources in order to discover something about the theoretical foundations of Ballerina. If that's the case, we've got you covered! Here are a load of videos from the same author(s) of the framework:
- a quick React primer might give you some insights on how to interpret React a bit differently than is usually done.
- if you need, brush up your Typescript a bit, it will come in handy.
- finally, learn advanced functional programming and category theory in Typescript and React to see the whole picture.
And something from the brilliant, ballerina-unrelated, Bartosw Milewski: his amazing course on Category Theory for programmers.
Ballerina works well with a monorepo, allowing you to share domain logic between applications.
We recommend yarn classic workspaces if you're working with a react and react native workspace.
To avoid issues with different React versions, you may need to use the no hoist option. You can start by setting nohoist to [**]
which will completely turn off package hoisting, and then work from there.
If you're using Typescript and React in your project, you may also run into type resolution issues when building your own app which looks like TS2300: Duplicate identifier 'ElementType'
.
This appears to be an issue with type resolution in workspaces where typescript does not see the @types/react
module in locally imported packages (i.e. packages in your workspaces) as the same types in your app project.
To fix this, you can add these lines to your compilerOptions of your app's tsconfig.json
file:
"baseUrl": "./",
"paths": { "*": ["node_modules/@types/*", "*"] }
This will direct typescript to attempt to resolve imports first from the root level node_modules/@types
folder before resolving elsewhere.
All dependent packages whose node_modules folder contains the same type definition packages as your project resolve from the root node modules.
I am Giuseppe. I started programming as a teenager, ruinously attempting to build videogames in C++ and DirectX. I did not manage, but I ended up discovering algorithms, linear algebra, complexity, as well as memory leaks and spaghetti code.
This made me realize that maybe, just maybe, I was not going to manage to learn all of this on my own, and that maybe, just maybe, learning from the masters was a humbling but necessary experience. And that I did! I acquired a BSc, then an MSc, and finally why not! Also a PhD in Computer Science at the University of Venice.
During my studies I discovered functional programming, and how code does not need to be broken and miserable all the time. And for a while, that was it.
I started teaching in the Netherlands, at RUAS in Rotterdam, where I kinda quickly became an Associate Professor (best translation I can think of for the word hoofddocent). Teaching is one of my passions, but I also like to build stuff, so I became CTO of Hoppinger in Rotterdam/Amsterdam, building web apps. CTOing is one of my passions, but I also like to build complex stuff, so I also became Senior Architect at BLP in Zurich, building a mega complicated SaaS app with machine learning and super big multi-tenant integrations with ERPs all over Europe.
During my career, I always tried to keep things in balance: code can be beautiful and elegant (Ballerina!!!) but at some point the software needs to be shipped with happy clients and happy business, buuuuuuut three years from now the code needs to still be as elegant and clean as when ti was shipped the first time because otherwise you die from the weight of your technical debt.
This open source repository has been entirely built during my tenure at BLP, together with BLP' amazing frontend team. A very heartfelt shout out goes to the brilliant David and Konrad: without you guys Ballerina would not exist. Have some ballerina shoes as a token of my appreciation: 🩰 and 🩰.
BLP is the only contributor and sponsor of the Ballerina framework, but it would be unfair to not mention that Ballerina stands on the shoulders of giants. One giant in particular is Hoppinger, where the previous iterations of React wrappers and coroutines have been extensively studied and played with:
A lot of little thankful ballerina shoes and hearts go to 🩰 Francesco ❤️, 🩰 Lucas ❤️, and 🩰 Marielle ❤️, as well as countless others who provided the critique and inspirations that gave me the knowledge to eventually build Ballerina at BLP. I would not be where I am without you.