Track habits and gamify them in a tycoon-game. Started with a web app, now working on an iOS mobile app.
Fullstack software engineer
I'm a detail-oriented software engineer with 6 years of experience in fullstack web development. I'm preferably targeting roles involving React, TypeScript and Go with an opportunity to transfer to the US.
Read about my professional contributions, skills, hobby project and more below or see my CV.
I have driven increased product demand by developing a calculator that helps founders model unpriced funding rounds. See more... The solution involved an RFC-like process to come up with the best solution as this was the first public page (i.e. not behind auth, obviously excluding a sign in/sign up flow). At the time we were also in a rebranding transition where the homepage was already transitioned and the idea was to use the new design on all public facing pages so the calculator should also follow the new design. It uses many primitive components and other more complex components some of which I helped building. Check it out for yourself!
I enabled better sales offering and streamlined customer billing management by revamping billing in a Ruby on Rails app integration with Stripe and HubSpot. See more... This was the longest running project I worked on and wrote around 90% of it when it comes solely to the changes in the app codebase. First step was a migration from one Stripe instance to another. The next step involved changing how the app is interacting with Stripe where previously it was the app recording changes to customer subscriptions using Stripe's API and now the idea was to make these changes from the self-service customer portal and update the app data via webhook events from Stripe. It also involved a change to the pricing packages where there was previously a single package and additional customer features were enabled on an ad-hoc basis and now there would be multiple packages with certain pre-enabled features.
This project helps early-stage startups in fundraising and offering them inexpensive assistance on tax relief to ensure their success. See more... Services included are round modelling calculator, creating and filing advance assurance, SEIS & EIS, and templates for ASAs & SAFEs. Major challenge in this project was a tight deadline that was agreed on few months in advance and that had to be met which I achieved without sacrificing quality.
The in-app workflows I implemented increased number of requests for an additional paid service from ~5 to 20+ a month increasing revenue in this area by 300%. See more... These included a backoffice part, where our colleagues fulfilling the request were able to manage and update progress on the requests, and the end customer part, where customers were able to request these services.
I provided affordable currency exchange rate solution for 200+ currencies. See more... While the app had a multi-currency support from the time I joined, the currencies that were supported were those available from European Central Bank which contains 30 currencies. Meanwhile, there are 180 currently circulating currencies recognized by UN and some of those started to appear in the app and it became cumbersome to deal with them manually. I used Exchangerates API to get a more comprehensive list of the exchange rates. As we didn't call the API everytime we needed the exchange rate but instead stored it in a database (we also needed historical exchange rates in some cases), new exchange rates were automatically added to the database on a daily basis.
Addressed invalid links by automatically checking and reporting them. See more... An external link can change and suddenly you're linking to a non-existent page. At some point we encountered a few links that were invalid and instead of just fixing those and moving on with our life I wrote a script to check external links that I set up to run on GitHub Actions on a weekly basis and send a result to an #automated-checks Slack channel. In this way, these issues became more visible and otherwise invalid links that would be in the app for many months were fixed within a week.
Improved both UX and DX by improving a modal component. See more... The way we were writing most of the modals was by directly using the
design system components as opposed to a more complete modal
implementation. Every colleague therefore had a lot of freedom when it
came to implementation of a modal. Properly implementing a modal so
that it's consistent with design and behaves in the least obstructive
way possible then becomes hard.
Our modals were incosistent with how disabled and loading states
were handled after clicking cancel and submit buttons, layout of
those buttons was sometimes different and when it came to modals
with forms, the form being reset after a successful submit caused a
flicker while the modal was fading out which was visually
distracting.
I implemented a modal component to make implementing consistent
modals simpler, including support for forms, where all of the
aforementioned concerns were handled by default. These models can be
seen in action in the funding rounds calculator above.
Maintained product quality by implementing data integrity checks. See more... My colleague implemented a framework for adding data integrity checks to notify us of instances where invalid data got into the database and also automatically fix them if an automatic fix is possible. I wrote checks dealing with detecting and autofixing trailing whitespace in attributes like names, emails and domain specific identifiers which could cause subtle bugs in filtering and matching items.
Before
key :required, [:id, :created_at] property :id do key :type, :integer end property :created_at do key :type, :string key :format, 'date-time' end property :status do key :'$ref', :StatusEnum end property :values do items do key :type, :number end end
After
DocsApiHelper.properties({ id: :integer, created_at: { type: :string, format: 'date-time' }, status?: :StatusEnum, values?: { type: :number, array: true } })
"This::Constant::Exists" ... "But::ThisOne::DoesNot"
Maintained coding conventions and simplified codebase using linting rules as well as prevented subtle bugs. This involved playing around with ASTs. I made linter rules for both frontend (ESLint) and backend (RuboCop). Frontend one was about linting a stylistic error to follow our coding conventions (i.e. no more pointing it out in code reviews). In the backend, I made two linting rules which are a tad more complex:
Before
key :required, [:id, :created_at] property :id do key :type, :integer end property :created_at do key :type, :string key :format, 'date-time' end property :status do key :'$ref', :StatusEnum end property :values do items do key :type, :number end end
After
DocsApiHelper.properties({ id: :integer, created_at: { type: :string, format: 'date-time' }, status?: :StatusEnum, values?: { type: :number, array: true } })
"This::Constant::Exists" ... "But::ThisOne::DoesNot"
Frontend type:
type Example = { id: number kind: 'good' | 'bad' }
Server response:
{ id: '123' kind: 'neutral' unknown: 'smth' }
Type validation output:
Property 'id' is not of type 'number'.
Property 'kind' does not match allowed values.
Unknown property 'unknown' found.
The commit message "example message" is autoformatted based on the branch name "feat/JIRA-123-example" to a final message reading "feat(JIRA-123): example message"
Freed up engineering lead time by automating app release announcements in Slack. See more... We used Heroku for hosting and deployed the latest Git main branch on every new change. We had a process in place where once a week one of the engineering leads compiled a list of all the changes with their Jira ID and a short description and posted this in a #release Slack channel. I implemented a task that ran on every deployment to Heroku which compiled a list of Jiras from the last deployment, got the Jira tasks descriptions, the author (developer) of that change, found a related Slack account and tagged them in the automated Slack message (developers were encouraged to provide additional context, especially when it was related to a bug they fixed for someone from a support of other department).
Improved copy quality through automated grammar and spellchecking. See more... Using LanguageTool I implemented a check in GitHub Actions that checked grammar including spellchecks. This involved compiling the list of translations from a YML file into a human-readable form which is what LanguageTool expects.
Example output for copy "This is highlight by diferent colour.":
highlight - possible grammar mistake highlighted diferent - possible spelling mistake different colour - colour is British English color
Enabled proactive identification and resolution of frontend issues by setting up Sentry for error tracking.
Laid the foundation for future enhancements such as automated TypeScript type generation by adding OpenAPI. See more... Found the tooling and implemented the helpers for adding OpenAPI 2.0 (later upgraded to version 3.0) and implemented it on a few regular cases as well as some exceptional ones. This was used later by the team for generating TypeScript types for frontend which increased our efficiency. It led to other initiatives, such as my work on linters and frontend to backend type validation later on to improve our workflow and get as much as possible from the coupling of backend to frontend this afforded us.
Accelerated developer velocity for hospital task system app for nurses and orderlies by streamlining dependency management and build system. See more... The team had a strict policy for dependency management and actually maintained their own repository of dependencies. This was initially done through quite a large Git submodule. I've made it so that the team was able to use our own repository of dependencies as if they were using the original repository which simplified the workflow to a large extent.
Track habits and gamify them in a tycoon-game. Started with a web app, now working on an iOS mobile app.