catalog_product_save_afterとcatalog_product_save_commit_afterの違いは?


8

誰もがこれらのイベントの違いを説明できますか。早くて汚いだけです。ありがとうございました。

私はそのようなオブザーバーメソッドを持っています:

public function detectProductChanges($observer)
    {
        $product = $observer->getProduct();
        $old = $product->getOrigData();
        $new = $product->getData();
        if ($product->hasDataChanges() && $old['status'] == 1 && $new['status'] == 2) {
            $this->_sendStatusMail($product);
        }
    }

に到達していません sendStatusMail()

私はイベントにフックしています:

        <events>
            <catalog_product_save_after>
                <observers>
                    <productchange>
                        <type>singleton</type>
                        <class>A_ProductNotification_Model_Observer</class>
                        <method>detectProductChanges</method>
                    </productchange>
                </observers>
            </catalog_product_save_after>
        </events>

私は使用する必要があります: catalog_product_save_commit_after

ゴール:

製品が無効になった後にメールを送信します。

private function _sendStatusMail($product)
    {
        if (!Mage::getStoreConfig('trans_email/ident_custom3/email')) return false;
        $emailTemplate = Mage::getModel('core/email_template');
        $emailTemplate->loadDefault('elec_productnotification_tpl');
        $emailTemplate->setTemplateSubject('Product has been disabled');
        $emailTemplate->setSenderEmail($salesData['email']);
        $emailTemplateVariables['style_number']   = $product->getElecStyle();
        $emailTemplateVariables['frame_color']    = $product->getAttributeText('frame_color');
        $emailTemplateVariables['size']           = $product->getAttributeText('size');
        $emailTemplateVariables['elec_color'] = $product->getAttributeText('elec_color');
        $emailTemplateVariables['store_name']   = Mage::getModel('core/store')->load($product->getStoreId())->getName();
        $emailTemplateVariables['product_name'] = Mage::getModel('catalog/product')->load($product->getId())->getName();
        $emailTemplateVariables['product_sku']  = $product->getSku();
        $emailTemplateVariables['dates']        = date("F jS Y h:i:sA", strtotime('-7 hours'));
        // Get General email address (Admin->Configuration->General->Store Email Addresses)
        $emails = explode(',', Mage::getStoreConfig('trans_email/ident_custom3/email'));
        foreach ($emails as $email) $emailTemplate->send($email, $product->getStoreId(), $emailTemplateVariables);
    }
}

イベントを使用する必要があります <catalog_product_status_update>
Nickool

それが発砲しない理由ですか?単に間違ったイベントを使用していますか?@Nickool
thismethod

回答:


14

保存はMySQLトランザクションでsave_after行われ、トランザクションがコミットされる前にイベントがトリガーされるため、同じトランザクション内でデータベースに追加の更新を行うことができます。

save_commit_afterトランザクションがコミットされた後にイベントがトリガされた、すなわち変更がデータベースに書き込まれたとき。

また、でsave_commit_after_hasDataChangesプロパティは既ににリセットされているfalseため、チェックは機能しません。一方、変更がなければ、Mage_Core_Model_Abstract :: save()はデータの変更がない場合は何もしないため、両方のイベントはトリガーされません。

if (!$this->_hasModelChanged()) {
    return $this;
}

とはいえ、コードが機能しない理由はわかりません。


答えてくれてありがとう。sendStatusMail()の代わりにMage :: log()を追加すると、ログメッセージが正しく表示されます。ただし、メールは送信されません。[メール通信を無効にする]が[いいえ]に設定されていることと、メールアドレスがカスタムメール>ストアのメールアドレスにあることを確認しました なぜそれが機能しないのですか?@fschmengler
thismethod

sendStatusMailメソッドを知らなければ、違います。それはおそらく別の質問にとって重要です。または、別のコンテキストから呼び出された場合、同じメソッドは機能しますか?
Fabian Schmengler 2016年

元の質問を更新してsendStatusMailメソッドを表示しました。あなたがさらに援助をしてもかまわない場合。ありがとうございました。
メソッドは、2016年

私のsendStatusMail($ product)メソッドについてあなたの意見を教えていただけませんか?
方法は、

私は申し訳ありませんが、そこにすべてのエラーを発見することはできません
ファビアンSchmengler

0

vendor / magento / framework / Model / ResourceModel / Db / AbstractDb.php

public function save(\Magento\Framework\Model\AbstractModel $object)
{
    // ...

    $this->beginTransaction();

    try {
        // ...
        if ($object->isSaveAllowed()) {
            // ...
            $this->_beforeSave($object);
            // ...
            if ($this->isObjectNotNew($object)) {
                $this->updateObject($object);
            } else {
                $this->saveNewObject($object);
            }
            // ...
            $this->processAfterSaves($object);
        }
        $this->addCommitCallback([$object, 'afterCommitCallback'])->commit();
        // ...
    } catch (\Exception $e) {
        $this->rollBack();
        $object->setHasDataChanges(true);
        throw $e;
    }
    return $this;
}

製品エンティティの保存を見てみましょう。

-product_model save
|-product_resource save
|--begin transaction (0 lvl)
|---before product save events
|---creating new product or updating existing one
|---after product save events
|----one of event is saving another entity CatalogInventory Stock
|-----catalog_inventory_stock resource save
|------begin another transaction (1 lvl)
|-------before stock save events
|-------updating / creating stock item
|-------after product save events (here could be one more 
        dependable entity which could cause one more save
        operation and begin another transaction)
|------commit of 1st level !!! No callbacks executed
|--commit of 0 level ALL CALLBACKS ARE EXECUTED

コミット機能のコードは次のとおりです。

/**
 * Commit resource transaction
 *
 * @return $this
 * @api
 */
public function commit()
{
    $this->getConnection()->commit();
    /**
     * Process after commit callbacks
     */
    if ($this->getConnection()->getTransactionLevel() === 0) {
        $callbacks = CallbackPool::get(spl_object_hash($this->getConnection()));
        try {
            foreach ($callbacks as $callback) {
                call_user_func($callback);
            }
        } catch (\Exception $e) {
            $this->getLogger()->critical($e);
        }
    }
    return $this;
}

私たちの例をもっと詳しく見てみましょう。

  1. $this->getConnection()->commit();1番目のレベル(ストック)の値をDBに入力します。ここで何か悪いことが起こった場合、例外がスローされ、すべての変更がロールバックされます。

  2. 次に、コールバックを処理します。現在第1レベルであるため、コールバックは呼び出されません。そして、製品の変更(0レベル)をコミットするために、catalog_product_after_saveイベントから移動します。

  3. $this->getConnection()->commit();0レベルの値をDBに入れます(製品自体です)。ここで何か悪いことが起こった場合、例外もスローされ、すべての変更もロールバックされます。

  4. 次に、コールバックの実行に移ります。これで0レベルになり、コールバックが実行されます。内部であなたが悪いcall_user_func($callback);ことはすべて追いつき、ただログに記録されます。コールバックで例外が発生しても、何もロールバックされません

弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.