it-swarm.com.ru

Как получить построитель запросов для вывода его необработанного SQL-запроса в виде строки?

Учитывая следующий код:

DB::table('users')->get();

Я хочу получить необработанную строку запроса SQL, которую сгенерирует построитель запросов базы данных выше. В этом примере это будет SELECT * FROM users.

Как мне это сделать?

355
meiryo

Для вывода на экран последних выполненных запросов вы можете использовать это

dd(DB::getQueryLog());

Я считаю, что самые последние запросы будут в нижней части массива.

У вас будет что-то вроде этого:

array(1) {
  [0]=>
  array(3) {
    ["query"]=>
    string(21) "select * from "users""
    ["bindings"]=>
    array(0) {
    }
    ["time"]=>
    string(4) "0.92"
  }
}

В соответствии с комментарием Джошуа ниже, по умолчанию это отключено. Чтобы использовать, вам нужно включить его вручную, используя:

DB::enableQueryLog();
421
jfortunato

Используйте метод toSql() в экземпляре QueryBuilder.

DB::table('users')->toSql() вернет:

выберите * из `users`

Это проще, чем подключить прослушиватель событий, а также позволяет проверить, как будет выглядеть запрос в любой момент, пока вы его создаете.

578
Steven Mercatante

Вы можете прослушать событие «световые запросы». Перед запросом добавьте следующий прослушиватель событий:

Event::listen('illuminate.query', function($query, $params, $time, $conn) 
{ 
    dd(array($query, $params, $time, $conn));
});

DB::table('users')->get();

Это распечатает что-то вроде:

array(4) {
  [0]=>
  string(21) "select * from "users""
  [1]=>
  array(0) {
  }
  [2]=>
  string(4) "0.94"
  [3]=>
  string(6) "sqlite"
}
53
Rubens Mariuzzo

Если вы пытаетесь получить Журнал с помощью Illuminate без использования Laravel:

\Illuminate\Database\Capsule\Manager::getQueryLog();

Вы также можете добавить быструю функцию, например:

function logger() {
    $queries = \Illuminate\Database\Capsule\Manager::getQueryLog();
    $formattedQueries = [];
    foreach( $queries as $query ) :
        $prep = $query['query'];
        foreach( $query['bindings'] as $binding ) :
            $prep = preg_replace("#\?#", is_numeric($binding) ? $binding : "'" . $binding . "'", $prep, 1);
        endforeach;
        $formattedQueries[] = $prep;
    endforeach;
    return $formattedQueries;
}

ПРАВКА

обновленные версии, по-видимому, по умолчанию отключают ведение журнала запросов (приведенное выше возвращает пустой массив). Чтобы снова включить, при инициализации Capsule Manager захватите экземпляр соединения и вызовите метод enableQueryLog

$capsule::connection()->enableQueryLog();

Правка СНОВА

Принимая во внимание фактический вопрос, вы могли бы фактически сделать следующее, чтобы преобразовать текущий одиночный запрос вместо всех предыдущих запросов:

$sql = $query->toSql();
$bindings = $query->getBindings();
45
Luke Snowden

DB::QueryLog() работает только после выполнения запроса $builder->get(). если вы хотите получить запрос до его выполнения, вы можете использовать метод $builder->toSql(). Вот пример, как получить sql и связать его:

    $query = str_replace(array('?'), array('\'%s\''), $builder->toSql());
    $query = vsprintf($query, $builder->getBindings());
    dump($query);

    $result = $builder->get();
41
Kakashi

В eloquent есть метод для получения строки запроса. 

toSql ()

в нашем случае

 DB::table('users')->toSql(); 

вернуть

select * from users

это точное решение, которое возвращает строку запроса SQL .. Надеюсь, это полезно ...

34
CelinVeronicca
$data = User::toSql();
echo $data; //this will retrun select * from users. //here User is model
26
Kuldeep Mishra

Если вы используете laravel 5.1 и MySQL, вы можете использовать эту функцию, сделанную мной:

/*
 *  returns SQL with values in it
 */
function getSql($model)
{
    $replace = function ($sql, $bindings)
    {
        $needle = '?';
        foreach ($bindings as $replace){
            $pos = strpos($sql, $needle);
            if ($pos !== false) {
                if (gettype($replace) === "string") {
                     $replace = ' "'.addslashes($replace).'" ';
                }
                $sql = substr_replace($sql, $replace, $pos, strlen($needle));
            }
        }
        return $sql;
    };
    $sql = $replace($model->toSql(), $model->getBindings());

    return $sql;
}

В качестве входного параметра вы можете использовать любой из этих

Осветить\Database\красноречивый\Builder

