It has been around 3 months since we rolled out the current version of Text Reload. Until then, we spent several more months prototyping different features, different subscription models and different UIs. But one thing was constant all this time; we were able to iterate fast. We could prototype features overnight, try them out against our beta testing users, get feedback and refine. There was a technology that made these rapid iterations possible; Frappe - the open source low code framework. After 100+ registrations and 25,000+ SMS messages being sent, we thought of sharing a deep dive into the underlying technology at Text Reload to show how powerful, yet agile it is.
ℹ️ Fun Fact
The current subscription model you see in Text Reload was preceded by 2 different implementations. We even had a wallet built in where users can top up for SMS credit at one point.
We can’t talk about how Frappe helped us build Text Reload, without explaining what Text Reload is. The following diagram shows the different services/components Text Reload interacts with in order to provide its services.
In summary, Text Reload allows you to import your contacts (and invoices) from different sources, create campaigns against those contacts based on different scenarios like:
Text Reload allows you to use fields in contacts and invoices to filter out the target audience based on the requirement. For example, you can choose to set up a campaign to send SMS only to the “contacts who have purchased services worth more than $1000 in the past year”. Additionally, Text Reload can receive and notify on the replies sent by your contacts for the SMS you send to them, enabling meaningful 2-way communication. Additionally, there are some cool AI features to build and refine campaigns.
A reply received to an SMS sent via a campaign
Fundamentally, Text Reload is a SaaS solution. Users can create as many organizations as needed to represent the different companies they own/work with. Then they can invite other members to these individual organizations. As developers, the security and usability implications are very high in these kinds of scenarios.
Now let’s look at how we utilized the power of Frappe framework to build Text Reload.
Frappe is a full stack framework. When you define the data model (doctypes), the related views (list view, report view, kanban view, etc.), forms and REST APIs get automatically created. If the default forms and UIs are not suitable for your scenario (which was our case), you can build your own frontend with your favorite frontend technologies (ex: React.js, Vue.js, Vanilla JS + HTML). In our case, we wanted to build a custom user experience for our SaaS. As a result, we used React.js as the frontend technology. Below are some screenshots from the Text Reload dashboard.
Text Reload Dashboard (for end users)
Message composer in campaign creator
Then we used the Doctypes, which is the basic building block of the Frappe framework that allowed us to define the data models and their relationships using the UI itself. The framework manages the underlying database tables and relationships. These models are then easily accessible via the Frappe built-in ORM in Python or via the REST API. Doctypes are easily customizable using the UI. Different permissions to a doctype can be defined in the same doctype definition.
UI to customize a doctype
When the data model changes, we just have to add or update the fields within doctypes; and Frappe will manage the migrations automatically. Only if there’s a data migration, we can write patches
to explicitly migrate the data. Apart from that, database structural changes are managed by the framework automatically. This helped us a lot in iterating fast between different prototypes and experimental features. The doctype definitions are saved to JSON files (metadata) which can be committed and will be used to track structural changes and to migrate databases when deployed.
Both the frontend (React.js app) and the backend reside in the same frappe app (therefore within the same codebase/repository). We have used Doppio to set up the React.js dashboard inside the frappe app and to integrate with frappe build system. Doppio CLI helps us set up the developer environment with live reloading, tailwind.css and API access. The frontend is then communicating with the backend using the frappe-js-sdk. It uses the Frappe REST API to interact with the backend using the permissions of the user currently logged into the dashboard.
export async function getWorkflowList(
account: string
): Promise<WorkflowType[]> {
try {
const frappe = new FrappeApp('');
const workflows = await frappe.db().getDocList('SMS Campaign', {
fields: [
'name',
'title',
'message',
'frequency',
'disabled',
'campaign_based_on',
'xero_integration'
],
filters: [['xero_integration', '=', account]],
orderBy: { field: 'modified', order: 'desc' }
});
return workflows;
} catch (err) {
console.error('Failed to fetch workflow list', err);
throw err;
}
}
When all these things are combined, we have a frappe site that contains the full Text Reload app running. Frappe comes with the ready-to-host development environment named Frappe Bench
. It manages all the support services like caching, migrations, backups, updates and app dependencies. For production deployment, we have used Frappe Cloud which allows us to manage Text Reload Frappe site including upgrades, monitoring and deployments.
Source: https://frappe.io/framework/deployment
To manage all the users, configuration, admin tools, charts and reports we have the Desk UI. Below is a screenshot of the main workspace that shows a summary of the platform to the admins. Creating visualizations (charts) can be done from the UI with Frappe’s low code capabilities.
Getting this initial setup done will take around 10 minutes (create a frappe app, create a new site, setup react app for development). Once set up, we have a full stack application that’s ready for rapid development.
User management is the single most important part when building a SaaS. There can be multiple organizations where each organization can have its own users. Then we had to make sure the organizations are properly isolated and organization data is restricted only to the members of that organization. On the other hand, a user can be a member of multiple organizations. Therefore, we had to provide the ability for the users to switch between organizations too.
While having all these restrictions, we had to make the user registration and onboarding processes seamless. Allowing the users to login/register via passwords or via social logins (i.e. via external identity providers like Google and Xero) was part of that requirement. Then we have the password resets, welcome emails, etc. as trivial requirements. Let’s explore how Frappe made it seamless to implement those requirements in rapid iterations.
Frappe has this inbuilt Social Login Key feature that makes it easy to integrate any OAuth2 based identity provider as a login provider to a frappe site. As Text Reload was focusing on Xero in the initial days, we quickly got the Login with Xero
feature added with the help of Social Login Key.
Continue with Xero on Text Reload login with page
We have tested out adding support for Login with Google
and other providers as well. But we are yet to roll them out to the live platform.
We had 2 types of access control requirements:
We could use Roles
in Frappe combined with permissions defined in DocTypes
to define different user roles to restrict access to different activities inside an organization. Then we used User Permissions
(Read more on Frappe permissions model) to allow access only to the member organizations for the users.
An example user permissions restricting access to an organization
Frappe has an inbuilt password reset mechanism. We simply reused that for password resets. We could define the content of the password reset email via an Email Template
.
One of the key features of Text Reload is that it allows users to connect their Xero accounts and import the contacts into Text Reload to setup campaigns against them. Users should grant read only access to Xero invoices and contacts via OAuth2 for Text Reload to import the contacts and invoices. Remember that this happens across different organizations in Text Reload. The granted access should be tracked per organization. Once access is granted, we have to manage access tokens
and refresh tokens
. When tokens expire, we have to make sure we refresh the tokens as well. If we were to do that manually, it would have been a lengthy process. And if we wanted to integrate other platforms like HubSpot, MYOB, Google Calendar, etc; we would have to implement the same process again and again.
Frappe has this inbuilt concept called Connected Apps that simplifies integrating with 3rd party applications which support OAuth2. And Frappe manages these connections per user isolating connections made to different Xero accounts by different users. Frappe has the Token Cache
concept to handle this. Frappe also manages the tokens and handles token refreshing when an access token
expires.
An example connected app used in Text Reload for Xero integration
As you can see, all the scopes related to requested permissions can be configured in the connected app. It provides us with the redirect URIs to be used as well. If we wanted to add an integration to another app like M365 Calendar, we could simply setup an OAuth2 app in M365 app, obtain client ID
and client secret
, then setup a connected app like above. Connected Apps
simplify the step of requesting access to users’ accounts on behalf of the user using OAuth2. At the end of the day, accessing a user’s access token for a service they have granted access for is as simple as below. Frappe handles the token expiry and refreshing in the background.
token_cache = connected_app.get_active_token(user)
access_token = token_cache.get_password("access_token")
Once the access is granted, we may require to write some python code to fetch the relevant data in the backend and store in the relevant doctypes.
At Text Reload, we have heavy use cases for background jobs:
x
days after due date, annual followups should be sent 1 year after the invoice date, etc.).
Weekly summary email example
Frappe has an inbuilt Background Jobs concept that consists of:
short
, default
and long
) by default dedicated to short, medium and long jobs respectively. We can even add more workers if needed.scheduler_events = {
"daily": ["textreload_app.api.subscription.process_subscriptions"],
"cron": {
"0 22 * * 5": [ # Runs at 10:00 PM (22:00) every Friday (day 5)
"textreload_app.api.emails.email_summary.send_weekly_summary_emails"
],
},
}
These queues and background jobs become very helpful when sending 1000s of messages at one go. We can use one worker (long
queue) to send out the messages while the message delivery status responses are queued in the same queue to be processed later once the SMS sending job is completed. When multiple user registrations happen at once, if they all connect their third party services like Xero to import contacts, we again queue those data syncing tasks while the UI/frontend is operating in a non-blocking manner. Users can go ahead and setup their campaigns while the data is being synced in the background.
Following features from the Frappe framework should be mentioned for making our lives easier during the Text Reload journey so far.
Some charts on Text Reload Stats
As shown in the above screenshot, we can create visualizations based on desired doctypes using low code/no code capabilities. We can even create new charts based on the report output. We can setup custom dashboards for targeted metrics based on the focus area (like subscriptions) as well. We use such visualizations and dashboards for internal purposes.
Reporting is built into the framework allowing us to setup reports using the report builder (for simple reports). For more advanced reports, we have used script reports and query reports.
Frappe has these single doctypes to represent objects that are supposed to have a single instance. These are useful to store the global settings or global states.
Storing sensitive data like access tokens and refresh tokens (obtained via OAuth2 connections or setup by users manually) is very important, especially in the context of a SaaS. We are utilizing such fields across different doctypes to store sensitive information in an encrypted manner.
We rely on webhooks to receive message delivery status notifications from SMS gateways and for data update notifications from different connected services like Xero. We can create whitelisted functions that are authenticated or not (guest access allowed) depending on the requirement.
We have a requirement to use different email providers. In the past we used to rely on SendGrid. Recently we moved to Amazon SES for transactional emails. We still use SendGrid for marketing emails. The email accounts feature in Frappe combined with email queue and frappe.sendmail()
util function, give us the flexibility to send emails at different occasions using a chosen email provider. Automated Notifications
in frappe allows us to explicitly select the sender where we can specify which email account to use based on the type of notification (ex: Send as [email protected] vs [email protected]).
Email templates are useful to setup templates for different occasions (password resets, new account registrations, payment receipts, new SMS replies, etc.). We are heavily using email templates to setup different notifications that we send to our users. Email templates support templating with jinja2
which comes in very handy when building emails with dynamic content (weekly account summaries for example).
We can setup automated notifications to be sent out on different events using the Notifications in Frappe. We extensively use this to send email notifications to the users on:
Request for feedback
campaigns.Notifications in Frappe is a powerful way to keep your audience (users and internal stakeholders) informed on key events happening in the system.
Frappe has this concept called Print Formats that allows us to define how the printed version (or PDF) of any given doctype looks like. We are using print formats to build the payment receipts and invoice PDFs. We can either use html
and jinja2
or use the print format builder to design the print format.
There are scenarios where caching needs to be done at the application level. There are some external API calls that are expensive or can be rate limited when invoked continuously. We use Redis based caching provided by Frappe to optimize such API calls and improve the application performance where applicable.
Our journey with Text Reload has been a testament to how the right technology stack can empower developers to build and iterate on complex, feature-rich SaaS platforms rapidly. Frappe has been instrumental in this process, allowing us to focus on solving business problems rather than reinventing technical wheels.
In just a few months, we’ve built a multi-tenant SaaS application with enterprise-grade features like:
What would have taken us significantly longer with traditional frameworks was accelerated by Frappe’s batteries-included approach. The low-code aspects for data modeling, job scheduling, and user management meant we could prototype features overnight and refine them based on real user feedback. At the same time, the framework’s flexibility allowed us to build custom React interfaces when the standard UIs weren’t sufficient for our vision.
For developers and technical teams evaluating framework options, Frappe deserves serious consideration. It strikes an impressive balance between convention and configuration - providing structure and built-in solutions for common problems while remaining flexible enough for customization. The documentation is comprehensive, and the community is active and supportive. We particularly appreciated how quickly new developers on our team could become productive with the framework.
If you’re building a SaaS product or internal business application that requires complex data relationships, workflow automation, or integrations with third-party services, we highly recommend giving Frappe a try. The learning curve is surprisingly gentle given its power, and the productivity gains are substantial. You can get started with the official Frappe documentation.
At HighFlyer, we’re excited to continue exploring what’s possible with Frappe and to share our learnings with the developer community. Feel free to reach out if you have questions about our implementation or want to discuss how we tackled specific challenges along the way.
Head of Engineering
Imesha is the Head of Engineering at HighFlyer and a member of the Apache Software Foundation with 10+ years of experience across integration, cloud, and AI. He led the engineering efforts for Text Reload, focusing on creating a scalable SaaS platform using open source technologies.
We are excited to share that our team at HighFlyer recently engaged in initial discussions with Alibaba Cloud and the...
Read MoreWe are excited to share that our Head of Engineering Imesha Sudasingha and Principal Consultant (IT) Pasan Thilakasiri recently visited...
Read MoreWe are excited to announce that our Principal Consultant (IT), Dr. Pasan Thilakasiri, recently delivered a guest lecture as an...
Read MoreSubscribe to our newsletter to receive the latest insights and updates directly in your inbox.