2020-01-07 17:36:49 +01:00
|
|
|
package meta
|
|
|
|
|
|
|
|
import (
|
|
|
|
"fmt"
|
|
|
|
"math"
|
2020-05-15 15:29:56 +02:00
|
|
|
"path/filepath"
|
2020-07-13 10:41:45 +02:00
|
|
|
"runtime/debug"
|
2020-01-07 17:36:49 +01:00
|
|
|
"strconv"
|
|
|
|
"strings"
|
2020-11-21 23:43:53 +01:00
|
|
|
"sync"
|
2020-01-07 17:36:49 +01:00
|
|
|
"time"
|
|
|
|
|
2020-07-17 09:41:37 +02:00
|
|
|
"github.com/dsoprea/go-exif/v3"
|
2022-03-25 16:31:09 +01:00
|
|
|
"gopkg.in/photoprism/go-tz.v2/tz"
|
|
|
|
|
2020-07-17 09:41:37 +02:00
|
|
|
exifcommon "github.com/dsoprea/go-exif/v3/common"
|
2022-03-25 16:31:09 +01:00
|
|
|
|
2022-04-15 09:42:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/clean"
|
2020-07-19 16:39:43 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/fs"
|
2022-04-15 09:42:07 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/projection"
|
2020-07-23 15:34:20 +02:00
|
|
|
"github.com/photoprism/photoprism/pkg/rnd"
|
2022-03-25 16:31:09 +01:00
|
|
|
"github.com/photoprism/photoprism/pkg/txt"
|
2020-01-07 17:36:49 +01:00
|
|
|
)
|
|
|
|
|
2020-07-16 20:32:45 +02:00
|
|
|
var exifIfdMapping *exifcommon.IfdMapping
|
|
|
|
var exifTagIndex = exif.NewTagIndex()
|
2020-11-21 23:43:53 +01:00
|
|
|
var exifMutex = sync.Mutex{}
|
2022-07-22 11:46:53 +02:00
|
|
|
var exifDateTimeTags = []string{"DateTimeOriginal", "DateTimeCreated", "CreateDate", "DateTime", "DateTimeDigitized"}
|
|
|
|
var exifSubSecTags = []string{"SubSecTimeOriginal", "SubSecTime", "SubSecTimeDigitized"}
|
2020-05-15 11:15:15 +02:00
|
|
|
|
2020-07-16 20:32:45 +02:00
|
|
|
func init() {
|
2020-07-17 09:41:37 +02:00
|
|
|
exifIfdMapping = exifcommon.NewIfdMapping()
|
2020-07-16 20:32:45 +02:00
|
|
|
|
2020-07-17 09:41:37 +02:00
|
|
|
if err := exifcommon.LoadStandardIfds(exifIfdMapping); err != nil {
|
2020-07-16 20:32:45 +02:00
|
|
|
log.Errorf("metadata: %s", err.Error())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-28 15:57:29 +02:00
|
|
|
// Exif parses an image file for Exif metadata and returns as Data struct.
|
2022-04-15 09:42:07 +02:00
|
|
|
func Exif(fileName string, fileType fs.Type, bruteForce bool) (data Data, err error) {
|
2022-03-28 15:57:29 +02:00
|
|
|
err = data.Exif(fileName, fileType, bruteForce)
|
2020-05-13 20:53:15 +02:00
|
|
|
|
|
|
|
return data, err
|
|
|
|
}
|
|
|
|
|
2022-03-28 15:57:29 +02:00
|
|
|
// Exif parses an image file for Exif metadata and returns as Data struct.
|
2022-04-15 09:42:07 +02:00
|
|
|
func (data *Data) Exif(fileName string, fileFormat fs.Type, bruteForce bool) (err error) {
|
2020-11-21 23:43:53 +01:00
|
|
|
exifMutex.Lock()
|
|
|
|
defer exifMutex.Unlock()
|
|
|
|
|
2020-01-07 17:36:49 +01:00
|
|
|
defer func() {
|
|
|
|
if e := recover(); e != nil {
|
2022-04-15 09:42:07 +02:00
|
|
|
err = fmt.Errorf("metadata: %s in %s (exif panic)\nstack: %s", e, clean.Log(filepath.Base(fileName)), debug.Stack())
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
}()
|
|
|
|
|
2022-07-06 23:01:54 +02:00
|
|
|
// Resolve file name e.g. in case it's a symlink.
|
|
|
|
if fileName, err = fs.Resolve(fileName); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-02-02 17:41:45 +01:00
|
|
|
// Extract raw Exif block.
|
2022-04-12 13:28:28 +02:00
|
|
|
rawExif, err := RawExif(fileName, fileFormat, bruteForce)
|
2020-01-14 01:47:26 +01:00
|
|
|
|
2020-08-28 09:27:25 +02:00
|
|
|
if err != nil {
|
|
|
|
return err
|
2020-05-15 11:15:15 +02:00
|
|
|
}
|
2020-01-14 01:47:26 +01:00
|
|
|
|
2022-04-15 09:42:07 +02:00
|
|
|
logName := clean.Log(filepath.Base(fileName))
|
2020-01-07 17:36:49 +01:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
// Enumerate data.exif in Exif block.
|
2020-07-17 09:41:37 +02:00
|
|
|
opt := exif.ScanOptions{}
|
|
|
|
entries, _, err := exif.GetFlatExifData(rawExif, &opt)
|
2020-01-07 17:36:49 +01:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
// Create large enough map for values.
|
|
|
|
if data.exif == nil {
|
|
|
|
data.exif = make(map[string]string, len(entries))
|
|
|
|
}
|
|
|
|
|
|
|
|
// Ignore IFD1 data.exif with existing IFD0 values.
|
|
|
|
// see https://github.com/photoprism/photoprism/issues/2231
|
|
|
|
for _, tag := range entries {
|
2022-11-17 08:28:30 +01:00
|
|
|
s := strings.Split(tag.FormattedFirst, "\x00")
|
|
|
|
if tag.TagName == "" || len(s) == 0 {
|
|
|
|
// Do nothing.
|
|
|
|
} else if s[0] != "" && (data.exif[tag.TagName] == "" || tag.IfdPath != exif.ThumbnailFqIfdPath) {
|
|
|
|
data.exif[tag.TagName] = s[0]
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
2020-07-16 20:32:45 +02:00
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
// Abort if no values were found.
|
|
|
|
if len(data.exif) == 0 {
|
|
|
|
return fmt.Errorf("metadata: no exif data in %s", logName)
|
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
var ifdIndex exif.IfdIndex
|
|
|
|
_, ifdIndex, err = exif.Collect(exifIfdMapping, exifTagIndex, rawExif)
|
2020-01-07 17:36:49 +01:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
// Find and parse GPS coordinates.
|
2020-07-16 20:32:45 +02:00
|
|
|
if err != nil {
|
2022-11-17 08:28:30 +01:00
|
|
|
log.Debugf("metadata: %s in %s (exif collect)", err, logName)
|
2020-07-16 20:32:45 +02:00
|
|
|
} else {
|
2022-04-09 19:56:38 +02:00
|
|
|
var ifd *exif.Ifd
|
|
|
|
if ifd, err = ifdIndex.RootIfd.ChildWithIfdPath(exifcommon.IfdGpsInfoStandardIfdIdentity); err == nil {
|
|
|
|
var gi *exif.GpsInfo
|
|
|
|
if gi, err = ifd.GpsInfo(); err != nil {
|
2022-11-17 08:28:30 +01:00
|
|
|
log.Debugf("metadata: %s in %s (exif gps-info)", err, logName)
|
2020-12-23 18:43:52 +01:00
|
|
|
} else {
|
2022-04-09 19:56:38 +02:00
|
|
|
if !math.IsNaN(gi.Latitude.Decimal()) && !math.IsNaN(gi.Longitude.Decimal()) {
|
2022-12-28 20:12:30 +01:00
|
|
|
data.Lat, data.Lng = NormalizeGPS(gi.Latitude.Decimal(), gi.Longitude.Decimal())
|
2022-04-09 19:56:38 +02:00
|
|
|
} else if gi.Altitude != 0 || !gi.Timestamp.IsZero() {
|
2022-04-15 09:42:07 +02:00
|
|
|
log.Warnf("metadata: invalid exif gps coordinates in %s (%s)", logName, clean.Log(gi.String()))
|
2022-04-09 19:56:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if gi.Altitude != 0 {
|
2023-02-09 11:51:26 +01:00
|
|
|
data.Altitude = float64(gi.Altitude)
|
2022-04-09 19:56:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
if !gi.Timestamp.IsZero() {
|
2022-04-13 22:17:59 +02:00
|
|
|
data.TakenGps = gi.Timestamp
|
2022-04-09 19:56:38 +02:00
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["Artist"]; ok {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.Artist = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["Copyright"]; ok {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.Copyright = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2023-10-06 02:22:48 +02:00
|
|
|
// Ignore numeric model names as they are probably invalid.
|
|
|
|
if value, ok := data.exif["CameraModel"]; ok && !txt.IsUInt(value) {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.CameraModel = SanitizeString(value)
|
2023-10-06 02:22:48 +02:00
|
|
|
} else if value, ok = data.exif["Model"]; ok && !txt.IsUInt(value) {
|
|
|
|
data.CameraModel = SanitizeString(value)
|
|
|
|
} else if value, ok = data.exif["UniqueCameraModel"]; ok && !txt.IsUInt(value) {
|
2020-05-15 11:15:15 +02:00
|
|
|
data.CameraModel = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2023-10-06 02:22:48 +02:00
|
|
|
if value, ok := data.exif["CameraMake"]; ok && !txt.IsUInt(value) {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.CameraMake = SanitizeString(value)
|
2023-10-06 02:22:48 +02:00
|
|
|
} else if value, ok = data.exif["Make"]; ok && !txt.IsUInt(value) {
|
2020-05-15 11:15:15 +02:00
|
|
|
data.CameraMake = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["CameraOwnerName"]; ok {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.CameraOwner = SanitizeString(value)
|
2020-02-07 14:40:06 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["BodySerialNumber"]; ok {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.CameraSerial = SanitizeString(value)
|
2020-02-07 14:40:06 +01:00
|
|
|
}
|
|
|
|
|
2023-10-06 02:22:48 +02:00
|
|
|
if value, ok := data.exif["LensMake"]; ok && !txt.IsUInt(value) {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.LensMake = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2023-10-06 02:22:48 +02:00
|
|
|
// Ignore numeric model names as they are probably invalid.
|
|
|
|
if value, ok := data.exif["LensModel"]; ok && !txt.IsUInt(value) {
|
|
|
|
data.LensModel = SanitizeString(value)
|
|
|
|
} else if value, ok = data.exif["Lens"]; ok && !txt.IsUInt(value) {
|
2020-04-29 16:42:06 +02:00
|
|
|
data.LensModel = SanitizeString(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["Software"]; ok {
|
|
|
|
data.Software = SanitizeString(value)
|
|
|
|
}
|
|
|
|
|
|
|
|
if value, ok := data.exif["ExposureTime"]; ok {
|
2020-04-26 12:17:49 +02:00
|
|
|
if n := strings.Split(value, "/"); len(n) == 2 {
|
|
|
|
if n[0] != "1" && len(n[0]) < len(n[1]) {
|
|
|
|
n0, _ := strconv.ParseUint(n[0], 10, 64)
|
|
|
|
if n1, err := strconv.ParseUint(n[1], 10, 64); err == nil && n0 > 0 && n1 > 0 {
|
2020-04-26 14:31:33 +02:00
|
|
|
value = fmt.Sprintf("1/%d", n1/n0)
|
2020-04-26 12:17:49 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-01-07 17:36:49 +01:00
|
|
|
data.Exposure = value
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["FNumber"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
values := strings.Split(value, "/")
|
|
|
|
|
|
|
|
if len(values) == 2 && values[1] != "0" && values[1] != "" {
|
|
|
|
number, _ := strconv.ParseFloat(values[0], 64)
|
|
|
|
denom, _ := strconv.ParseFloat(values[1], 64)
|
|
|
|
|
2020-04-26 11:41:54 +02:00
|
|
|
data.FNumber = float32(math.Round((number/denom)*1000) / 1000)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["ApertureValue"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
values := strings.Split(value, "/")
|
|
|
|
|
|
|
|
if len(values) == 2 && values[1] != "0" && values[1] != "" {
|
|
|
|
number, _ := strconv.ParseFloat(values[0], 64)
|
|
|
|
denom, _ := strconv.ParseFloat(values[1], 64)
|
|
|
|
|
2020-04-26 11:41:54 +02:00
|
|
|
data.Aperture = float32(math.Round((number/denom)*1000) / 1000)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["FocalLengthIn35mmFilm"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.FocalLength = i
|
|
|
|
}
|
2022-11-17 08:28:30 +01:00
|
|
|
} else if v, ok := data.exif["FocalLength"]; ok {
|
|
|
|
values := strings.Split(v, "/")
|
2020-01-07 17:36:49 +01:00
|
|
|
|
|
|
|
if len(values) == 2 && values[1] != "0" && values[1] != "" {
|
|
|
|
number, _ := strconv.ParseFloat(values[0], 64)
|
|
|
|
denom, _ := strconv.ParseFloat(values[1], 64)
|
|
|
|
|
|
|
|
data.FocalLength = int(math.Round((number/denom)*1000) / 1000)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["ISOSpeedRatings"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Iso = i
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["ImageUniqueID"]; ok {
|
2020-07-23 15:34:20 +02:00
|
|
|
if id := rnd.SanitizeUUID(value); id != "" {
|
|
|
|
data.DocumentID = id
|
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["PixelXDimension"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Width = i
|
|
|
|
}
|
2022-04-09 19:56:38 +02:00
|
|
|
} else if value, ok := data.exif["ImageWidth"]; ok {
|
2020-02-07 14:40:06 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Width = i
|
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["PixelYDimension"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Height = i
|
|
|
|
}
|
2022-04-09 19:56:38 +02:00
|
|
|
} else if value, ok := data.exif["ImageLength"]; ok {
|
2020-02-07 14:40:06 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Height = i
|
|
|
|
}
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["Orientation"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil {
|
|
|
|
data.Orientation = i
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
data.Orientation = 1
|
|
|
|
}
|
|
|
|
|
|
|
|
if data.Lat != 0 && data.Lng != 0 {
|
|
|
|
zones, err := tz.GetZone(tz.Point{
|
2020-04-26 11:41:54 +02:00
|
|
|
Lat: float64(data.Lat),
|
|
|
|
Lon: float64(data.Lng),
|
2020-01-07 17:36:49 +01:00
|
|
|
})
|
|
|
|
|
2020-04-29 16:12:09 +02:00
|
|
|
if err == nil && len(zones) > 0 {
|
|
|
|
data.TimeZone = zones[0]
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-25 16:31:09 +01:00
|
|
|
takenAt := time.Time{}
|
2020-05-15 11:15:15 +02:00
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
for _, name := range exifDateTimeTags {
|
2024-01-05 16:31:07 +01:00
|
|
|
if dateTime := txt.ParseTime(data.exif[name], data.TimeZone); !dateTime.IsZero() {
|
2022-03-25 16:31:09 +01:00
|
|
|
takenAt = dateTime
|
|
|
|
break
|
|
|
|
}
|
2020-05-15 11:15:15 +02:00
|
|
|
}
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
// Fallback to GPS timestamp.
|
|
|
|
if takenAt.IsZero() && !data.TakenGps.IsZero() {
|
|
|
|
takenAt = data.TakenGps.UTC()
|
|
|
|
}
|
|
|
|
|
|
|
|
// Nanoseconds.
|
|
|
|
if data.TakenNs <= 0 {
|
|
|
|
for _, name := range exifSubSecTags {
|
|
|
|
if s := data.exif[name]; txt.IsPosInt(s) {
|
|
|
|
data.TakenNs = txt.Int(s + strings.Repeat("0", 9-len(s)))
|
|
|
|
break
|
|
|
|
}
|
2022-04-09 19:56:38 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-15 09:42:07 +02:00
|
|
|
// UniqueID time found in Exif metadata?
|
2022-03-25 16:31:09 +01:00
|
|
|
if !takenAt.IsZero() {
|
|
|
|
if takenAtLocal, err := time.ParseInLocation("2006-01-02T15:04:05", takenAt.Format("2006-01-02T15:04:05"), time.UTC); err == nil {
|
|
|
|
data.TakenAtLocal = takenAtLocal
|
2020-01-07 17:36:49 +01:00
|
|
|
} else {
|
2022-03-25 16:31:09 +01:00
|
|
|
data.TakenAtLocal = takenAt
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
2022-03-25 16:31:09 +01:00
|
|
|
|
|
|
|
data.TakenAt = takenAt.UTC()
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-13 22:17:59 +02:00
|
|
|
// Add nanoseconds to the calculated UTC and local time.
|
|
|
|
if data.TakenAt.Nanosecond() == 0 {
|
|
|
|
if ns := time.Duration(data.TakenNs); ns > 0 && ns <= time.Second {
|
|
|
|
data.TakenAt.Truncate(time.Second).UTC().Add(ns)
|
|
|
|
data.TakenAtLocal.Truncate(time.Second).Add(ns)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["Flash"]; ok {
|
2020-01-07 17:36:49 +01:00
|
|
|
if i, err := strconv.Atoi(value); err == nil && i&1 == 1 {
|
2021-04-25 14:17:34 +02:00
|
|
|
data.AddKeywords(KeywordFlash)
|
2020-01-07 17:36:49 +01:00
|
|
|
data.Flash = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["ImageDescription"]; ok {
|
2020-07-13 14:20:43 +02:00
|
|
|
data.AutoAddKeywords(value)
|
2020-06-27 14:15:25 +02:00
|
|
|
data.Description = SanitizeDescription(value)
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|
|
|
|
|
2022-04-09 19:56:38 +02:00
|
|
|
if value, ok := data.exif["ProjectionType"]; ok {
|
2021-04-25 14:17:34 +02:00
|
|
|
data.AddKeywords(KeywordPanorama)
|
2022-04-15 09:42:07 +02:00
|
|
|
data.Projection = projection.New(SanitizeString(value)).String()
|
2020-07-16 13:02:48 +02:00
|
|
|
}
|
|
|
|
|
2020-10-19 11:50:54 +02:00
|
|
|
data.Subject = SanitizeMeta(data.Subject)
|
|
|
|
data.Artist = SanitizeMeta(data.Artist)
|
|
|
|
|
2020-05-13 20:53:15 +02:00
|
|
|
return nil
|
2020-01-07 17:36:49 +01:00
|
|
|
}
|