在没有数据库连接的情况下实例化 ActiveRecord 模型

Instantiating ActiveRecord models without a database connection

我目前正在用 php-activerecord 替换自制的 hack-y MVC 方法的模型组件。我有一个单独的页面,在数据库服务器出现故障时显示,该页面用于显示带有来宾头像、姓名和角色的用户徽章。代码如下:

if (Auth::$signed_in)
    echo Auth::$user->getAvatarWrap();
else echo (new \App\Models\User([
    'name' => 'Guest',
    'role' => 'guest',
    'avatar_url' => GUEST_AVATAR
]))->getAvatarWrap();
当用户登录时,

Auth::$signed_in 设置为 true,这在数据库中断的情况下是不可能的,因此 else 分支使用预定义的数据执行.

Pre-ActiveRecord 这将简单地将属性添加到对象,并且对 getAvatarWrap 方法的调用将毫无问题地执行。现在,模型由 ActiveRecord 控制,无论出于何种原因都会发生一组额外的调用。这可能是因为模型定义了关系,但我不确定。

ActiveRecord\DatabaseException: PDOException: SQLSTATE[08006] [7] could not connect to server: Connection refused
    Is the server running on host "localhost" (127.0.0.1) and accepting
    TCP/IP connections on port 5432? in /var/www/vendor/php-activerecord/php-activerecord/lib/Connection.php:260
Stack trace:
#0 /var/www/vendor/php-activerecord/php-activerecord/lib/Connection.php(260): PDO->__construct('pgsql:host=loca...', '<username>', '<password>', Array)
#1 /var/www/vendor/php-activerecord/php-activerecord/lib/Connection.php(122): ActiveRecord\Connection->__construct(Object(stdClass))
#2 /var/www/vendor/php-activerecord/php-activerecord/lib/ConnectionManager.php(33): ActiveRecord\Connection::instance('pgsql://databas...')
#3 /var/www/vendor/php-activerecord/php-activerecord/lib/Table.php(114): ActiveRecord\ConnectionManager::get_connection('pgsql')
#4 /var/www/vendor/php-activerecord/php-activerecord/lib/Table.php(90): ActiveRecord\Table->reestablish_connection(false)
#5 /var/www/vendor/php-activerecord/php-activerecord/lib/Table.php(71): ActiveRecord\Table->__construct('App\Models\User')
#6 /var/www/vendor/php-activerecord/php-activerecord/lib/Model.php(765): ActiveRecord\Table::load('App\Models\User')
#7 /var/www/vendor/php-activerecord/php-activerecord/lib/Model.php(271): ActiveRecord\Model::table()
#8 /var/www/includes/views/_sidebar.php(20): ActiveRecord\Model->__construct(Array)
#9 /var/www/includes/views/_layout.php(148): include('/var/www...')
#10 /var/www/includes/views/fatalerr.php(46): require('/var/www...')
#11 /var/www/includes/init.php(32): require('/var/www...')
#12 /var/www/includes/do.php(3): require('/var/www...')
#13 /var/www/public/index.php(1): require('/var/www...')
#14 {main} in /var/www/vendor/php-activerecord/php-activerecord/lib/Connection.php on line 262

当我知道它不会找到一个连接并且它应该对我提供的数据感到满意时,我如何告诉 ActiveRecord 停止寻找连接?我是否必须在停电期间告别使用我的任何模型?

感谢 I found out about SQLite memory databases which put me on the path to success. Given that the pull request 启用对这种数据库的支持已经收集了将近一年的灰尘,我决定自己动手。
如果您稍后阅读此内容,请务必检查所述 PR 是否已合并,如果已合并请发表评论,以便我更新此答案。

首先,我从 PR 的提交者的 fork 中 fork 了 original repository repository on GitHub. I'm intentionally not linking my fork, because it can get outdated or deleted eventually, so I suggest you make a fork for yourself. Then, I used a combination of this answer and this answer 以将 magic 分支放入我的 fork 中。您可以通过按分叉存储库主页上的 "Clone or download" 按钮来获取 <url>

$ git clone <url>
$ git remote add target https://github.com/claytonrcarter/php-activerecord.git
$ git fetch --all
$ git checkout master
$ git merge --squash target/sqlite-memory-support
$ git commit -m "Add support for SQLite :memory: databases"
$ git push

现在我的 fork 有了最新的更改,我对项目的 composer.json 进行了更改,并将 require 块中 php-activerecord/php-activerecord 的版本更改为 "dev-master",然后添加一个 repositories 块,里面有我的叉子。再次,将 <url> 替换为分支的克隆 URL.

{
    // ...
    "require": {
        // ...
        "php-activerecord/php-activerecord": "dev-master",
        // ...
    },
    "repositories": [
        {
            "type": "vcs",
            "url":  "<url>"
        }
    ]
}

但是等等,乐趣并没有就此结束!现在我们已经支持内存 SQLite DB 适配器,我们实际上必须定义它。所以修改你的初始化配置:

ActiveRecord\Config::initialize(function ($cfg){
    $cfg->set_connections([
        'pgsql' => 'pgsql://'.DB_USER.':'.DB_PASS.'@'.DB_HOST.'/database?charset=utf8',

        // Add this line below. 'failsafe' can be changed to something else, of course
        'failsafe' => 'sqlite://:memory:',
    ], 'pgsql');
});

结果是 pretty difficult 热交换默认连接,所以我做了下一个最好的事情,做了一个 child class 设置了 $connection到前面定义的 failsafe。请注意,必须手动定义必填字段,因为库无法知道允许您从空数据库中使用哪些字段。有趣的是,主 class 上出现的 $has_many 和类似关系对子 class 的行为没有负面影响,因此我可以保持相对较短。

namespace App\Models;

/** @inheritdoc */
class FailsafeUser extends User {
    static $connection = 'failsafe';

    public $name, $role, $avatar_url;
}

最后,我不得不更改调用以使用新定义的 class:

if (Auth::$signed_in)
    echo Auth::$user->getAvatarWrap();
else echo (new \App\Models\FailsafeUser([
    'name' => 'Guest',
    'role' => 'guest',
    'avatar_url' => GUEST_AVATAR
]))->getAvatarWrap();

瞧,脚本不再中断,我终于可以继续使用我的代码库的 ActiveRecord-ification。如果你想自己经历这种折磨并且有多个模型你想要 "offline" 变体那么你可能想要制作一个自动设置属性的某种抽象 class您将构造函数与 pre-setting 和 $connection 一起传入,使 child 实例稍微不那么冗长。