package exif import ( "fmt" "strconv" "strings" "time" "github.com/dsoprea/go-logging" "github.com/dsoprea/go-exif/v2/common" "github.com/dsoprea/go-exif/v2/undefined" ) // ParseExifFullTimestamp parses dates like "2018:11:30 13:01:49" into a UTC // `time.Time` struct. func ParseExifFullTimestamp(fullTimestampPhrase string) (timestamp time.Time, err error) { defer func() { if state := recover(); state != nil { err = log.Wrap(state.(error)) } }() parts := strings.Split(fullTimestampPhrase, " ") datestampValue, timestampValue := parts[0], parts[1] dateParts := strings.Split(datestampValue, ":") year, err := strconv.ParseUint(dateParts[0], 10, 16) if err != nil { log.Panicf("could not parse year") } month, err := strconv.ParseUint(dateParts[1], 10, 8) if err != nil { log.Panicf("could not parse month") } day, err := strconv.ParseUint(dateParts[2], 10, 8) if err != nil { log.Panicf("could not parse day") } timeParts := strings.Split(timestampValue, ":") hour, err := strconv.ParseUint(timeParts[0], 10, 8) if err != nil { log.Panicf("could not parse hour") } minute, err := strconv.ParseUint(timeParts[1], 10, 8) if err != nil { log.Panicf("could not parse minute") } second, err := strconv.ParseUint(timeParts[2], 10, 8) if err != nil { log.Panicf("could not parse second") } timestamp = time.Date(int(year), time.Month(month), int(day), int(hour), int(minute), int(second), 0, time.UTC) return timestamp, nil } // ExifFullTimestampString produces a string like "2018:11:30 13:01:49" from a // `time.Time` struct. It will attempt to convert to UTC first. func ExifFullTimestampString(t time.Time) (fullTimestampPhrase string) { t = t.UTC() return fmt.Sprintf("%04d:%02d:%02d %02d:%02d:%02d", t.Year(), t.Month(), t.Day(), t.Hour(), t.Minute(), t.Second()) } // ExifTag is one simple representation of a tag in a flat list of all of them. type ExifTag struct { IfdPath string `json:"ifd_path"` TagId uint16 `json:"id"` TagName string `json:"name"` TagTypeId exifcommon.TagTypePrimitive `json:"type_id"` TagTypeName string `json:"type_name"` Value interface{} `json:"value"` ValueBytes []byte `json:"value_bytes"` ChildIfdPath string `json:"child_ifd_path"` } // String returns a string representation. func (et ExifTag) String() string { return fmt.Sprintf("ExifTag 0 { var ifd *Ifd ifd, q = q[0], q[1:] ti := NewTagIndex() for _, ite := range ifd.Entries { tagName := "" it, err := ti.Get(ifd.IfdPath, ite.TagId()) if err != nil { // If it's a non-standard tag, just leave the name blank. if log.Is(err, ErrTagNotFound) != true { log.PanicIf(err) } } else { tagName = it.Name } valueBytes, err := ite.RawBytes() log.PanicIf(err) value, err := ite.Value() if err != nil { if err == exifcommon.ErrUnhandledUnknownTypedTag { value = exifundefined.UnparseableUnknownTagValuePlaceholder } else { log.Panic(err) } } tagType := ite.TagType() et := ExifTag{ IfdPath: ifd.IfdPath, TagId: ite.TagId(), TagName: tagName, TagTypeId: tagType, TagTypeName: tagType.String(), Value: value, ValueBytes: valueBytes, ChildIfdPath: ite.ChildIfdPath(), } exifTags = append(exifTags, et) } for _, childIfd := range ifd.Children { q = append(q, childIfd) } if ifd.NextIfd != nil { q = append(q, ifd.NextIfd) } } return exifTags, nil }