fix(localize): serialize all the message locations to XLIFF (#39411)
Previously only the first message, for each id, was serialized which meant that additional message location information was lost. Now all the message locations are included in the serialized messages. Fixes #39330 PR Close #39411
This commit is contained in:
parent
35fdc6d92f
commit
57ead7aa53
|
@ -0,0 +1,37 @@
|
||||||
|
/**
|
||||||
|
* @license
|
||||||
|
* Copyright Google LLC All Rights Reserved.
|
||||||
|
*
|
||||||
|
* Use of this source code is governed by an MIT-style license that can be
|
||||||
|
* found in the LICENSE file at https://angular.io/license
|
||||||
|
*/
|
||||||
|
import {ɵMessageId, ɵParsedMessage, ɵSourceLocation} from '@angular/localize';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Consolidate an array of messages into a map from message id to an array of messages with that id.
|
||||||
|
*
|
||||||
|
* @param messages the messages to consolidate.
|
||||||
|
* @param getMessageId a function that will compute the message id of a message.
|
||||||
|
*/
|
||||||
|
export function consolidateMessages(
|
||||||
|
messages: ɵParsedMessage[],
|
||||||
|
getMessageId: (message: ɵParsedMessage) => string): Map<ɵMessageId, ɵParsedMessage[]> {
|
||||||
|
const consolidateMessages = new Map<ɵMessageId, ɵParsedMessage[]>();
|
||||||
|
for (const message of messages) {
|
||||||
|
const id = getMessageId(message);
|
||||||
|
if (!consolidateMessages.has(id)) {
|
||||||
|
consolidateMessages.set(id, [message]);
|
||||||
|
} else {
|
||||||
|
consolidateMessages.get(id)!.push(message);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return consolidateMessages;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Does the given message have a location property?
|
||||||
|
*/
|
||||||
|
export function hasLocation(message: ɵParsedMessage): message is ɵParsedMessage&
|
||||||
|
{location: ɵSourceLocation} {
|
||||||
|
return message.location !== undefined;
|
||||||
|
}
|
|
@ -11,6 +11,7 @@ import {ɵParsedMessage, ɵSourceLocation} from '@angular/localize';
|
||||||
import {FormatOptions, validateOptions} from './format_options';
|
import {FormatOptions, validateOptions} from './format_options';
|
||||||
import {extractIcuPlaceholders} from './icu_parsing';
|
import {extractIcuPlaceholders} from './icu_parsing';
|
||||||
import {TranslationSerializer} from './translation_serializer';
|
import {TranslationSerializer} from './translation_serializer';
|
||||||
|
import {consolidateMessages, hasLocation} from './utils';
|
||||||
import {XmlFile} from './xml_file';
|
import {XmlFile} from './xml_file';
|
||||||
|
|
||||||
/** This is the number of characters that a legacy Xliff 1.2 message id has. */
|
/** This is the number of characters that a legacy Xliff 1.2 message id has. */
|
||||||
|
@ -33,7 +34,7 @@ export class Xliff1TranslationSerializer implements TranslationSerializer {
|
||||||
}
|
}
|
||||||
|
|
||||||
serialize(messages: ɵParsedMessage[]): string {
|
serialize(messages: ɵParsedMessage[]): string {
|
||||||
const ids = new Set<string>();
|
const messageMap = consolidateMessages(messages, message => this.getMessageId(message));
|
||||||
const xml = new XmlFile();
|
const xml = new XmlFile();
|
||||||
xml.startTag('xliff', {'version': '1.2', 'xmlns': 'urn:oasis:names:tc:xliff:document:1.2'});
|
xml.startTag('xliff', {'version': '1.2', 'xmlns': 'urn:oasis:names:tc:xliff:document:1.2'});
|
||||||
// NOTE: the `original` property is set to the legacy `ng2.template` value for backward
|
// NOTE: the `original` property is set to the legacy `ng2.template` value for backward
|
||||||
|
@ -50,21 +51,19 @@ export class Xliff1TranslationSerializer implements TranslationSerializer {
|
||||||
...this.formatOptions,
|
...this.formatOptions,
|
||||||
});
|
});
|
||||||
xml.startTag('body');
|
xml.startTag('body');
|
||||||
for (const message of messages) {
|
for (const [id, duplicateMessages] of messageMap.entries()) {
|
||||||
const id = this.getMessageId(message);
|
const message = duplicateMessages[0];
|
||||||
if (ids.has(id)) {
|
|
||||||
// Do not render the same message more than once
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
ids.add(id);
|
|
||||||
|
|
||||||
xml.startTag('trans-unit', {id, datatype: 'html'});
|
xml.startTag('trans-unit', {id, datatype: 'html'});
|
||||||
xml.startTag('source', {}, {preserveWhitespace: true});
|
xml.startTag('source', {}, {preserveWhitespace: true});
|
||||||
this.serializeMessage(xml, message);
|
this.serializeMessage(xml, message);
|
||||||
xml.endTag('source', {preserveWhitespace: false});
|
xml.endTag('source', {preserveWhitespace: false});
|
||||||
if (message.location) {
|
|
||||||
this.serializeLocation(xml, message.location);
|
// Write all the locations
|
||||||
|
for (const {location} of duplicateMessages.filter(hasLocation)) {
|
||||||
|
this.serializeLocation(xml, location);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (message.description) {
|
if (message.description) {
|
||||||
this.serializeNote(xml, 'description', message.description);
|
this.serializeNote(xml, 'description', message.description);
|
||||||
}
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ import {ɵParsedMessage, ɵSourceLocation} from '@angular/localize';
|
||||||
import {FormatOptions, validateOptions} from './format_options';
|
import {FormatOptions, validateOptions} from './format_options';
|
||||||
import {extractIcuPlaceholders} from './icu_parsing';
|
import {extractIcuPlaceholders} from './icu_parsing';
|
||||||
import {TranslationSerializer} from './translation_serializer';
|
import {TranslationSerializer} from './translation_serializer';
|
||||||
|
import {consolidateMessages, hasLocation} from './utils';
|
||||||
import {XmlFile} from './xml_file';
|
import {XmlFile} from './xml_file';
|
||||||
|
|
||||||
/** This is the maximum number of characters that can appear in a legacy XLIFF 2.0 message id. */
|
/** This is the maximum number of characters that can appear in a legacy XLIFF 2.0 message id. */
|
||||||
|
@ -33,7 +34,7 @@ export class Xliff2TranslationSerializer implements TranslationSerializer {
|
||||||
}
|
}
|
||||||
|
|
||||||
serialize(messages: ɵParsedMessage[]): string {
|
serialize(messages: ɵParsedMessage[]): string {
|
||||||
const ids = new Set<string>();
|
const messageMap = consolidateMessages(messages, message => this.getMessageId(message));
|
||||||
const xml = new XmlFile();
|
const xml = new XmlFile();
|
||||||
xml.startTag('xliff', {
|
xml.startTag('xliff', {
|
||||||
'version': '2.0',
|
'version': '2.0',
|
||||||
|
@ -48,24 +49,23 @@ export class Xliff2TranslationSerializer implements TranslationSerializer {
|
||||||
// messages that come from a particular original file, and the translation file parsers may
|
// messages that come from a particular original file, and the translation file parsers may
|
||||||
// not
|
// not
|
||||||
xml.startTag('file', {'id': 'ngi18n', 'original': 'ng.template', ...this.formatOptions});
|
xml.startTag('file', {'id': 'ngi18n', 'original': 'ng.template', ...this.formatOptions});
|
||||||
for (const message of messages) {
|
for (const [id, duplicateMessages] of messageMap.entries()) {
|
||||||
const id = this.getMessageId(message);
|
const message = duplicateMessages[0];
|
||||||
if (ids.has(id)) {
|
|
||||||
// Do not render the same message more than once
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
ids.add(id);
|
|
||||||
xml.startTag('unit', {id});
|
xml.startTag('unit', {id});
|
||||||
if (message.meaning || message.description || message.location) {
|
const messagesWithLocations = duplicateMessages.filter(hasLocation);
|
||||||
|
if (message.meaning || message.description || messagesWithLocations.length) {
|
||||||
xml.startTag('notes');
|
xml.startTag('notes');
|
||||||
if (message.location) {
|
|
||||||
const {file, start, end} = message.location;
|
// Write all the locations
|
||||||
|
for (const {location: {file, start, end}} of messagesWithLocations) {
|
||||||
const endLineString =
|
const endLineString =
|
||||||
end !== undefined && end.line !== start.line ? `,${end.line + 1}` : '';
|
end !== undefined && end.line !== start.line ? `,${end.line + 1}` : '';
|
||||||
this.serializeNote(
|
this.serializeNote(
|
||||||
xml, 'location',
|
xml, 'location',
|
||||||
`${this.fs.relative(this.basePath, file)}:${start.line + 1}${endLineString}`);
|
`${this.fs.relative(this.basePath, file)}:${start.line + 1}${endLineString}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (message.description) {
|
if (message.description) {
|
||||||
this.serializeNote(xml, 'description', message.description);
|
this.serializeNote(xml, 'description', message.description);
|
||||||
}
|
}
|
||||||
|
|
|
@ -135,6 +135,101 @@ runInEachFileSystem(() => {
|
||||||
`</xliff>\n`,
|
`</xliff>\n`,
|
||||||
].join('\n'));
|
].join('\n'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should convert a set of parsed messages into an XML string', () => {
|
||||||
|
const messageLocation1: ɵSourceLocation = {
|
||||||
|
start: {line: 0, column: 5},
|
||||||
|
end: {line: 0, column: 10},
|
||||||
|
file: absoluteFrom('/project/file-1.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation2: ɵSourceLocation = {
|
||||||
|
start: {line: 3, column: 2},
|
||||||
|
end: {line: 4, column: 7},
|
||||||
|
file: absoluteFrom('/project/file-2.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation3: ɵSourceLocation = {
|
||||||
|
start: {line: 0, column: 5},
|
||||||
|
end: {line: 0, column: 10},
|
||||||
|
file: absoluteFrom('/project/file-3.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation4: ɵSourceLocation = {
|
||||||
|
start: {line: 3, column: 2},
|
||||||
|
end: {line: 4, column: 7},
|
||||||
|
file: absoluteFrom('/project/file-4.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messages: ɵParsedMessage[] = [
|
||||||
|
mockMessage('1234', ['message text'], [], {location: messageLocation1}),
|
||||||
|
mockMessage('1234', ['message text'], [], {location: messageLocation2}),
|
||||||
|
mockMessage('1234', ['message text'], [], {
|
||||||
|
location: messageLocation3,
|
||||||
|
legacyIds: ['87654321FEDCBA0987654321FEDCBA0987654321', '563965274788097516']
|
||||||
|
}),
|
||||||
|
mockMessage(
|
||||||
|
'1234', ['message text'], [], {location: messageLocation4, customId: 'other'}),
|
||||||
|
];
|
||||||
|
const serializer = new Xliff1TranslationSerializer(
|
||||||
|
'xx', absoluteFrom('/project'), useLegacyIds, options);
|
||||||
|
const output = serializer.serialize(messages);
|
||||||
|
|
||||||
|
// Note that in this test the third message will match the first two if legacyIds is
|
||||||
|
// false. Otherwise it will be a separate message on its own.
|
||||||
|
|
||||||
|
expect(output).toEqual([
|
||||||
|
`<?xml version="1.0" encoding="UTF-8" ?>`,
|
||||||
|
`<xliff version="1.2" xmlns="urn:oasis:names:tc:xliff:document:1.2">`,
|
||||||
|
` <file source-language="xx" datatype="plaintext" original="ng2.template"${
|
||||||
|
toAttributes(options)}>`,
|
||||||
|
` <body>`,
|
||||||
|
` <trans-unit id="1234" datatype="html">`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` <context-group purpose="location">`,
|
||||||
|
` <context context-type="sourcefile">file-1.ts</context>`,
|
||||||
|
` <context context-type="linenumber">1</context>`,
|
||||||
|
` </context-group>`,
|
||||||
|
` <context-group purpose="location">`,
|
||||||
|
` <context context-type="sourcefile">file-2.ts</context>`,
|
||||||
|
` <context context-type="linenumber">4,5</context>`,
|
||||||
|
` </context-group>`,
|
||||||
|
...useLegacyIds ?
|
||||||
|
[] :
|
||||||
|
[
|
||||||
|
` <context-group purpose="location">`,
|
||||||
|
` <context context-type="sourcefile">file-3.ts</context>`,
|
||||||
|
` <context context-type="linenumber">1</context>`,
|
||||||
|
` </context-group>`,
|
||||||
|
],
|
||||||
|
` </trans-unit>`,
|
||||||
|
...useLegacyIds ?
|
||||||
|
[
|
||||||
|
` <trans-unit id="87654321FEDCBA0987654321FEDCBA0987654321" datatype="html">`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` <context-group purpose="location">`,
|
||||||
|
` <context context-type="sourcefile">file-3.ts</context>`,
|
||||||
|
` <context context-type="linenumber">1</context>`,
|
||||||
|
` </context-group>`,
|
||||||
|
` </trans-unit>`,
|
||||||
|
] :
|
||||||
|
[],
|
||||||
|
` <trans-unit id="other" datatype="html">`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` <context-group purpose="location">`,
|
||||||
|
` <context context-type="sourcefile">file-4.ts</context>`,
|
||||||
|
` <context context-type="linenumber">4,5</context>`,
|
||||||
|
` </context-group>`,
|
||||||
|
` </trans-unit>`,
|
||||||
|
` </body>`,
|
||||||
|
` </file>`,
|
||||||
|
`</xliff>\n`,
|
||||||
|
].join('\n'));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
|
@ -158,6 +158,91 @@ runInEachFileSystem(() => {
|
||||||
`</xliff>\n`,
|
`</xliff>\n`,
|
||||||
].join('\n'));
|
].join('\n'));
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should convert a set of parsed messages into an XML string', () => {
|
||||||
|
const messageLocation1: ɵSourceLocation = {
|
||||||
|
start: {line: 0, column: 5},
|
||||||
|
end: {line: 0, column: 10},
|
||||||
|
file: absoluteFrom('/project/file-1.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation2: ɵSourceLocation = {
|
||||||
|
start: {line: 3, column: 2},
|
||||||
|
end: {line: 4, column: 7},
|
||||||
|
file: absoluteFrom('/project/file-2.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation3: ɵSourceLocation = {
|
||||||
|
start: {line: 0, column: 5},
|
||||||
|
end: {line: 0, column: 10},
|
||||||
|
file: absoluteFrom('/project/file-3.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messageLocation4: ɵSourceLocation = {
|
||||||
|
start: {line: 3, column: 2},
|
||||||
|
end: {line: 4, column: 7},
|
||||||
|
file: absoluteFrom('/project/file-4.ts'),
|
||||||
|
text: 'message text'
|
||||||
|
};
|
||||||
|
|
||||||
|
const messages: ɵParsedMessage[] = [
|
||||||
|
mockMessage('1234', ['message text'], [], {location: messageLocation1}),
|
||||||
|
mockMessage('1234', ['message text'], [], {location: messageLocation2}),
|
||||||
|
mockMessage('1234', ['message text'], [], {
|
||||||
|
location: messageLocation3,
|
||||||
|
legacyIds: ['87654321FEDCBA0987654321FEDCBA0987654321', '563965274788097516']
|
||||||
|
}),
|
||||||
|
mockMessage(
|
||||||
|
'1234', ['message text'], [], {location: messageLocation4, customId: 'other'}),
|
||||||
|
];
|
||||||
|
const serializer = new Xliff2TranslationSerializer(
|
||||||
|
'xx', absoluteFrom('/project'), useLegacyIds, options);
|
||||||
|
const output = serializer.serialize(messages);
|
||||||
|
|
||||||
|
// Note that in this test the third message will match the first two if legacyIds is
|
||||||
|
// false. Otherwise it will be a separate message on its own.
|
||||||
|
|
||||||
|
expect(output).toEqual([
|
||||||
|
`<?xml version="1.0" encoding="UTF-8" ?>`,
|
||||||
|
`<xliff version="2.0" xmlns="urn:oasis:names:tc:xliff:document:2.0" srcLang="xx">`,
|
||||||
|
` <file id="ngi18n" original="ng.template"${toAttributes(options)}>`,
|
||||||
|
` <unit id="1234">`,
|
||||||
|
` <notes>`,
|
||||||
|
` <note category="location">file-1.ts:1</note>`,
|
||||||
|
` <note category="location">file-2.ts:4,5</note>`,
|
||||||
|
...useLegacyIds ? [] : [' <note category="location">file-3.ts:1</note>'],
|
||||||
|
` </notes>`,
|
||||||
|
` <segment>`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` </segment>`,
|
||||||
|
` </unit>`,
|
||||||
|
...useLegacyIds ?
|
||||||
|
[
|
||||||
|
` <unit id="563965274788097516">`,
|
||||||
|
` <notes>`,
|
||||||
|
` <note category="location">file-3.ts:1</note>`,
|
||||||
|
` </notes>`,
|
||||||
|
` <segment>`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` </segment>`,
|
||||||
|
` </unit>`,
|
||||||
|
] :
|
||||||
|
[],
|
||||||
|
` <unit id="other">`,
|
||||||
|
` <notes>`,
|
||||||
|
` <note category="location">file-4.ts:4,5</note>`,
|
||||||
|
` </notes>`,
|
||||||
|
` <segment>`,
|
||||||
|
` <source>message text</source>`,
|
||||||
|
` </segment>`,
|
||||||
|
` </unit>`,
|
||||||
|
` </file>`,
|
||||||
|
`</xliff>\n`,
|
||||||
|
].join('\n'));
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
Loading…
Reference in New Issue