Update and create timestamps with MySQL

A lot of relational tables need created and update timestamps columns. I prefer having them for all tables with no exception. However, most of applications I am working on are running MySQL. MySQL has minor limitation on timestamps. Unfortunately you can create only one time stamp column that has DEFAULT NOW() value. Read more to see how to avoid this limitation.


The simplest way to do this is create the following columns in the table:
stamp_created timestamp default now(),
stamp_updated timestamp default now() on update now())

But MySQL will return the following error:

ERROR 1293 (HY000): Incorrect table definition; there can be only one TIMESTAMP column with CURRENT_ in DEFAULT or ON UPDATE clause
You can use only one of the definitions in one table. However here is the way how to create both timestamps columns:
 create table test_table( 
id integer not null auto_increment primary key,
stamp_created timestamp default '0000-00-00 00:00:00',
stamp_updated timestamp default now() on update now()
);

Note that it is necessary to enter nulls into both columns during ‘insert’:
 mysql> insert into test_table(stamp_created, stamp_updated) values(null, null); 
Query OK, 1 row affected (0.06 sec)
mysql> select * from t5;
+----+---------------------+---------------------+
| id | stamp_created | stamp_updated |
+----+---------------------+---------------------+
| 2 | 2009-04-30 09:44:35 | 2009-04-30 09:44:35 |
+----+---------------------+---------------------+
2 rows in set (0.00 sec)
mysql> update test_table set id = 3 where id = 2;
Query OK, 1 row affected (0.05 sec) Rows matched: 1 Changed: 1 Warnings: 0
mysql> select * from test_table;
+----+---------------------+---------------------+
| id | stamp_created | stamp_updated |
+----+---------------------+---------------------+
| 3 | 2009-04-30 09:44:35 | 2009-04-30 09:46:59 |
+----+---------------------+---------------------+
2 rows in set (0.00 sec)
mysql>

Komentar

  1. You can use triggers in tables and avoid inserting NULL in your insert statement. This would enforce the current timestamp in that column.
    eg.

    DELIMITER $$
    CREATE trigger trgrname
    before insert on test_table
    for each row
    begin
    set new.stamp_created = current_timestamp;
    end$$
    DELIMITER ;

    BalasHapus
  2. Agreed, but I prefer avoid triggers.
    Both solutions have weak points.

    BalasHapus
  3. "Note that it is necessary to enter nulls into both columns during ‘insert’:"

    Actually, you only need the null for the stamp_created as the stamp_updated will get updated automatically.

    mysql> insert into test_table(stamp_created) values(NULL);
    Query OK, 1 row affected (0.00 sec)

    mysql> select * from test_table;

    +----+---------------------+---------------------+
    | id | stamp_created | stamp_updated |
    +----+---------------------+---------------------+
    | 5 | 2009-06-17 15:54:18 | 2009-06-17 15:54:18 |
    +----+---------------------+---------------------+

    BalasHapus
  4. Oops, misread the default value thing. I just tried it, and by default it'll insert a timestamp value of '0000-00-00 00:00:00'.

    BalasHapus
  5. I was doing the same experiments that you mentioned in your blog. The trick above is the only one way to have both timestamps in current MySQL version.

    BalasHapus
  6. thanks a lot for the great tips...

    BalasHapus
  7. thnx Bogdan, good trick

    BalasHapus
  8. It's crazy to have to do this. With dynamic websites, blogs, and such, it would seem obviously intrinsic that the original creation date should be maintained automatically somehow.

    Oh the naivety of the `nub` :). Thanks for saving me several hours work figuring this out.

    Regards,
    sl

    BalasHapus
  9. I don't get it. If you already have a "workaround" which does force you to insert null, why don't you insert NOW() instead of null?

    With trigger workaround you gain the advantage to not mention "stamp_created" at all, so I can see a point there, although I personally refrain from such trigger usage, and I rather add NOW() into my inserts, which is IMHO much better then using "null" hack described here.

    Because honestly, this looks more like a bug to me. If you want 0000-00-00 as default, why don't you get it? I would be not shocked if MySQL would decide to change this behaviour later in future and "fix" it. (although I would rather welcome to have 2+ columns default NOW fix :) )

    BalasHapus
  10. hi! for the "insert into test_table(stamp_created, stamp_updated) values(null, null); " statement.. how would u write it in php codings so that whenever a new data is created, it will automatically include the date created..

    bcux right now i tried writing the insert in mysql codings but it only shows null null for other column info.. only the date modified n created are updated..

    as im using a webpage to add data it doesn shows the created time.. rather it shows the "0000-00-00 00:00:00". only my updated column can b updated..

    pls help thks!

    BalasHapus
  11. Your weblog is incredibly helpful,I need to connect with u,could i sent e-mail to you?

    BalasHapus
  12. Another solution is to use a trigger :

    CREATE TABLE sample_table (
    id TINYINT(4) NOT NULL AUTO_INCREMENT,
    xxx VARCHAR(255) NOT NULL DEFAULT 'value',
    date_insert TIMESTAMP NULL DEFAULT NULL,
    date_lastupdate TIMESTAMP NOT NULL DEFAULT CURRENT_TIMESTAMP ON UPDATE CURRENT_TIMESTAMP,
    PRIMARY KEY (id)
    );

    CREATE TRIGGER sample_table_on_insert BEFORE INSERT
    ON sample_table
    FOR EACH ROW
    SET NEW.date_insert = CURRENT_TIMESTAMP

    BalasHapus

Posting Komentar

Postingan populer dari blog ini

10 Tempat Yang Tidak Bisa Kamu Kunjungi

Kawasaki 150 KLX S

Musisi Pink Floyd Serukan Boikot Produk Israel