Pages

Sunday, June 30, 2013

Magento Advanced Collection Filters

I always seem to end up in a position where I need more specific information from a collection than addFilter() can give me. Functions like addAttributeToFilter() are great if you’re working with EAV-based collections like products and categories, but they don’t work with a lot of core Magento collections. In this post I’ll go into some detail on how to use the ->getSelect() function to filter collection results. This post comes with a warning: The methods detailed below are intended primarily for data output, not data manipulation. Be careful saving models returned from your original collection.

Calling $collection->getSelect() on your collection will return a model of class type Varien_Db_Select. This model represents the SQL query that is performed to select your collection. I’ve created a module called Mby_Testmodule, and within it I have a model of type testmodule/comment. Let’s get the Select element and play with it a bit.

<?php
        $collection = Mage::getModel('testmodule/comment')->getCollection();
        $select = $collection->getSelect();
        echo $select->__toString();

This will echo the following:

SELECT `main_table`.* FROM `mby_testmodule_comment` AS `main_table`

Essentially we’re returning all data. If I want to filter that result I can use a number of functions. Lets search for all comments by people whose name starts with “abcefg”:

<?php
    $name = "abcefg%";
    $select->where("author_name LIKE ?", $name);
    echo $select->__toString();
?>

Outputs:
SELECT `main_table`.* FROM `mby_testmodule_comment` AS `main_table` WHERE (author_name LIKE 'abcefg%')

If we want to reverse the order of the collection:

<?php
    $select->order("comment_id DESC");
    echo $select->__toString();
?>

Outputs:
SELECT `main_table`.* FROM `mby_testmodule_comment` AS `main_table` WHERE (author_name LIKE 'abcefg%') ORDER BY `comment_id` DESC

If we want to then limit the results to the first 20 comments, we can use:

<?php
    $select->limit(20);
    echo $select->__toString();
?>

Outputs:
SELECT `main_table`.* FROM `mby_testmodule_comment` AS `main_table` WHERE (author_name LIKE 'abcefg%') ORDER BY `comment_id` DESC LIMIT 20

Or if we wanted to limit the results to 20 comments, starting from comment 10:

<?php
    $select->limit(20, 10);
    echo $select->__toString();
?>

Outputs:
SELECT `main_table`.* FROM `mby_testmodule_comment` AS `main_table` WHERE (author_name LIKE 'abcefg%') ORDER BY `comment_id` DESC LIMIT 20 OFFSET 10

Say we also wanted to join these comments to the blog posts that they were made against, we could use an inner join like so:

<?php
    $select->joinInner(
        array(
            'blogpost_table' => 'mby_testmodule_blogpost'
        ),
        'blogpost_table.blogpost_id = main_table.blogpost_id'
    );
    echo $select->__toString();
?>

Outputs:
SELECT `main_table`.*, `blogpost_table`.* FROM `mby_testmodule_comment` AS `main_table` INNER JOIN `mby_testmodule_blogpost` AS `blogpost_table` ON blogpost_table.blogpost_id = main_table.blogpost_id WHERE (author_name LIKE 'abcefg%') ORDER BY `comment_id` DESC LIMIT 20 OFFSET 10

You get the idea.

Modifying the Select model will filter the data returned by your collection when you iterate through it. Implementing the process above will look something like the following:

<?php
        $name = "abcefg%";
        $collection = Mage::getModel('testmodule/comment')->getCollection();
        $collection->getSelect()
            ->where("author_name LIKE ?", $name)
            ->order("comment_id DESC")
            ->limit(20, 10)
            ->joinInner(
                array(
                    'blogpost_table' => 'mby_testmodule_blogpost'
                ),
                'blogpost_table.blogpost_id = main_table.blogpost_id'
            );
        foreach ($collection as $comment) {
            echo "Comment #".$comment->getCommentId();
            echo " by ".$comment->getAuthorName();
            echo " in response to ".$comment->getBlogpostTitle();
            echo "<br />";
            echo $comment->getContent();
        }

How to add Javascript to a Magento Admin page (AKA: How to Override Magento Blocks)

Recently I had to add some additional Javascript validation to a core part of the Magento Admin, and I thought I’d share the best way to do this without having to modify core files.

