Logo

dev-resources.site

for different kinds of informations.

Techniques for Synchronous DB Access in TypeScript

Published at
7/5/2024
Categories
typescript
node
orm
database
Author
koyopro
Categories
4 categories in total
typescript
open
node
open
orm
open
database
open
Author
7 person written this
koyopro
open
Techniques for Synchronous DB Access in TypeScript

I am developing an ORM library for TypeScript called Accel Record. Unlike other TypeScript/JavaScript ORM libraries, Accel Record adopts a synchronous API instead of an asynchronous one.

However, to execute DB access synchronously, it was necessary to conduct thorough technical research.

In this article, I will introduce the techniques Accel Record employs to achieve synchronous DB access.

Supported Databases

Accel Record supports the following databases:

  • SQLite
  • MySQL
  • PostgreSQL

In the early stages of development, priority was given to supporting SQLite and MySQL. Therefore, this article focuses on SQLite and MySQL.

SQLite

When using SQLite with Node.js, the better-sqlite3 library is commonly used. Other ORM libraries also frequently use better-sqlite3 to access SQLite.

Upon investigation, we found that better-sqlite3 inherently provides a synchronous API. Thus, executing queries to SQLite synchronously was easily achievable using better-sqlite3.

MySQL

The challenge was with MySQL.

When using MySQL with Node.js, the mysql2 library is commonly used. However, mysql2 only offers an asynchronous API, making it impossible to use a synchronous API. We searched for other MySQL libraries that offered a synchronous API but could not find any that were well-maintained recently.

Next, we investigated whether there was a way to execute asynchronous APIs synchronously.

We found several older libraries claiming to execute MySQL queries synchronously, and we examined how they achieved synchronous processing.

The first method involved using Atomics.wait(). This method employs two threads: one for performing asynchronous operations and one for synchronously waiting for the result. Libraries such as synckit wrap this functionality to make it more user-friendly. However, synckit cannot be used outside the main thread and is not easily usable in a multi-threaded environment. In the Accel Record project, we use Vitest for testing. Vitest performs parallel testing using Node.js worker_threads, making this constraint a barrier to adoption.

The second method led us to a library called sync-rpc. This library uses Node.js's child_process module to create a separate process for executing asynchronous operations and waits synchronously for the result. Upon testing, we found that we could use sync-rpc to leverage mysql2's asynchronous API synchronously. However, since sync-rpc itself is an older library and did not always perform as expected, we incorporated its source code and made necessary modifications to achieve the desired functionality.

How sync-rpc Works

sync-rpc operates as follows:

  1. The main process specifies the entry point file and starts a child process.
  2. The child process reads the entry point file and starts as a server.
  3. The main process requests function execution from the child process and waits synchronously for the result.
  4. The child process executes the asynchronous function and returns the result to the main process.
  5. The main process receives the result from the child process and continues processing synchronously.
  6. When the main process exits, the child process also terminates.

Using sync-rpc, we realized that any asynchronous process could be used synchronously from the perspective of the main process.

Current Implementation of Accel Record

Currently, by using sync-rpc, we can execute asynchronous processes synchronously. Therefore, regardless of the database engine, queries are executed through sync-rpc for SQLite and PostgreSQL as well.

Specifically, SQL construction is performed in the main process, and only the query execution is handled by the child process using sync-rpc.

Future Improvements

While the current implementation uses sync-rpc to execute asynchronous processes synchronously, it relies on launching a child process.

However, using child processes has its drawbacks:

  • Overhead of inter-process communication
    • There is overhead due to data exchange between the main process and the child process.
    • Generally, this overhead is not significantly large compared to DB access latency, so it may not be a major issue in this case.
  • Operational complexity
    • Launching child processes can complicate operations.
    • Currently, we depend on Node.js's child_process for launching child processes, which might make it difficult to operate in environments other than Node.js.
    • It is expected to work properly in typical Node.js environments and serverless environments where Node.js runs (e.g., AWS Lambda, Vercel Functions).

If we find a method that can overcome these drawbacks, we would consider adopting it.

Summary

We introduced the techniques Accel Record considered and adopted to achieve synchronous DB access. During the research phase, we explored methods to execute asynchronous processes synchronously using multi-threading and inter-process communication. Ultimately, we adopted sync-rpc, which spawns a separate process, to execute queries synchronously.

Please check out 'Introduction to "Accel Record": A TypeScript ORM Using the Active Record Pattern' and the README to see what kind of interface Accel Record can achieve by adopting a synchronous API.

orm Article's
30 articles in total
Favicon
Let's take a quick look at Drizzle ORM
Favicon
Prisma 101 baby.
Favicon
The Step-by-Step Process of Online Reputation Repair for Personal and Business Profiles
Favicon
The Best Strategies for Removing Negative News Articles from Search Engines
Favicon
Trysil - Multi-tenat API REST
Favicon
Typing Prisma Json Fields? Yes, You Can!
Favicon
What is ORM
Favicon
Top 5 ORM Services to Safeguard Your Online Reputation in 2024
Favicon
Gorm Pagination With Ease
Favicon
Introdução ao ORM do Django: Exercícios Práticos
Favicon
Explorando ORM: Facilitando o Desenvolvimento com Bancos de Dados
Favicon
Techniques for Synchronous DB Access in TypeScript
Favicon
Why We Adopted a Synchronous API for the New TypeScript ORM
Favicon
Data Access with Dapper: A Lightweight ORM for .NET Apps
Favicon
Writing code like this improves efficiency by 100 times compared to directly using MyBatis
Favicon
The Power of Less: Streamlining Dependencies with Remult
Favicon
Seeking a Type-Safe Ruby on Rails in TypeScript, I Started Developing an ORM
Favicon
TypeORM: O ORM que Você Precisa Conhecer para Trabalhar com Node.js e TypeScript
Favicon
SQL generation: Golang's builder pattern vs Clojure's persistent map
Favicon
How to Setting Default UUID for User Model in Peewee
Favicon
Introduction to "Accel Record": A TypeScript ORM Using the Active Record Pattern
Favicon
Difference Between ORM and ODM
Favicon
How to add update, and delete data in peewee Database Python
Favicon
Most popular Postgres ORMs
Favicon
Introducing Stalactite ORM
Favicon
Advanced Django ORM Features (Q-Objects, F-Expressions, Aggregations and Annotations)
Favicon
Less Code (-75%) & More Power (💪) with Remult
Favicon
A Software Engineer's Tips and Tricks #1: Drizzle
Favicon
Asgard saviour: Neurelo
Favicon
Prisma vs. Drizzle: A Comprehensive Guide for Your NextJS Project

Featured ones: