Category: 小小草

IT 技术领域学海无涯。其实任何领域都学海无涯,无非 IT 发展太快了,让我有更多嘘唏。希望我掌握的技术有如小小草,虽然渺小,却有旺盛的生命力。

  • Upgrade a server from Fedora 10 to 14

    今天终于把一台服务器从 Fedora 10 升到了 Fedora 14,一开始想用 yum 升级(在 fedora upgrader 的帮助下),尝试跳级升级的,不成功。然后用 Fedora 自身的 major upgrade,分别尝试了跳级升级和不跳级升级,都不成功,最后卡在 Fedora 11 拷贝完升级文件但进不了图形界面完成升级,彻底动弹不得。很懊恼,早知如此还不如一开始就全新安装,就是想绕开重建 server 的麻烦,但这看来是绕不开的。

    稍后我列出此次升级的 checklist(以前陆续也写过,但有两年没干这活了,变化还是很大)。

    此次升级损失了 DNS slave zones 的文件。本以为主 Server 有备份,但主 Server 几次搬家,什么时候把那些文件搬丢了也没印象了。

    升级的起因是我需要 php ssh2,却发现 Zend Server 暂不支持。不想经历自行往 Zend Server 添加 php ssh2 的麻烦,就想回到 Fedora 管理下的 php。不久前说过,我觉得 Zend Server 是鸡肋,这下更坚定我的看法。

  • Reinstate Magento order status after upgrade

    Recently I upgraded a Magento store from 1.4.0.1 to 1.5.0.1.

    Magento has changed order tables from key-value type to flat structure in 1.4.1.0, which caused a big problem for my upgrade – all orders lost their status and state value.

    I have not discovered what exact reason made it happen, but I know it is my store specific, because I have other stores successfully upgraded from 1.3 to 1.5.0.1 without any problems.

    Just in case you had the same issue, you can run this mysql query code to reinstate Magento order status(and state as well).

    
    UPDATE magento_sales_flat_order AS o
    INNER JOIN (SELECT s1.parent_id, s1.status, c.state
     FROM magento_sales_flat_order_status_history s1
     JOIN (
     SELECT parent_id, MAX(entity_id) AS entity_id
     FROM magento_sales_flat_order_status_history
     GROUP BY parent_id) AS s2
     ON s1.parent_id = s2.parent_id AND s1.entity_id = s2.entity_id
     LEFT JOIN magento_sales_order_status_state AS c
     ON s1.status = c.status
    ) as s3
    ON o.entity_id = s3.parent_id, magento_sales_flat_order_grid AS g
    SET o.status = s3.status, o.state = s3.state, g.status = s3.status
    WHERE g.entity_id = s3.parent_id;
    
    
  • Activate Magento ajax loading graphic using jQuery

    I extend Magento js using jQuery.
    I need ajax loading mask and grahpic for jQuery ajax request.
    I want to achieve consistent look and feel.
    I want to activate Magento ajax loading graphic initially used in prototype using jQuery.

    Although prototype register global event handlers for ajax create and complete events, the handlers will not fire by jQuery ajax request. For jQuery to activate ajax loading graphic, I manually fire the onCreate event in jQuery function.

    var r = {options:{loadArea:''}};
    varienLoaderHandler.handler.onCreate(r);
    

    to deactivate

    varienLoaderHandler.handler.onComplete();
    

    Let me know if you know a better way to tie jQuery and prototype together.

  • Magento Cush module is about to release

    Long long time ago, I installed Customshippingrate module, but never made it work on my site. The module sits on the disk but disabled. Recently I took some trouble and about three days developing my own module Cush for admin panel users to charge special shipping price when creating an order.

    Cush module does not override any Magento classes so it is virtually 100% compatible with other modules. Neither does Cush module override any Magento templates so basically it reuses Magento native interface. Cush is not a shipping method so it can work with any Magento shipping methods or 3rd party shipping methods.

    Cush module injects customisation logic by javascript. What it actually customise are values in magento_sales_flat_quote_shipping_rate where is pool of shipping quotes. Values customised belong to an individual quote so the customisation hits its point.

    When writing Cush module, I found a defect in Magento own code app/design/adminhtml/default/default/template/sales/order/create/shipping/method/form.phtml.

    
    echo $this->getCarrierName($_rate->getCarrier());
    
    

    The above code reads value from configuration and shows it as carrier title. If I was not injecting shipping method customisation logic, I would not realise it is a defect. Carrier title value, like method title and shipping rate, if read from magento_sales_flat_quote_shipping_rate using $_rate->getCarrierTitle(), will make more sense. Values from configuration will not reflect carrier title changes done by Cush. I respect Magento own templates, especially adminhtml templates. Instead of overriding this template, I wrote some additional javascript in Cush module to correct carrier title. I stick to my green principle although it costs much more time developing.

    Upon installing Cush module, I removed all files of Customshippingrate module. A small accident happened – can not create shipment. It turns out during creating shipment, Magento is gathering all config paths start with “carriers/” and working out all carriers between two slashes. It is a bad logic, or I can call it a bug. It does not check whether this carrier exists or is active. So, very bad.

    Just add to my Magento caveats: after removing a shipping method, remove all entries of “carriers/CARRIER_CODE/*” in magento_core_config_data.

  • Magento datetime picker is not picking up time value

    Magento 用了 dynarch.com 的 calendar 1.0 javascript,有个 bug:无法得到 time 的值。

    dynarch.com calendar 已经是 2.0 了,单独使用的话,能显示和修改 time 的值。

    我暂时没想好该怎么办:我倾向于用 jQuery 去增强 Magento(prototype 我也用不好,其他的就更不要说了),但 jQuery 现下的版本只有 datepicker,官方还没有 datetimepicker。试过很多第三方 jQuery datetimepicker plugin,没觉得某一款有 jQuery 的神韵。

  • I could use Google apps account as my OpenId

    Years ago, I knew OpenId.
    Years ago, I knew Google account supports OpenId as OpenId provider.
    Years ago, I knew Google apps premium version account support SSO.

    And I was misled by many threads that Google apps standard version account does not support OpenId. I did try several times without success, but I did not try hard.

    Today, a great post How to Setup OpenID with Google Apps gave me a big confidence that I can do it. So I tried hard – spent a whole evening trying, and successfully logged into SourceForge with my Google apps acount, and I am using Google apps standard version!

  • Controller override and request rewrite in Magento

    There are three ways to override controller in Magento. They fit for various purposes.

    The first and easiest way can be used to route the request to more than one module. When a request arrives on a frontName, it usually is rounted to a module. For example, /cms/page/view is routed to cms module page controller view action. If I have developed a cms related module called “faq” with a brand new controller “QuestionController”, but I want it share the same cms frontName with cms module, i.e., I want /cms/question/any_action be routed to faq module.

    It is very easy to achieve by a config.xml like the following:

    <config>
    	<frontend>
    		<routers>
    			<cms>
    				<args>
    					<modules>
    						<any_name>MyNamespace_Faq</any_name>
    					</modules>
    				</args>
    			</cms>
    		</routers>
    	</frontend>
    </config>
    

    Strictly speaking, no overriding in above example because it only activates a brand new controller. In a truly overriding example, if I want PageController of faq module override PageController in cms module, I can add before=”Mage_Cms” to make sure PageController of faq module supersede the same name controller in cms module. The complete configuration is shown below:

    <config>
    	<frontend>
    		<routers>
    			<cms>
    				<args>
    					<modules>
    						<any_name before="Mage_Cms">MyNamespace_Faq</any_name>
    					</modules>
    				</args>
    			</cms>
    		</routers>
    	</frontend>
    </config>
    

    The second way can be used to mass override controllers.

    <config>
    	<global>
    		<rewrite>
    			<any_name>
    				<from><![CDATA[#^/cms/#]]></from>
    				<to>/faq/</to>
    				<complete></complete>
    			</any_name>
    		</rewrite>
    	</global>
    </config>
    

    The above configuration makes all controllers in faq module override same name controllers in cms module. In class Mage_Core_Controller_Varien_Front, there is

    $pathInfo = preg_replace($from, $to, $request->getPathInfo());
    

    doing path info string replacement based on regular expression. And because it is based on regular expression, I can do mass replacement at a time.

    The tag flags whether requested path info should be turned into new module path info. In other words, when the request arrives on /cms/page/view, it is rewritten to /faq/page/view. Without tag, the action layout handle is cms_page_view; with tag, the action layout handle is faq_page_view.

    I can use this method to route the request to a brand new controller, i.e.

    <config>
    	<global>
    		<rewrite>
    			<any_name>
    				<from><![CDATA[#^/cms/question/#]]></from>
    				<to>/faq/question/</to>
    				<complete></complete>
    			</any_name>
    		</rewrite>
    	</global>
    </config>
    

    Note that cms module does not have question controller, but the above configuration will not cause any error.

    The third way can be used to override individual actions.

    <config>
    	<global>
    		<routers>
    			<cms>
    				<rewrite>
    					<page>
    						<to>faq/question</to>
    						<override_actions>true</override_actions>
    						<actions>
    							<view_action><to>new_module/new_controller/new_action</view_action>
    						</actions>
    					</page>
    				</rewrite>
    			</cms>
    		</routers>
    	</global>
    </config>
    

    This method is documented in class Mage_Core_Controller_Varien_Action.

    * This will override:
    * 1. cms_module/page_controller/view_action to new_module/new_controller/new_action
    * 2. all other actions of cms_module/page_controller to faq_module/question_module

    It is very handy to precisely control the request rewrite to action level, but it can not route to a brand new controller. The following code will cause an error.

    <config>
    	<global>
    		<routers>
    			<cms>
    				<rewrite>
    					<question><!-- Error: cms module does not have question controller -->
    						<to>faq/question</to>
    						<override_actions>true</override_actions>
    						<actions>
    							<view_action><to>new_module/new_controller/new_action</view_action>
    						</actions>
    					</question>
    				</rewrite>
    			</cms>
    		</routers>
    	</global>
    </config>
    

  • Bxgy 0.1.2 release

    Bxgy 0.1.0 and Bxgy 0.1.1 packaged a layout file bxgy.xml to a wrong place. Bxgy 0.1.2 is a quick release not on schedule.

    Thanks to Carsten for pointing out the error. Sorry to John, Hamichok and Tsk for reporting the bug but I pointed them into wrong direction.

    Down BuyXGetY.tar.gz

  • Is it a serious security leak for WordPress bloggers?

    By WordPress default settings, anyone who knows WordPress blog administrator’s email address can put this address in comment required field, and make comment without being moderated as if he was the administrator. I think it is very bad design of authentication. It worries me a lot although none of my blog users pretend to be me so far.

    I never publish my email address on the web to protect it from spam. But you can find my email address in many places – I print it on business cards, write it on CV, give it to people first met in pubs… Now I have to rethink how to use my email address.

    I think password check is essential if someone is acting like the administrator in WordPress blog. If it can be bypassed, it means I have to keep my email address as my privacy.  Birthday is a privacy, residence address is a privacy. There are so many privacies to keep. And now, email address?

  • jQuery dialog widget moves dialog DOM to the very end of body

    我在 checkout page 上使用 jQuery dialog widget 来显示 terms and conditions。Terms and conditions 的内容不随 checkout 页面加载(因为大部分顾客不会去读它),而在首次点击 Read terms and conditions 链接后以 ajax 获取内容后用 jQuery dialog 展示。再次点击则不必再用 ajax 获取,而直接用 jQuery dialog 展示。

    因为不同种类的产品会对应不同的 terms and conditions,所以 checkout page 会根据 shopping cart 里的所有产品列示出对应的 terms and conditions,可能会有多条。我原想把每条 Read terms and conditions 链接获取的 terms and conditions 内容插入在该链接的后面,在点击事件发生后就可以判断其后的内容 DOM 存在与否决定是否要执行 ajax post 操作。这样可以避免重复获取每个 terms and conditions,以加快用户响应和减少服务器负担。

    可如意算盘打了一通,写出最初稿一运行,发现每点一次 Read terms and conditions,ajax post 都要执行一次。一开始我还以为内容 DOM 插入位置不正确,被判断为没加载过而去执行 ajax post。查了好久才发现,内容 DOM 插入位置是正确的,只是每次执行 jQuery dialog(),它就把 dialog DOM(也就是内容 DOM)移到了 body DOM 的最后。

    这意味着我不能在 Read terms and conditions 链接后寻取对应的内容 DOM,而必须用另外的办法链接和内容的关系。当然办法有很多,这里就不多说了。