Showing how to integrate SQLite into a Blazor wasm app which runs completely inside the browser!
This fork addresses issues with the original project:
- .NET 6 supports the Module object in JS Interop, but .NET 8 and above do not
- .NET 8 and above use the Blazor.runtime object to provide access to the Mono and Emscripten APIs
- The Sqlite Nuget had to be updated for .NET 8+ compatibility
- The Javascript code for syncing the file actually is* necessary, so I fixed it as well and kept it in
- More info about the .NET breaking changes can be found here: https://learn.microsoft.com/en-us/dotnet/core/compatibility/aspnet-core/9.0/legacy-apis
-
.NET Core 8 or 9
-
wasm-tools
workload$ sudo dotnet workload install wasm-tools
-
Microsoft Visual Studio 2022
-
JetBrains Rider
-
Visual Studio Code
-
Google Chrome or Microsoft Edge
- should work in other browsers which support wasm
$ git clone https://github.com/TrevorDArcyEvans/BlazorSQLiteWasm.git
$ cd BlazorSQLiteWasm
$ dotnet build
$ dotnet run
open BlazorSQLiteWasm
- open BlazorSQLiteWasm.sln in Visual Studio 2022
- F5 to run will open a browser and load wasm
Debugging should also work with the latest JetBrains Rider
This project is largely based on demo code from BlazeOrbital and requires a number of coordinated and cooperating parts.
- application code
- Entity Framework Core
- EF Core SQlite provider
is all compiled into the final wasm file.
A small piece of helper code is required to create an SQLite database file in the browser. This is called from C# application code via .NET-javascript interop, IJSRuntime. The database file is created on a per-user-per-url basis and is persistent between browser sessions.
There is some additional code:
setInterval(() => {
const path = `/${filename}`;
if (FS.analyzePath(path).exists) {
const mtime = FS.stat(path).mtime;
if (mtime.valueOf() !== lastModifiedTime.valueOf()) {
lastModifiedTime = mtime;
const data = FS.readFile(path);
db.result.transaction('Files', 'readwrite').objectStore('Files').put(data, 'file');
}
}
}, 1000);
which runs every second. This is an artefact from the original BlazeOrbital project which required the data to be synchronised every second. This code is actually required otherwise all data can be lost on reload.
SQLite driver is provided by SQLitePCLRaw.bundle_e_sqlite3 nuget package and is linked into the final wasm file.
If more properties are added to Car class, the application will throw EF Core exception. This is because the class and and underlying database schema are now mismatched. The database needs to be rebuilt and, during testing and development, this can be done by running:
await db.Database.EnsureDeletedAsync();
await db.Database.EnsureCreatedAsync();
For production, you could follow the guide here
There are several workarounds required to successfully publish this:
- various options for emscripten
AllowUnsafeBlocks
EmccExtraLDFlags
- force all types to be included in wasm file
Once published, it can be served by any webserver capable of serving static content eg python test server
$ dotnet publish
$ cd cd bin/Debug/net6.0/publish/wwwroot/
$ python3 -m http.server
open BlazorSQLiteWasm
Note that there are specific workarounds in index.html for hosting on GitHub.io pages
-
Icons made by Freepik from www.flaticon.com
-
Icons made by Eucalyp from www.flaticon.com