LoginSignup
1
0

ReactでSQLite Wasmを実行して、localStorageに永続化する最小のサンプル

Last updated at Posted at 2024-03-02

はじめに

SQLiteの正式なWebAssembly版「SQLite3 WASM/JS」が登場 - Publickey

SQLite3をWebAssembly化した「SQLite3 WASM/JS」が気になっていたので、React上で動作するサンプルを作成してみました。
(Web標準になれなかったWeb SQL Databaseの代わりに利用できればと)

SQLite Wasmの実行方法は3つあります。今回は一番単純な「メインスレッドで実行」パターンを試しました

  1. ラップされたWorderをメインスレッドで実行(推奨)
  2. Web Workerで実行
  3. メインスレッドで実行

サンプルプログラムの動作確認はこちらをクリック

img00.png

  • React環境はViteで作成しています
  • localStorageに保存しているため、ブラウザを閉じても永続化されます(最大10MB)

作成したソースはこちら
https://github.com/murasuke/react_wasm_sqlite_step1
GitHub Pagesのページ
https://murasuke.github.io/react_wasm_sqlite_step1/

サンプルソースの内容

  • WASM版SQLite3をブラウザ上で実行
  • localStorageにDBを作成
  • テーブル作成
  • InsertとSelectを実行してクエリが想定通りに実行されることを確認
  • リロードしてもデータが消えないことを確認

作成手順

プロジェクト作成

$ npm create vite@latest wasm_sqlite_step1  -- --template react-ts
$ cd wasm_sqlite_step1
$ npm install @sqlite.org/sqlite-wasm
  • vite.config.tsを修正

headersoptimizeDepsを追加します

vite.config.ts
import { defineConfig } from 'vite';
import react from '@vitejs/plugin-react';

// https://vitejs.dev/config/
export default defineConfig({
  plugins: [react()],
  server: {
    headers: {
      'Cross-Origin-Opener-Policy': 'same-origin',
      'Cross-Origin-Embedder-Policy': 'require-corp',
    },
  },
  optimizeDeps: {
    exclude: ['@sqlite.org/sqlite-wasm'],
  },
});

SQLite初期化モジュール

  • sqlite3InitModule()でSQLiteのロード後、Databaseの読み込みを行う
  • 実行したクエリをconsoleに出力する

localStorageを簡単に利用できるJsStorageDb()は、SQLのトレースができなかったためやめました

  // JsStorageDb()はlocalStorageを簡単に利用できるが、SQLのトレースが利用できない
  db = new sqlite3.oo1.JsStorageDb('local');
./src/database.ts
import sqlite3InitModule, {
  Database,
  Sqlite3Static,
} from '@sqlite.org/sqlite-wasm';

const log = (...args: any[]) => console.log(...args); // eslint-disable-line
const error = (...args: any[]) => console.error(...args); // eslint-disable-line

let db: Database | null = null;

/**
 * sqlite3に接続してDBを作成
 * ・https://sqlite.org/wasm/doc/tip/api-oo1.md
 * @param sqlite3
 * @returns
 */
const connectDB = (sqlite3: Sqlite3Static) => {
  log('Running SQLite3 version', sqlite3.version.libVersion);

  // localStorageに保存(永続化)
  // (c: DBがなければ作成する, t: 実行したクエリをConsoleへ出力(trace on))
  db = new sqlite3.oo1.DB('file:local?vfs=kvvfs', 'ct');

  // DBサイズ(CスタイルのAPI経由)
  console.log(`DB size: ${sqlite3.capi.sqlite3_js_kvvfs_size()}`);

  // 永続化しない場合
  // db = new sqlite3.oo1.DB('/mydb.sqlite3', 'ct');
  // JsStorageDb()はlocalStorageを簡単に利用できるが、SQLのトレースが利用できない
  // db = new sqlite3.oo1.JsStorageDb('local');
  return db;
};

/**
 * DBが初期化済みであれば閉じる
 */
export const closeDB = () => {
  db?.close();
  db = null;
};

/**
 * DBの初期化と接続を行う
 * @returns Database
 */
export const getDatabase = async (): Promise<Database> => {
  if (db) {
    return db;
  }

  log('Loading and initializing SQLite3 module...');

  try {
    // sqlite3の初期化
    const sqlite3 = await sqlite3InitModule({
      print: log,
      printErr: error,
    });

    // DBに接続
    db = connectDB(sqlite3);

    return db;
  } catch (err: unknown) {
    if (err instanceof Error) {
      error(err.name, err.message);
      throw err;
    }
  }

  throw new Error('unknown error');
};

画面(App.tsx)でクエリを実行する

  • 初回ボタンクリック時、データ登録の前にDBの初期化を行います。
  • 2回目以降はデータ追加のみ
./src/App.tsx
import { useEffect } from 'react';
import './App.css';
import { getDatabase, closeDB } from './database';

function App() {
  useEffect(() => {
    return () => {
      // clean up
      closeDB();
    };
  }, []);

  const executeQuery = async () => {
    // 初回実行時に生成、以降は生成済みのconnectionを返す
    const db = await getDatabase();

    // テーブル作成
    db.exec('CREATE TABLE IF NOT EXISTS users(id INTEGER, name TEXT)');

    const select_max = 'SELECT max(id) as max_count FROM users';
    const max = (db.selectValue(select_max) as number) ?? 0;
    console.log(`row count: ${max}`);

    // 行追加(exec)
    db.exec({
      sql: 'insert into users values(?,?)',
      bind: [max + 1, `Alice${max + 1}`],
    });

    // 行追加(prepare & bind)
    const stmt = db.prepare('insert into users values(?, ?)');
    stmt.bind([max + 2, `Bob${max + 2}`]).stepReset();
    stmt.finalize();

    // 結果出力
    const values = db.exec({
      sql: 'SELECT * FROM users',
      rowMode: 'object',
      returnValue: 'resultRows',
    });
    console.log(values);
  };

  return (
    <>
      <div>
        <button id="exec" onClick={() => executeQuery()}>
          SQLite Wasm実行
        </button>
        <p>実行結果はDevToolsのConsoleに出力されます</p>
      </div>
    </>
  );
}

export default App;

動作確認

$ npm run dev
  • 初回実行時

    • DBサイズsizeは0byteです
    • テーブル作成+2行追加されたことが確認できます

    img10.png

  • 画面をリロードして、再度実行

    • DBサイズsizeは652byteです
    • 前回追加した2行+今回追加分の2行が表示されます

    img20.png

おまけ GitHub Pagesにデプロイ

vite.config.tsbase:を追加

react_wasm_sqlite_step1 の部分は、リポジトリ名です

vite.config.ts
export default defineConfig({
  base: process.env.GITHUB_PAGES ? 'react_wasm_sqlite_step1' : './',

package.json の`build'を変更

  • distdocsにコピー
package.json
    "build": "tsc && vite build && cp -r dist docs",

GitHub Pagesで公開するための設定

  • Setting ⇒ ②Pages をクリック
  • ③公開するブランチ(main or master)と、公開するディレクトリdocsを選択してSaveをクリック

img30.png

ビルドとデプロイ

  • ビルドの際docsディレクトリが作成されることを確認(公開用)
$ npm run build
  • デプロイ

コミットしてから、通常通りpushを行う

$ git push

GitHub Pagesで公開されたことを確認

数分待ってからhttps://murasuke.github.io/react_wasm_sqlite_step1/へアクセスすると、GitHub Pagesで公開されたことが確認できる

img40.png

(※想定通り動作していることが確認できた)

参考サイト

1
0
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
1
0