0

同じ 2 つのクエリを 3 つのテーブルに適用するので、ループしてテーブル名を置き換えます。最初の INSERT クエリは、常にテスト サーバーと運用環境で機能します。2 番目の DELETE クエリは 2 つの異なるテスト サーバーで動作しますが、本番環境では失敗し、mysql エラー 1046 No database selected がスローされます。

影響を受けるコードの抜粋:

    $queries = array(
        "INSERT INTO knd_bkg.@table@
            SELECT DISTINCT SQL_NO_CACHE
            B.ASIN, Author, Title, EditorialReview, DetailPageURL, current_price, salesrank, number_reviews, erotica, G.BrowseNodeId, price_salesrank_checked_on
            FROM bkd_books.club_books B
            USE INDEX (free_quality)
            RIGHT JOIN bkd_books.book_genre G on G.ASIN=B.ASIN
            WHERE EXISTS
            (
              SELECT 1 FROM knd_bkg.genres K WHERE K.browsenode=G.BrowseNodeId
            )
            AND @priceWhere@
            AND price_salesrank_checked_on > '@newerThan@'
            AND B.number_reviews >= 4
            AND B.rating >= 4.0
            AND B.language = 'english'
            AND B.hide_knd=0
            AND B.public_domain=0;",
        "DELETE T FROM knd_bkg.@table@ T
         INNER JOIN
         (
             SELECT N.* FROM knd_bkg.@table@ N
             INNER JOIN
             (
                 SELECT DISTINCT ASIN, BrowseNodeId FROM knd_bkg.@table@
                 INNER JOIN knd_bkg.genres ON knd_bkg.genres.browsenode=knd_bkg.@table@.BrowseNodeId
                 WHERE isFiction=1
             ) F
             ON F.ASIN=N.ASIN
             INNER JOIN knd_bkg.genres
             ON knd_bkg.genres.browsenode=N.BrowseNodeId
             WHERE knd_bkg.genres.isNonFiction=1
         ) D
         USING (ASIN, BrowseNodeId)
         WHERE D.ASIN=T.ASIN AND D.BrowseNodeId=T.BrowseNodeId"
    );

    $tables = array(
        "free_books" => array('priceWhere' => "B.current_price = 0", 'timePeriod' => $freeTimePeriod),
        "99_books" => array('priceWhere' => "B.current_price > 0 AND B.current_price < 100", 'timePeriod' => $paidTimePeriod),
        "399_books" => array('priceWhere' => "B.current_price >= 100 AND B.current_price < 400", 'timePeriod' => $paidTimePeriod),
    );

    connectSlaveDB();

    foreach ($tables as $table => $data) {
        $newerThan = date("Y-m-d H:i:s", strtotime("-" . $data['timePeriod'] . " hours"));
        foreach ($queries as $query) {
            $query = str_replace('@table@', $table, $query);
            $query = str_replace('@newerThan@', $newerThan, $query);
            $query = str_replace('@priceWhere@', $data['priceWhere'], $query);
            Logger::_write(LOG_VERBOSE, "API - api/bkg/rebuild query: $query");
            $result = mysql_query($query);
            if (!$result) {
                Logger::_write(LOG_ERR, "API - api/bkg/rebuild FAILED: " . mysql_error());
                $success = false;
            } else {
                Logger::_write(LOG_VERBOSE, "API - api/bkg/rebuild SUCCESS");
            }
        }
    }

そして connectSlaveDB()

function connectSlaveDB()
{
    global $dbSlave, $dbUser, $dbPass, $dbName;
    $connSlave = mysql_connect($dbSlave, $dbUser, $dbPass, TRUE);
    if (!$connSlave) {
        Logger::_write(LOG_CRIT, "Unable to connect to slave DB");
        include 'screens/error.inc.php';
        exit();
   }
    mysql_set_charset('utf8', $connSlave);
    mysql_select_db($dbName, $connSlave);
    return TRUE;
}

テーブルはMEMORYテーブルですが、MYISAMで同じエラーが発生し、本番環境でのみ発生します。このクエリはレプリケーション スレーブで実行されており、マスターは関与していませんが、前のコードはテーブルを TRUNCATES にしているため、パーミッションに依存しているとは思えません。クエリがログ ファイルから取り出されると、phpMyAdmin で正常に動作します。そして、テストサーバーでは問題ありません。

1046 No database selected エラーの理由がわかりません。そして 3 時間後には、本当に明白な何かを見落としている可能性があります。特に、1 つの DB 接続の場合、障害パターンは次のようになります。

INSERT on free_books - success
DELETE on free_books - fail
INSERT on 99_books - success
DELETE on 99_books - fail
INSERT on 399_books - success
DELETE on 399_books - fail

醜いグローバルと非推奨の mysql* 関数をお許しください。これはかなり時代遅れのレガシーアプリです。知っている...

4

2 に答える 2

0

DELETE の T エイリアスが問題を引き起こしているように見えますが、その理由はよくわかりません。この問題は、DELETE クエリを次のようにすることで修正されました。

"DELETE knd_bkg.@table@ FROM knd_bkg.@table@
     INNER JOIN
     (
         SELECT N.* FROM knd_bkg.@table@ N
         INNER JOIN
         (
             SELECT DISTINCT ASIN, BrowseNodeId FROM knd_bkg.@table@
             INNER JOIN knd_bkg.genres ON knd_bkg.genres.browsenode=knd_bkg.@table@.BrowseNodeId
             WHERE isFiction=1
         ) F
         ON F.ASIN=N.ASIN
         INNER JOIN knd_bkg.genres
         ON knd_bkg.genres.browsenode=N.BrowseNodeId
         WHERE knd_bkg.genres.isNonFiction=1
     ) D
     USING (ASIN, BrowseNodeId)
     WHERE D.ASIN=knd_bkg.@table@.ASIN AND D.BrowseNodeId=knd_bkg.@table@.BrowseNodeId"
于 2013-07-18T09:43:45.647 に答える
0

DELETE クエリの構文が問題を引き起こしている可能性があります

DELETE T FROM

これにより、mysql のバグが原因で、UPDATE クエリで問題が発生することがありました。

参照を回避するようにクエリを書き直すと、Tこの問題が解決するはずです。

于 2013-07-18T10:06:08.257 に答える