2017-05-24 15:10:37 +02:00
/ *
This file is part of TALER
( C ) 2016 GNUnet e . V .
TALER is free software ; you can redistribute it and / or modify it under the
terms of the GNU General Public License as published by the Free Software
Foundation ; either version 3 , or ( at your option ) any later version .
TALER is distributed in the hope that it will be useful , but WITHOUT ANY
WARRANTY ; without even the implied warranty of MERCHANTABILITY or FITNESS FOR
A PARTICULAR PURPOSE . See the GNU General Public License for more details .
You should have received a copy of the GNU General Public License along with
TALER ; see the file COPYING . If not , see < http : //www.gnu.org/licenses/>
* /
/ * *
* Generate . po file from list of source files .
*
* Note that duplicate message IDs are NOT merged , to get the same output as
* you would from xgettext , just run msguniq .
*
* @ author Florian Dold
* /
"use strict" ;
2017-05-27 15:50:21 +02:00
Object . defineProperty ( exports , "__esModule" , { value : true } ) ;
2017-05-24 15:10:37 +02:00
var fs _1 = require ( "fs" ) ;
var ts = require ( "typescript" ) ;
function wordwrap ( str , width ) {
if ( width === void 0 ) { width = 80 ; }
var regex = '.{1,' + width + '}(\\s|$)|\\S+(\\s|$)' ;
return str . match ( RegExp ( regex , 'g' ) ) ;
}
function processFile ( sourceFile ) {
processNode ( sourceFile ) ;
var lastTokLine = 0 ;
var preLastTokLine = 0 ;
function getTemplate ( node ) {
switch ( node . kind ) {
case ts . SyntaxKind . FirstTemplateToken :
return node . text ;
case ts . SyntaxKind . TemplateExpression :
var te = node ;
var textFragments = [ te . head . text ] ;
for ( var _i = 0 , _a = te . templateSpans ; _i < _a . length ; _i ++ ) {
var tsp = _a [ _i ] ;
textFragments . push ( "%" + ( ( textFragments . length - 1 ) / 2 + 1 ) + "$s" ) ;
textFragments . push ( tsp . literal . text . replace ( /%/g , "%%" ) ) ;
}
return textFragments . join ( '' ) ;
default :
return "(pogen.ts: unable to parse)" ;
}
}
function getComment ( node ) {
var lc = ts . getLineAndCharacterOfPosition ( sourceFile , node . pos ) ;
var lastComments ;
for ( var l = preLastTokLine ; l < lastTokLine ; l ++ ) {
var pos = ts . getPositionOfLineAndCharacter ( sourceFile , l , 0 ) ;
var comments = ts . getTrailingCommentRanges ( sourceFile . text , pos ) ;
if ( comments ) {
lastComments = comments ;
}
}
if ( ! lastComments ) {
return ;
}
var candidate = lastComments [ lastComments . length - 1 ] ;
var candidateEndLine = ts . getLineAndCharacterOfPosition ( sourceFile , candidate . end ) . line ;
if ( candidateEndLine != lc . line - 1 ) {
return ;
}
var text = sourceFile . text . slice ( candidate . pos , candidate . end ) ;
switch ( candidate . kind ) {
case ts . SyntaxKind . SingleLineCommentTrivia :
// Remove comment leader
text = text . replace ( /^[/][/]\s*/ , "" ) ;
break ;
case ts . SyntaxKind . MultiLineCommentTrivia :
// Remove comment leader and trailer,
// handling white space just like xgettext.
text = text
. replace ( /^[/][*](\s*?\n|\s*)?/ , "" )
. replace ( /(\n[ \t]*?)?[*][/]$/ , "" ) ;
break ;
}
return text ;
}
function getPath ( node ) {
switch ( node . kind ) {
case ts . SyntaxKind . PropertyAccessExpression :
var pae = node ;
return Array . prototype . concat ( getPath ( pae . expression ) , [ pae . name . text ] ) ;
case ts . SyntaxKind . Identifier :
var id = node ;
return [ id . text ] ;
}
return [ "(other)" ] ;
}
function arrayEq ( a1 , a2 ) {
if ( a1 . length != a2 . length ) {
return false ;
}
for ( var i = 0 ; i < a1 . length ; i ++ ) {
if ( a1 [ i ] != a2 [ i ] ) {
return false ;
}
}
return true ;
}
function processTaggedTemplateExpression ( tte ) {
var lc = ts . getLineAndCharacterOfPosition ( sourceFile , tte . pos ) ;
if ( lc . line != lastTokLine ) {
preLastTokLine = lastTokLine ;
lastTokLine = lc . line ;
}
var path = getPath ( tte . tag ) ;
var res = {
path : path ,
line : lc . line ,
comment : getComment ( tte ) ,
template : getTemplate ( tte . template ) . replace ( /"/g , '\\"' ) ,
} ;
return res ;
}
function formatMsgComment ( line , comment ) {
if ( comment ) {
for ( var _i = 0 , _a = comment . split ( '\n' ) ; _i < _a . length ; _i ++ ) {
var cl = _a [ _i ] ;
console . log ( "#. " + cl ) ;
}
}
console . log ( "#: " + sourceFile . fileName + ":" + ( line + 1 ) ) ;
console . log ( "#, c-format" ) ;
}
function formatMsgLine ( head , msg ) {
// Do escaping, wrap break at newlines
var parts = msg
. match ( /(.*\n|.+$)/g )
. map ( function ( x ) { return x . replace ( /\n/g , '\\n' ) ; } )
. map ( function ( p ) { return wordwrap ( p ) ; } )
. reduce ( function ( a , b ) { return a . concat ( b ) ; } ) ;
if ( parts . length == 1 ) {
console . log ( head + " \"" + parts [ 0 ] + "\"" ) ;
}
else {
console . log ( head + " \"\"" ) ;
for ( var _i = 0 , parts _1 = parts ; _i < parts _1 . length ; _i ++ ) {
var p = parts _1 [ _i ] ;
console . log ( "\"" + p + "\"" ) ;
}
}
}
function getJsxElementPath ( node ) {
var path ;
var process = function ( childNode ) {
switch ( childNode . kind ) {
case ts . SyntaxKind . JsxOpeningElement :
{
var e = childNode ;
return path = getPath ( e . tagName ) ;
}
default :
break ;
}
} ;
ts . forEachChild ( node , process ) ;
return path ;
}
function translateJsxExpression ( node , h ) {
switch ( node . kind ) {
case ts . SyntaxKind . StringLiteral :
{
var e = node ;
return e . text ;
}
default :
return "%" + h [ 0 ] ++ + "$s" ;
}
}
function trim ( s ) {
return s . replace ( /^[ \n\t]*/ , "" ) . replace ( /[ \n\t]*$/ , "" ) ;
}
function getJsxContent ( node ) {
var fragments = [ ] ;
var holeNum = [ 1 ] ;
var process = function ( childNode ) {
switch ( childNode . kind ) {
case ts . SyntaxKind . JsxText :
{
var e = childNode ;
2018-02-22 08:58:48 +01:00
var s = e . getText ( ) ;
var t = s . split ( "\n" ) . map ( trim ) . join ( "\n" ) ;
if ( s . length >= 1 && ( s [ 0 ] === "\n" || s [ 0 ] === " " ) ) {
t = " " + t ;
}
if ( s . length >= 1 && ( s [ s . length - 1 ] === "\n" || s [ s . length - 1 ] === " " ) ) {
t = t + " " ;
}
console . log ( "got" , JSON . stringify ( s ) ) ;
2017-05-24 15:10:37 +02:00
fragments . push ( t ) ;
}
case ts . SyntaxKind . JsxOpeningElement :
break ;
case ts . SyntaxKind . JsxElement :
fragments . push ( "%" + holeNum [ 0 ] ++ + "$s" ) ;
break ;
case ts . SyntaxKind . JsxExpression :
{
var e = childNode ;
fragments . push ( translateJsxExpression ( e . expression , holeNum ) ) ;
break ;
}
case ts . SyntaxKind . JsxClosingElement :
break ;
default :
var lc = ts . getLineAndCharacterOfPosition ( childNode . getSourceFile ( ) , childNode . getStart ( ) ) ;
console . error ( "unrecognized syntax in JSX Element " + ts . SyntaxKind [ childNode . kind ] + " (" + childNode . getSourceFile ( ) . fileName + ":" + ( lc . line + 1 ) + ":" + ( lc . character + 1 ) ) ;
break ;
}
} ;
ts . forEachChild ( node , process ) ;
return fragments . join ( "" ) ;
}
function getJsxSingular ( node ) {
var res ;
var process = function ( childNode ) {
switch ( childNode . kind ) {
case ts . SyntaxKind . JsxElement :
{
var path = getJsxElementPath ( childNode ) ;
if ( arrayEq ( path , [ "i18n" , "TranslateSingular" ] ) ) {
res = getJsxContent ( childNode ) ;
}
}
default :
break ;
}
} ;
ts . forEachChild ( node , process ) ;
return res ;
}
function getJsxPlural ( node ) {
var res ;
var process = function ( childNode ) {
switch ( childNode . kind ) {
case ts . SyntaxKind . JsxElement :
{
var path = getJsxElementPath ( childNode ) ;
if ( arrayEq ( path , [ "i18n" , "TranslatePlural" ] ) ) {
res = getJsxContent ( childNode ) ;
}
}
default :
break ;
}
} ;
ts . forEachChild ( node , process ) ;
return res ;
}
function processNode ( node ) {
switch ( node . kind ) {
case ts . SyntaxKind . JsxElement :
var path = getJsxElementPath ( node ) ;
if ( arrayEq ( path , [ "i18n" , "Translate" ] ) ) {
var content = getJsxContent ( node ) ;
var line = ts . getLineAndCharacterOfPosition ( sourceFile , node . pos ) . line ;
var comment = getComment ( node ) ;
formatMsgComment ( line , comment ) ;
formatMsgLine ( "msgid" , content ) ;
console . log ( "msgstr \"\"" ) ;
console . log ( ) ;
return ;
}
if ( arrayEq ( path , [ "i18n" , "TranslateSwitch" ] ) ) {
var line = ts . getLineAndCharacterOfPosition ( sourceFile , node . pos ) . line ;
var comment = getComment ( node ) ;
formatMsgComment ( line , comment ) ;
var singularForm = getJsxSingular ( node ) ;
if ( ! singularForm ) {
console . error ( "singular form missing" ) ;
process . exit ( 1 ) ;
}
var pluralForm = getJsxPlural ( node ) ;
if ( ! pluralForm ) {
console . error ( "plural form missing" ) ;
process . exit ( 1 ) ;
}
formatMsgLine ( "msgid" , singularForm ) ;
formatMsgLine ( "msgid_plural" , pluralForm ) ;
console . log ( "msgstr[0] \"\"" ) ;
console . log ( "msgstr[1] \"\"" ) ;
console . log ( ) ;
return ;
}
break ;
case ts . SyntaxKind . CallExpression :
{
// might be i18n.plural(i18n[.X]`...`, i18n[.X]`...`)
var ce = node ;
var path _1 = getPath ( ce . expression ) ;
if ( ! arrayEq ( path _1 , [ "i18n" , "plural" ] ) ) {
break ;
}
if ( ce . arguments [ 0 ] . kind != ts . SyntaxKind . TaggedTemplateExpression ) {
break ;
}
if ( ce . arguments [ 1 ] . kind != ts . SyntaxKind . TaggedTemplateExpression ) {
break ;
}
var line = ts . getLineAndCharacterOfPosition ( sourceFile , ce . pos ) . line ;
var t1 = processTaggedTemplateExpression ( ce . arguments [ 0 ] ) ;
var t2 = processTaggedTemplateExpression ( ce . arguments [ 1 ] ) ;
var comment = getComment ( ce ) ;
formatMsgComment ( line , comment ) ;
formatMsgLine ( "msgid" , t1 . template ) ;
formatMsgLine ( "msgid_plural" , t2 . template ) ;
console . log ( "msgstr[0] \"\"" ) ;
console . log ( "msgstr[1] \"\"" ) ;
console . log ( ) ;
// Important: no processing for child i18n expressions here
return ;
}
case ts . SyntaxKind . TaggedTemplateExpression :
{
var tte = node ;
var _a = processTaggedTemplateExpression ( tte ) , comment = _a . comment , template = _a . template , line = _a . line , path _2 = _a . path ;
if ( path _2 [ 0 ] != "i18n" ) {
break ;
}
formatMsgComment ( line , comment ) ;
formatMsgLine ( "msgid" , template ) ;
console . log ( "msgstr \"\"" ) ;
console . log ( ) ;
break ;
}
}
ts . forEachChild ( node , processNode ) ;
}
}
exports . processFile = processFile ;
var fileNames = process . argv . slice ( 2 ) ;
console . log ( "# SOME DESCRIPTIVE TITLE.\n# Copyright (C) YEAR THE PACKAGE'S COPYRIGHT HOLDER\n# This file is distributed under the same license as the PACKAGE package.\n# FIRST AUTHOR <EMAIL@ADDRESS>, YEAR.\n#\n#, fuzzy\nmsgid \"\"\nmsgstr \"\"\n\"Project-Id-Version: PACKAGE VERSION\\n\"\n\"Report-Msgid-Bugs-To: \\n\"\n\"POT-Creation-Date: 2016-11-23 00:00+0100\\n\"\n\"PO-Revision-Date: YEAR-MO-DA HO:MI+ZONE\\n\"\n\"Last-Translator: FULL NAME <EMAIL@ADDRESS>\\n\"\n\"Language-Team: LANGUAGE <LL@li.org>\\n\"\n\"Language: \\n\"\n\"MIME-Version: 1.0\\n\"\n\"Content-Type: text/plain; charset=UTF-8\\n\"\n\"Content-Transfer-Encoding: 8bit\\n\"" ) ;
console . log ( ) ;
fileNames . sort ( ) ;
fileNames . forEach ( function ( fileName ) {
var sourceFile = ts . createSourceFile ( fileName , fs _1 . readFileSync ( fileName ) . toString ( ) , ts . ScriptTarget . ES2016 , /*setParentNodes */ true ) ;
processFile ( sourceFile ) ;
} ) ;