11

我正在尝试运行我的单元测试并在安装过程中创建一个数据库。出于某种原因,我收到了错误Unknown database 'coretest'。如果我手动创建数据库并运行测试,那么我会得到Can't create database 'coretest'; database exists.

drop database 语句刚刚在 create database 中起作用。

这是我的 setUP 和 tearDown 方法:

class TestCase extends Illuminate\Foundation\Testing\TestCase {
    /**
     * Default preparation for each test
     */

    public function setUp() {
        parent::setUp();

        DB::statement('create database coretest;');
        Artisan::call('migrate');
        $this->seed();
        Mail::pretend(true);
    }

    public function tearDown() {
        parent::tearDown();
        DB::statement('drop database coretest;');
    }
}
4

5 回答 5

12

您收到此错误的原因仅仅是因为 laravel 尝试连接到配置中指定的数据库,该数据库不存在。

解决方案是在不指定数据库的情况下从设置中构建您自己的 PDO 连接(PDO 允许这样做)并CREATE DATABASE $dbname使用它运行语句。

我们在项目中使用这种方法进行测试,没有任何问题。


这里有一些代码:

<?php

/**
 * Bootstrap file for (re)creating database before running tests
 *
 * You only need to put this file in "bootstrap" directory of the project
 * and change "bootstrap" phpunit parameter within "phpunit.xml"
 * from "bootstrap/autoload.php" to "bootstap/testing.php"
 */

$testEnvironment = 'testing';

$config = require("app/config/{$testEnvironment}/database.php");

extract($config['connections'][$config['default']]);

$connection = new PDO("{$driver}:user={$username} password={$password}");
$connection->query("DROP DATABASE IF EXISTS ".$database);
$connection->query("CREATE DATABASE ".$database);

require_once('app/libraries/helpers.php');

// run migrations for packages
foreach(glob('vendor/*/*', GLOB_ONLYDIR) as $package) {
    $packageName = substr($package, 7); // drop "vendor" prefix
    passthru("./artisan migrate --package={$packageName} --env={$testEnvironment}");
}
passthru('./artisan migrate --env='.$testEnvironment);

require('autoload.php'); // run laravel's original bootstap file
于 2013-09-09T06:05:20.820 回答
2

neoascetic 有最好的答案,因为基本上你必须再次启动 laravel 的数据库配置文件。

因此,一个聪明的技巧是在删除数据库后再次创建数据库。无需触摸配置/数据库。

public function setUp() {
    parent::setUp();

    Artisan::call('migrate');
    $this->seed();
    Mail::pretend(true);
}

public function tearDown() {
    parent::tearDown();

    DB::statement('drop database coretest;');
    DB::statement('create database coretest;');
}
于 2015-11-19T15:00:51.653 回答
1

我觉得我有一种更清洁的方式来做到这一点。只需通过 shell 正常执行命令即可。

$host      = Config::get('database.connections.mysql.host');
$database  = Config::get('database.connections.mysql.database');
$username  = Config::get('database.connections.mysql.username');
$password  = Config::get('database.connections.mysql.password');
echo shell_exec('mysql -h ' . $host . ' -u ' . $username . ' -p' . $password . ' -e "DROP DATABASE ' . $database . '"');
echo shell_exec('mysql -h ' . $host . ' -u ' . $username . ' -p' . $password . ' -e "CREATE DATABASE ' . $database . '"');
于 2014-09-25T03:57:03.387 回答
0

在 Laravel 5 中,可以在内部调用迁移到 Laravel 进程,最终运行起来比使用外部命令快得多。

在 TestCase::setUp (或更早版本)中,调用迁移命令:

$kernel = app('Illuminate\Contracts\Console\Kernel');
$kernel->call('migrate');
于 2015-08-30T05:42:37.473 回答
0

像这样创建它:


public function setUp() {
        parent::setUp();

        $pdo = new PDO(
                "mysql:host=localhost",
                config('database.connections.mysql.username'),
                config('database.connections.mysql.password')
            );
       $pdo->query('CREATE DATABASE test_database');

       // set up the new database as the default
       config()->set('database.connections.mysql.database', 'test_database');
    }

于 2022-02-24T23:24:59.990 回答