Осветить\Database\красноречивый\Отношения\HasMany

Осветить\Database\Query\Builder

23
Yevgeniy Afanasyev

Первый способ:

Просто вы можете выполнять следующие действия, используя метод toSql(),

$query = DB::table('users')->get();

echo $query->toSql();

Если он не работает, вы можете настроить его из документации laravel .

Второй способ:

Еще один способ сделать это

DB::getQueryLog()

но если он возвращает пустой массив, то по умолчанию он отключен посетите это ,

просто включите с DB::enableQueryLog(), и это будет работать :)

для получения дополнительной информации посетите Github Issue , чтобы узнать больше об этом.

Надеюсь, поможет :)

10
Sagar Naliyapara

От Laravel 5.2 и далее. Вы можете использовать DB::listen для получения выполненных запросов.

DB::listen(function ($query) {
    // $query->sql
    // $query->bindings
    // $query->time
});

Или, если вы хотите отладить один экземпляр Builder, вы можете использовать метод toSql.

DB::table('posts')->toSql(); 
8
Zayn Ali

использовать пакет debugbar 

composer require "barryvdh/laravel-debugbar": "2.3.*"

 enter image description here

5
潘庆强

Чтобы увидеть выполненный запрос Laravel, используйте журнал запросов laravel

DB::enableQueryLog();

$queries = DB::getQueryLog();
5
Jasim Juwel

Это функция, которую я поместил в свой базовый класс модели. Просто передайте объект построителя запросов в него, и строка SQL будет возвращена.

function getSQL($builder) {
  $sql = $builder->toSql();
  foreach ( $builder->getBindings() as $binding ) {
    $value = is_numeric($binding) ? $binding : "'".$binding."'";
    $sql = preg_replace('/\?/', $value, $sql, 1);
  }
  return $sql;
}
5
BoogieBug

A 'macroable' замена, чтобы получить SQL-запрос с привязками.

  1. Добавьте ниже функцию макроса вAppServiceProviderboot()метод.

    \Illuminate\Database\Query\Builder::macro('toRawSql', function(){
        return array_reduce($this->getBindings(), function($sql, $binding){
            return preg_replace('/\?/', is_numeric($binding) ? $binding : "'".$binding."'" , $sql, 1);
        }, $this->toSql());
    });
    
  2. Добавьте псевдоним для Eloquent Builder. ( Laravel 5.4+ )

    \Illuminate\Database\Eloquent\Builder::macro('toRawSql', function(){
        return ($this->getQuery()->toRawSql());
    });
    
  3. Затем отлаживайте как обычно. ( Laravel 5.4+ )

    Например, Query Builder

    \Log::debug(\DB::table('users')->limit(1)->toRawSql())
    

    Например, Eloquent Builder

    \Log::debug(\App\User::limit(1)->toRawSql());
    

Примечание: с Laravel 5.1 до 5.3, поскольку Eloquent Builder не использует черту Macroable, не может добавить псевдоним toRawSql к Eloquent Builder на лету. Следуйте приведенному ниже примеру, чтобы добиться того же.

Например, Eloquent Builder ( Laravel 5.1 - 5.3 )

\Log::debug(\App\User::limit(1)->getQuery()->toRawSql());
4
Ijas Ameenudeen

Для laravel 5.5.X

Если вы хотите получать каждый SQL-запрос, выполняемый вашим приложением, вы можете использовать метод listen. Этот метод полезен для регистрации запросов или отладки. Вы можете зарегистрировать свой приемник запросов у поставщика услуг:

<?php

namespace App\Providers;

use Illuminate\Support\Facades\DB;
use Illuminate\Support\ServiceProvider;

class AppServiceProvider extends ServiceProvider
{
    /**
     * Bootstrap any application services.
     *
     * @return void
     */
    public function boot()
    {
        DB::listen(function ($query) {
            // $query->sql
            // $query->bindings
            // $query->time
        });
    }

    /**
     * Register the service provider.
     *
     * @return void
     */
    public function register()
    {
        //
    }
}

Источник

4
scre_www

Сначала вам нужно будет включить журнал запросов, вызвав:

DB::enableQueryLog();

после запросов с использованием фасада БД вы можете написать:

dd(DB::getQueryLog());

вывод будет как ниже:

array:1 [▼
  0 => array:3 [▼
    "query" => "select * from `users` left join `website_user` on `users`.`id` = `website_user`.`user_id` left join `region_user` on `users`.`id` = `region_user`.`user_id` left ▶"
    "bindings" => array:5 [▶]
    "time" => 3.79
  ]
]
4
Ravi Mane

