package MooseX::Timestamp; our $VERSION = '0.05'; =head1 NAME MooseX::Timestamp - simple timestamp type for Moose =head1 SYNOPSIS use MooseX::Timestamp; print timestamp; # 2007-12-06 23:15:42 print timestamp 0; # 1970-01-01 12:00:00 print timestamp gmtime 0; # 1970-01-01 00:00:00 use POSIX qw(strftime); print strftime("%a", posixtime "2007-12-06 23:15"); # Thu #... package MyClass; use Moose; has 'stamp' => isa => "Timestamp", is => "rw", coerce => 1; package main; my $obj = MyClass->new(stamp => "2007-01-02 12:00:12"); # ok $obj->stamp("2007-01-02 12:01"); $obj->stamp("2007-01-02 12"); $obj->stamp("2007-01-02 12:00:00Gibbons"); #fail =head1 DESCRIPTION Tired of bulky date modules? Don't put up with them any longer. This module provides floating dates on the Gregorian calendar without much code. It operates in (one or two particular variants of) ISO-8601 date format, and POSIX-style 6-number lists. Note: you probably want the functions provided by MooseX::TimestampTZ most of the time, as they deal in unix epoch times. =cut use Moose::Util::TypeConstraints; my @exports; use Sub::Exporter -setup => { exports => [ qw(timestamp posixtime valid_posixtime) ], groups => { default => [qw(timestamp posixtime)] }, }; use Carp; #use MooseX::Timestamp::__version; subtype Timestamp => as Str => where { m{^\d{4}-\d{2}-\d{2} \d{2}:\d{2}:\d{2}$} and eval { valid_posixtime(posixtime($_)) }; }; use POSIX qw(strftime); sub timestamp { if ( @_ == 0 ) { @_ = time; } if ( @_ == 1 ) { my $time = shift; @_ = localtime($time); } valid_posixtime(@_); strftime("%Y-%m-%d %H:%M:%S", @_ ); } my @short = qw(0 1 0 1 0 1 0 0 1 0 1 0); sub valid_posixtime { my @lt = @_; croak "invalid month ".($lt[4]+1) if $lt[4]<0 or $lt[4]>11; croak "invalid day $lt[3]" if !$lt[3] or $lt[3]>31 or (($lt[3]==31 and $short[$lt[4]]) or ($lt[3] > 28 and $lt[4] == 1 and !($lt[3] == 29 and (($lt[5]%4) == 0 and ($lt[5]%100 != 0 or ($lt[5]+300)%400 == 0))))); croak "invalid hour $lt[2]" if $lt[2]<0 or $lt[2]>23; croak "invalid minute $lt[1]" if $lt[1]<0 or $lt[1]>59; croak "invalid second $lt[0]" if ($lt[0]<0 or $lt[0]>60 or ($lt[0]==60 and $lt[1]!=59)); 1; } sub posixtime { return localtime time unless @_; my @lt = ($_[0] =~ m{^(\d{4})(-\d{1,2}|\d{2})(-\d{1,2}|\d{2})T? \s*(?:(\d{1,2}) (?::(\d{2}) (?::(\d{2}))? )? )?$}x) or croak "bad timestamp '$_[0]'"; $lt[1]=abs($lt[1]); $lt[2]=abs($lt[2]); $lt[0]-=1900; $lt[1]--; $_ ||= 0 for (@lt[3..5]); reverse @lt; } coerce Timestamp => from Timestamp => via { $_ }, => from Str => via { timestamp posixtime $_ }; =head1 FUNCTIONS The following functions are available for import. If you want to import them all, use the C<:all> import group, as below: use MooseX::Timestamp qw(:all); =head2 timestamp(time_t $time = time()) =head2 timestamp(@posixtime) Converts from a POSIX-style array of time components, or an epoch time, into a Timestamp. If an epoch time is passed, the local timezone rules are used for conversion into a wallclock time. See L for a version which returns the time zone as well. =head2 posixtime() Alias for the built-in C =head2 posixtime(Timestamp) Converts a Timestamp into a POSIX-style array of time components. They are B guaranteed to be valid. This accepts a similar set of input values to C; see its documentation (L) for a list. The defining difference is that Timestamps passed into this function MUST NOT have a time zone (or "Z") attached. =head2 valid_posixtime(@posixtime) This function croaks with an error if the passed POSIX-style array of time components are found to be out of range in any way. This function contains leap year rules and passes through leap seconds. =head1 TYPES AND COERCIONS One type is defined by this module. =head2 Timestamp This is a subtype of C which conforms to the normalized form of a Timestamp. Rules exist to coerce C objects to this type, and are available by using the C 1> flag on a Moose attribute declaration: package Widget; use MooseX::Timestamp; has 'created' => isa => Timestamp, is => "rw", coerce => 1; package main; my $widget = new Widget; $widget->created("2007-12-07"); print $widget->created; # 2007-12-07 00:00:00 With the above, if you set C to a value such as automatically get converted into a TimestampTZ in the current time zone. =head2 EXPORTS The default exporting action of this module is to export the C and C methods. To avoid this, pass an empty argument list to the use statement: use MooseX::Timestamp (); =head1 BUGS This module is relatively slow, as conversions and calls to C and friends happen far too often, really - especially with coercion. =head1 AUTHOR AND LICENSE Sam Vilain, Copyright 2007, Sam Vilain. All Rights Reserved. This program is Free Software; you may use it and/or redistribute it under the terms of Perl itself. =cut 1;