Because I often have to modify small parts of default Magento functionality, I have created a module that exists for the sole purpose of overriding models and blocks. In this post I’ll go through the steps required to override blocks properly, without modifying core code. The process for models is almost identical. I’m going to use as an example the block that I was required to override: Mage_Adminhtml_Block_Promo_Quote_Edit. All I want to do is add some additional Javascript to the page that hijacks the “Save” event and asks the user if they are sure.

First things first: build a custom module (or pick an existing one) to work from.

in {namespace}/{modulename}/etc/config.xml, add the following:

<config>
   ...
   <global>
      ...
      <blocks>
         <adminhtml>
            <rewrite>
               <promo_quote_edit>{namespace}_{modulename}_Block_Adminhtml_Promo_Quote_Edit</promo_quote_edit>
            </rewrite>
         </adminhtml>
      </blocks>
      ...
   </global>
   ...
</config>

What we’ve just done is tell Magento that when it looks for the block identified as “adminhtml/promo_quote_edit“, load this block class: {namespace}_{modulename}_Block_Adminhtml_Promo_Quote_Edit

We are therefore required to define this class. We do so in path that the class suggests. In this case, it’s: app/code/local/{namespace}/{modulename}/Block/Adminhtml/Promo/Quote/Edit.php. Create this file, and write the following:

