mirror of
https://github.com/xuthus83/pigallery2.git
synced 2025-01-14 14:43:17 +08:00
Merge branch 'master' of https://github.com/grasdk/pigallery2 into feature/Clear-DateTime-Tag-Priority
This commit is contained in:
commit
e628f6adde
14
package-lock.json
generated
14
package-lock.json
generated
@ -20,13 +20,12 @@
|
||||
"express": "4.18.2",
|
||||
"express-unless": "2.1.3",
|
||||
"fluent-ffmpeg": "2.1.2",
|
||||
"image-size": "1.0.2",
|
||||
"image-size": "1.1.1",
|
||||
"locale": "0.1.0",
|
||||
"node-geocoder": "4.2.0",
|
||||
"nodemailer": "6.9.4",
|
||||
"reflect-metadata": "0.1.13",
|
||||
"sharp": "0.31.3",
|
||||
"ts-node-iptc": "1.0.11",
|
||||
"typeconfig": "2.2.11",
|
||||
"typeorm": "0.3.12",
|
||||
"xml2js": "0.6.2"
|
||||
@ -12234,8 +12233,9 @@
|
||||
}
|
||||
},
|
||||
"node_modules/image-size": {
|
||||
"version": "1.0.2",
|
||||
"license": "MIT",
|
||||
"version": "1.1.1",
|
||||
"resolved": "https://registry.npmjs.org/image-size/-/image-size-1.1.1.tgz",
|
||||
"integrity": "sha512-541xKlUw6jr/6gGuk92F+mYM5zaFAc5ahphvkqvNe2bQ6gVBkd6bfrmVJ2t4KDAfikAYZyIqTnktX3i6/aQDrQ==",
|
||||
"dependencies": {
|
||||
"queue": "6.0.2"
|
||||
},
|
||||
@ -12243,7 +12243,7 @@
|
||||
"image-size": "bin/image-size.js"
|
||||
},
|
||||
"engines": {
|
||||
"node": ">=14.0.0"
|
||||
"node": ">=16.x"
|
||||
}
|
||||
},
|
||||
"node_modules/immutable": {
|
||||
@ -29978,7 +29978,9 @@
|
||||
}
|
||||
},
|
||||
"image-size": {
|
||||
"version": "1.0.2",
|
||||
"version": "1.1.1",
|
||||
"resolved": "https://registry.npmjs.org/image-size/-/image-size-1.1.1.tgz",
|
||||
"integrity": "sha512-541xKlUw6jr/6gGuk92F+mYM5zaFAc5ahphvkqvNe2bQ6gVBkd6bfrmVJ2t4KDAfikAYZyIqTnktX3i6/aQDrQ==",
|
||||
"requires": {
|
||||
"queue": "6.0.2"
|
||||
}
|
||||
|
@ -47,13 +47,12 @@
|
||||
"express": "4.18.2",
|
||||
"express-unless": "2.1.3",
|
||||
"fluent-ffmpeg": "2.1.2",
|
||||
"image-size": "1.0.2",
|
||||
"image-size": "1.1.1",
|
||||
"locale": "0.1.0",
|
||||
"node-geocoder": "4.2.0",
|
||||
"nodemailer": "6.9.4",
|
||||
"reflect-metadata": "0.1.13",
|
||||
"sharp": "0.31.3",
|
||||
"ts-node-iptc": "1.0.11",
|
||||
"typeconfig": "2.2.11",
|
||||
"typeorm": "0.3.12",
|
||||
"xml2js": "0.6.2"
|
||||
|
@ -12,7 +12,6 @@ import { FfprobeData } from 'fluent-ffmpeg';
|
||||
import { FileHandle } from 'fs/promises';
|
||||
import * as util from 'node:util';
|
||||
import * as path from 'path';
|
||||
import { IptcParser } from 'ts-node-iptc';
|
||||
import { Utils } from '../../../common/Utils';
|
||||
import { FFmpegFactory } from '../FFmpegFactory';
|
||||
import { ExtensionDecorator } from '../extension/ExtensionDecorator';
|
||||
@ -181,7 +180,7 @@ export class MetadataLoader {
|
||||
icc: false,
|
||||
jfif: false, //not needed and not supported for png
|
||||
ihdr: true,
|
||||
iptc: false, //exifr reads UTF8-encoded data wrongly, using IptcParser instead
|
||||
iptc: true,
|
||||
exif: true,
|
||||
gps: true,
|
||||
reviveValues: false, //don't convert timestamps
|
||||
@ -206,6 +205,13 @@ export class MetadataLoader {
|
||||
} catch (e) {
|
||||
//in case of failure, set dimensions to 0 so they may be read via tags
|
||||
metadata.size = { width: 0, height: 0 };
|
||||
} finally {
|
||||
if (isNaN(metadata.size.width) || metadata.size.width == null) {
|
||||
metadata.size.width = 0;
|
||||
}
|
||||
if (isNaN(metadata.size.height) || metadata.size.height == null) {
|
||||
metadata.size.height = 0;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
@ -221,46 +227,6 @@ export class MetadataLoader {
|
||||
await fileHandle.close();
|
||||
}
|
||||
try {
|
||||
|
||||
|
||||
try { //Parse iptc data using the IptcParser, which works correctly for both UTF-8 and ASCII
|
||||
const iptcData = IptcParser.parse(data);
|
||||
if (iptcData.country_or_primary_location_name) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.country =
|
||||
iptcData.country_or_primary_location_name
|
||||
.replace(/\0/g, '')
|
||||
.trim();
|
||||
}
|
||||
if (iptcData.province_or_state) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.state = iptcData.province_or_state
|
||||
.replace(/\0/g, '')
|
||||
.trim();
|
||||
}
|
||||
if (iptcData.city) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.city = iptcData.city
|
||||
.replace(/\0/g, '')
|
||||
.trim();
|
||||
}
|
||||
if (iptcData.object_name) {
|
||||
metadata.title = iptcData.object_name.replace(/\0/g, '').trim();
|
||||
}
|
||||
if (iptcData.caption) {
|
||||
metadata.caption = iptcData.caption.replace(/\0/g, '').trim();
|
||||
}
|
||||
if (Array.isArray(iptcData.keywords)) {
|
||||
metadata.keywords = iptcData.keywords;
|
||||
}
|
||||
|
||||
if (iptcData.date_time) {
|
||||
metadata.creationDate = iptcData.date_time.getTime();
|
||||
}
|
||||
} catch (err) {
|
||||
// Logger.debug(LOG_TAG, 'Error parsing iptc data', fullPath, err);
|
||||
}
|
||||
|
||||
try {
|
||||
const exif = await exifr.parse(data, exifrOptions);
|
||||
MetadataLoader.mapMetadata(metadata, exif);
|
||||
@ -338,10 +304,10 @@ export class MetadataLoader {
|
||||
|
||||
private static mapImageDimensions(metadata: PhotoMetadata, exif: any, orientation: number) {
|
||||
if (metadata.size.width <= 0) {
|
||||
metadata.size.width = exif.ifd0?.ImageWidth || exif.exif?.ExifImageWidth;
|
||||
metadata.size.width = exif.ifd0?.ImageWidth || exif.exif?.ExifImageWidth || metadata.size.width;
|
||||
}
|
||||
if (metadata.size.height <= 0) {
|
||||
metadata.size.height = exif.ifd0?.ImageHeight || exif.exif?.ExifImageHeight;
|
||||
metadata.size.height = exif.ifd0?.ImageHeight || exif.exif?.ExifImageHeight || metadata.size.height;
|
||||
}
|
||||
metadata.size.height = Math.max(metadata.size.height, 1); //ensure height dimension is positive
|
||||
metadata.size.width = Math.max(metadata.size.width, 1); //ensure width dimension is positive
|
||||
@ -370,20 +336,35 @@ export class MetadataLoader {
|
||||
}
|
||||
}
|
||||
}
|
||||
if (exif.iptc &&
|
||||
exif.iptc.Keywords &&
|
||||
exif.iptc.Keywords.length > 0) {
|
||||
const subj = Array.isArray(exif.iptc.Keywords) ? exif.iptc.Keywords : [exif.iptc.Keywords];
|
||||
if (metadata.keywords === undefined) {
|
||||
metadata.keywords = [];
|
||||
}
|
||||
for (let kw of subj) {
|
||||
kw = Utils.asciiToUTF8(kw);
|
||||
if (metadata.keywords.indexOf(kw) === -1) {
|
||||
metadata.keywords.push(kw);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
private static mapTitle(metadata: PhotoMetadata, exif: any) {
|
||||
metadata.title = exif.dc?.title?.value || metadata.title || exif.photoshop?.Headline || exif.acdsee?.caption; //acdsee caption holds the title when data is saved by digikam. Used as last resort if iptc and dc do not contain the data
|
||||
metadata.title = exif.dc?.title?.value || Utils.asciiToUTF8(exif.iptc?.ObjectName) || metadata.title || exif.photoshop?.Headline || exif.acdsee?.caption; //acdsee caption holds the title when data is saved by digikam. Used as last resort if iptc and dc do not contain the data
|
||||
}
|
||||
|
||||
private static mapCaption(metadata: PhotoMetadata, exif: any) {
|
||||
metadata.caption = exif.dc?.description?.value || metadata.caption || exif.ifd0?.ImageDescription || exif.exif?.UserComment?.value || exif.Iptc4xmpCore?.ExtDescrAccessibility?.value ||exif.acdsee?.notes;
|
||||
metadata.caption = exif.dc?.description?.value || Utils.asciiToUTF8(exif.iptc?.Caption) || metadata.caption || exif.ifd0?.ImageDescription || exif.exif?.UserComment?.value || exif.Iptc4xmpCore?.ExtDescrAccessibility?.value ||exif.acdsee?.notes;
|
||||
}
|
||||
|
||||
private static mapTimestampAndOffset(metadata: PhotoMetadata, exif: any) {
|
||||
metadata.creationDate = Utils.timestampToMS(exif?.photoshop?.DateCreated, null) ||
|
||||
Utils.timestampToMS(exif?.xmp?.CreateDate, null) ||
|
||||
Utils.timestampToMS(exif?.xmp?.ModifyDate, null) ||
|
||||
Utils.timestampToMS(Utils.toIsoTimestampString(exif?.iptc?.DateCreated, exif?.iptc?.TimeCreated), null) ||
|
||||
metadata.creationDate;
|
||||
|
||||
metadata.creationDateOffset = Utils.timestampToOffsetString(exif?.photoshop?.DateCreated) ||
|
||||
@ -490,24 +471,15 @@ export class MetadataLoader {
|
||||
|
||||
private static mapToponyms(metadata: PhotoMetadata, exif: any) {
|
||||
//Function to convert html code for special characters into their corresponding character (used in exif.photoshop-section)
|
||||
const unescape = (tag: string) => {
|
||||
return tag.replace(/&#([0-9]{1,3});/gi, function (match, numStr) {
|
||||
return String.fromCharCode(parseInt(numStr, 10));
|
||||
});
|
||||
}
|
||||
//photoshop section sometimes has City, Country and State
|
||||
if (exif.photoshop) {
|
||||
if (!metadata.positionData?.country && exif.photoshop.Country) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.country = unescape(exif.photoshop.Country);
|
||||
}
|
||||
if (!metadata.positionData?.state && exif.photoshop.State) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.state = unescape(exif.photoshop.State);
|
||||
}
|
||||
if (!metadata.positionData?.city && exif.photoshop.City) {
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.city = unescape(exif.photoshop.City);
|
||||
|
||||
metadata.positionData = metadata.positionData || {};
|
||||
metadata.positionData.country = Utils.asciiToUTF8(exif.iptc?.Country) || Utils.decodeHTMLChars(exif.photoshop?.Country);
|
||||
metadata.positionData.state = Utils.asciiToUTF8(exif.iptc?.State) || Utils.decodeHTMLChars(exif.photoshop?.State);
|
||||
metadata.positionData.city = Utils.asciiToUTF8(exif.iptc?.City) || Utils.decodeHTMLChars(exif.photoshop?.City);
|
||||
if (metadata.positionData) {
|
||||
Utils.removeNullOrEmptyObj(metadata.positionData);
|
||||
if (Object.keys(metadata.positionData).length === 0) {
|
||||
delete metadata.positionData;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
118
src/common/HTMLCharCodes.ts
Normal file
118
src/common/HTMLCharCodes.ts
Normal file
@ -0,0 +1,118 @@
|
||||
interface HTMLCharDictionary {
|
||||
[key: string]: string;
|
||||
}
|
||||
|
||||
export const HTMLChar: HTMLCharDictionary = {
|
||||
""": "\"",
|
||||
"&": "&",
|
||||
"<": "<",
|
||||
">": ">",
|
||||
" ": " ",
|
||||
"¡": "¡",
|
||||
"¢": "¢",
|
||||
"£": "£",
|
||||
"¤": "¤",
|
||||
"¥": "¥",
|
||||
"¦": "¦",
|
||||
"§": "§",
|
||||
"¨": "¨",
|
||||
"©": "©",
|
||||
"®": "®",
|
||||
"™": "™",
|
||||
"ª": "ª",
|
||||
"«": "«",
|
||||
"¬": "¬",
|
||||
"­": "",
|
||||
"¯": "¯",
|
||||
"°": "°",
|
||||
"±": "±",
|
||||
"²": "²",
|
||||
"³": "³",
|
||||
"´": "´",
|
||||
"µ": "µ",
|
||||
"¶": "¶",
|
||||
"·": "·",
|
||||
"¸": "¸",
|
||||
"¹": "¹",
|
||||
"º": "º",
|
||||
"»": "»",
|
||||
"¼": "¼",
|
||||
"½": "½",
|
||||
"¾": "¾",
|
||||
"¿": "¿",
|
||||
"×": "×",
|
||||
"÷": "÷",
|
||||
"Ð": "Ð",
|
||||
"ð": "ð",
|
||||
"Þ": "Þ",
|
||||
"þ": "þ",
|
||||
"Æ": "Æ",
|
||||
"æ": "æ",
|
||||
"Œ": "Œ",
|
||||
"œ": "œ",
|
||||
"Å": "Å",
|
||||
"Ø": "Ø",
|
||||
"Ç": "Ç",
|
||||
"ç": "ç",
|
||||
"ß": "ß",
|
||||
"Ñ": "Ñ",
|
||||
"ñ": "ñ",
|
||||
"Á": "Á",
|
||||
"À": "À",
|
||||
"Â": "Â",
|
||||
"Ä": "Ä",
|
||||
"Ã": "Ã",
|
||||
"á": "á",
|
||||
"à": "à",
|
||||
"â": "â",
|
||||
"ä": "ä",
|
||||
"ã": "ã",
|
||||
"å": "å",
|
||||
"É": "É",
|
||||
"È": "È",
|
||||
"Ê": "Ê",
|
||||
"Ë": "Ë",
|
||||
"&Etilde;": "Ẽ",
|
||||
"é": "é",
|
||||
"è": "è",
|
||||
"ê": "ê",
|
||||
"ë": "ë",
|
||||
"Í": "Í",
|
||||
"Ì": "Ì",
|
||||
"Î": "Î",
|
||||
"Ï": "Ï",
|
||||
"Ĩ": "Ĩ",
|
||||
"í": "í",
|
||||
"ì": "ì",
|
||||
"î": "î",
|
||||
"ï": "ï",
|
||||
"ĩ": "ĩ",
|
||||
"Ó": "Ó",
|
||||
"Ò": "Ò",
|
||||
"Ô": "Ô",
|
||||
"Ö": "Ö",
|
||||
"Õ": "Õ",
|
||||
"ó": "ó",
|
||||
"ò": "ò",
|
||||
"ô": "ô",
|
||||
"ö": "ö",
|
||||
"õ": "õ",
|
||||
"Ú": "Ú",
|
||||
"Ù": "Ù",
|
||||
"Û": "Û",
|
||||
"Ü": "Ü",
|
||||
"Ũ": "Ũ",
|
||||
"Ů": "Ů",
|
||||
"ú": "ú",
|
||||
"ù": "ù",
|
||||
"û": "û",
|
||||
"ü": "ü",
|
||||
"ũ": "ũ",
|
||||
"ů": "ů",
|
||||
"Ý": "Ý",
|
||||
"Ŷ": "Ŷ",
|
||||
"Ÿ": "Ÿ",
|
||||
"ý": "ý",
|
||||
"ŷ": "ŷ",
|
||||
"ÿ": "ÿ"
|
||||
};
|
@ -1,3 +1,5 @@
|
||||
import { HTMLChar } from './HTMLCharCodes';
|
||||
|
||||
export class Utils {
|
||||
static GUID(): string {
|
||||
const s4 = (): string =>
|
||||
@ -97,6 +99,25 @@ export class Utils {
|
||||
return d.getUTCFullYear() + '-' + d.getUTCMonth() + '-' + d.getUTCDate();
|
||||
}
|
||||
|
||||
static toIsoTimestampString(YYYYMMDD: string, hhmmss: string): string {
|
||||
if (YYYYMMDD && hhmmss) {
|
||||
// Regular expression to match YYYYMMDD format
|
||||
const dateRegex = /^(\d{4})(\d{2})(\d{2})$/;
|
||||
// Regular expression to match hhmmss+/-ohom format
|
||||
const timeRegex = /^(\d{2})(\d{2})(\d{2})([+-]\d{2})?(\d{2})?$/;
|
||||
const [, year, month, day] = YYYYMMDD.match(dateRegex);
|
||||
const [, hour, minute, second, offsetHour, offsetMinute] = hhmmss.match(timeRegex);
|
||||
const isoTimestamp = `${year}-${month}-${day}T${hour}:${minute}:${second}`;
|
||||
if (offsetHour && offsetMinute) {
|
||||
return isoTimestamp + `${offsetHour}:${offsetMinute}`;
|
||||
} else {
|
||||
return isoTimestamp;
|
||||
}
|
||||
} else {
|
||||
return undefined;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
static makeUTCMidnight(d: number | Date) {
|
||||
if (!(d instanceof Date)) {
|
||||
@ -125,7 +146,7 @@ export class Utils {
|
||||
}
|
||||
|
||||
//function to convert timestamp into milliseconds taking offset into account
|
||||
static timestampToMS(timestamp: string, offset: string) {
|
||||
static timestampToMS(timestamp: string, offset: string): number {
|
||||
if (!timestamp) {
|
||||
return undefined;
|
||||
}
|
||||
@ -371,6 +392,31 @@ export class Utils {
|
||||
return curr;
|
||||
}
|
||||
|
||||
public static asciiToUTF8(text: string): string {
|
||||
if (text) {
|
||||
return Buffer.from(text, 'ascii').toString('utf-8');
|
||||
} else {
|
||||
return text;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
|
||||
public static decodeHTMLChars(text: string): string {
|
||||
if (text) {
|
||||
const newtext = text.replace(/&#([0-9]{1,3});/gi, function (match, numStr) {
|
||||
return String.fromCharCode(parseInt(numStr, 10));
|
||||
});
|
||||
return newtext.replace(/&[^;]+;/g, function (match) {
|
||||
const char = HTMLChar[match];
|
||||
return char ? char : match;
|
||||
});
|
||||
} else {
|
||||
return text;
|
||||
}
|
||||
}
|
||||
|
||||
|
||||
public static isUInt32(value: number, max = 4294967295): boolean {
|
||||
value = parseInt('' + value, 10);
|
||||
return !isNaN(value) && value >= 0 && value <= max;
|
||||
|
BIN
test/backend/assets/parsingfromheic.heic
Normal file
BIN
test/backend/assets/parsingfromheic.heic
Normal file
Binary file not shown.
15
test/backend/assets/parsingfromheic.json
Normal file
15
test/backend/assets/parsingfromheic.json
Normal file
@ -0,0 +1,15 @@
|
||||
{
|
||||
"positionData": {
|
||||
"GPSData": {
|
||||
"latitude": 9.061331,
|
||||
"longitude": 38.761711
|
||||
}
|
||||
},
|
||||
"creationDate": 1706438594000,
|
||||
"creationDateOffset": "+03:00",
|
||||
"fileSize": 2158564,
|
||||
"size": {
|
||||
"height": 512,
|
||||
"width": 512
|
||||
}
|
||||
}
|
@ -41,9 +41,9 @@
|
||||
"latitude": 37.871093,
|
||||
"longitude": -122.25678
|
||||
},
|
||||
"city": "test city őúéáűóöí-.,)(=",
|
||||
"city": "test city őúéáűóöí-.,)(=/%!+\"'",
|
||||
"country": "test country őúéáűóöí-.,)(=/%!+\"'",
|
||||
"state": "test state őúéáűóöí-.,)("
|
||||
"state": "test state őúéáűóöí-.,)(=/%!+\"'"
|
||||
},
|
||||
"rating": 3,
|
||||
"size": {
|
||||
|
@ -23,6 +23,12 @@ describe('MetadataLoader', () => {
|
||||
});
|
||||
|
||||
|
||||
it('should load heic', async () => {
|
||||
const data = await MetadataLoader.loadPhotoMetadata(path.join(__dirname, '/../../../assets/parsingfromheic.heic'));
|
||||
const expected = require(path.join(__dirname, '/../../../assets/parsingfromheic.json'));
|
||||
expect(Utils.clone(data)).to.be.deep.equal(expected);
|
||||
});
|
||||
|
||||
it('should load png', async () => {
|
||||
const data = await MetadataLoader.loadPhotoMetadata(path.join(__dirname, '/../../../assets/test_png.png'));
|
||||
const expected = require(path.join(__dirname, '/../../../assets/test_png.json'));
|
||||
|
Binary file not shown.
Loading…
x
Reference in New Issue
Block a user