Самый простой способ - сделать преднамеренную ошибку . Например, я хочу увидеть полный запрос SQL следующего отношения:

 public function jobs()
        {
            return $this->belongsToMany(Job::class, 'eqtype_jobs')
                   ->withPivot(['created_at','updated_at','id'])
                   ->orderBy('pivot_created_at','desc');
        }

Я просто хочу, чтобы столбец не был найден, здесь я выбираю created_at, и я изменил его на created_ats, добавив в конце s:

public function jobs()
            {
                return $this->belongsToMany(Job::class, 'eqtype_jobs')
                       ->withPivot(['created_ats','updated_at','id'])
                       ->orderBy('pivot_created_at','desc');
            }

Итак, отладчик вернет следующую ошибку:

(4/4) ErrorException SQLSTATE [42S22]: столбец не найден: 1054 Неизвестно столбец 'eqtype_jobs.created_ats' в 'списке полей' (SQL: выберите jobs. *, eqtype_jobs.set_id как pivot_set_id, eqtype_jobs.job_id как pivot_job_id, eqtype_jobs.created_ats Как pivot_created_ats, eqtype_jobs.updated_at как pivot_updated_at, eqtype_jobs.id as pivot_id из jobs inner Join eqtype_jobs в jobs.id = eqtype_jobs.job_id где eqtype_jobs.set_id = 56 упорядочить по pivot_created_at desc предел 20 Смещение 0) (Просмотр: /Home/said/www/factory/resources/views/set/show. blade.php)

Приведенное выше сообщение об ошибке возвращает полный запрос SQL с ошибкой

SQL: select  jobs.*, eqtype_jobs.set_id as pivot_set_id,  eqtype_jobs.job_id as pivot_job_id, eqtype_jobs.created_ats as pivot_created_ats, eqtype_jobs.updated_at as  pivot_updated_at, eqtype_jobs.id as pivot_id from jobs inner join eqtype_jobs on jobs.id = eqtype_jobs.job_id where  eqtype_jobs.set_id = 56 order by pivot_created_at desc limit 20 offset 0

Теперь просто удалите лишний s из созданного_атеста и протестируйте этот SQL, как вам нравится, в любом редакторе SQL, таком как phpMyAdmin SQL editor!

Обратите внимание:

Решение было протестировано с Laravel 5.4 .

3
SaidbakR

Если вы не используете Laravel, а используете пакет Eloquent, тогда:

use \Illuminate\Database\Capsule\Manager as Capsule;
use \Illuminate\Events\Dispatcher;
use \Illuminate\Container\Container;

$capsule = new Capsule;

$capsule->addConnection([
    // connection details
]);
// Set the event dispatcher used by Eloquent models... (optional)
$capsule->setEventDispatcher(new Dispatcher(new Container));

// Make this Capsule instance available globally via static methods... (optional)
$capsule->setAsGlobal();

// Setup the Eloquent ORM...(optional unless you've used setEventDispatcher())
$capsule->bootEloquent();

// Listen for Query Events for Debug
$events = new Dispatcher;
$events->listen('illuminate.query', function($query, $bindings, $time, $name)
{
    // Format binding data for sql insertion
    foreach ($bindings as $i => $binding) {
        if ($binding instanceof \DateTime) {
            $bindings[$i] = $binding->format('\'Y-m-d H:i:s\'');
        } else if (is_string($binding)) {
            $bindings[$i] = "'$binding'";`enter code here`
        }
    }

    // Insert bindings into query
    $query = str_replace(array('%', '?'), array('%%', '%s'), $query);
    $query = vsprintf($query, $bindings);

    // Debug SQL queries
    echo 'SQL: [' . $query . ']';
});

$capsule->setEventDispatcher($events);
2
Salman Ahmed

Я создал несколько простых функций для получения SQL и привязок из некоторых запросов.

/**
 * getSql
 *
 * Usage:
 * getSql( DB::table("users") )
 * 
 * Get the current SQL and bindings
 * 
 * @param  mixed  $query  Relation / Eloquent Builder / Query Builder
 * @return array          Array with sql and bindings or else false
 */
function getSql($query)
{
    if( $query instanceof Illuminate\Database\Eloquent\Relations\Relation )
    {
        $query = $query->getBaseQuery();
    }

    if( $query instanceof Illuminate\Database\Eloquent\Builder )
    {
        $query = $query->getQuery();
    }

    if( $query instanceof Illuminate\Database\Query\Builder )
    {
        return [ 'query' => $query->toSql(), 'bindings' => $query->getBindings() ];
    }

    return false;
}

