Math-Formula

 view release on metacpan or  search on metacpan

lib/Math/Formula/Type.pod  view on Meta::CPAN

  date = 2006-11-21T12:23:34.56+0110
  dt.year     => INTEGER  2006
  dt.month    => INTEGER  11
  dt.day      => INTEGER  21
  dt.hour     => INTEGER  12
  dt.minute   => INTEGER  23
  dt.second   => INTEGER  34
  dt.fracsec  => FLOAT    34.56
  dt.timezone => TIMEZONE +0110
  dt.time     => TIME     12:23:34.56
  dt.date     => DATE     2006-11-21+0110

=head2 MF::DATE, refers to a day in some timezone

A date has format 'YYYY-MM-DD+TZ', for instance C<2023-02-20+0100>.

The date may be cast into a datetime, where the time is set to C<00:00:00>.
This transformation is actually slightly problematic: a date means "anywhere
during the day, where a datetime is a specific moment.

You may add (C<+>) and subtract (C<->) durations from a date, which result in
a new date.  Those durations cannot contain a time component.

An subtract (C<->) from a date produces a duration.

You may also add a time to a date, forming a datetime.  Those may be in different
time-zones.  You may also numerically compare dates, but only when they are not
in the same timezone, this will return C<false>.

B<. Example: for date>

  1966-12-21        # without timezone, default from context
  1966-12-21+0200   # with explicit timezone info

  2023-02-21+0200 - P3D            => DATE     2023-02-18+0200
  2012-03-08+0100 + 06:07:08+0200  => DATETIME 2012-03-08T06:07:08+0300
  2023-02-26 - 2023-01-20          => DURATION P1M6D
  2023-02-22 < 1966-04-05          => BOOLEAN  false
  2023-02-22 <=> 1966-04-05        => INTEGER 1      # -1, 0 1

  4 + 2000-10-20    => INTEGER 1974  # parser accident repaired

Attributes:

  date = 2006-11-21+0700
  date.year                        => INTEGER   2006
  date.month                       => INTEGER   11
  date.day                         => INTEGER   21
  date.timezone                    => TIMEZONE  +0700

=head2 MF::TIME, a moment during any day

Useful to indicate a daily repeating event.  For instance, C<start-backup: 04:00:12>.
Times do no have a timezone: it only gets a meaning when added to a (local) time.

Time supports numeric comparison.  When you add (C<+>) a (short) duration to a time, it
will result in a new time (modulo 24 hours).

When you subtract (C<->) one time from another, you will get a duration modulo 24 hours.
This does not take possible jumps to and from Daylight Savings time into
account.  When you care about that, than create a formula involving
the actual date:

  bedtime = 23:00:00
  wakeup  = 06:30:00
  now     = #system.now
  sleep   = ((now+P1D).date + wakeup) - (now.date + bedtime)  #DURATION

B<. Example: for time>

  12:00:34      # lunch-time, in context default time-zone
  23:59:61      # with max leap seconds
  09:11:11.111  # precise time (upto nanoseconds)

  12:00:34 + PT30M      => TIME 12:30:34   # end of lunch
  12:00:34 - PT15M      => TIME 11:45:34   # round-up coworkers
  23:40:00 + PT7H       => TIME 06:40:00   # early rise
  07:00:00 - 23

  18:00:00 ==  17:00:00 => BOOLEAN
  18:00:00 <=> 17:00:00 => INTEGER

Attributes:

  time = 12:23:34.56
  time.hour             => INTEGER 12
  time.minute           => INTEGER 23
  time.second           => INTEGER 34
  time.fracsec          => FLOAT   34.56

=head2 MF::TIMEZONE, time-zone.

Clock difference to UTC (Zulu).  The format requires a plus (C<+>) or minus (C<->) followed by
two digits representing hours and two digits for minutes.

Timezone supports numeric comparison.  When you add (C<+>) and subtract (C<->) a (short)
duration to a timezone.

When you subtract (C<->) one timezone from another, you will get a duration.

B<. Example: for timezone>

  +0000
  -0612

  -0600 + PT1H     => -0500
  +0230 - PT3H30M  => -0100

Attributes:

  tz = -1236
  tz.in_minutes    => INTEGER -756
  tz.in_seconds    => INTEGER -45360

=head2 MF::DURATION, a period of time

Durations are usually added to datetimes, and may be negative.  They are formatted in
ISO8601 standard format, which is a bit awkward, to say the least.

Durations can be added (C<+>) together, subtracted (C<->) together,
or multiplied by an integer factor.  The prefix C<+> and C<-> are also supported.

lib/Math/Formula/Type.pod  view on Meta::CPAN

day length differs per month.  This will only be resolved when the duration is added
to an actual datetime.

Two durations can be compared numerically.  However: a bit of care must be taken.
Sometimes, it is only clear what the ordering is when relative to a specific datetime.
For instance, which is longer: P1M or P30D?  The current time ("now") is used as
reference point.  Otherwise, add some other datetime to both durations before
comparison.

