質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

Q&A

解決済

2回答

5779閲覧

Laravelで動的にOrder byしたい。

aglkjggg

総合スコア769

Laravel

LaravelとはTaylor Otwellによって開発された、オープンソースなPHPフレームワークです。Laravelはシンプルで表現的なシンタックスを持ち合わせており、ウェブアプリケーション開発の手助けをしてくれます。

0グッド

0クリップ

投稿2016/08/14 13:28

$app->get('/data/ranking/{sortType}', function($sortType) { $results = DB::select('SELECT ... FROM ... ORDER BY ?',[$sortType]); echo json_encode(results); }

LaravelでランキングのAPIサーバーを作っています。

  • 攻撃力順

/data/ranking/attack

  • 防御力順

/data/ranking/defence

とリクエストされるとそれぞれ、

  • ORDER BY attack DESC
  • ORDER BY defence DESC

としたいです。

しかし、?で自動的にパースしてもらえるはずなのですが、
うまく動作しません。

以下の方法だとうまくいきました。

  • ? を使わずに文字列の結合をすると動作する

(適切な処理をしてないのでセキュリティ的にまずいですが)

どのようにすればよいでしょうか。

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

ベストアンサー

恐らく、

? を使わずに文字列の結合をする

または kaji様の回答にあるように ORM を使うしかありません。

Laravel のマニュアルによると Laravel は内部的に PDO を使用しているようですが、
https://laravel.com/docs/5.2/queries

Note: The Laravel query builder uses PDO parameter

PDO ではパラメータマーカー(?のこと)で識別子(テーブル名・カラム名などのこと)を指定することはできないからです。
http://php.net/manual/ja/pdo.prepare.php

パラメータマーカーが表せるのは、データリテラルだけです。 リテラルの一部やキーワード、識別子、その他のクエリのパーツをパラメータにバインドすることはできません。


それとは別に、API で指定されたパラメータを直接 ORDER BY 句に適用することには問題がある、と考えます。

例え適切にエスケープできたとしても、存在しないカラム名を指定されると SQL 文がシンタックスエラーになるからです。

代替案として、以下のように ORDER BY 句に指定可能な文字列をホワイトリスト形式で持っておき、それに一致したものだけをクエリに含める、というのはいかがでしょうか?

php

1$app->get('/data/ranking/{sortType}', function($sortType) { 2 3 // order by 句に指定可能な文字列のリスト 4 $allowOrderBy = [ 5 'attack', 6 'defence', 7 ... 8 ]; 9 10 if (in_array($sortType, $allowOrderBy)) { 11 $orderBy = $sortType; 12 } else { 13 $orderBy = 'id'; // 一致しない場合はデフォルトの並び順 14 } 15 16 $results = DB::select('SELECT ... FROM ... ORDER BY ' . $orderBy); 17 18 echo json_encode(results); 19}

これならインジェクション対策とシンタックスエラー防止を同時に実現できますので。

投稿2016/08/15 07:44

編集2016/08/15 17:16
KiyoshiMotoki

総合スコア4791

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

aglkjggg

2016/08/15 09:54

なるほど! SQL文に代入する値をあらかじめハードコードしておく方法がありましたね! ありがとうございます。今回はその方法を参考にして実装していこうと思います。
guest

0

文字列はパースするのでsql文の結果は
SELECT ... FROM ... ORDER BY 'attack';
となりエラーとなっているのではないですか?
?のパースは値に使うものでカラム名には使いません。

ormを使えばカラム名のsqlインジェクションチェックもライブラリでしてくれる思います。
https://readouble.com/laravel/5.dev/ja/eloquent.html
Sample::orderBy($scoretype,'desc')->get();

投稿2016/08/14 23:46

編集2016/08/14 23:51
kaji

総合スコア648

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

aglkjggg

2016/08/15 04:29 編集

エラーにはならないのですが、文字列によるORDER BYが無意味になってしまう為、 期待したソートが行われませんでした。 非常にクエリが複雑に、長くなっているのでできればDB::selectで生で書きたかったのですが難しそうですね…
kaji

2016/08/15 05:27

DB::selectで生で書く方法は危なっかしいので、なんとも言えません。 他の方に回答を譲ります。 私だったらどんなに複雑でもormに書きかえるかもしれません。 お力になれずすいません。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問