Skip to content

Instantly share code, notes, and snippets.

@leek
Created February 29, 2012 16:19
Show Gist options
  • Star 12 You must be signed in to star a gist
  • Fork 0 You must be signed in to fork a gist
  • Save leek/1942116 to your computer and use it in GitHub Desktop.
Save leek/1942116 to your computer and use it in GitHub Desktop.
INSERT ... ON DUPLICATE KEY UPDATE with Zend Framework (Zend_Db_Table))
<?php

/**
 * @method DbTable_Row_Foo createRow()
 */
class DbTable_Foo extends Zend_Db_Table_Abstract
{
    protected $_primary  = 'foo_id';
    protected $_name     = 'foo';
    protected $_rowClass = 'DbTable_Row_Foo';

    /**
     * @param array $insertData
     * @param array $updateData
     * @return int
     * @throws Zend_Db_Adapter_Exception
     */
    public function insertOrUpdate(array $insertData, array $updateData)
    {
        $db    = $this->getAdapter();
        $table = ($this->_schema ? $this->_schema . '.' : '') . $this->_name;

        // extract and quote col names from the array keys
        $i           = 0;
        $bind        = array();
        $insert_cols = array(); $insert_vals = array();
        $update_cols = array(); $update_vals = array();
        foreach (array('insert', 'update') as $type) {
            $data = ${"{$type}Data"};
            $cols = array();
            $vals = array();
            foreach ($data as $col => $val) {
                $cols[] = $db->quoteIdentifier($col, true);
                if ($val instanceof Zend_Db_Expr) {
                    $vals[] = $val->__toString();
                } else {
                    if ($db->supportsParameters('positional')) {
                        $vals[] = '?';
                        $bind[] = $val;
                    } else {
                        if ($db->supportsParameters('named')) {
                            $bind[':col' . $i] = $val;
                            $vals[] = ':col'.$i;
                            $i++;
                        } else {
                            /** @see Zend_Db_Adapter_Exception */
                            require_once 'Zend/Db/Adapter/Exception.php';
                            throw new Zend_Db_Adapter_Exception(get_class($db) ." doesn't support positional or named binding");
                        }
                    }
                }
            }
            ${"{$type}_cols"} = $cols; unset($cols);
            ${"{$type}_vals"} = $vals; unset($vals);
        }

        // build the statement
        $set = array();
        foreach ($update_cols as $i => $col) {
            $set[] = sprintf('%s = %s', $col, $update_vals[$i]);
        }

        $sql = sprintf(
            'INSERT INTO %s (%s) VALUES (%s) ON DUPLICATE KEY UPDATE %s;',
                $db->quoteIdentifier($table, true),
                implode(', ', $insert_cols),
                implode(', ', $insert_vals),
                implode(', ', $set)
        );

        // execute the statement and return the number of affected rows
        if ($db->supportsParameters('positional')) {
            $bind = array_values($bind);
        }
        $stmt   = $db->query($sql, $bind);
        $result = $stmt->rowCount();
        return $result;
    }
}

Usage

<?php

// ...

$fooTable = new DbTable_Foo();
$fooTable->insertOrUpdate(array(
    'field_1' => 'value',
    'field_2' => new Zend_Db_Expr('NOW()'),
), array(
    'field_3' => 'other value',
    'field_4' => new Zend_Db_Expr('NOW()'),
));
@mattparker
Copy link

Shouldn't the call to query be:

$stmt = $db->query($sql, $bind);

I don't think the table has a query() method, it's on the adapter?

@leek
Copy link
Author

leek commented Nov 26, 2012

@mattparker - You're right, my apologies. I've since fixed that but never came back to update this Gist.

@the-eater
Copy link

Nice job!

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment