So, I implemented all permuations of string '{Пожалуйста|Просто} сделайте так, чтобы это {удивительное|крутое|простое} тестовое предложение {изменялось {быстро|мгновенно} случайным образом|менялось каждый раз}.' with code above. I used parser from geektimes.ru/post/87754/#comment_2638276 comment, where author returns only first random variation of string, and modified it:
function printTree(t, index, r, delim) { var out = [], prefix, res; r = r || ''; delim = delim || ' '; index = index || 0; if (index < t.length) { t = t.map(function(el) { return el instanceof Array ? el : [el]; }); prefix = r ? r + delim : ''; for (var i = 0; i < t[index].length; i++) { if (t[index][i] instanceof Array) { res = printTree(t[index][i], 0, r, delim); } else { res = [prefix + t[index][i]]; } for (var j = 0; j < res.length; j++) { out = out.concat(printTree(t, index + 1, res[j], delim)); } } } else { return [r]; } return out; } function Token(type, pos, value) { this.type = type; this.pos = pos; this.value = value; } Token.END_OF_LINE = 0; Token.OPERATOR = 1; Token.TEXT = 2; function Parser(text) { this._text = text; this._pos = 0; this._len = text.length; } Parser.prototype = { operators: { '{': true, '}': true, '|': true }, nextToken: function() { if (this._pos >= this._len) return new Token(Token.END_OF_LINE); if (this._text[this._pos] in this.operators) { return new Token(Token.OPERATOR, this._pos, this._text[this._pos++]); } var text = '', start = this._pos; while ((this._pos < this._len) && !(this._text[this._pos] in this.operators)) { text += this._text[this._pos]; this._pos++; } return new Token(Token.TEXT, start, text); }, getNextToken: function() { var pos = this._pos, result = this.nextToken(); this._pos = pos; return result; } }; function Interpretter(text) { this._parser = new Parser(text); } Interpretter.prototype = { value: function() { var result = [], token = this._parser.getNextToken(); while (token.type == Token.TEXT || (token.type == Token.OPERATOR && token.value == '{')) { token = this._parser.nextToken(); if (token.type == Token.OPERATOR) { if (token.value == '{') { result.push(this.exprеssion()); } else { throw 'Syntax error at pos ' + token.pos; } } else { result.push(token.value); } token = this._parser.getNextToken(); } return result; }, includeValue: function(variants, value) { var hasNested = false; for (var i = 0; i < value.length; i++) { if (value[i] instanceof Array) { hasNested = true; break; } } if (hasNested) { variants.push(value); } else { variants = variants.concat(value); } return variants; }, expression: function() { var variants = [], value = this.value(), token = this._parser.nextToken(); variants = this.includeValue(variants, value); while (token.value == '|') { value = this.value(); variants = this.includeValue(variants, value); token = this._parser.nextToken(); } if (!token.type == '}') throw 'Syntax error at pos ' + token.pos; return variants; } }; var text = '{Пожалуйста|Просто} сделайте так, чтобы это ' + '{удивительное|крутое|простое} тестовое предложение {изменялось {быстро|мгновенно}' + ' случайным образом|менялось каждый раз}.'; var interpretter = new Interpretter(text); var v = interpretter.value(); var r = printTree(v); r.sort(); print(r.join('\n'));
You can to check it in action at: ideone.com/hxJP3F