<?php
    class {namespace}_{modulename}_Block_Adminhtml_Promo_Quote_Edit
        extends Mage_Adminhtml_Block_Promo_Quote_Edit
    {

By using “extends” and giving the class name of the original block, our new class inherits all of the functions and variables of it’s parent class. If we left the file like this, Magento would act no differently from a user’s perspective.

To give that block new functionality, add as many new functions as you like. But to override a function, as I am doing here, redefine that function within your new class. One very important thing to remember, however, is to ensure that any overridden functions perform the same tasks as the parent, or you risk interfering unintentionally with Magento’s core functionality.

I want to add more Javascript, and I happen to know that Javascript is added into the $_formScripts variable that belongs to this block. I also know that Javascript tends to be added on __construct(). To ensure the class acts the same way as it always did, I define the __construct function like this:

<?php
    public function __construct()
    {
        parent::__construct();
       
        $this->_formScripts[] = "
            editForm.secondarySubmit = editForm.submit;
            editForm.submit = function(url) {
                if (document.getElementById('rule_coupon_type').value == 1) {
                    var answer = confirm('WARNING: This price rule has no Coupon Code.
                    Are you sure you want to do this?');
                } else {
                    var answer = true;
                }
                if (answer) {
                    editForm.secondarySubmit(url);
                } else {
                    return false;
                }
            }
        ";
    }
?>

Note the parent::__construct(); call, that will run the __construct() function of Mage_Adminhtml_Block_Promo_Quote_Edit

In this case, the Javascript I added uses the prototype library to redefine what happens when the form is saved.

Pretty simple really!

Programmatically create Shopping Cart Price Rules with Conditions and Actions

Rather than going into detail of creating the Shopping Cart Price Rule from scratch, this post is specifically about conditions and actions.



If you have ever needed to programmatically create Magento Shopping Cart Price Rules, then you may have wondered how you attach conditions and actions.

Prior to saving your Mage_Salesrule_Model_Rule object, you will need to assign the following data:

<?php
    $rule->setData('actions',$actions);       
    $rule->setData('conditions',$conditions);
?>

Where $actions and $conditions are Array’s of a very specific format.

We then call Mage_Salesrule_Model_Rule::loadPost():

<?php
    $rule->loadPost($rule->getData());
?>

The format that is required for conditions and actions is significantly outside Magento’s EAV model, and is not documented as far as I can see. So to find it out we are required to trace it a little further.

Conditions and actions are processed within the Mage_Salesrule_Model_Rule::loadPost() function, and are converted to a recursive format that you find serialized in the database as conditions_serialized and actions_serialized table columns. This is done via Mage_Rule_Model_Rule::_convertFlatToRecursive() which is called within Mage_Salesrule_Model_Rule::loadPost().

But I digress, a little. What this means for us is that we need to put our data in a specific format prior to saving. I’ve established that format for you below. If you hijack Mage_Salesrule_Model_Rule::loadPost() and print_r() the content of a rule as it’s being saved, you’ll see the format:


    [conditions] => Array
        (
            [1] => Array
                (
                    [type] => salesrule/rule_condition_combine
                    [aggregator] => all
                    [value] => 1
                    [new_child] =>
                )
            [1--1] => Array
                (
                    [type] => salesrule/rule_condition_product_found
                    [value] => 1
                    [aggregator] => all
                    [new_child] =>
                )
            [1--1--1] => Array
                (
                    [type] => salesrule/rule_condition_product
                    [attribute] => sku
                    [operator] => ==
                    [value] => SKU123
                )
        )

    [actions] => Array
        (
            [1] => Array
                (
                    [type] => salesrule/rule_condition_product_combine
                    [aggregator] => all
                    [value] => 1
                    [new_child] =>
                )
            [1--1] => Array
                (
                    [type] => salesrule/rule_condition_product
                    [attribute] => sku
                    [operator] => ==
                    [value] => SKU123
                )
        )

In Layman’s terms, this is saying:


    [conditions][1]        
        "If ALL(aggregator) of these conditions are TRUE(value)"
    [conditions][1--1]     
        "If an item is FOUND(type, value) with ALL(aggregator) of these conditions true"
    [conditions][1--1--1]  
        "SKU(attribute) is(operator) SKU123(value)"
   
    [actions][1]           
        "if ALL(aggregator) of these conditions are TRUE(value)"
    [actions][1--1]        
        "SKU(attribute) is(operator) SKU123(value)"

So what are we really looking at? We’re looking at an array of arrays that uses the array keys to establish the heirarchy. The array keys are exploded by “--” in Mage_Rule_Model_Rule::_convertFlatToRecursive() and are converted to recursive arrays before being stored in the database.

This is how we can write it:

<?php
    $rule = Mage::getModel('salesrule/rule')->load($my_rule_id);
    $conditions = array(
        "1"         => array(
                "type"          => "salesrule/rule_condition_combine",
                "aggregator"    => "all",
                "value"         => "1",
                "new_child"     => null
            ),
        "1--1"      => array(
                "type"          => "salesrule/rule_condition_product_found",
                "aggregator"    => "all",
                "value"         => "1",
                "new_child"     => null
            ),
        "1--1--1"   => array(
                "type"          => "salesrule/rule_condition_product",
                "attribute"     => "sku",
                "operator"      => "==",
                "value"         => "SKU123"
            )
    );
    $actions = array(
        "1"         => array(
                "type"          => "salesrule/rule_condition_product",
                "aggregator"    => "all",
                "value"         => "1",
                "new_child"     => false
            ),
        "1--1"      => array(
                "type"          => "salesrule/rule_condition_product",
                "attribute"     => "sku",
                "operator"      => "==",
                "value"         => "SKU123"
            )
    );
    $rule->setData("conditions",$conditions);
    $rule->setData("actions",$actions);
    $rule->loadPost($rule->getData());
    $rule->save();
?>

How to get a grouped product’s associated products in Magento

This is a question that seems to be asked quite often by new Magento developers.

There is no simple Mage_Catalog_Model_Product::getAssociatedProducts() method, or similar to return all simple products assigned to a grouped product. I outline here how Magento gains access to the collection that we need.

In

/magento/app/design/frontend/base/default/template/catalog/product/view/type/grouped.phtml

you’ll see that they use this:

<?php
    $_associatedProducts = $this->getAssociatedProducts();

Since that .phtml file is of type Mage_Catalog_Block_Product_View_Type_Grouped, we can go to

/magento/app/code/core/Mage/Catalog/Block/Product/View/Type/Grouped.php

and see in Mage_Catalog_Block_Product_View_Type_Grouped::getAssociatedProducts() that they have done this:

<?php
    $this->getProduct()->getTypeInstance(true)->getAssociatedProducts($this->getProduct());

So we can safely assume that $this->getProduct() returns a product object, and replace it with your $product variable like so:

<?php
    $associatedProducts = $product->getTypeInstance(true)->getAssociatedProducts($product);

So a total solution would look something like this:

<?php
    $products = Mage::getModel('catalog/product')
        ->getCollection()
        ->addAttributeToFilter('type_id', array('eq' => 'grouped'));
    foreach ($products as $product) {
        $associatedProducts = $product->getTypeInstance(true)->getAssociatedProducts($product);
        // Do something with the $associatedProducts collection
    }

Adding Surcharges in Magento

Recently I needed to add a surcharge to orders using a particular payment type. Surcharges are what is called a “Total”. They need to be created in the same way as Delivery, Discounts, etc. This post assumes knowledge of module creation.

Please note that this post is incomplete! I’m pressed for time and I’ll be back to edit this with more detailed information on WHY we do it.

So first off, create a new Module for Surcharges in magento/app/code/local/<namespace>/Surcharges/ with the usual subfolders. Don’t forget to create the module’s xml file in /app/etc/modules/.

in <namespace>/Surcharges/etc/config.xml:

<?php
    <config>
        ...
        <global>
            ...
            <sales>
                <quote>
                    <totals>
                        <surcharge>
                            <class>surcharges/quote_address_total_surcharge</class>
                            <after>tax</after>
                        </surcharge>
                    </totals>
                </quote>
            </sales>
            <fieldsets>
                <sales_convert_quote>
                    <surcharge><to_order>*</to_order></surcharge>
                    <surcharge_tax><to_order>*</to_order></surcharge_tax>
                </sales_convert_quote>
            </fieldsets>
            ...
        </global>
        ...
    </config>
?>

in <namespace>/Surcharges/sql/surcharges_setup/mysql4-install-0.1.0.php

<?php

    $installer = $this;

    $installer->startSetup();

    $installer->addAttribute(
        'order',
        'surcharge',
        array(
            'type' => 'float',
            'grid' => false
        )
    );
    $installer->addAttribute(
        'order',
        'surcharge_tax',
        array(
            'type' => 'float',
            'grid' => false
        )
    );

    $installer->addAttribute(
        'quote',
        'surcharge',
        array(
            'type' => 'float',
            'grid' => false
        )
    );
    $installer->addAttribute(
        'quote',
        'surcharge_tax',
        array(
            'type' => 'float',
            'grid' => false
        )
    );

    $installer->endSetup();

in <namespace>/Surcharges/Model/Quote/Address/Total/Surcharge.php:

<?php

class {{namespace}}_Surcharges_Model_Quote_Address_Total_Surcharge
    extends Mage_Sales_Model_Quote_Address_Total_Abstract
{
   
    public function __construct()
    {
        $this->setCode('surcharge');
    }

    public function collect(Mage_Sales_Model_Quote_Address $address)
    {       
        $amount = $address->getShippingAmount();
        if ($amount != 0 || $address->getShippingDescription()) {
                       
            $address->setSurchargeAmount($this->getSurchargeAmount());
            $address->setSurchargeTaxAmount($this->getSurchargeTaxAmount());
           
            $address->setSurcharge($this->getSurchargeAmount());
            $address->getQuote()->setData('surcharge', $this->getSurchargeAmount());
            $address->getQuote()->setData('surcharge_tax', $this->getSurchargeTaxAmount());
           
            $address->setTaxAmount($address->getTaxAmount() + $address->getSurchargeTaxAmount());
            $address->setBaseTaxAmount(
                $address->getBaseTaxAmount() + $address->getSurchargeTaxAmount()
            );
            $address->setSubtotal($address->getSubtotal() + $this->getSurchargeAmount(true));
            $address->setBaseSubtotal(
                $address->getBaseSubtotal() + $this->getSurchargeAmount(true)
            );
            $address->setGrandTotal($address->getGrandTotal() + $address->getSurchargeAmount());
            $address->setBaseGrandTotal(
                $address->getBaseGrandTotal() + $address->getSurchargeAmount()
            );
           
        }
        return $this;
    }
   
    public function fetch(Mage_Sales_Model_Quote_Address $address)
    {
        $amount = $address->getShippingAmount();
        if ($amount != 0 || $address->getShippingDescription()) {
            if ($address->getSurchargeAmount()) {
                $address->addTotal(array(
                    'code'  => $this->getCode(),
                    'title' => $this->getSurchargeTitle(),
                    'value' => $address->getSurchargeAmount()
                ));
            }
        }
        return $this;
    }
   
    public function getSurchargeTitle()
    {
        $title = "Surcharges";
        return $title;
    }
   
    public function getSurchargeAmount()
    {
        $amount = (float)10;
        return $amount;
    }
   
    public function getSurchargeTaxAmount()
    {
        $taxpercent = (float)10;
        $tax = ($this->getSurchargeAmount()/100)*$taxpercent;
        return $tax;
    }
   
}

Upgrading Magento Enterprise

Unfortunately there is no magic button you can push to upgrade your version of Enterprise.

If you have enterprise, you’ll be paying for support. I recommend you contact Magento Support before going any further. There’s no documentation on how to do it yourself. Also, if you’re not a System Admin or a Developer, turn back now.

It shouldn’t need to be said, but here I go anyway: Do this on a development server or at least a staging environment first. Do not just upgrade your live site without doing it elsewhere first. Also, for the record this post will be vague. There is a lot of assumed knowledge within. I use examples from an Apache server as I imagine many people will be using Apache. I know I do.

Log in to your enterprise account on www.magentocommerce.com and click on “My Account”. Click “Downloads”, and the latest versions of Magento Enterprise will be listed on the right, among other things such as changelog files. Download your desired version.

Essentially there are only a few key steps to the process:

1) Back up your database.

2) Turn off Cron! You don’t want things re-indexing while Magento modifies database tables.

