source: branches/version-2_5-dev/data/class/db/dbfactory/SC_DB_DBFactory_MYSQL.php @ 20167

Revision 20167, 17.5 KB checked in by nanasess, 11 years ago (diff)

#981([フロント]MYページ)

  • 受注取得処理を抽象化

#642(共通ロジックの機能向上)

  • SC_Query::extractOnlyColsOf() を追加

#628(未使用処理・定義などの削除)

  • SC_Helper_DB クラスの未使用関数を削除
  • Property svn:eol-style set to LF
  • Property svn:keywords set to Id
  • Property svn:mime-type set to text/x-httpd-php; charset=UTF-8
Line 
1<?php
2/*
3 * This file is part of EC-CUBE
4 *
5 * Copyright(c) 2000-2010 LOCKON CO.,LTD. All Rights Reserved.
6 *
7 * http://www.lockon.co.jp/
8 *
9 * This program is free software; you can redistribute it and/or
10 * modify it under the terms of the GNU General Public License
11 * as published by the Free Software Foundation; either version 2
12 * of the License, or (at your option) any later version.
13 *
14 * This program is distributed in the hope that it will be useful,
15 * but WITHOUT ANY WARRANTY; without even the implied warranty of
16 * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
17 * GNU General Public License for more details.
18 *
19 * You should have received a copy of the GNU General Public License
20 * along with this program; if not, write to the Free Software
21 * Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
22 */
23
24// {{{ requires
25require_once(CLASS_REALDIR . "db/SC_DB_DBFactory.php");
26
27/**
28 * MySQL 固有の処理をするクラス.
29 *
30 * このクラスを直接インスタンス化しないこと.
31 * 必ず SC_DB_DBFactory クラスを経由してインスタンス化する.
32 * また, SC_DB_DBFactory クラスの関数を必ずオーバーライドしている必要がある.
33 *
34 * @package DB
35 * @author LOCKON CO.,LTD.
36 * @version $Id:SC_DB_DBFactory_MYSQL.php 15267 2007-08-09 12:31:52Z nanasess $
37 */
38class SC_DB_DBFactory_MYSQL extends SC_DB_DBFactory {
39
40    /** SC_Query インスタンス */
41    var $objQuery;
42
43    /**
44     * DBのバージョンを取得する.
45     *
46     * @param string $dsn データソース名
47     * @return string データベースのバージョン
48     */
49    function sfGetDBVersion($dsn = "") {
50        $objQuery =& SC_Query::getSingletonInstance();
51        $val = $objQuery->getOne("select version()");
52        return "MySQL " . $val;
53    }
54
55    /**
56     * MySQL 用の SQL 文に変更する.
57     *
58     * @access private
59     * @param string $sql SQL 文
60     * @return string MySQL 用に置換した SQL 文
61     */
62    function sfChangeMySQL($sql){
63        // 改行、タブを1スペースに変換
64        $sql = preg_replace("/[\r\n\t]/"," ",$sql);
65        // view表をインラインビューに変換する
66        $sql = $this->sfChangeView($sql);
67        // ILIKE検索をLIKE検索に変換する
68        $sql = $this->sfChangeILIKE($sql);
69        // RANDOM()をRAND()に変換する
70        $sql = $this->sfChangeRANDOM($sql);
71        // TRUNCをTRUNCATEに変換する
72        $sql = $this->sfChangeTrunc($sql);
73        // ARRAY_TO_STRINGをGROUP_CONCATに変換する
74        $sql = $this->sfChangeArrayToString($sql);
75        return $sql;
76    }
77
78    /**
79     * 文字コード情報を取得する
80     *
81     * @return array 文字コード情報
82     */
83    function getCharSet() {
84        $objQuery =& SC_Query::getSingletonInstance();
85        $arrRet = $objQuery->getAll("SHOW VARIABLES LIKE 'char%'");
86        return $arrRet;
87    }
88
89    /**
90     * 昨日の売上高・売上件数を算出する SQL を返す.
91     *
92     * @param string $method SUM または COUNT
93     * @return string 昨日の売上高・売上件数を算出する SQL
94     */
95    function getOrderYesterdaySql($method) {
96        return "SELECT ".$method."(total) FROM dtb_order "
97              . "WHERE del_flg = 0 "
98                . "AND cast(create_date as date) = DATE_ADD(current_date, interval -1 day) "
99                . "AND status <> " . ORDER_CANCEL;
100    }
101
102    /**
103     * 当月の売上高・売上件数を算出する SQL を返す.
104     *
105     * @param string $method SUM または COUNT
106     * @return string 当月の売上高・売上件数を算出する SQL
107     */
108    function getOrderMonthSql($method) {
109        return "SELECT ".$method."(total) FROM dtb_order "
110              . "WHERE del_flg = 0 "
111                . "AND date_format(create_date, '%Y/%m') = ? "
112                . "AND date_format(create_date, '%Y/%m/%d') <> date_format(now(), '%Y/%m/%d') "
113                . "AND status <> " . ORDER_CANCEL;
114    }
115
116    /**
117     * 昨日のレビュー書き込み件数を算出する SQL を返す.
118     *
119     * @return string 昨日のレビュー書き込み件数を算出する SQL
120     */
121    function getReviewYesterdaySql() {
122        return "SELECT COUNT(*) FROM dtb_review AS A "
123          . "LEFT JOIN dtb_products AS B "
124                 . "ON A.product_id = B.product_id "
125              . "WHERE A.del_flg = 0 "
126                . "AND B.del_flg = 0 "
127                . "AND cast(A.create_date as date) = DATE_ADD(current_date, interval -1 day) "
128                . "AND cast(A.create_date as date) != current_date";
129    }
130
131    /**
132     * メール送信履歴の start_date の検索条件の SQL を返す.
133     *
134     * @return string 検索条件の SQL
135     */
136    function getSendHistoryWhereStartdateSql() {
137        return "start_date BETWEEN date_add(now(),INTERVAL -5 minute) AND date_add(now(),INTERVAL 5 minute)";
138    }
139
140    /**
141     * ダウンロード販売の検索条件の SQL を返す.
142     *
143     * @param string $dtb_order_alias
144     * @return string 検索条件の SQL
145     */
146    function getDownloadableDaysWhereSql($dtb_order_alias = 'dtb_order') {
147        return "(SELECT IF((SELECT d1.downloadable_days_unlimited FROM dtb_baseinfo d1)=1, 1, DATE(NOW()) <= DATE(DATE_ADD(" . $dtb_order_alias . ".payment_date, INTERVAL (SELECT downloadable_days FROM dtb_baseinfo) DAY))))";
148    }
149
150
151    /**
152     * 売上集計の期間別集計のSQLを返す
153     *
154     * @param mixed $type
155     * @return string 検索条件のSQL
156     */
157    function getOrderTotalDaysWhereSql($type) {
158        switch($type){
159        case 'month':
160            $format = '%m';
161            break;
162        case 'year':
163            $format = '%Y';
164            break;
165        case 'wday':
166            $format = '%a';
167            break;
168        case 'hour':
169            $format = '%H';
170            break;
171        default:
172            $format = '%Y-%m-%d';
173            break;
174        }
175
176        return " date_format(create_date, '".$format."') AS str_date,
177            COUNT(order_id) AS total_order,
178            SUM(CASE WHEN order_sex = 1 THEN 1 ELSE 0 END) AS men,
179            SUM(CASE WHEN order_sex = 2 THEN 1 ELSE 0 END) AS women,
180            SUM(CASE WHEN customer_id <> 0 AND order_sex = 1 THEN 1 ELSE 0 END) AS men_member,
181            SUM(CASE WHEN customer_id <> 0 AND order_sex = 2 THEN 1 ELSE 0 END) AS women_member,
182            SUM(CASE WHEN customer_id = 0 AND order_sex = 1 THEN 1 ELSE 0 END) AS men_nonmember,
183            SUM(CASE WHEN customer_id = 0 AND order_sex = 2 THEN 1 ELSE 0 END) AS women_nonmember,
184            SUM(total) AS total,
185            AVG(total) AS total_average";
186    }
187
188
189    /**
190     * 文字列連結を行う.
191     *
192     * @param array $columns 連結を行うカラム名
193     * @return string 連結後の SQL 文
194     */
195    function concatColumn($columns) {
196        $sql = "concat(";
197        $i = 0;
198        $total = count($columns);
199        foreach ($columns as $column) {
200            $sql .= $column;
201            if ($i < $total -1) {
202                $sql .= ", ";
203            }
204            $i++;
205        }
206        $sql .= ")";
207        return $sql;
208    }
209
210    /**
211     * テーブルのカラム一覧を取得する.
212     *
213     * @deprecated SC_Query::listTableFields() を使用してください
214     * @param string $table_name テーブル名
215     * @return array テーブルのカラム一覧の配列
216     */
217    function sfGetColumnList($table_name) {
218        $objQuery =& SC_Query::getSingletonInstance();
219        $sql = "SHOW COLUMNS FROM " . $table_name;
220        $arrColList = $objQuery->getAll($sql);
221        $arrColList = SC_Utils_Ex::sfswaparray($arrColList);
222        return $arrColList["Field"];
223    }
224
225    /**
226     * テーブルを検索する.
227     *
228     * 引数に部分一致するテーブル名を配列で返す.
229     *
230     * @param string $expression 検索文字列
231     * @return array テーブル名の配列
232     */
233    function findTableNames($expression = "") {
234        $objQuery =& SC_Query::getSingletonInstance();
235        $sql = "SHOW TABLES LIKE ". $objQuery->quote("%" . $expression . "%");
236        $arrColList = $objQuery->getAll($sql);
237        $arrColList = SC_Utils_Ex::sfswaparray($arrColList, false);
238        return $arrColList[0];
239    }
240
241    /**
242     * View の WHERE 句を置換する.
243     *
244     * @param string $target 置換対象の文字列
245     * @param string $where 置換する文字列
246     * @param array $arrval WHERE 句の要素の配列
247     * @param string $option SQL 文の追加文字列
248     * @return string 置換後の SQL 文
249     */
250    function sfViewWhere($target, $where = "", $arrval = array(), $option = ""){
251
252        $arrWhere = split("[?]", $where);
253        $where_tmp = " WHERE " . $arrWhere[0];
254        for($i = 1; $i < count($arrWhere); $i++){
255            $where_tmp .= SC_Utils_Ex::sfQuoteSmart($arrval[$i - 1]) . $arrWhere[$i];
256        }
257        $arrWhere = $this->getWhereConverter();
258        $arrWhere[$target] = $where_tmp . " " . $option;
259        return $arrWhere[$target];
260    }
261
262    /**
263     * View をインラインビューに変換する.
264     *
265     * @access private
266     * @param string $sql SQL 文
267     * @return string インラインビューに変換した SQL 文
268     */
269    function sfChangeView($sql){
270
271        $arrViewTmp = $this->viewToSubQuery();
272
273            // viewのwhereを変換
274        foreach($arrViewTmp as $key => $val){
275            $arrViewTmp[$key] = strtr($arrViewTmp[$key], $this->getWhereConverter());
276        }
277
278            // viewを変換
279        $changesql = strtr($sql, $arrViewTmp);
280
281        return $changesql;
282    }
283
284    /**
285     * ILIKE句 を LIKE句へ変換する.
286     *
287     * @access private
288     * @param string $sql SQL文
289     * @return string 変換後の SQL 文
290     */
291    function sfChangeILIKE($sql){
292        $changesql = eregi_replace("(ILIKE )", "LIKE ", $sql);
293        return $changesql;
294    }
295
296    /**
297     * RANDOM() を RAND() に変換する.
298     *
299     * @access private
300     * @param string $sql SQL文
301     * @return string 変換後の SQL 文
302     */
303    function sfChangeRANDOM($sql){
304        $changesql = eregi_replace("( RANDOM)", " RAND", $sql);
305        return $changesql;
306    }
307
308    /**
309     * TRUNC() を TRUNCATE() に変換する.
310     *
311     * @access private
312     * @param string $sql SQL文
313     * @return string 変換後の SQL 文
314     */
315    function sfChangeTrunc($sql){
316        $changesql = eregi_replace("( TRUNC)", " TRUNCATE", $sql);
317        return $changesql;
318    }
319
320    /**
321     * ARRAY_TO_STRING(ARRAY(A),B) を GROUP_CONCAT() に変換する.
322     *
323     * @access private
324     * @param string $sql SQL文
325     * @return string 変換後の SQL 文
326     */
327    function sfChangeArrayToString($sql){
328        if(strpos(strtoupper($sql), 'ARRAY_TO_STRING') !== FALSE) {
329            preg_match_all('/ARRAY_TO_STRING.*?\(.*?ARRAY\(.*?SELECT (.+?) FROM (.+?) WHERE (.+?)\).*?\,.*?\'(.+?)\'.*?\)/is', $sql, $match, PREG_SET_ORDER);
330
331            foreach($match as $item) {
332                $replace = 'GROUP_CONCAT(' . $item[1] . ' SEPARATOR \'' . $item[4] . '\') FROM ' . $item[2] . ' WHERE ' . $item[3];
333                $sql = str_replace($item[0], $replace, $sql);
334            }
335        }
336        return $sql;
337    }
338
339    /**
340     * WHERE 句置換用の配列を返す.
341     *
342     * @access private
343     * @return array WHERE 句置換用の配列
344     */
345    function getWhereConverter() {
346        return array(
347            "&&crscls_where&&" => "",
348            "&&crsprdcls_where&&" =>"",
349            "&&noncls_where&&" => "",
350            "&&allcls_where&&" => "",
351            "&&allclsdtl_where&&" => "",
352            "&&prdcls_where&&" => "",
353            "&&catcnt_where&&" => ""
354        );
355    }
356
357    /**
358     * View をサブクエリに変換するための配列を返す.
359     *
360     * @access private
361     * @return array View をサブクエリに変換するための配列
362     */
363    function viewToSubQuery() {
364
365        static $sql = array();
366
367        if (empty($sql)) {
368
369
370            $sql['vw_products_allclass_detail'] = <<< __EOS__
371                (
372                    SELECT
373                        dtb_products.product_id,
374                        dtb_products.name,
375                        dtb_products.maker_id,
376                        dtb_products.status,
377                        dtb_products.comment1,
378                        dtb_products.comment2,
379                        dtb_products.comment3,
380                        dtb_products.comment4,
381                        dtb_products.comment5,
382                        dtb_products.comment6,
383                        dtb_products.note,
384                        dtb_products.main_list_comment,
385                        dtb_products.main_list_image,
386                        dtb_products.main_comment,
387                        dtb_products.main_image,
388                        dtb_products.main_large_image,
389                        dtb_products.sub_title1,
390                        dtb_products.sub_comment1,
391                        dtb_products.sub_image1,
392                        dtb_products.sub_large_image1,
393                        dtb_products.sub_title2,
394                        dtb_products.sub_comment2,
395                        dtb_products.sub_image2,
396                        dtb_products.sub_large_image2,
397                        dtb_products.sub_title3,
398                        dtb_products.sub_comment3,
399                        dtb_products.sub_image3,
400                        dtb_products.sub_large_image3,
401                        dtb_products.sub_title4,
402                        dtb_products.sub_comment4,
403                        dtb_products.sub_image4,
404                        dtb_products.sub_large_image4,
405                        dtb_products.sub_title5,
406                        dtb_products.sub_comment5,
407                        dtb_products.sub_image5,
408                        dtb_products.sub_large_image5,
409                        dtb_products.sub_title6,
410                        dtb_products.sub_comment6,
411                        dtb_products.sub_image6,
412                        dtb_products.sub_large_image6,
413                        dtb_products.del_flg,
414                        dtb_products.creator_id,
415                        dtb_products.create_date,
416                        dtb_products.update_date,
417                        dtb_products.deliv_date_id,
418                        T4.product_code_min,
419                        T4.product_code_max,
420                        T4.price01_min,
421                        T4.price01_max,
422                        T4.price02_min,
423                        T4.price02_max,
424                        T4.stock_min,
425                        T4.stock_max,
426                        T4.stock_unlimited_min,
427                        T4.stock_unlimited_max,
428                        T4.class_count
429                    FROM
430                        dtb_products
431                    JOIN
432                            (
433                                SELECT
434                                    product_id,
435                                    MIN(product_code) AS product_code_min,
436                                    MAX(product_code) AS product_code_max,
437                                    MIN(price01) AS price01_min,
438                                    MAX(price01) AS price01_max,
439                                    MIN(price02) AS price02_min,
440                                    MAX(price02) AS price02_max,
441                                    MIN(stock) AS stock_min,
442                                    MAX(stock) AS stock_max,
443                                    MIN(stock_unlimited) AS stock_unlimited_min,
444                                    MAX(stock_unlimited) AS stock_unlimited_max,
445                                    COUNT(*) as class_count
446                                FROM dtb_products_class
447                                GROUP BY product_id
448                            ) AS T4
449                            ON dtb_products.product_id = T4.product_id
450                )
451__EOS__;
452
453            $sql['vw_products_allclass'] = <<< __EOS__
454                (
455                    SELECT
456                        alldtl.*,
457                        dtb_category.rank AS category_rank,
458                        T2.category_id,
459                        T2.rank AS product_rank
460                    FROM
461                        {$sql['vw_products_allclass_detail']} AS alldtl
462                        LEFT JOIN
463                            dtb_product_categories AS T2
464                            ON alldtl.product_id = T2.product_id
465                        LEFT JOIN
466                            dtb_category
467                            ON T2.category_id = dtb_category.category_id
468                )
469__EOS__;
470        }
471
472        return $sql;
473
474    }
475
476    /**
477     * インデックス作成の追加定義を取得する
478     *
479     * 引数に部分一致するテーブル名を配列で返す.
480     *
481     * @param string $table 対象テーブル名
482     * @param string $name 対象カラム名
483     * @return array インデックス設定情報配列
484     */
485    function sfGetCreateIndexDefinition($table, $name, $definition) {
486        $objQuery =& SC_Query::getSingletonInstance();
487        $arrTblInfo = $objQuery->getTableInfo($table);
488        foreach($arrTblInfo as $fieldInfo) {
489            if(array_key_exists($fieldInfo['name'], $definition['fields'])) {
490                if($fieldInfo['nativetype'] == 'text') {
491                    // TODO: text型フィールドの場合に255文字以内決めうちでインデックス列のサイズとして
492                    //       指定して良いか確認は必要。
493                    $definition['fields'][$fieldInfo['name']]['length'] = '255';
494                }
495            }
496        }
497        return $definition;
498    }
499}
500?>
Note: See TracBrowser for help on using the repository browser.