2014-06-19 14:16:30 +02:00
|
|
|
var fs = require('fs');
|
|
|
|
var isArray = require('lodash.isarray');
|
2011-12-21 22:50:01 -08:00
|
|
|
|
2012-01-02 15:52:05 -08:00
|
|
|
function trim(string) {
|
2013-12-16 16:36:32 +01:00
|
|
|
return string.replace(/^\s+|\s+$/g, '');
|
|
|
|
}
|
2011-12-21 22:50:01 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
var PO = function () {
|
|
|
|
this.comments = [];
|
|
|
|
this.headers = {};
|
|
|
|
this.items = [];
|
2012-01-01 17:02:19 -08:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.prototype.save = function (filename, callback) {
|
2013-12-17 14:30:07 +01:00
|
|
|
fs.writeFile(filename, this.toString(), callback);
|
2013-12-16 16:36:32 +01:00
|
|
|
};
|
2012-01-02 15:52:05 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.prototype.toString = function () {
|
2014-06-19 14:16:30 +02:00
|
|
|
var lines = [];
|
2012-01-02 15:52:05 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
if (this.comments) {
|
|
|
|
this.comments.forEach(function (comment) {
|
|
|
|
lines.push('# ' + comment);
|
|
|
|
});
|
|
|
|
}
|
2012-01-01 17:02:19 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
lines.push('msgid ""');
|
|
|
|
lines.push('msgstr ""');
|
|
|
|
|
|
|
|
var keys = Object.keys(this.headers);
|
2014-06-19 14:16:30 +02:00
|
|
|
var self = this;
|
2013-12-16 16:36:32 +01:00
|
|
|
keys.forEach(function (key) {
|
2014-06-19 14:16:30 +02:00
|
|
|
lines.push('"' + key + ': ' + self.headers[key] + '\\n"');
|
2013-12-16 16:36:32 +01:00
|
|
|
});
|
2012-01-01 17:02:19 -08:00
|
|
|
|
2011-12-22 16:12:01 -08:00
|
|
|
lines.push('');
|
2012-01-01 17:02:19 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
this.items.forEach(function (item) {
|
|
|
|
lines.push(item.toString());
|
|
|
|
lines.push('');
|
|
|
|
});
|
|
|
|
|
2014-06-19 14:16:30 +02:00
|
|
|
return lines.join('\n');
|
2011-12-21 22:50:01 -08:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.load = function (filename, callback) {
|
|
|
|
fs.readFile(filename, 'utf-8', function (err, data) {
|
|
|
|
if (err) {
|
2013-12-17 14:30:07 +01:00
|
|
|
return callback(err);
|
2013-12-16 16:36:32 +01:00
|
|
|
}
|
|
|
|
var po = PO.parse(data);
|
2013-12-17 14:30:07 +01:00
|
|
|
callback(null, po);
|
2013-12-16 16:36:32 +01:00
|
|
|
});
|
2012-01-02 15:52:05 -08:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.parse = function (data) {
|
2013-09-12 14:01:37 +03:00
|
|
|
//support both unix and windows newline formats.
|
|
|
|
data = data.replace(/\r\n/g, '\n');
|
2014-06-19 14:16:30 +02:00
|
|
|
var po = new PO();
|
|
|
|
var sections = data.split(/\n\n/);
|
|
|
|
var headers = sections.shift();
|
|
|
|
var lines = sections.join('\n').split(/\n/);
|
2013-12-16 16:36:32 +01:00
|
|
|
|
|
|
|
po.headers = {
|
|
|
|
'Project-Id-Version': '',
|
|
|
|
'Report-Msgid-Bugs-To': '',
|
|
|
|
'POT-Creation-Date': '',
|
|
|
|
'PO-Revision-Date': '',
|
|
|
|
'Last-Translator': '',
|
|
|
|
'Language': '',
|
|
|
|
'Language-Team': '',
|
|
|
|
'Content-Type': '',
|
|
|
|
'Content-Transfer-Encoding': '',
|
|
|
|
'Plural-Forms': '',
|
|
|
|
};
|
|
|
|
|
2014-03-21 10:14:00 +01:00
|
|
|
headers.split(/\n/).reduce(function (acc, line) {
|
|
|
|
if (acc.merge) {
|
|
|
|
//join lines, remove last resp. first "
|
|
|
|
line = acc.pop().slice(0, -1) + line.slice(1);
|
|
|
|
delete acc.merge;
|
|
|
|
}
|
|
|
|
if (/^".*"$/.test(line) && !/^".*\\n"$/.test(line)) {
|
|
|
|
acc.merge = true;
|
|
|
|
}
|
|
|
|
acc.push(line);
|
|
|
|
return acc;
|
|
|
|
}, []).forEach(function (header) {
|
2013-12-16 16:36:32 +01:00
|
|
|
if (header.match(/^#/)) {
|
|
|
|
po.comments.push(header.replace(/^#\s*/, ''));
|
|
|
|
}
|
|
|
|
if (header.match(/^"/)) {
|
|
|
|
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
|
2014-06-19 14:16:30 +02:00
|
|
|
var p = header.split(/:/);
|
|
|
|
var name = p.shift().trim();
|
|
|
|
var value = p.join(':').trim();
|
2013-12-16 16:36:32 +01:00
|
|
|
po.headers[name] = value;
|
|
|
|
}
|
|
|
|
});
|
2012-01-02 15:52:05 -08:00
|
|
|
|
2014-06-19 14:16:30 +02:00
|
|
|
var item = new PO.Item();
|
|
|
|
var context = null;
|
|
|
|
var plural = 0;
|
|
|
|
var obsoleteCount = 0;
|
|
|
|
var noCommentLineCount = 0;
|
2012-01-01 17:02:19 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
function finish() {
|
|
|
|
if (item.msgid.length > 0) {
|
2014-06-19 13:38:07 +02:00
|
|
|
if (obsoleteCount >= noCommentLineCount) {
|
|
|
|
item.obsolete = true;
|
|
|
|
}
|
|
|
|
obsoleteCount = 0;
|
|
|
|
noCommentLineCount = 0;
|
2013-12-16 16:36:32 +01:00
|
|
|
po.items.push(item);
|
|
|
|
item = new PO.Item();
|
|
|
|
}
|
2012-01-02 15:52:05 -08:00
|
|
|
}
|
2013-12-16 16:36:32 +01:00
|
|
|
|
|
|
|
function extract(string) {
|
|
|
|
string = trim(string);
|
|
|
|
string = string.replace(/^[^"]*"|"$/g, '');
|
2012-09-23 01:39:44 +02:00
|
|
|
string = string.replace(/\\([abtnvfr'"\\?]|([0-7]{3})|x([0-9a-fA-F]{2}))/g, function (match, esc, oct, hex) {
|
|
|
|
if (oct) {
|
|
|
|
return String.fromCharCode(parseInt(oct, 8));
|
|
|
|
}
|
|
|
|
if (hex) {
|
|
|
|
return String.fromCharCode(parseInt(hex, 16));
|
|
|
|
}
|
|
|
|
switch (esc) {
|
|
|
|
case 'a':
|
|
|
|
return '\x07';
|
|
|
|
case 'b':
|
|
|
|
return '\b';
|
|
|
|
case 't':
|
|
|
|
return '\t';
|
|
|
|
case 'n':
|
|
|
|
return '\n';
|
|
|
|
case 'v':
|
|
|
|
return '\v';
|
|
|
|
case 'f':
|
|
|
|
return '\f';
|
|
|
|
case 'r':
|
|
|
|
return '\r';
|
|
|
|
default:
|
|
|
|
return esc;
|
|
|
|
}
|
|
|
|
});
|
2013-12-16 16:36:32 +01:00
|
|
|
return string;
|
2012-01-02 15:52:05 -08:00
|
|
|
}
|
2013-12-16 16:36:32 +01:00
|
|
|
|
|
|
|
while (lines.length > 0) {
|
2014-06-19 14:16:30 +02:00
|
|
|
var line = trim(lines.shift());
|
|
|
|
var lineObsolete = false;
|
|
|
|
var add = false;
|
2014-03-07 10:15:50 +01:00
|
|
|
|
|
|
|
if (line.match(/^#\~/)) { // Obsolete item
|
2014-06-19 13:38:07 +02:00
|
|
|
//only remove the obsolte comment mark, here
|
|
|
|
//might be, this is a new item, so
|
|
|
|
//only remember, this line is marked obsolete, count after line is parsed
|
2014-03-07 10:15:50 +01:00
|
|
|
line = trim(line.substring(2));
|
2014-06-19 13:38:07 +02:00
|
|
|
lineObsolete = true;
|
2014-03-07 10:15:50 +01:00
|
|
|
}
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
if (line.match(/^#:/)) { // Reference
|
|
|
|
finish();
|
|
|
|
item.references.push(trim(line.replace(/^#:/, '')));
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^#,/)) { // Flags
|
2013-12-17 15:04:44 +01:00
|
|
|
finish();
|
2014-06-19 14:16:30 +02:00
|
|
|
var flags = trim(line.replace(/^#,/, '')).split(',');
|
2013-12-17 15:04:44 +01:00
|
|
|
for (var i = 0; i < flags.length; i++) {
|
|
|
|
item.flags[flags[i]] = true;
|
|
|
|
}
|
2014-06-19 17:04:03 +02:00
|
|
|
} else if (line.match(/^#($|\s+)/)) { // Translator comment
|
2013-12-16 16:36:32 +01:00
|
|
|
finish();
|
2014-06-19 17:04:03 +02:00
|
|
|
item.comments.push(trim(line.replace(/^#($|\s+)/, '')));
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^#\./)) { // Extracted comment
|
2014-03-02 13:11:37 -05:00
|
|
|
finish();
|
|
|
|
item.extractedComments.push(trim(line.replace(/^#\./, '')));
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^msgid_plural/)) { // Plural form
|
2013-12-16 16:36:32 +01:00
|
|
|
item.msgid_plural = extract(line);
|
|
|
|
context = 'msgid_plural';
|
2014-06-19 13:38:07 +02:00
|
|
|
noCommentLineCount++;
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^msgid/)) { // Original
|
2013-12-16 16:36:32 +01:00
|
|
|
finish();
|
|
|
|
item.msgid = extract(line);
|
|
|
|
context = 'msgid';
|
2014-06-19 13:38:07 +02:00
|
|
|
noCommentLineCount++;
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^msgstr/)) { // Translation
|
2013-12-16 16:36:32 +01:00
|
|
|
var m = line.match(/^msgstr\[(\d+)\]/);
|
|
|
|
plural = m && m[1] ? parseInt(m[1]) : 0;
|
|
|
|
item.msgstr[plural] = extract(line);
|
|
|
|
context = 'msgstr';
|
2014-06-19 13:38:07 +02:00
|
|
|
noCommentLineCount++;
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (line.match(/^msgctxt/)) { // Context
|
2014-01-21 13:44:52 +01:00
|
|
|
finish();
|
|
|
|
item.msgctxt = extract(line);
|
2014-06-19 13:38:07 +02:00
|
|
|
noCommentLineCount++;
|
2014-06-19 14:16:30 +02:00
|
|
|
} else { // Probably multiline string or blank
|
2013-12-16 16:36:32 +01:00
|
|
|
if (line.length > 0) {
|
2014-06-19 13:38:07 +02:00
|
|
|
noCommentLineCount++;
|
2013-12-16 16:36:32 +01:00
|
|
|
if (context === 'msgstr') {
|
|
|
|
item.msgstr[plural] += extract(line);
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (context === 'msgid') {
|
2013-12-16 16:36:32 +01:00
|
|
|
item.msgid += extract(line);
|
2014-06-19 14:16:30 +02:00
|
|
|
} else if (context === 'msgid_plural') {
|
2013-12-16 16:36:32 +01:00
|
|
|
item.msgid_plural += extract(line);
|
|
|
|
}
|
|
|
|
}
|
2012-01-02 15:52:05 -08:00
|
|
|
}
|
2014-06-19 14:16:30 +02:00
|
|
|
|
2014-06-19 13:38:07 +02:00
|
|
|
if (lineObsolete) {
|
2014-06-19 14:16:30 +02:00
|
|
|
// Count obsolete lines for this item
|
2014-06-19 13:38:07 +02:00
|
|
|
obsoleteCount++;
|
|
|
|
}
|
2012-01-02 15:52:05 -08:00
|
|
|
}
|
2013-12-16 16:36:32 +01:00
|
|
|
finish();
|
2012-01-02 15:52:05 -08:00
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
return po;
|
2011-12-21 22:50:01 -08:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.Item = function () {
|
|
|
|
this.msgid = '';
|
2014-01-22 14:10:15 +01:00
|
|
|
this.msgctxt = null;
|
2013-12-16 16:36:32 +01:00
|
|
|
this.references = [];
|
|
|
|
this.msgid_plural = null;
|
|
|
|
this.msgstr = [];
|
2014-03-02 13:11:37 -05:00
|
|
|
this.comments = []; // translator comments
|
|
|
|
this.extractedComments = [];
|
2013-12-17 15:04:44 +01:00
|
|
|
this.flags = {};
|
2014-03-07 10:15:50 +01:00
|
|
|
this.obsolete = false;
|
2011-12-21 22:50:01 -08:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
PO.Item.prototype.toString = function () {
|
2014-06-19 14:16:30 +02:00
|
|
|
var lines = [];
|
|
|
|
var self = this;
|
2013-12-16 16:36:32 +01:00
|
|
|
|
2014-03-05 10:13:30 +01:00
|
|
|
// reverse what extract(string) method during PO.parse does
|
|
|
|
var _escape = function (string) {
|
2014-06-23 18:24:02 +02:00
|
|
|
// don't unescape \n, since string can never contain it
|
|
|
|
// since split('\n') is called on it
|
|
|
|
string = string.replace(/[\x07\b\t\v\f\r"\\]/g, function (match) {
|
|
|
|
switch (match) {
|
|
|
|
case '\x07':
|
|
|
|
return '\\a';
|
|
|
|
case '\b':
|
|
|
|
return '\\b';
|
|
|
|
case '\t':
|
|
|
|
return '\\t';
|
|
|
|
case '\v':
|
|
|
|
return '\\v';
|
|
|
|
case '\f':
|
|
|
|
return '\\f';
|
|
|
|
case '\r':
|
|
|
|
return '\\r';
|
|
|
|
default:
|
|
|
|
return '\\' + match;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return string;
|
2014-03-05 10:13:30 +01:00
|
|
|
};
|
|
|
|
|
2013-12-16 16:36:32 +01:00
|
|
|
var _process = function (keyword, text, i) {
|
2014-06-19 14:16:30 +02:00
|
|
|
var lines = [];
|
|
|
|
var parts = text.split(/\n/);
|
|
|
|
var index = typeof i !== 'undefined' ? '[' + i + ']' : '';
|
2013-12-16 16:36:32 +01:00
|
|
|
if (parts.length > 1) {
|
2013-12-18 16:46:09 +01:00
|
|
|
lines.push(keyword + index + ' ""');
|
2013-12-16 16:36:32 +01:00
|
|
|
parts.forEach(function (part) {
|
2014-03-05 10:13:30 +01:00
|
|
|
lines.push('"' + _escape(part) + '"');
|
2013-12-16 16:36:32 +01:00
|
|
|
});
|
2014-06-19 14:16:30 +02:00
|
|
|
} else {
|
2014-03-05 10:13:30 +01:00
|
|
|
lines.push(keyword + index + ' "' + _escape(text) + '"');
|
2013-12-16 16:36:32 +01:00
|
|
|
}
|
|
|
|
return lines;
|
|
|
|
};
|
2012-01-01 17:02:19 -08:00
|
|
|
|
2014-03-02 13:11:37 -05:00
|
|
|
// https://www.gnu.org/software/gettext/manual/html_node/PO-Files.html
|
|
|
|
// says order is translator-comments, extracted-comments, references, flags
|
|
|
|
|
2014-03-07 10:17:52 +01:00
|
|
|
this.comments.forEach(function (c) {
|
|
|
|
lines.push('# ' + c);
|
|
|
|
});
|
2014-03-01 17:21:18 -05:00
|
|
|
|
2014-03-07 10:17:52 +01:00
|
|
|
this.extractedComments.forEach(function (c) {
|
|
|
|
lines.push('#. ' + c);
|
|
|
|
});
|
2014-03-02 13:11:37 -05:00
|
|
|
|
2014-03-07 10:17:52 +01:00
|
|
|
this.references.forEach(function (ref) {
|
|
|
|
lines.push('#: ' + ref);
|
|
|
|
});
|
2012-01-02 16:06:32 -08:00
|
|
|
|
2013-12-17 16:35:40 +01:00
|
|
|
var flags = Object.keys(this.flags);
|
|
|
|
if (flags.length > 0) {
|
2014-06-19 14:16:30 +02:00
|
|
|
lines.push('#, ' + flags.join(','));
|
2013-12-17 16:35:40 +01:00
|
|
|
}
|
2014-06-19 13:38:07 +02:00
|
|
|
var mkObsolete = this.obsolete ? '#~ ' : '';
|
2013-12-17 16:35:40 +01:00
|
|
|
|
2014-01-21 15:06:11 +01:00
|
|
|
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
|
2014-06-19 14:16:30 +02:00
|
|
|
var text = self[keyword];
|
2013-12-16 16:36:32 +01:00
|
|
|
if (text != null) {
|
2013-12-18 16:46:09 +01:00
|
|
|
if (isArray(text) && text.length > 1) {
|
2013-12-16 16:36:32 +01:00
|
|
|
text.forEach(function (t, i) {
|
2014-06-19 13:38:07 +02:00
|
|
|
lines = lines.concat(mkObsolete + _process(keyword, t, i));
|
2013-12-16 16:36:32 +01:00
|
|
|
});
|
2014-06-19 14:16:30 +02:00
|
|
|
} else {
|
2013-12-18 16:46:09 +01:00
|
|
|
text = isArray(text) ? text.join() : text;
|
2014-06-23 12:17:40 +02:00
|
|
|
var processed = _process(keyword, text);
|
2014-06-23 18:24:02 +02:00
|
|
|
//handle \n in single-line texts (can not be handled in _escape)
|
2014-06-23 12:52:10 +02:00
|
|
|
for (var i = 1; i < processed.length - 1; i++) {
|
|
|
|
processed[i] = processed[i].slice(0, -1) + '\\n"';
|
|
|
|
}
|
2014-06-23 12:17:40 +02:00
|
|
|
lines = lines.concat(mkObsolete + processed.join('\n' + mkObsolete));
|
2013-12-16 16:36:32 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-06-19 14:16:30 +02:00
|
|
|
return lines.join('\n');
|
2011-12-22 16:17:04 -08:00
|
|
|
};
|
|
|
|
|
2013-10-31 14:13:25 +01:00
|
|
|
module.exports = PO;
|