3) Back up all files from your Magento working directory, and move the files to a different folder (eg: rsync -aPzx var/www/magento var/www/magento_backup).

4) Extract the files from the downloaded version to var/www/magento

5) Make a list of all custom content that needs to be installed on the new version. My list of things generally includes:

    everything in app/code/local,
    files in app/etc/modules,
    app/etc/local.xml
    custom themes & skins in app/design/frontend/enterprise and app/design/adminhtml/default/default, and
    a lot of things in the media, js & skin folders.

If you’ve been a good little developer and haven’t edited any core files, you’ll be patting yourself on the back at this very moment. If not, you’re out of luck, really. Go back and do things properly. Don’t copy across any core files you’ve edited as you’re likely to break lots of things or undo new bug fixes.

6) Copy the things in your list from var/www/magento_backup to var/www/magento

7) Load the site in a browser window. This could take a while. There are a number of things that can go wrong at this point. For example:

    You may get some fatal errors. If this is the case, you’ll need to track them back to their source and find out why they are being thrown. Did you forget to migrate some files? Are you overriding core classes that have now changed? Are you using functionality that has been deprecated? Have you reached this step previously and now cached data is breaking your shit? Get in there and have a look.
    Your mysql server may get stuck in an infinite loop. With the new version will come a lot of new setup files that modify the database. They could be adding new tables, new columns, new keys or simply modifying elements. For instance, if Magento tries to add UNIQUE to an existing field that contains duplicate values, it will fail and repeatedly retry.

8) Once the site loads, and loads with no fatal errors, TEST! Test all of your modules, place test orders, just go nuts! This is by far the most important step and cannot be taken seriously enough.

Finally, when you’re happy with the level that you’re at:

9) Do it all again for your live site, or if this is a staging server, make it live for all to see.

Magento :: get URL paths for skin, media, Js OR base URL

If anyone of you facing problems to fetch the Magento URL paths of  skin, media, Js or simple base URL of Magento while customization/programming, then following lines should be helpful for you:

Mage::getBaseUrl(Mage_Core_Model_Store::URL_TYPE_JS);
//http://magento.demo/js/

Mage::getBaseUrl(Mage_Core_Model_Store::URL_TYPE_LINK);
//http://magento.demo/index.php/

Mage::getBaseUrl(Mage_Core_Model_Store::URL_TYPE_MEDIA);
//http://magento.demo/media/

Mage::getBaseUrl(Mage_Core_Model_Store::URL_TYPE_SKIN);
//http://magento.demo/skin/

Mage::getBaseUrl(Mage_Core_Model_Store::URL_TYPE_WEB);
//http://magento.demo/

Friday, June 28, 2013

Add Javascript/CSS to page Head from within a Magento Block

This is a pretty easy one, but it’s something that can come in very handy at times.

Say you want to load a particular script on the page, but only if a particular block loads on that page. As it stands, Magento’s core/template blocks don’t support this.

To do it, include this function from within your custom module’s Block file in app/code/local/Namespace/Modulename/Block

public function addItem($type, $path)
{
    $head = $this->getLayout()->getBlock('head');
    return $type == 'css' ? $head->addCss($path) : $type == 'javascript' ? $head->addJs($path) : $this ;
}

Then to use it, just include a call to this function from within your block declaration in the layout:

<block type="modulename/blockname" name="modulename.blockname" as="blockname" template="path/to/template/file.phtml">
    <action method="addItem"><type>css</type><path>css/path/to/file.css</path></action>
    <action method="addItem"><type>javascript</type><path>path/to/file.js</path></action>
</block>

When that block is constructed, it will run the function and your custom scripts will load in the head.

Automatically refresh Magento cache

In Magento, whenever you make changes to products, static blocks, etc, it recognizes that the data in the database is no longer the same as what it has in the cache. Unfortunately, Magento doesn’t realize what cache data is different, just that something is different.

Traditionally, you will need to go into System > Cache Management and refresh the invalidated cache types, but I’ve overcome this by using a cron job that runs every time cron runs on the server, and calls a function to refresh the cache automatically.


Whenever you make changes, magento fires events. There are listeners to these events that invalidate the relevant cache. As for why it does this (and why it doesn’t automatically refresh) this is ultimately a design decision, but probably has something to do with being able to stage content. For example, you could make changes to several products that all relate to one another, and can then refresh the cache.

Before using this functionality, you’ll need to ensure that cron is set up and configured correctly on your server. I’m not the right person to ask regarding this, but I’m sure google has some tutorials relevant to your server configuration.

Create a module (or use an existing module) that you can use to set up a cron job for refreshing the cache.

Create a file: {{namespace}}/{{modulename}}/Model/Observer.php

Inside that file:

  <?php

  class <namespace>_<modulename>_Model_Observer {

    public function refreshCache() {
      try {
        $allTypes = Mage::app()->useCache();
        foreach($allTypes as $type => $blah) {
          Mage::app()->getCacheInstance()->cleanType($type);
        }
      } catch (Exception $e) {
        // do something
        error_log($e->getMessage());
      }
    }

  }

In your module’s etc/config.xml:

  <config>
    ...
    <crontab>
      <jobs>
        <{{modulename}}_refresh_cache>
          <schedule><cron_expr>* * * * *</cron_expr></schedule>
          <run><model>{{modulename}}/observer::refreshCache</model></run>
        </{{modulename}}_refresh_cache>
      </jobs>
    </crontab>
    ...
  </config>

Now as long as cron is configured correctly on your server, the cache will update automatically as often as cron runs.

Tuesday, June 25, 2013

Magento (How to fix): One or more of the Cache Types are invalidated: Blocks HTML output.


Somewhere around Magento 1.5, message from the title of this post begun to pop on every product save.
Although quite anoying, it is quite easy to fix and it seems that’s not a BUG, it is a feature – implemented without automatic block html cache refresh :)
I have tested it on Professional Edition and to be completely honest, I’m not sure if it will actually work on Magento CE,
but there is no reason why not (Please comment if it does).
Ok, what is the catch?!
Here we go:
Add this in your module config.xml:

<global>

<models>

<catalogrule>

    <rewrite>

        <rule>Yourpackage_Yourmodule_Model_Rule</rule>

    </rewrite>

</catalogrule>

</models>

</global>

Create file called Rule.php in YOUR module Model directory.

Add this code in newly created file:

class Yourpackage_Yourmodule_Model_Rule extends Mage_CatalogRule_Model_Rule

{

   /**

     * Apply all price rules to product

     *

     * @param int|Mage_Catalog_Model_Product $product

     * @return Mage_CatalogRule_Model_Rule

     */

    public function applyAllRulesToProduct($product)

    {

        $this->_getResource()->applyAllRulesForDateRange(NULL, NULL, $product);

        $this->_invalidateCache();

        //Notice this little line

    Mage::app()->getCacheInstance()->cleanType('block_html');

        $indexProcess = Mage::getSingleton('index/indexer')->getProcessByCode('catalog_product_price');

        if ($indexProcess) {

            $indexProcess->reindexAll();

        }

    }

}

Well…. that’s it :)

I hope it works for you as it works for me.