"невозможно разобрать" дату IPTC для XMP DateTime в exiv2

Я немного взломал exiv2, пытаясь расширить функции преобразования даты / времени между EXIF, IPTC и XMP. http://dev.exiv2.org/issues/864

Дата IPTC и время IPTC хранятся в отдельных полях, поэтому сначала я пытаюсь проанализировать дату IPTC, затем в своем коде я пытаюсь проанализировать время.

Я новичок в программировании, так что это будет довольно простой проблемой, но здесь все. Код добавляется в файл exiv2 convert.cpp внутри функции cnvIptcValue.

Точную ошибку я получаю из командной строки:

$ exiv2 -eiX iptc.jpg 
Warning: Failed to convert Iptc.Application2.DateCreated to Xmp.photoshop.DateCreated, unable to parse '1944-09-08'
Warning: Failed to convert Iptc.Application2.DigitizationDate to Xmp.xmp.CreateDate, unable to parse '2015-03-06'

Сначала я подумал, что код преобразования даты не работает:

if (std::string(from) == "Iptc.Application2.DateCreated" || std::string(from) == "Iptc.Application2.DigitizationDate")  {

int year, month, day, hour, min, sec, tzHour, tzMinute;
char tzSign;
char buf[30];


    if (sscanf(value.c_str(), "%d-%d-%d", &year, &month, &day) != 3) {

    EXV_WARNING << "Failed to convert " << from << " to " << to
                << ", unable to parse '" << value << "'\n";
    return;
}

Но дальнейшее тестирование показало, что это был код преобразования времени:

    const char* iptcTime = 0;  
if (std::string(from) == "Iptc.Application2.DateCreated") {
    iptcTime = "Iptc.Application2.TimeCreated";
}
else if (std::string(from) == "Iptc.Application2.DigitizationDate") {
    iptcTime = "Iptc.Application2.DigitizationTime";
}
if (iptcTime) {              
    if (sscanf(value.c_str(), "%d:%d:%d%1c%d:%02d", &hour, &min, &sec, &tzSign, &tzHour, &tzMinute) != 6) {

        EXV_WARNING << "Failed to convert " << from << " to " << to
                    << ", unable to parse '" << value << "'\n";

        return;
    }

Поэтому временной код пытался проанализировать поле даты. Это было изменено в поле Time путем добавления пары строк кода:

    Exiv2::IptcData::iterator iptctime_pos = iptcData_->findKey(IptcKey(from));
    std::string value = iptctime_pos->toString();

Я также расширил код сообщения об ошибке, чтобы увидеть, что было в поле Time. результат сейчас:

Warning: Failed to convert Iptc.Application2.TimeCreated to Xmp.photoshop.DateCreated, unable to parse '11:11:14+00:00'
Year Month Day:          1944 9 8
Hour Minute Second:      32767 36913992 1978972555
Zone Sign Hour Minute:   "U+007F"  -184143665 6457088
Warning: Failed to convert Iptc.Application2.DigitizationTime to Xmp.xmp.CreateDate, unable to parse '16:26:47+00:00'
Year Month Day:          2015 3 6
Hour Minute Second:      32767 36909480 36909144
Zone Sign Hour Minute:   "U+007F"  -184143665 6457088

Он правильно разбирает поле Date, но не Time. Почему это? Разве это неправильно?

1 ответ

Решено! Были два изменения, необходимые.

Первым делом нужно было проверить тот же ключ, который был установлен ранее. Мои попытки проб и ошибок заставили меня сканировать из "iptctime", а не из "value", что дало ошибку разбора.

std::string value = iptctime_pos->toString();

if (sscanf(value.c_str(), "%d:%d:%d%1c%d:%d", &hour, &min, &sec, &tzSign, &tzHour, &tzMinute) != 6) {

Вторым было добавить else if предотвращение перезаписи свойств DateTime с помощью общего кода назначения.

Полный рабочий результат:

if (std::string(from) == "Iptc.Application2.DateCreated" || std::string(from) == "Iptc.Application2.DigitizationDate")  {

    int year, month, day, hour, min, sec, tzHour, tzMinute;
    char tzSign;
    std::string iptctime;
    char buf[30];


    (sscanf(value.c_str(), "%d-%d-%d", &year, &month, &day)

    const char* iptcTime = 0;  
    if (std::string(from) == "Iptc.Application2.DateCreated") {
        iptcTime = "Iptc.Application2.TimeCreated";
    }
    else if (std::string(from) == "Iptc.Application2.DigitizationDate") {
        iptcTime = "Iptc.Application2.DigitizationTime";
    }
    if (iptcTime) {              
        Exiv2::IptcData::iterator iptctime_pos = iptcData_->findKey(IptcKey(iptcTime));
        if (iptctime_pos == iptcData_->end()) return;
            if (iptctime_pos->key() == iptcTime) {
                std::string value = iptctime_pos->toString();

                sscanf(value.c_str(), "%d:%d:%d%1c%d:%d", &hour, &min, &sec, &tzSign, &tzHour, &tzMinute)
        }
    }
    snprintf(buf, sizeof(buf), "%04d-%02d-%02dT%02d:%02d:%02d%1c%02d:%02d",
         year, month, day, hour, min, sec, tzSign, tzHour, tzMinute);
    buf[sizeof(buf) - 1] = 0;
    (*xmpData_)[to] = buf;
}

else if (std::string(from) != "Iptc.Application2.DateCreated" || std::string(from) != "Iptc.Application2.DigitizationDate")  {

        (*xmpData_)[to] = value;
}

Немного больше тестирования, и я загружу его в репозиторий exiv2, который будет представлен в следующем выпуске exiv2 0.25!

PS: Спасибо @tivn за вашу помощь.

Другие вопросы по тегам