2010年9月10日金曜日

SafariのopenDatabase

数年前にSafariにSQLiteのデータベースがついて一時期話題になりましたが、その後も色々と改善が続いているようです。その解説が以下のURLにあります。

Safari Client-Side Storage and Offline Applications Programming Guide: Introduction
http://developer.apple.com/safari/library/documentation/iPhone/Conceptual/SafariJSDatabaseGuide/Introduction/Introduction.html

ちなみに、この資料の日付は2010/01/20となっていますが、PDF版の作成日にはいくつかのバージョンがあるようです。

この資料では"HTML 5 Offline Application Cache"や"Key-Value Storage"の解説が追加されています。

データベースに関する解説も手が加えられています。"Relational Database Basics"はSQLの初心者を対象にデータベースの解説があり、"Using the JavaScript Database"でJavaScriptからデータベースを利用する方法が解説されています。"APPENDIX A: Database Example: A Simple Text Editor"にはサンプルコードを元に解説があります。

解説中のサンプル コードはわかりやすい書き方になっていますが、まだ完成には時間が必要ではないかという気がします。例えば"Listing 4-1 Creating and opening a database"にあるデータベースを開く時の処理です。

try {
 if (!window.openDatabase) {
  alert('not supported');
 } else {
  var shortName = 'mydatabase';
  var version = '1.0';
  var displayName = 'My Important Database';
  var maxSize = 65536; // in bytes
  var db = openDatabase(shortName, version, displayName, maxSize);
  // You should have a database instance in db.
 }
} catch(e) {
 // Error handling code goes here.
 if (e == 2) {
  // Version number mismatch.
  alert("Invalid database version.");
 } else {
  alert("Unknown error "+e+".");
 }
 return;
}
alert("Database is: "+db);

この書き方であれば、データベースの名前やバージョンなど後から調べる時に便利そうです。

ただし、このコードは関数内に入っていません。「最初に実行されるコード」の項目の1stの方法と同じようにすればこのままでも動作します。しかし、最初に一度しか実行しないコードをdeleteせずにメモリ上に置いておくのは無駄です。関数の中に入れて最初に実行させ最後にdeleteするのが妥当だと思います。ただし、この時には変数宣言のvar dbは式の外に置いてグローバル変数としないと他の関数でトランザクション処理ができなくなります。

また、サンプルにある// You should have a database instance in db.というコメントは具体的に何をしようとしているのかが分かりません。データベースと同時にtaransaction()メソッドで'CREATE TABLE'を使ったテーブルの処理も考えているのかもしれませんが、この解説ではテーブルの処理は別の関数で処理しています。

古いサンプルコードと比べるとopenDatabaseでデータベースが開けない時の処理がなくなり、try-catchcatchでエラー処理ができるようになったようです。エラー処理は"Handling Errors"と言う項目で解説されていますが、この通りには動いていないように思います。例えば上記のデータベースを開く時のcatchでの処理です。試しにversion1.12.0にすると、アラートに"Invalid database version."が表示されずに、以下のようなエラーとなりました。

"Unknown error Error: INVALID_STATE_ERR: DOM Exception 11."

さらに、"APPENDIX A: Database Example: A Simple Text Editor"にあるエラー処理では2の代わりに変数"INVALID_STATE_ERR"を置いていますが未定義のエラーになります。

エラーのメッセージやエラーコードの処理が紹介されています。ただ、後半部分が何を意味するのか分かりません。
function errorHandler(transaction, error)
{
 // error.message is a human-readable string.
 // error.code is a numeric error code
 alert('Oops.  Error was '+error.message+' (Code '+error.code+')');

 // Handle errors here
 var we_think_this_error_is_fatal = true;
 if (we_think_this_error_is_fatal) return true;
 return false;
}

先にも書いた'CREATE TABLE'を使ったテーブルの処理は"Listing 4-2 Creating a SQL table"に書かれています。しかし、'CREATE TABLE'のSQL文は長文になりやすく、読みにくいものです。以下のような書き方の方が後から調べやすいと思います。また、複数のテーブルを扱う場合には、この方法を発展させると便利になります。

db.transaction(
 function (transaction) {
  var mySQL = 'CREATE TABLE people(' +
   'id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, ' +
   'name TEXT NOT NULL DEFAULT "John Doe", ' +
   'shirt TEXT NOT NULL DEFAULT "Purple");';
  transaction.executeSql(mySQL, [], nullDataHandler, errorHandler);
 }
);

以上の内容をまとめると、今のところデータベースを開く関数は以下のような書き方が便利そうです。

function initDB() {
 try {
  if (!window.openDatabase) {
   alert('not supported');
  } else {
   var shortName = 'mydatabase';
   var version = '1.0';
   var displayName = 'My Important Database';
   var maxSize = 65536; // in bytes
   db = openDatabase(shortName, version, displayName, maxSize);
   // You should have a database instance in db.
   db.transaction(
    function (transaction) {
     var mySQL = 'CREATE TABLE people(' +
      'id INTEGER NOT NULL PRIMARY KEY AUTOINCREMENT, ' +
      'name TEXT NOT NULL DEFAULT "John Doe", ' +
      'shirt TEXT NOT NULL DEFAULT "Purple");';
     transaction.executeSql(mySQL, [], nullDataHandler, errorHandler);
    }
   );
  }
 } catch(e) {
  // Error handling code goes here.
  if (e == 2) {
   // Version number mismatch.
   alert("Invalid database version.");
  } else {
   alert("Unknown error "+e+".");
  }
  return;
 }
 // alert("Database is: "+db);
 delete initDB;
}

0 件のコメント: