訂正
そもそもPDOのsqliteドライバはfile:をサポートしていないことがわかりました。(#74184 Sqlite PDO doesn't support URI filenames) mysqlと比べると速くなりましたが、in-memoryになっていたわけではないようです。すいません。
なんとなくlsしたらfile::memory:?cache=shared
なる名前のファイルが生成されていて気がつきました。
ということで以下は無視してください。
はじめに
Lumenでテストが増えてきたのでsqliteにしてin memory databaseにすると少しは速くなるという話だったのですが一筋縄ではいかなかったのでメモ。
:memory:だとうまくいかない
まず単純に以下の設定で試してみました。
DB_CONNECTION=sqlite
DB_DATABASE=:memory:
しかし、なぜかテスト時にテーブルが存在しないというエラーが出ます。よくよく調べるとmigrationの時とテストのときでそれぞれデータベースへのconnectを行っていました。:memory:は接続のたびに新しいdbをメモリ内に作る(=内容は引き継がれない)ということなのでこれではだめそう。(DatabaseMigrations
でなければ多分問題ないんだろうけど)
file::memory:?cache=sharedを使う
sqliteのマニュアルによればfile::memory:?cache=shared
と指定すれば複数の接続から同じin-memoryデータベースを開くことができるとありました。これを使えば良さそうです。
しかし、
DB_CONNECTION=sqlite
DB_DATABASE=file::memory:?cache=shared
とやってみると、
Illuminate\Database\QueryException : Database (file::memory:?cache=shared) does not exist. (SQL: select * from sqlite_master where type = 'table' and name = migrations)
というエラーになってしまいました。メモリ内にこれから作られるはずのものが存在しないというのも変な話なのでこのエラーを出しているところを調べると以下のようになっていました。
Illuminate\Database\Connectors\SQLiteConnector
クラス:
public function connect(array $config)
{
$options = $this->getOptions($config);
// SQLite supports "in-memory" databases that only last as long as the owning
// connection does. These are useful for tests or for short lifetime store
// querying. In-memory databases may only have a single open connection.
if ($config['database'] == ':memory:') {
return $this->createConnection('sqlite::memory:', $config, $options);
}
$path = realpath($config['database']);
// Here we'll verify that the SQLite database exists before going any further
// as the developer probably wants to know if the database exists and this
// SQLite driver will not throw any exception if it does not by default.
if ($path === false) {
throw new InvalidArgumentException("Database (${config['database']}) does not exist.");
}
return $this->createConnection("sqlite:{$path}", $config, $options);
}
つまり、:memory:
という文字列だけは特別扱いするけどそれ以外はrealpath
で存在を確認してなければエラーになります。
file:
の考慮もないしこれ余計なお世話なんじゃないの...?
SQLiteConnectorを置き換える
DB_CONNECTION
に対応するconnectorのクラスはdb.connector.<DB_CONNECTIONの名前>
のコンテナがあればそれを使うようになっているので、入れ替えることが出来ます。(Illuminate\Database\Connectors\ConnectionFactory
のcreateConnector
メソッド)
そこで余計なチェックを一切しない以下のようなApp\Database\SqliteConnector
クラスを作成し、
namespace App\Database;
use Illuminate\Database\Connectors\Connector;
use Illuminate\Database\Connectors\ConnectorInterface;
class SqliteConnector extends Connector implements ConnectorInterface
{
/**
* Establish a database connection.
*
* @param array $config
* @return \PDO
*/
public function connect(array $config)
{
$options = $this->getOptions($config);
return $this->createConnection('sqlite:'.$config['database'], $config, $options);
}
}
AppServiceProvider
のregister
メソッドの中で以下のように登録しました。
$this->app->bind('db.connector.sqlite', SqliteConnector::class);
試したところ問題なく動作するようです。
おわりに
これでmysqlで1分46秒かかっていたテストが28秒になりました。(docker for mac上)