B<. Example: for duration>

  P1Y2M5D           # duration one year, 2 months, 5 days
  PT1M              # mind the "T" before smaller than day values!
  P3MT5M            # confusing: 3 months + 5 minutes
  PT3H4M8.2S        # duration 3 hours, 4 minutes, just over 8 seconds

  - -P1Y            # prefix + and =
  P3Y2M + P1YT3M5S  => DURATION P4Y2MT3M5S
  P1Y2MT3H5M - P3Y8MT5H13M14S -> DURATION -P2Y6MT2H8M14S
  P1DT2H * 4        => DURATION P4DT8H
  4 * P1DT2H        => DURATION P4DT8H

  P1Y > P1M         => BOOLEAN true
  PT20M <=> PT19M   => INTEGER 1     # -1, 0, 1

Attributes

  P1Y.in_days       => INTEGER  365  (4yrs will have extra day)
  P1H.in_seconds    => INTEGER  3600

=over 4

=item $obj-E<gt>B<inSeconds>()

Returns the duration in seconds, nanoseconds ignored.  This is exact when the duration
only contains minutes and seconds.  When hours are involved, do you care about switching
from and to Daylight Savings time?  Months are taken as 1/12 of a year.  A year is taken
as 365.256 year.  For exact calculations, add the duration to a DATETIME first.

=back

=head2 MF::NAME, refers to something in the context

The L<Math::Formula::Context|Math::Formula::Context> object contains translations for names to
contextual objects.  Names are the usual tokens: Unicode alpha-numeric
characters and underscore (C<_>), where the first character cannot be
a digit.

On the right-side of a fragment indicator C<#> or method indicator C<.>,
the name will be lookup in the features of the object on the left of that
operator.

A name which is not right of a hash (C<#>) or dot (C<.>) can be cast
into an object from the context.

Names are symbol: are not a value by themselves, so have no values to
be ordered.  However, they may exist however: test it with prefix operator
C<exists>.

The more complicated concept is the 'defaults to' operator (C<//>).
When there is no formula with the name on the left side, the right side
is taken.  This is often stacked with a constant default value at the end.

B<. Example: of names>

  tic
  route66
  the_boss
  _42             # and '_' works as a character
  αβΩ             # Unicode alpha nums allowed

  7eleven         # not allowed: no start with number

  See "Math::Formula::Context" for the following
  #frag           # (prefix #) fragment of default object
  .method         # (prefix .) method on default object
  name#frag       # fragment of object 'name'
  file.size       # method 'size' on object 'file'

Attributes on names

  exists live     => BOOLEAN    # does formula 'live' exist?
  not exists live => BOOLEAN

  live // dead // false         # pick first rule which exists

=over 4

=item $class-E<gt>B<validated>($string, $where)

Create a MF::NAME from a C<$string> which needs to be validated for being a valid
name.  The C<$where> will be used in error messages when the C<$string> is invalid.

=item $obj-E<gt>B<value>()

Z<>

=back

=head2 MF::PATTERN, pattern matching

This type implements pattern matching for the C<like> and C<unlike> operators.
The patterns are similar to file-system patterns.  However, there is no special meaning
to leading dots and '/'.

Pattern matching constructs are C<*> (zero or more characters), C<?> (any single
character), and C<[abcA-Z]> (one of a, b, c, or capital), C<[!abc]> (not a, b, or c).
Besides, it supports curly alternatives like C<*.{jpg,gif,png}>.

B<. Example: of patterns>

  "abc" like "b"     => BOOLEAN false
  "abc" like "*b*"   => BOOLEAN false
  "abc" like "*c"    => BOOLEAN true

  "abc" unlike "b"   => BOOLEAN true
  "abc" unlike "*b*" => BOOLEAN true
  "abc" unlike "*c"  => BOOLEAN false

=over 4

=item $class-E<gt>B<new>( $token|undef, [$value] )

Z<>

=back

=head2 MF::REGEXP, Regular expression

This type implements regular expressions for the C<=~> and C<!~> operators.

The base of a regular expression is a single or double quote string. When the
operators are detected, those will automatically be cast into a regexp.

B<. Example: of regular expressions>

  "abc" =~ "b"       => BOOLEAN true
  "abc" =~ "c$"      => BOOLEAN true
  "abc" !~ "b"       => BOOLEAN false
  "abc" !~ "c$"      => BOOLEAN false

=over 4

=item $obj-E<gt>B<regexp>()

Returns the regular expression as compiled object (qr).

=back

=head2 MF::FRAGMENT, access to externally provided data

The used of this type is explained in L<Math::Formula::Context|Math::Formula::Context>.

=head1 DIAGNOSTICS

=over 4

=item Error: Illegal name '$name' in '$where'

Z<>

=item Warning: cannot convert qr back to pattern, do $regexp

Z<>

=item Warning: cannot find attribute '$attr' for $class '$token'

Z<>

=item Error: cannot find prefx operator '$op' on a $child

Z<>

=item Error: cannot match infix operator '$op' for ($left -> $right)

Z<>

=item Error: name '$name' cannot be used as value.

Z<>

=item Error: rvalue name '$name' for operator '$op' is not a formula

Z<>

=back

=head1 SEE ALSO

This module is part of Math-Formula version 0.18,
built on August 19, 2025. Website: F<http://perl.overmeer.net/CPAN/>

=head1 LICENSE

For contributors see file ChangeLog.

This software is copyright (c) 2023-2025 by Mark Overmeer.

This is free software; you can redistribute it and/or modify it under
the same terms as the Perl 5 programming language system itself.



( run in 1.097 second using v1.01-cache-2.11-cpan-13bb782fe5a )