pofile/lib/po.js

272 lines
7.5 KiB
JavaScript
Raw Normal View History

var fs = require('fs'),
isArray = require('lodash.isarray');
2011-12-21 22:50:01 -08:00
function trim(string) {
return string.replace(/^\s+|\s+$/g, '');
}
2011-12-21 22:50:01 -08:00
var PO = function () {
this.comments = [];
this.headers = {};
this.items = [];
2012-01-01 17:02:19 -08:00
};
PO.prototype.save = function (filename, callback) {
fs.writeFile(filename, this.toString(), callback);
};
PO.prototype.toString = function () {
var lines = [],
that = this;
if (this.comments) {
this.comments.forEach(function (comment) {
lines.push('# ' + comment);
});
}
2012-01-01 17:02:19 -08:00
lines.push('msgid ""');
lines.push('msgstr ""');
var keys = Object.keys(this.headers);
keys.forEach(function (key) {
lines.push('"' + key + ': ' + that.headers[key] + '\\n"');
});
2012-01-01 17:02:19 -08:00
lines.push('');
2012-01-01 17:02:19 -08:00
this.items.forEach(function (item) {
lines.push(item.toString());
lines.push('');
});
return lines.join("\n");
2011-12-21 22:50:01 -08:00
};
PO.load = function (filename, callback) {
fs.readFile(filename, 'utf-8', function (err, data) {
if (err) {
return callback(err);
}
var po = PO.parse(data);
callback(null, po);
});
};
PO.parse = function (data) {
//support both unix and windows newline formats.
data = data.replace(/\r\n/g, '\n');
var po = new PO(),
sections = data.split(/\n\n/),
headers = sections.shift(),
lines = sections.join("\n").split(/\n/);
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': '',
};
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) {
if (header.match(/^#/)) {
po.comments.push(header.replace(/^#\s*/, ''));
}
if (header.match(/^"/)) {
header = header.trim().replace(/^"/, '').replace(/\\n"$/, '');
var p = header.split(/:/),
name = p.shift().trim(),
value = p.join(':').trim();
po.headers[name] = value;
}
});
var item = new PO.Item(),
context = null,
plural = 0,
obsolete = false;
2012-01-01 17:02:19 -08:00
function finish() {
if (item.msgid.length > 0) {
po.items.push(item);
item = new PO.Item();
item.obsolete = obsolete;
obsolete = false;
}
}
function extract(string) {
string = trim(string);
string = string.replace(/^[^"]*"|"$/g, '');
string = string.replace(/\\"/g, '"');
string = string.replace(/\\\\/g, '\\');
return string;
}
while (lines.length > 0) {
var line = trim(lines.shift()),
add = false;
if (line.match(/^#\~/)) { // Obsolete item
obsolete = true;
line = trim(line.substring(2));
} else {
obsolete = false;
}
if (line.match(/^#:/)) { // Reference
finish();
item.references.push(trim(line.replace(/^#:/, '')));
}
2013-12-17 15:04:44 +01:00
else if (line.match(/^#,/)) { // Flags
finish();
var flags = trim(line.replace(/^#,/, '')).split(",");
for (var i = 0; i < flags.length; i++) {
item.flags[flags[i]] = true;
}
}
2014-03-02 13:11:37 -05:00
else if (line.match(/^#\s+/)) { // Translator comment
finish();
2014-03-02 13:11:37 -05:00
item.comments.push(trim(line.replace(/^#\s+/, '')));
}
else if (line.match(/^#\./)) { // Extracted comment
finish();
item.extractedComments.push(trim(line.replace(/^#\./, '')));
}
else if (line.match(/^msgid_plural/)) { // Plural form
item.msgid_plural = extract(line);
context = 'msgid_plural';
}
else if (line.match(/^msgid/)) { // Original
finish();
item.msgid = extract(line);
context = 'msgid';
}
else if (line.match(/^msgstr/)) { // Translation
var m = line.match(/^msgstr\[(\d+)\]/);
plural = m && m[1] ? parseInt(m[1]) : 0;
item.msgstr[plural] = extract(line);
context = 'msgstr';
}
else if (line.match(/^msgctxt/)) { // Context
finish();
item.msgctxt = extract(line);
}
else { // Probably multiline string or blank
if (line.length > 0) {
if (context === 'msgstr') {
item.msgstr[plural] += extract(line);
}
else if (context === 'msgid') {
item.msgid += extract(line);
}
else if (context === 'msgid_plural') {
item.msgid_plural += extract(line);
}
}
}
}
finish();
return po;
2011-12-21 22:50:01 -08:00
};
PO.Item = function () {
this.msgid = '';
this.msgctxt = null;
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 = {};
this.obsolete = false;
2011-12-21 22:50:01 -08:00
};
PO.Item.prototype.toString = function () {
var lines = [],
that = this;
// reverse what extract(string) method during PO.parse does
var _escape = function (string) {
string = string.replace(/\\/g, '\\\\');
return string.replace(/"/g, '\\"');
};
var _process = function (keyword, text, i) {
var lines = [],
parts = text.split(/\n/),
index = typeof i !== 'undefined' ? '[' + i + ']' : '';
if (parts.length > 1) {
lines.push(keyword + index + ' ""');
parts.forEach(function (part) {
lines.push('"' + _escape(part) + '"');
});
}
else {
lines.push(keyword + index + ' "' + _escape(text) + '"');
}
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-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) {
lines.push('#, ' + flags.join(","));
2013-12-17 16:35:40 +01:00
}
['msgctxt', 'msgid', 'msgid_plural', 'msgstr'].forEach(function (keyword) {
var text = that[keyword];
if (text != null) {
if (isArray(text) && text.length > 1) {
text.forEach(function (t, i) {
lines = lines.concat(_process(keyword, t, i));
});
}
else {
text = isArray(text) ? text.join() : text;
lines = lines.concat(_process(keyword, text));
}
}
});
if (this.obsolete) {
return "#~ " + lines.join("\n#~ ");
} else {
return lines.join("\n");
}
};
module.exports = PO;