Получите формат продолжительности ISO-8601 (SCORM 2004) в секундах, используя MySQL

У меня есть столбец sql со значениями, такими как

PT2M52.37S
PT21.79S
PT53M29.68S
P
PT9S

Возможно ли с помощью некоторых функций MySQL преобразовать указанный выше формат в секунды?

Я искал все, но не нашел что-то точно в том же формате, что и выше. Любая функция mysql даты, которую я пробовал, не работает.

Дополнительная информация об этом формате: http://www.ostyn.com/standards/scorm/samples/ISOTimeForSCORM.htm

Функция PHP (это всегда будет P, первый символ)

function scorm_format_duration($duration) {
    // Fetch date/time strings.
    $stryears = get_string('years');
    $strmonths = get_string('nummonths');
    $strdays = get_string('days');
    $strhours = get_string('hours');
    $strminutes = get_string('minutes');
    $strseconds = get_string('seconds');

    if ($duration[0] == 'P') {
        // If timestamp starts with 'P' - it's a SCORM 2004 format
        // this regexp discards empty sections, takes Month/Minute ambiguity into consideration,
        // and outputs filled sections, discarding leading zeroes and any format literals
        // also saves the only zero before seconds decimals (if there are any) and discards decimals if they are zero.
        $pattern = array( '#([A-Z])0+Y#', '#([A-Z])0+M#', '#([A-Z])0+D#', '#P(|\d+Y)0*(\d+)M#',
                            '#0*(\d+)Y#', '#0*(\d+)D#', '#P#', '#([A-Z])0+H#', '#([A-Z])[0.]+S#',
                            '#\.0+S#', '#T(|\d+H)0*(\d+)M#', '#0*(\d+)H#', '#0+\.(\d+)S#',
                            '#0*([\d.]+)S#', '#T#' );
        $replace = array( '$1', '$1', '$1', '$1$2 '.$strmonths.' ', '$1 '.$stryears.' ', '$1 '.$strdays.' ',
                            '', '$1', '$1', 'S', '$1$2 '.$strminutes.' ', '$1 '.$strhours.' ',
                            '0.$1 '.$strseconds, '$1 '.$strseconds, '');
    } else {
        // Else we have SCORM 1.2 format there
        // first convert the timestamp to some SCORM 2004-like format for conveniency.
        $duration = preg_replace('#^(\d+):(\d+):([\d.]+)$#', 'T$1H$2M$3S', $duration);
        // Then convert in the same way as SCORM 2004.
        $pattern = array( '#T0+H#', '#([A-Z])0+M#', '#([A-Z])[0.]+S#', '#\.0+S#', '#0*(\d+)H#',
                            '#0*(\d+)M#', '#0+\.(\d+)S#', '#0*([\d.]+)S#', '#T#' );
        $replace = array( 'T', '$1', '$1', 'S', '$1 '.$strhours.' ', '$1 '.$strminutes.' ',
                            '0.$1 '.$strseconds, '$1 '.$strseconds, '' );
    }

    $result = preg_replace($pattern, $replace, $duration);

    return $result;
}

person OhGodWhy    schedule 16.10.2015    source источник
comment
вышеуказанный формат является ISO. Я спрашиваю, есть ли готовые функции. Если это вручную, я даже не знаю, какие у нас есть разные возможности. В первом примере это 2 минуты 52 секунды и 37 миллисекунд.   -  person OhGodWhy    schedule 16.10.2015
comment
@GordonLinoff да, именно так.   -  person OhGodWhy    schedule 16.10.2015
comment
ISO-8601 относится к множеству форматов. Они выглядят как продолжительность. IMX, большую часть времени люди говорят об ISO-8601 и базах данных, они говорят о датах ISO или дате и времени. Возможно, вы сможете использовать STR_TO_DATE(), но, вероятно, это будет некрасиво с таким меняющимся форматом.   -  person Bacon Bits    schedule 16.10.2015
comment
Кажется, кто-то заставил его работать с STR_TO_DATE() здесь их полная команда была SELECT STR_TO_DATE('PT13M22S', 'PT%iM%sS');, но, к сожалению, она не работает для меня, не знаю почему.   -  person cardamom    schedule 30.03.2017


Ответы (3)


Чтобы STR_TO_DATE работало, вам нужно добавить %f для микросекунд.

SELECT duration,
CASE
    WHEN duration LIKE 'P%DT%H%M%.%S' THEN STR_TO_DATE(duration, 'P%dDT%HH%iM%s.%fS')
    WHEN duration LIKE 'P%DT%H%M%S' THEN STR_TO_DATE(duration, 'P%dDT%HH%iM%sS')
    WHEN duration LIKE 'PT%H%M%.%S' THEN STR_TO_DATE(duration, 'PT%HH%iM%s.%fS')
    WHEN duration LIKE 'PT%H%M%S' THEN STR_TO_DATE(duration, 'PT%HH%iM%sS')
    WHEN duration LIKE 'PT%M%.%S' THEN STR_TO_DATE(duration, 'PT%iM%s.%fS')
    WHEN duration LIKE 'PT%M%S' THEN STR_TO_DATE(duration, 'PT%iM%sS')
    WHEN duration LIKE 'PT%.%S' THEN STR_TO_DATE(duration, 'PT%s.%fS')
    WHEN duration LIKE 'PT%S' THEN STR_TO_DATE(duration, 'PT%sS')
