From CRUD to Offline-First

Using Workbox and Dexie to make a CRUD application offline-first

Over the past few days, we’ve thought about the power of offline-first development. How it works in favour of the user and allows them to use our applications as they need them regardless of their current connection status.

Today, I want to take a regular CRUD web application and make it offline-first. This will require two things:

  • Adding a service worker so that when I refresh the page in an offline state I have access to the site
  • Updating the data layer to make it accessible offline and across devices

This is an exciting time in Local First development as there are so many creative and dedicated teams and individuals working to solve these problems. At Local First Academy, we want to spotlight some of those amazing technologies while also making it easier for developers to get up an running. For this tutorial, I’m going to be using Dexie for my data layer and Workbox to simplify the generation of the service workers.

The App

I read a lot of books and want to be able to keep track of what I’ve read and when I read it. So, I’ve created BookShelf. At the moment it’s a basic CRUD application that allows me to store the title, author and date finished. In future, I’m going to have it automatically add details like genre, cover art, number of pages, etc.

For now though, I want to make it local first. I’ve built it in Vue and, if you want to follow along there’s a folder called start in this repo that is the basis for this tutorial. Pull it down, run npm install and npm run dev in the start folder and you’ll be presented with:

TODO: Add screenshot from above here

Not yet a thing of beauty but let’s get on with making this local first.

Adding a Service Worker

A service worker is a small JavaScript file that acts as a background worker in the browser. It gets registered when you first visit a site and operates separately from the main web page. Service workers can add various capabilities to a web application, such as caching assets for offline availability, enabling push notifications, and improving performance. In our case, we will use a service worker to make our site available offline.

To add a service worker, I’m going to use Workbox, a set of open-source tools developed by Google. Workbox simplifies the process of creating and managing service workers, especially as your application grows and its requirements become more complex. It provides features like pre-caching assets, runtime caching strategies, and tools for handling updates efficiently.

Step 1: Adding dependencies First thing we’ll do is add workbox-build and workbox-cli to our project:

npm install workbox-build workbox-cli

Step 2: Generate the service worker Now, we’ll create a file called workbox-config.cjs at the root of the project and add the following configuration:

module.exports = {
	globDirectory: 'dist/',
	globPatterns: ['**/*.{html,javascript,css,json,ico,png,jpg,jpeg,svg,woff2,woff,eot,ttf,otf}'],
	swDest: 'dist/service-worker.javascript',
	runtimeCaching: [
		{
			urlPattern: ({ request }) => request.mode === 'navigate',
			handler: 'NetworkFirst',
			options: {
				cacheName: 'pages-cache',
				expiration: {
					maxEntries: 50,
					maxAgeSeconds: 30 * 24 * 60 * 60 // 30 days
				}
			}
		},
		{
			urlPattern: /\.(?:javascript|css|html|json)$/,
			handler: 'StaleWhileRevalidate',
			options: {
				cacheName: 'static-resources-cache'
			}
		},
		{
			urlPattern: /\.(?:png|jpg|jpeg|svg|ico)$/,
			handler: 'CacheFirst',
			options: {
				cacheName: 'image-cache',
				expiration: {
					maxEntries: 50,
					maxAgeSeconds: 60 * 60 * 24 * 7 // 1 week
				}
			}
		}
	]
};

Step 3: Modify package.json

Add a new script to build the service worker:

"scripts": {
	"build:withsw": "vite build && npx workbox generateSW workbox-config.cjs"
}

Step 4: Register the Service Worker Modify src/main.ts to register the service worker:

if ('serviceWorker' in navigator) {
	window.addEventListener('load', () => {
		navigator.serviceWorker
			.register('/service-worker.javascript')
			.then((registration) => {
				console.log('Service Worker registered with scope:', registration.scope);
			})
			.catch((error) => {
				console.error('Service Worker registration failed:', error);
			});
	});
}

Step 5: Build and test Build the application with npm run build:withsw and serve the dist directory with a tool like serve (npx serve dist).

Now open the app in your browser and refresh the page. Go offline and refresh, checking that you get the regular functionality.

You should have full CRUD functionality in your application now whether you are online or offline. Woohoo!! Local first for the win! :)

Adding in other devices

We could stop now but I want to show how using a tool like Dexie allows the data to be synced and consolidated after you come back online.

To do that, I’m going to use Dexie Cloud and add in user authentication in a few lines of code.

Step 1: Create your database in the cloud Run this in the terminal,

npx dexie-cloud create

and you’ll be prompted for email verification. The URL of your database will output to the console and stored in a new local file called dexie-cloud.json.

Step 2: Whitelist your app origin In order to allow your application to communicate with Dexie Cloud, you’ll need to explicitly whitelist it.

npx dexie-cloud whitelist http://localhost:5173

Make sure the port is right for where your app is running. When you move into production, you’ll need to add those origins as well.

Step 3: Update dexie and install dexie-cloud-addon Run this command to get that ready to go:

npm install dexie@latest dexie-cloud-addon

Step 4: Update the database declaration

Open src/stores/db.ts and import dexieCloud. This should be added to the Dexie initialisation options object with the addons key.

import Dexie from 'dexie';
import dexieCloud from 'dexie-cloud-addon';

const db = new Dexie('BookVault', { addons: [dexieCloud] });

The schema can largely stay the same but we’ll swap out the ++id to @id. The cloud database prefers a string id and this will autogenerate it on store.

db.version(1).stores({
	books: '@id, title, author, dateFinished, isbn, genre, coverArt'
});

Finally, we’ll configure the cloud by adding our databaseUrl from step 1. We’ll also add requireAuth which will authenticate users and require an email verification before they can interact with the data.

db.cloud.configure({
	databaseUrl: '<add-your-cloud-url>',
	requireAuth: true
});

export default db;

Step 5: Fire it up! That’s it. Now, the data generated by a user will be automatically only be accessible to them. They can login on multiple devices and use this application offline and online.

When they come online, Dexie will do the work of synchronising the local IndexDb version of the data with the cloud version.

Conclusion

Developing applications in an offline way helps us democratise our software, making it more useful to our users wherever they are.

Hopefully, this tutorial has helped to give you a head start to begin developing in this way.

As I mentioned earlier, there are so many excellent tools and libraries in this space. Each of them have slightly different approaches - giving you more or less control, doing more or less of the work for you. At Local First Academy, we will spotlight different tools and help you get onboarded with them as quickly as possible. Then, as you develop local first software, you’ll have a better idea of which tool will fit your needs.

So far we’ve thought about synchronisation and offline-first. Tomorrow, we’ll spotlight and amazing application that uses both of these technologies before we explore the third theme of this series - collaboration.

We hope you’ve been enjoying this series - reach out to us by email or on Bluesky if you have any thoughts or comments.