/**
 * logQuery
 *
 * Get the SQL from a query in a closure
 *
 * Usage:
 * logQueries(function() {
 *     return User::first()->applications;
 * });
 * 
 * @param  closure $callback              function to call some queries in
 * @return Illuminate\Support\Collection  Collection of queries
 */
function logQueries(closure $callback) 
{
    // check if query logging is enabled
    $logging = DB::logging();

    // Get number of queries
    $numberOfQueries = count(DB::getQueryLog());

    // if logging not enabled, temporarily enable it
    if( !$logging ) DB::enableQueryLog();

    $query = $callback();

    $lastQuery = getSql($query);

    // Get querylog
    $queries = new Illuminate\Support\Collection( DB::getQueryLog() );

    // calculate the number of queries done in callback
    $queryCount = $queries->count() - $numberOfQueries;

    // Get last queries
    $lastQueries = $queries->take(-$queryCount);

    // disable query logging
    if( !$logging ) DB::disableQueryLog();

    // if callback returns a builder object, return the sql and bindings of it
    if( $lastQuery )
    {
        $lastQueries->Push($lastQuery);
    }

    return $lastQueries;
}

Использование:

getSql( DB::table('users') );
// returns 
// [
//     "sql" => "select * from `users`",
//     "bindings" => [],
// ]

getSql( $project->rooms() );
// returns
// [
//     "sql" => "select * from `rooms` where `rooms`.`project_id` = ? and `rooms`.`project_id` is not null",
//     "bindings" => [ 7 ],
// ]
2
blablabla

вы можете использовать часовой механизм

Clockwork - это расширение Chrome для разработки PHP, расширяющее Инструменты разработчика новой панелью, предоставляющей все виды информации, полезной для отладки и профилирования ваших приложений PHP, включая информацию о запросах, заголовках, получении и публикации данных, файлы cookie, данные сеанса, запросы к базе данных, маршруты, визуализация времени выполнения приложения и многое другое.

но работает и в Firefox

2
wdog

Вы можете использовать этот пакет для получения всех запросов, которые выполняются при загрузке вашей страницы

https://github.com/barryvdh/laravel-debugbar
2
Lucky Saini

Столько, сколько я люблю эту структуру, я ненавижу, когда она действует как .

DB::enableQueryLog() совершенно бесполезен. DB::listen одинаково бесполезен. Он показал часть запроса, когда я сказал $query->count(), но если я сделаю $query->get(), ему нечего сказать.

Единственное решение, которое, по-видимому, работает согласованно, состоит в том, чтобы преднамеренно поместить некоторый синтаксис или другую ошибку в параметры ORM, например, несуществующее имя столбца/таблицы, запустить код в командной строке в режиме отладки, и он выдаст ошибку SQL. наконец, с полным проклятым запросом. В противном случае, мы надеемся, что ошибка появится в файле журнала при запуске с веб-сервера.

1
Spencer Williams

Распечатать последний запрос

DB::enableQueryLog();

$query        = DB::getQueryLog();
$lastQuery    = end($query);
print_r($lastQuery);
1
Sohomdeep Paul

Вот решение, которое я использую: 

DB::listen(function ($sql, $bindings, $time) {
    $bound = preg_replace_callback("/\?/", function($matches) use ($bindings) {
        static $localBindings;
        if (!isset($localBindings)) {
            $localBindings = $bindings;
        }
        $val = array_shift($localBindings);

        switch (gettype($val)) {
            case "boolean":
                $val = ($val === TRUE) ? 1 : 0;  // mysql doesn't support BOOL data types, ints are widely used
                // $val = ($val === TRUE) ? "'t'" : "'f'";   // todo: use this line instead of the above for postgres and others
                break;

            case "NULL":
                $val = "NULL";
                break;

            case "string":
            case "object":
                $val = "'". addslashes($val). "'";   // correct escaping would depend on the RDBMS
                break;
        }
        return $val;
    }, $sql);
    array_map(function($x) { 
        (new \Illuminate\Support\Debug\Dumper)->dump($x); 
    }, [$sql, $bindings, $bound]);
});

Пожалуйста, прочитайте комментарии в коде. Я знаю, это не идеально, но для моей ежедневной отладки это нормально. Он пытается построить связанный запрос с более или менее надежной. Однако, не доверяйте этому полностью, механизмы баз данных избегают значений по-другому, которые не реализует эта короткая функция. Итак, внимательно отнеситесь к результату.

0
Csongor Halmai

Использование:

$data = DB::select('select * from users where id = :id', ['id' => 1]);
print_r($data);

Вывод будет как ниже:

Array ( [0] => stdClass Object ( [id] => 1 [name] => parisa [last] => naderi [username] => png [password] => 2132 [role] => 0 ) )
0
ParisaN