END as session
FROM db

PT27M59.08S => 00:27:59.080000

PT2H27M37.11S => 02:27:37.110000

P4DT19H46M18.22S => 115:46:18.220000

person Fu Danny    schedule 29.05.2018

Я создал 2 функции для преобразования результата cmi.session_time и cmi.total_time из ISO8601 в секунды:

CREATE FUNCTION 'ISO8601_duration_delete_miliseconds'('isoDuration' VARCHAR(512)) RETURNS varchar(512) CHARSET latin1
BEGIN
  DECLARE strIndex INT;
  DECLARE strRes varchar(512);

  SET strIndex = INSTR(isoDuration, '.');
  if (strIndex > 0) then
    set strRes = substring(isoDuration, 1, strIndex-1);
    set strRes = concat(strRes, 'S');
  else
    set strRes = isoDuration;
  end if;

  RETURN strRes;
END

И последняя функция:

CREATE FUNCTION `iso8601duration_to_seconds`(`isoDuration` VARCHAR(512)) RETURNS int(11)
BEGIN
  DECLARE strIndex INT;
  DECLARE strRes varchar(512);
  DECLARE intRes INT;

  set strRes =  ISO8601_duration_delete_miliseconds(isoDuration);

  if (INSTR(isoDuration, 'H') > 0) then
    set intRes = time_to_sec(str_to_date(strRes, 'PT%HH%iM%sS'));
  elseif (INSTR(isoDuration, 'M') > 0) then
    set intRes = time_to_sec(str_to_date(strRes, 'PT%iM%sS'));
  else
    set intRes = time_to_sec(str_to_date(strRes, 'PT%sS'));
  end if;

  RETURN intRes;
END
person Aitor López    schedule 06.09.2016

Я не думаю, что есть стандартная функция для этого. Использование str_to_date() возможно, но оно не будет работать из-за отсутствующих компонентов времени. Если у вас есть только минуты и секунды, вы можете сделать это болезненно со строковыми операциями.

select ((case when p like '%M%S'
              then substring_index(substring_index(p, 'M', 2), 'M', -1)
              when p like '%T%S'
              then substring_index(substring_index(p, 'T', 2), 'T', -1)
              else ''
         end) +
        (case when p like 'T%M'
              then substring_index(substring_index(p, 'T', 2), 'T', -1) + 0
              else 0
         end) * 60
        ) as seconds

Это не очень легко обобщить, поскольку добавляется больше компонентов времени. Компонент секунд, например, является вторым элементом, когда разделителем является «M» или «T». Иногда вам просто нужно задаться вопросом, о чем думает комитет ISO, когда они создают эти стандарты.

Обратите внимание, что MySQL преобразует строку в числа, используя начальные цифры (если они есть). Вышеупомянутое использует эту функцию.

Если вы поддерживаете более сложные форматы периодов, я бы предложил написать пользовательскую функцию и опубликовать ее здесь в качестве ответа.

Также отмечу, что SAS поддерживает этот формат. Более того, у них действительно хорошие документация, объясняющая это.

РЕДАКТИРОВАТЬ:

С этим кладжем вам будет легче работать со всеми форматами. Он вставляет пробел после каждого компонента, а затем использует substring_index():

select (case when newp like '%S'
             then substring_index(substring_index(newp, 'S', 1), ' ', -1) + 1
             else 0
        end) as seconds,
       (case when newp like '%M%'
             then substring_index(substring_index(newp, 'M', 1), ' ', -1) + 1
             else 0
        end) as minutes,
       . . .
from (select replace(replace(replace(replace(replace(p, 'T', 'T '
                                                    ), 'Y', 'Y '
                                            ), 'D', 'D '
                                    ), 'H', 'H '
                            ), 'S', 'S '
                    ) as newp
      from t
     ) t
person Gordon Linoff    schedule 16.10.2015
comment
Здравствуйте, я отредактировал свой первый пост и прикрепил его рабочую версию кода PHP. Не могли бы вы проверить это? Благодарность - person OhGodWhy; 16.10.2015
comment
Это очень сложно и не работает с секундами. Потому что у него тоже есть миллисекунды. Разве мы не можем сделать это с помощью регулярных выражений, как в PHP? (первый пост) - person OhGodWhy; 16.10.2015
comment
@О, Боже, почему . . . Это должно работать нормально в течение миллисекунд. MySQL преобразует значение до S в число, даже если есть десятичная точка. MySQL не предлагает ничего похожего на замену регулярным выражением. - person Gordon Linoff; 16.10.2015