No OneTemporary

File Metadata

Created
Thu, Mar 12, 8:45 PM
This file is larger than 256 KB, so syntax highlighting was skipped.
diff --git a/.gitignore b/.gitignore
index 04f629e..b3fe7b7 100644
--- a/.gitignore
+++ b/.gitignore
@@ -1,67 +1,68 @@
# Compiled Object files
*.slo
*.lo
*.o
*.obj
# Compiled Dynamic libraries
*.so
*.so.*
*.dylib
*.dll
# Compiled Static libraries
*.lai
*.la
*.a
*.lib
# Executables
*.exe
*.out
*.app
*.class
# Mobile Tools for Java (J2ME)
.mtj.tmp/
# Package Files #
*.jar
*.war
*.ear
# virtual machine crash logs, see http://www.java.com/en/download/help/error_hotspot.xml
hs_err_pid*
# Qt-es
/.qmake.cache
/.qmake.stash
*.pro.user
*.pro.user.*
*.moc
moc_*.cpp
qrc_*.cpp
ui_*.h
Makefile*
*-build-*
# QtCreator
*.autosave
coco/*.old
coco/*~
*~
cpp/build-*/*
cpp/xreate-debug/*
cpp/xreate-release/*
cpp/.idea
cpp/CMakeLists.txt.user*
hs/*
project/*
nb*.xml
.*
target/*
+/tools/phabricator/xreate-frontend/nbproject/private/
\ No newline at end of file
diff --git a/coco/gen-grammar b/coco/gen-grammar
index 132dff9..814c770 100755
--- a/coco/gen-grammar
+++ b/coco/gen-grammar
@@ -1 +1 @@
-cococpp -frames /usr/share/coco-cpp/ ./xreate.ATG
+cococpp -frames ./generator-frames ./xreate.ATG
diff --git a/coco/xreate.ATG b/coco/xreate.ATG
index 3bfb0e9..242d665 100644
--- a/coco/xreate.ATG
+++ b/coco/xreate.ATG
@@ -1,518 +1,518 @@
//TODO add ListLiteral
//TODO ExprTyped: assign default(none) type
#include "ast.h"
#include "ExternLayer.h"
#include <string>
#include <stack>
#define wprintf(format, ...) \
char __buffer[100]; \
wcstombs(__buffer, format, 100); \
fprintf(stderr, __buffer, __VA_ARGS__)
using namespace xreate;
using namespace std;
COMPILER Xreate
xreate::AST root; // current program unit
struct {
std::stack<xreate::CodeScope*> scopesOld;
xreate::CodeScope* scope = nullptr;
} context;
void pushContextScope(CodeScope* scope){
context.scopesOld.push(context.scope);
context.scope = scope;
}
void popContextScope(){
context.scope = context.scopesOld.top();
context.scopesOld.pop();
}
int skipIdent()
{
int kind = 0;
scanner->ResetPeek();
while ((kind = scanner->Peek()->kind) == _colon)
if (scanner->Peek()->kind != _ident)
return 0;
return kind;
}
bool checkParametersList()
{
return la->kind == _ident && skipIdent() == _lparen;
}
bool checkInfix()
{
return la->kind == _ident && skipIdent() == _ident;
}
bool checkIndex()
{
return la->kind == _ident && skipIdent() == _lbrack;
}
bool checkFuncDecl()
{
if (la->kind != _ident) return false;
int xkind = skipIdent();
Token* y = scanner->Peek();
return xkind == _assign && (y->kind == _function || y->kind == _pre);
}
bool checkAssignment()
{
scanner->ResetPeek();
Token* x = scanner->Peek();
return la->kind == _ident && x->kind == _assign;
}
CHARACTERS
letter = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz".
any = ANY - '"'.
digit = "0123456789".
cr = '\r'.
lf = '\n'.
tab = '\t'.
TOKENS
ident = (letter | '_') {letter | digit | '_'}.
number = digit {digit}.
string = '"' { any } '"'.
function = "function".
pre = "pre".
lparen = '('.
rparen = ')'.
lbrack = '['.
rbrack = ']'.
lcurbrack = '{'.
rcurbrack = '}'.
equal = "==".
assign = '='.
implic = '-' '>'.
colon = ':'.
context = "context".
tagcolon = "::".
COMMENTS FROM "/*" TO "*/" NESTED
COMMENTS FROM "//" TO lf
IGNORE cr + lf + tab
PRODUCTIONS
Xreate = (. Function* function; .)
{ ( RuleDecl | InterfaceData | Imprt | ContextSection
| IF(checkFuncDecl()) FDecl<function> (. root.add(function); .)
| TDecl ) }.
Ident<std::wstring& name>
= ident {':' ident} (. name = t->val; .).
FDecl<Function* &f> = (. std::wstring fname; std::wstring varname; TypeAnnotation typIn; TypeAnnotation typOut; bool flagIsPrefunct = false; .)
Ident<fname> assign
[pre (. flagIsPrefunct = true; .)]
function (. f = new Function(fname); f->isPrefunction = flagIsPrefunct; CodeScope* entry = f->getEntryScope(); .)
['(' Ident<varname> tagcolon Type<typIn> (. f->addArg(std::move(varname), move(typIn)); .)
{',' Ident<varname> tagcolon Type<typIn> (. f->addArg(std::move(varname), move(typIn));.)
} ')']
[ tagcolon
( IF(flagIsPrefunct) FnTag<f>
| Type<typOut> (. f->setReturnType(typOut); .)
)
{';' FnTag<f> }]
BDecl<entry> (. entry->__body.bindType(move(typOut));.)
.
ContextSection<>= (. Expression context; Function* f; .)
"case" "context" tagcolon MetaSimpExpr<context>
lcurbrack { FDecl<f> (. f->guardContext = context; root.add(f); .)
} rcurbrack.
/**
* TYPES
*
*/
TypeTerm<TypeAtom& typ> = (. std::wstring tid; .)
("string" | "int" | "num" | "float" | "bool") (. typ = Atom<Type_t>(t->val); .)
.
Type<TypeAnnotation& typ> = (. TypeAnnotation typ2; TypeAtom typ3; std::wstring tid, field; .)
(
TList<typ>
| TStruct<typ>
| TypeTerm<typ3> (. typ = TypeAnnotation(typ3); .)
|IF (checkIndex()) Ident<tid> lbrack
Ident<field> (. typ = TypeAnnotation(TypeOperator::ACCESS, {}); typ.__valueCustom = Atom<Identifier_t>(tid).get(); typ.fields.push_back(Atom<Identifier_t>(field).get()); .)
{',' Ident<field> (. typ.fields.push_back(Atom<Identifier_t>(field).get()); .)
} rbrack
| Ident<tid> (. typ = TypeAnnotation(TypeOperator::CUSTOM, {}); typ.__valueCustom = Atom<Identifier_t>(tid).get(); .)
['(' Type<typ2> (. typ.__operator = TypeOperator::CALL; typ.__operands.push_back(typ2); .)
{',' Type<typ2> (. typ.__operands.push_back(typ2); .)
} ')']
) .
TList<TypeAnnotation& typ> = (. TypeAnnotation ty; .)
'[' Type<ty> (. typ = TypeAnnotation(TypeOperator::ARRAY, {ty}); .)
{',' Type<ty> (. typ.__operator = TypeOperator::TUPLE; typ.__operands.push_back(ty); .)
}']'
.
TStruct<TypeAnnotation& typ> = (. TypeAnnotation t; std::wstring field; .)
'{'
Ident<field> tagcolon Type<t> (. typ = TypeAnnotation(TypeOperator::STRUCT, {t}); typ.fields.push_back(Atom<Identifier_t>(field).get()); .)
{',' Ident<field> tagcolon Type<t>} (. typ.__operands.push_back(t); typ.fields.push_back(Atom<Identifier_t>(field).get()); .)
'}'.
TDecl = (. std::wstring ttag; TypeAnnotation t, t1; std::wstring tname, arg; std::vector<Atom<Identifier_t>> args; .)
Ident<tname> assign "type"
(
"alias" Type<t> (. root.add(move(t), Atom<Identifier_t>(tname)); .)
| "variant" lparen Ident<arg> (. t = TypeAnnotation(TypeOperator::VARIANT, {}); args.push_back(Atom<Identifier_t>(arg)); .)
{',' Ident<arg> (. args.push_back(Atom<Identifier_t>(arg)); .)
} rparen (. t.addFields(move(args)); root.add(move(t), Atom<Identifier_t>(tname)); .)
| Ident<ttag>
['(' Ident<arg> (. args.push_back(Atom<Identifier_t>(arg)); .)
{',' Ident<arg> (. args.push_back(Atom<Identifier_t>(arg)); .)
} ')']
Type<t> (. t.addBindings(move(args)); root.add(move(t), Atom<Identifier_t>(tname)); .)
) '.'
.
VDecl<CodeScope* f> = (. std::wstring vname; Expression e; TypeAnnotation typ;.)
Ident<vname> assign ExprTyped<e, typ>
(. f->addDeclaration(move(vname), move(typ), move(e)); .)
.
ContextDecl<CodeScope * scope> = (. Expression tag; .)
context tagcolon
MetaSimpExpr<tag> (. scope->tags.push_back(move(tag)); .)
{';' MetaSimpExpr<tag> (. scope->tags.push_back(move(tag)); .)
}.
//TODO forbid multiple body declaration (ExprTyped)
BDecl<CodeScope* scope> = (. Expression body; TypeAnnotation typ; pushContextScope(scope); .)
'{' { (RuleContextDecl<scope> | ContextDecl<scope> '.'
| IF(checkAssignment()) VDecl<scope> '.'
| ExprTyped<body, typ> (. scope->setBody(body); popContextScope();.)
)} '}'.
IfDecl<Expression& e, TypeAnnotation typ> = (. Expression cond; ManagedScpPtr blockTrue = root.add(new xreate::CodeScope(context.scope)); ManagedScpPtr blockFalse = root.add(new xreate::CodeScope(context.scope)); TypeAnnotation typIf; .)
"if" '(' Expr<cond> ')' tagcolon Type<typIf> BDecl<&*blockTrue> "else" BDecl<&*blockFalse>
(. e = Expression(Operator::IF, {cond}); e.addBlock(blockTrue); e.addBlock(blockFalse); e.bindType(move(typIf)); .)
.
LoopDecl<Expression& e, TypeAnnotation& typOut> =
(. Expression eIn, eAcc, eFilters; std::wstring varEl, varAcc, contextClass; TypeAnnotation typEl, typAcc;
ManagedScpPtr block = root.add(new xreate::CodeScope(context.scope)); .)
"loop"
("map" '(' Expr<eIn> implic Ident<varEl> tagcolon Type<typEl> ')' tagcolon Type<typOut> BDecl<&*block>
(. e = Expression(Operator::MAP, {eIn});
e.addBindings({Atom<Identifier_t>(varEl)});
block->addArg(Atom<Identifier_t>(varEl), move(typEl));
e.addBlock(block); .)
- |"fold" '(' Expr<eIn> implic Ident<varEl> tagcolon Type<typEl> ['|' Expr<eFilters> ] ',' Expr<eAcc> implic Ident<varAcc> tagcolon Type<typAcc> ')' tagcolon Type<typOut> BDecl<&*block>
+ |"fold" '(' Expr<eIn> implic Ident<varEl> tagcolon Type<typEl> ['|' Expr<eFilters> ] ',' Expr<eAcc> implic Ident<varAcc>')' tagcolon Type<typOut> BDecl<&*block>
(. e = Expression(Operator::FOLD, {eIn, eAcc});
e.addBindings({Atom<Identifier_t>(varEl), Atom<Identifier_t>(varAcc)});
block->addArg(Atom<Identifier_t>(varEl), move(typEl));
- block->addArg(Atom<Identifier_t>(varAcc), move(typAcc));
+ block->addArg(Atom<Identifier_t>(varAcc), move(typOut));
e.addBlock(block);
.)
| "context" '(' string (. contextClass = t->val; .)
')' BDecl<&*block>
(. e = Expression(Operator::LOOP_CONTEXT, {Expression(Atom<String_t>(std::move(contextClass)))});
e.addBlock(block);
.)
).
SwitchDecl<Expression& eSwitch> = (. TypeAnnotation typ; eSwitch = Expression(Operator::SWITCH, {}); Expression eCondition; Expression tag;.)
["switch"
("ad" "hoc" lparen Expr<eCondition> tagcolon MetaSimpExpr<tag> rparen (. eSwitch.op = Operator::SWITCH_ADHOC; eSwitch.operands.push_back(eCondition); eSwitch.tags.emplace(tag.getValueString(), move(tag)); .)
| lparen Expr<eCondition> rparen tagcolon Type<typ> (. eSwitch.operands.push_back(eCondition); eSwitch.type = typ; .)
)
]
CaseDecl<eSwitch> {CaseDecl<eSwitch>}
.
CaseDecl<Expression& outer> = (. ManagedScpPtr scope = root.add(new xreate::CodeScope(context.scope)); .)
"case"
( "default" BDecl<&*scope> (. Expression exprCase(Operator::CASE_DEFAULT, {});
exprCase.addBlock(scope);
outer.operands.insert(++outer.operands.begin(), exprCase); .)
| CaseParams<&*scope> (. ManagedScpPtr scopeBody = root.add(new xreate::CodeScope(&*scope)); .)
BDecl<&*scopeBody> (. Expression exprCase(Operator::CASE, {});
exprCase.addBlock(scope); exprCase.addBlock(scopeBody); outer.addArg(move(exprCase)); .)
).
CaseParams<CodeScope* scope> = (. Expression e; Expression guard(Operator::LOGIC_AND, {}); pushContextScope(scope); .)
CaseParam<scope, guard>
{',' CaseParam<scope, guard>
} (. scope->setBody(guard); popContextScope(); .).
CaseParam<CodeScope* scope, Expression& guard> = (. TypeAnnotation argtyp; Expression condition; .)
(
IF(checkAssignment()) VDecl<scope>
| ExprTyped<condition, argtyp> (. guard.addArg(move(condition)); .)
).
SequenceDecl<Expression& sequence> =
"sequence" ListLiteral<sequence> (. sequence.setOp(Operator::SEQUENCE); .).
/*============================ INTERFACES ===============================*/
Imprt<> =
"import" "raw" lparen string (. root.__rawImports.push_back(Atom<String_t>(t->val).get()); .)
rparen .
InterfaceData<> = "interface" '('
( "dfa" ')' InterfaceDFA
| "extern-c" ')' InterfaceExternC
| "cfa" ')' InterfaceCFA
| "adhoc" ')' InterfaceAdhoc
).
InterfaceAdhoc<> =
'{' [ PrefunctionSchemeDecl ] '}'.
PrefunctionSchemeDecl<> = (. TypeAnnotation typReturn; std::wstring prefName; Expression exprCases; .)
pre function Ident<prefName> tagcolon Type<typReturn>
'{' SwitchDecl<exprCases> '}'
(. Expression prefData(Operator::CALL, {Atom<Identifier_t>(prefName), exprCases});
prefData.type = typReturn;
root.addInterfaceData(Adhoc, move(prefData));
.).
InterfaceExternC<> = (.xreate::ExternData data; .)
'{' {IncludeExternDecl<data> | LibExternDecl<data> } '}'
(. root.addExternData(move(data)); .)
.
LibExternDecl<xreate::ExternData& data> = (. std::wstring pkgname, libname; .)
Ident<libname> assign "library" tagcolon "pkgconfig"
'(' string (. pkgname = t->val; .)
')' '.' (. data.addLibrary(Atom<Identifier_t>(libname), Atom<String_t>(pkgname)); .)
.
IncludeExternDecl<xreate::ExternData& data> = (. Expression inc; .)
"include" StructLiteral<inc> '.' (. data.addIncludeDecl(move(inc)); .)
.
InterfaceDFA<> = '{' { InstructDecl } '}' .
InstructDecl = (.Operator op; Expression tag;
Expression scheme;
std::vector<Expression>& tags = scheme.operands;
tags.push_back(Expression()); /* return value */ .)
"operator" InstructAlias<op> tagcolon '(' (.scheme.setOp(op); .)
[
MetaSimpExpr<tag> (. tags.push_back(tag); .)
{
',' MetaSimpExpr<tag> (. tags.push_back(tag); .)
}
] ')' [ implic MetaSimpExpr<tag> (. tags[0] = tag; .)
] (. root.addDFAData(move(scheme)); .)
'.'.
InstructAlias<Operator& op> =
(
"map" (. op = Operator::MAP; .)
| "list_range" (. op = Operator::LIST_RANGE; .)
| "list" (. op = Operator::LIST; .)
| "fold" (. op = Operator::FOLD; .)
| "index" (. op = Operator::INDEX; .)
).
InterfaceCFA<> = '{' { InstructCFADecl } '}' .
InstructCFADecl<> = (.Operator op; Expression tag;
Expression scheme;
std::vector<Expression>& tags = scheme.operands; .)
"operator" InstructAlias<op> tagcolon (. scheme.setOp(op); .)
[
MetaSimpExpr<tag> (. tags.push_back(tag); .)
{
',' MetaSimpExpr<tag> (. tags.push_back(tag); .)
}
] '.' (. root.addInterfaceData(CFA, move(scheme)); .).
/*============================ METAPROGRAMMING ===============================*/
// TagsDecl<CodeScope* f> = (. Expression tag; TagModifier mod = TagModifier::NONE; .)
// ':' { MetaSimpExpr<tag> (. /*f.addTag(std::move(tag), mod); */ .)
// }.
FnTag<Function* f> = (. Expression tag; TagModifier mod = TagModifier::NONE; .)
MetaSimpExpr<tag>
['-' TagMod<mod>] (. f->addTag(std::move(tag), mod); .).
TagMod<TagModifier& mod> =
( "assert" (. mod = TagModifier::ASSERT; .)
| "require" (. mod = TagModifier::REQUIRE; .)
).
RuleDecl<> =
"rule" tagcolon (. RuleArguments args; RuleGuards guards; DomainAnnotation typ; std::wstring arg; .)
'(' Ident<arg> tagcolon Domain<typ> (. args.add(arg, typ); .)
{',' Ident<arg> tagcolon Domain<typ> (. args.add(arg, typ); .)
} ')'
["case" RGuard<guards> {',' RGuard<guards>}]
'{' RBody<args, guards> '}' .
/* - TODO use RGuard for guards-*/
RuleContextDecl<CodeScope* scope> = (.Expression eHead, eGuards, eBody; .)
"rule" "context" tagcolon MetaSimpExpr<eHead>
"case" MetaSimpExpr<eGuards>
'{' MetaSimpExpr<eBody> '}' (.scope->contextRules.push_back(Expression(Operator::CONTEXT_RULE, {eHead, eGuards, eBody})); .).
Domain<DomainAnnotation& dom> =
(
"function" (. dom = DomainAnnotation::FUNCTION; .)
| "variable" (. dom = DomainAnnotation::VARIABLE; .)
).
RGuard<RuleGuards& guards>= (. Expression e; .)
MetaExpr<e> (. guards.add(std::move(e)); .).
MetaExpr<Expression& e>= (.Operator op; Expression e2; .)
MetaExpr2<e>
[MetaOp<op> MetaExpr2<e2> (. e = Expression(op, {e, e2}); .)
].
MetaExpr2<Expression& e>=
(
'(' MetaExpr<e> ')'
| MetaSimpExpr<e>
).
MetaSimpExpr<Expression& e>= (. std::wstring i1, infix; Expression e2; .)
( '-' MetaSimpExpr<e2> (. e = Expression(Operator::NEG, {e2}); .)
| IF(checkParametersList()) Ident<i1> (. e = Expression(Operator::CALL, {Expression(Atom<Identifier_t>(i1))}); .)
'(' [ CalleeParams<e> ] ')'
| IF(checkInfix()) Ident<i1> Ident<infix> MetaSimpExpr<e2>
(. e = Expression(Operator::CALL, {Expression(Atom<Identifier_t>(infix))});
e.addArg(Expression(Atom<Identifier_t>(i1)));
e.addArg(std::move(e2));
.)
| Ident<i1> (. e = Expression(Atom<Identifier_t>(i1)); .)
).
RBody<const RuleArguments& args, const RuleGuards& guards> =
(. Expression e; std::wstring msg; .)
"warning" MetaExpr<e> ["message" string (. msg = t->val; .)
] (. root.add(new RuleWarning(RuleArguments(args), RuleGuards(guards), std::move(e), Atom<String_t>(msg))); .)
.
MetaOp< Operator& op> =
implic (. op = Operator::IMPL; .)
.
/*============================ Expressions ===============================*/
ExprTyped<Expression&e, TypeAnnotation& typ> = (. Expression tag; .)
Expr<e> [tagcolon Type<typ> (. e.bindType(move(typ)); .)
{';' MetaSimpExpr<tag> (. e.tags.emplace(tag.getValueString(), move(tag)); .)
}]
.
Expr< Expression& e> (. Operator op; Expression e2; .)
= SimExpr< e>
[ RelOp< op>
SimExpr< e2> (. e = Expression(op, {e, e2}); .)
].
SimExpr< Expression& e> (. Operator op; Expression e2; .)
= Term< e>
{ AddOp< op>
Term< e2> (. e = Expression(op, {e, e2});.)
}.
Term< Expression& e> (. Operator op; Expression e2; .)
= Factor< e>
{ MulOp< op>
Factor< e2> (. e = Expression(op, {e, e2}); .)
}.
Factor< Expression& e> (. std::wstring name; TypeAnnotation typ; .)
=
(IF (checkParametersList()) Ident< name>
(. e = Expression(Operator::CALL, {Atom<Identifier_t>(name)}); .)
'(' [CalleeParams<e>] ')'
|IF (checkIndex()) Ident<name>
lbrack CalleeParams<e> rbrack (. e.setOp(Operator::INDEX); e.setValue({Atom<Identifier_t>(name)}); .)
| Ident< name> (. e = Expression(Atom<Identifier_t>(name)); root.recognizeVariantIdentifier(e); .)
| ListLiteral<e> (. /* tuple */.)
| StructLiteral<e> (. /* struct */.)
| SequenceDecl<e>
| LoopDecl<e, typ>
| IfDecl<e, typ>
| SwitchDecl<e>
| AdhocDecl<e>
| number (. e = Expression(Atom<Number_t>(t->val)); .)
| string (. e = Expression(Atom<String_t>(t->val)); .)
| '-' Factor< e> (. e = Expression(Operator::NEG, {e}); .)
| '(' Expr<e> ')'
).
StructLiteral<Expression& e> = (. std::wstring key; Expression val; std::list<Atom<Identifier_t>> keys; .)
'{' Ident<key> '=' Expr<val> (. keys.push_back(Atom<Identifier_t>(key)); e = Expression(Operator::LIST_NAMED, {val}); .)
{',' Ident<key> '=' Expr<val> (.e.addArg(move(val)); keys.push_back(Atom<Identifier_t>(key)); .)
} '}' (. e.addBindings(keys.begin(), keys.end()); .)
.
ListLiteral<Expression& e> = (. Expression eFrom, eTo; .)
'['
[ Expr<eFrom> (. e.addArg(std::move(eFrom)); .)
(".." Expr<eTo> (. e.addArg(std::move(eTo)); e.setOp(Operator::LIST_RANGE); .)
|{',' Expr<eFrom> (. e.addArg(std::move(eFrom)); .)
} (. e.setOp(Operator::LIST); .)
) ] ']'.
AdhocDecl<Expression& e> = (. Expression command; .)
"ad" "hoc" Expr<command> (. e.setOp(Operator::ADHOC); e.addArg(std::move(command)); .).
CalleeParams<Expression& e> = (. Expression e2; TypeAnnotation typ; .)
ExprTyped<e2, typ> (. e.addArg(std::move(e2)); .)
{',' ExprTyped<e2, typ> (. e.addArg(std::move(e2)); .)
}.
AddOp< Operator& op>
= (. op = Operator::ADD; .)
( '+'
| '-' (. op = Operator::SUB; .)
).
MulOp< Operator& op>
= (. op = Operator::MUL; .)
( '*'
| '/' (. op = Operator::DIV; .)
).
RelOp< Operator& op>
= (. op = Operator::EQU; .)
( equal
| '<' (. op = Operator::LSS; .)
| '>' (. op = Operator::GTR; .)
).
END Xreate.
diff --git a/config/default.json b/config/default.json
index 80e5eb5..08e9bd6 100644
--- a/config/default.json
+++ b/config/default.json
@@ -1,58 +1,59 @@
{
"containers": {
"id": {
"implementations": "impl_fulfill_cluster",
"clusters": "var_cluster",
"prototypes": "proto_cluster",
"linkedlist": "linkedlist"
},
"impl": {
"solid": "solid",
"onthefly": "on_the_fly"
}
},
"logging": {
"id": "logging"
},
"function-entry": "entry",
"clasp": {
"bindings" : {
"variable": "bind",
"function": "bind_func",
"scope": "bind_scope",
"function_demand" : "bind_function_demand",
"scope_decision": "bind_scope_decision"
},
"nonevalue": "nonevalue",
"ret": {
"symbol": "retv",
"tag": "ret"
}
},
"tests": {
- "template": "installation",
+ "template": "context",
"templates": {
"basic": "EntryFunction*",
"default": "*-",
"types": "Types*-",
"containers": "Containers*-",
"ast": "AST*",
"non-containers": "*-Containers*",
"log": "Logging*",
"clang": "ClangAPI*",
"cfg": "CFG.*",
"skip": "SkipDetection*",
"raw-xml": "libxml2*",
"xml": "Xml.*",
"adhocs": "ExpressionSerializer.*:Adhoc.*:Context.*",
+ "context": "ExpressionSerializer.*:Context.*",
"installation": "Compilation.*:Sprint1.*"
}
}
}
diff --git a/core/control-context.lp b/core/control-context.lp
index 27a27e4..62237fe 100644
--- a/core/control-context.lp
+++ b/core/control-context.lp
@@ -1,24 +1,24 @@
% context propagation
bind_scope_decision(Scope, Fn, Resolution):- cfa_call(Scope, Fn), cfa_function_specializations(Fn, Resolution), bind_scope(Scope, Resolution).
bind_scope_decision(Scope, Fn, Resolution):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, Fn), cfa_function_specializations(Fn, Resolution), bind_scope(Scope, Resolution).
bind_scope_demand(Scope, FnCallee):- cfa_call(Scope, FnCallee), cfa_function_specializations(FnCallee, _), not bind_scope_decision(Scope, FnCallee, _).
-%demand propagation
+ %demand propagation
bind_scope_demand(Scope, Subject):- bind_scope_demand(ScopeChild, Subject), cfa_parent(ScopeChild, scope(Scope)).
bind_scope_demand(Scope, Subject):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, Subject), not bind_scope_decision(Scope, Subject, _).
bind_function_demand(Fn, Subject):- bind_scope_demand(Scope, Subject), cfa_parent(Scope, function(Fn)).
bind_scope(Scope, Context) :- bind_scope(ScopeParent, Context), cfa_parent(Scope, scope(ScopeParent)).
bind_scope(Scope, Context) :- bind_scope(ScopeParent, Context): cfa_call(ScopeParent, FnCurrent); cfa_call(_, FnCurrent)
, cfa_parent(Scope, function(FnCurrent)), bind_scope(_, Context), scope(Scope).
% adhoc classes(unfinished):
%bind_func(Fn, adhoc_class(Context)) :- bind_func(Fn, adhoc(Context)), bind_scope(Scope, Context), cfa_parent(Scope, function(Fn)).
%scope_parent(Scope, ScopeParent) :- cfa_parent(Scope, scope(ScopeParent)).
%scope_parent(Scope, ScopeParent2) :- cfa_parent(Scope, scope(ScopeParent)), scope_parent(ScopeParent, ScopeParent2).
%scope_function(Scope, Fn) :- cfa_parent(Scope, function(Fn)).
%scope_function(Scope, Fn) :- cfa_parent(Scope, scope(ScopeParent)), scope_function(ScopeParent, Fn).
diff --git a/cpp/CMakeLists.txt b/cpp/CMakeLists.txt
index 909759a..1c12f2e 100644
--- a/cpp/CMakeLists.txt
+++ b/cpp/CMakeLists.txt
@@ -1,183 +1,16 @@
cmake_minimum_required(VERSION 2.8.11)
-project(xreate)
-find_package(LLVM REQUIRED CONFIG)
-
-# MACRO (ADD_PCH_RULE _header_filename _src_list)
-# SET(_gch_filename "${_header_filename}.gch")
-# LIST(APPEND ${_src_list} ${_gch_filename})
-# SET (_args ${CMAKE_CXX_FLAGS})
-# LIST(APPEND _args -c ${_header_filename} -o ${_gch_filename})
-# GET_DIRECTORY_PROPERTY(DIRINC INCLUDE_DIRECTORIES)
-# foreach (_inc ${DIRINC})
-# LIST(APPEND _args "-I" ${_inc})
-# endforeach(_inc ${DIRINC})
-# SEPARATE_ARGUMENTS(_args)
-# add_custom_command(OUTPUT ${_gch_filename}
-# COMMAND rm -f ${_gch_filename}
-# COMMAND ${CMAKE_CXX_COMPILER} ${CMAKE_CXX_COMPILER_ARG1} ${_args}
-# DEPENDS ${_header_filename})
-# ENDMACRO(ADD_PCH_RULE _header_filename _src_list)
-
-set (CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake-tools")
-include(PCH_GCC4_v2)
-
-set(LLVM_VERSION ${LLVM_VERSION_MAJOR}.${LLVM_VERSION_MINOR})
-
-link_directories(${LLVM_LIBRARY_DIRS})
-
-set(CMAKE_MODULE_PATH ${CMAKE_CURRENT_LIST_DIR})
-message ("MPATH:" ${CMAKE_MODULE_PATH})
-
-message(STATUS "Found LLVM ${LLVM_PACKAGE_VERSION}")
-message(STATUS "Using LLVMConfig.cmake in: ${LLVM_DIR}")
-#find_package(Clang REQUIRED clangTooling libClang)
+set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -Wall -fprofile-arcs -ftest-coverage -O0")
set(CMAKE_BUILD_TYPE Debug)
-set(POTASSCO_PATH "/opt/potassco/gringo" CACHE PATH "Path to gringo sources")
-set (LIBCLASP_PATH ${POTASSCO_PATH}/build/debug)
-link_directories(${LIBCLASP_PATH})
-
-FILE (GLOB TEST_FILES ./tests/*.cpp)
-message("TEST: " ${TEST_FILES})
-set(SOURCE_FILES
- ./src/pass/compilepass.cpp
- ./src/compilation/latecontextcompiler2.cpp
-
- ${TEST_FILES}
-
- #./src/compilation/latecontextcompiler.cpp
-
- ./src/serialization/expressionserializer.cpp
- ./src/serialization/expressionserializer2.cpp
- ./src/query/context.cpp
-
- ./src/ast.cpp
- ./src/llvmlayer.cpp ./src/clasplayer.cpp
-
- ./src/utils.cpp
- ./src/passmanager.cpp
-
- ./src/pass/abstractpass.cpp ./src/pass/dfgpass.cpp
- ./src/pass/cfgpass.cpp
- ./src/pass/loggerpass.cpp
- ./src/pass/adhocpass.cpp
- #./src/pass/rulespass.cpp #
-
- ./src/compilation/instr-containers.cpp
-
- ./src/query/containers.cpp
- ./src/query/ptrvalid.cpp
-
-
- ./src/attachments.cpp
- ./src/contextrule.cpp
-
- #${POTASSCO_PATH}/app/shared/src/clingocontrol.cc
- #${POTASSCO_PATH}/app/pyclingo/src/clingo_lib.cc
-)
-
-set(COCO_PATH ${CMAKE_HOME_DIRECTORY}/../coco/)
-set(COCO_SOURCE_FILES
- ${COCO_PATH}/Parser.h
- ${COCO_PATH}/Scanner.h
- ${COCO_PATH}/Parser.cpp
- ${COCO_PATH}/Scanner.cpp
-)
-
-INCLUDE_DIRECTORIES(${LLVM_INCLUDE_DIRS})
-INCLUDE_DIRECTORIES("/usr/include/libxml2")
-INCLUDE_DIRECTORIES(${COCO_PATH} ./src)
-INCLUDE_DIRECTORIES(${POTASSCO_PATH}/libgringo
-${POTASSCO_PATH}/libclasp
-${POTASSCO_PATH}/libclingo
-${POTASSCO_PATH}/libprogram_opts
-)
-
-include_directories(${CMAKE_HOME_DIRECTORY}/../vendors/jeayeson/include/)
-
-#execute_process(COMMAND ${COCO_PATH}/gen-xreate WORKING_DIRECTORY OUTPUT_VARIABLE COCO_OUTPUT)
-
-
-message(STATUS "COCO GRAMMAR BUILD STATUS:" ${COCO_OUTPUT})
-
-include_directories(${LLVM_INCLUDE_DIRS})
-add_definitions(${LLVM_DEFINITIONS})
-#add_definitions()
-set(CMAKE_CXX_FLAGS "${CMAKE_CXX_FLAGS} -Winvalid-pch -std=c++14 -D_GNU_SOURCE -D__STDC_CONSTANT_MACROS -D__STDC_FORMAT_MACROS -D__STDC_LIMIT_MACROS")
-set(CMAKE_CXX_FLAGS_DEBUG "${CMAKE_CXX_FLAGS_DEBUG} -g -Wall -fprofile-arcs -ftest-coverage -O0 -DWITH_THREADS=1")
-
-set(RESOURCES_DBUS_INTERFACES
- resources/hamster.xml
-)
-# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/ast.h SOURCE_FILES)
-# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/llvmlayer.h SOURCE_FILES)
-# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/clasplayer.h SOURCE_FILES)
-# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/pass/abstractpass.h SOURCE_FILES)
-
-add_custom_command(OUTPUT ${COCO_PATH}/Parser.cpp
- COMMAND ${COCO_PATH}/gen-grammar
- WORKING_DIRECTORY ${COCO_PATH}
- DEPENDS ${COCO_PATH}/xreate.ATG
-)
-
-add_executable(xreate ${SOURCE_FILES} ${COCO_SOURCE_FILES} src/ExternLayer.cpp src/ExternLayer.h)
-
-add_custom_target (coverage
- COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/src/code-coverage.sh
- WORKING_DIRECTORY ${CMAKE_BINARY_DIR})
-
-message(STATUS "BINARY DIR" ${CMAKE_BINARY_DIR})
-
-#add_dependencies(${PROJECT_NAME} coverageTarget)
-
-llvm_map_components_to_libnames(llvm_libs support core irreader all native nativecodegen)
-message(STATUS "LLVM LIBS: " ${llvm_libs})
-
-set(LIBCLASP_LIBS
- clingo
- clasp
- gringo
- program_opts
- reify
-)
-
-message(STATUS "CLANG LIBS: " ${CLANG_LIBS})
-message(STATUS "CLASP LIBS: " ${LIBCLASP_LIBS})
-
-
-
-
-
-FUNCTION(PREPEND var prefix)
- SET(listVar "")
- FOREACH(f ${ARGN})
- LIST(APPEND listVar "${prefix}/${f}")
- ENDFOREACH(f)
- SET(${var} "${listVar}" PARENT_SCOPE)
-ENDFUNCTION(PREPEND)
-
-
-find_package(GTest REQUIRED)
-INCLUDE_DIRECTORIES(${GTEST_INCLUDE_DIRS})
+add_subdirectory(src)
-set(CLANG_LIBS
- clangCodeGen
- clangASTMatchers
- clangQuery
- clangTooling
- clangFrontend
- clangSerialization
- clangDriver
- clangParse
- clangSema
- clangAnalysis
- clangAST
- clangEdit
- clangLex
- clangBasic
- )
+if (BUILD_XREATE_TESTS)
+ message ("Building xreate tests")
+ add_subdirectory(tests)
+endif ()
-target_link_libraries(xreate ${GTEST_LIBRARIES} LLVM-${LLVM_VERSION} ${LIBCLASP_LIBS} ${CLANG_LIBS} pthread xml2 gcov tbb)
-#set(COTIRE_UNITY_SOURCE_MAXIMUM_NUMBER_OF_INCLUDES "-j4")
-#cotire(xreate)
+if (BUILD_XREATE_SERVER)
+ message ("Building xreate server")
+ add_subdirectory(../tools/execution-server execution-server)
+endif ()
diff --git a/cpp/src/CMakeLists.txt b/cpp/src/CMakeLists.txt
new file mode 100644
index 0000000..3a28174
--- /dev/null
+++ b/cpp/src/CMakeLists.txt
@@ -0,0 +1,221 @@
+cmake_minimum_required(VERSION 2.8.11)
+project(xreate)
+cmake_policy(SET CMP0022 NEW)
+
+# BUILD OPTIONS
+#======================
+set(XREATE_DEFINITIONS
+ -D_GNU_SOURCE -D__STDC_CONSTANT_MACROS -D__STDC_FORMAT_MACROS -D__STDC_LIMIT_MACROS -DWITH_THREADS=1
+)
+
+add_definitions(${XREATE_DEFINITIONS})
+add_compile_options(-Winvalid-pch -std=c++14 -fPIC)
+
+
+
+# LLVM
+#======================
+find_package(LLVM REQUIRED CONFIG)
+set (CMAKE_MODULE_PATH "${CMAKE_SOURCE_DIR}/cmake-tools")
+include(PCH_GCC4_v2)
+
+set(LLVM_VERSION ${LLVM_VERSION_MAJOR}.${LLVM_VERSION_MINOR})
+
+link_directories(${LLVM_LIBRARY_DIRS})
+
+set(CMAKE_MODULE_PATH ${CMAKE_CURRENT_LIST_DIR})
+message ("MPATH:" ${CMAKE_MODULE_PATH})
+
+message(STATUS "Found LLVM ${LLVM_PACKAGE_VERSION}")
+message(STATUS "Using LLVMConfig.cmake in: ${LLVM_DIR}")
+INCLUDE_DIRECTORIES(${LLVM_INCLUDE_DIRS})
+message("LLVM defs: " ${LLVM_DEFINITIONS})
+add_definitions(${LLVM_DEFINITIONS})
+#llvm_map_components_to_libnames(llvm_libs support core irreader all native nativecodegen)
+
+
+
+# CLANG
+#======================
+set(CLANG_LIBS
+ clangCodeGen
+ clangASTMatchers
+ clangQuery
+ clangTooling
+ clangFrontend
+ clangSerialization
+ clangDriver
+ clangParse
+ clangSema
+ clangAnalysis
+ clangAST
+ clangEdit
+ clangLex
+ clangBasic
+)
+#find_package(Clang REQUIRED clangTooling libClang)
+message(STATUS "CLANG LIBS: " ${CLANG_LIBS})
+
+
+
+# POTASSCO
+#======================
+set(POTASSCO_PATH "/opt/potassco/gringo" CACHE PATH "Path to gringo sources")
+set(POTASSCO_INCLUDE_PATH
+ ${POTASSCO_PATH}/libgringo
+ ${POTASSCO_PATH}/libclasp
+ ${POTASSCO_PATH}/libclingo
+ ${POTASSCO_PATH}/libprogram_opts
+)
+
+INCLUDE_DIRECTORIES(${POTASSCO_INCLUDE_PATH})
+
+set (LIBCLASP_PATH ${POTASSCO_PATH}/build/debug)
+
+set(LIBCLASP_LIBS
+ clingo
+ clasp
+ gringo
+ program_opts
+ reify
+)
+message(STATUS "CLASP LIBS: " ${LIBCLASP_LIBS})
+link_directories(${LIBCLASP_PATH})
+
+
+
+# OTHER DEPENDENCIES
+#===========================
+set(JEAYESON_INCLUDE_PATH
+ ${CMAKE_HOME_DIRECTORY}/../vendors/jeayeson/include/
+)
+INCLUDE_DIRECTORIES(${JEAYESON_INCLUDE_PATH})
+
+
+
+# COCO
+#===========================
+set(COCO_PATH ${CMAKE_HOME_DIRECTORY}/../coco/)
+set(COCO_SOURCE_FILES
+ ${COCO_PATH}/Parser.cpp
+ ${COCO_PATH}/Scanner.cpp
+)
+
+INCLUDE_DIRECTORIES(${COCO_PATH} ./src)
+add_custom_command(OUTPUT COCO_OUTPUT
+ COMMAND ${COCO_PATH}/gen-grammar
+ WORKING_DIRECTORY ${COCO_PATH}
+ DEPENDS ${COCO_PATH}/xreate.ATG
+)
+message(STATUS "COCO GRAMMAR BUILD STATUS:" ${COCO_OUTPUT})
+
+
+
+# XREATE
+#======================
+set(SOURCE_FILES
+ pass/compilepass.cpp
+ compilation/latecontextcompiler2.cpp
+ #compilation/latecontextcompiler.cpp
+ serialization/expressionserializer.cpp
+ serialization/expressionserializer2.cpp
+ query/context.cpp
+
+ ast.cpp
+ llvmlayer.cpp clasplayer.cpp
+
+ utils.cpp
+ passmanager.cpp
+
+ pass/abstractpass.cpp pass/dfgpass.cpp
+ pass/cfgpass.cpp
+ pass/loggerpass.cpp
+ pass/adhocpass.cpp
+ #pass/rulespass.cpp #
+
+ compilation/instr-containers.cpp
+
+ query/containers.cpp
+ query/ptrvalid.cpp
+
+
+ attachments.cpp
+ contextrule.cpp
+ ExternLayer.cpp
+
+ #${POTASSCO_PATH}/app/shared/src/clingocontrol.cc
+ #${POTASSCO_PATH}/app/pyclingo/src/clingo_lib.cc
+)
+
+set(XREATE_INCLUDE_DIRS
+ ${CMAKE_CURRENT_SOURCE_DIR}/
+)
+
+INCLUDE_DIRECTORIES(${XREATE_INCLUDE_DIRS})
+
+set(XREATE_PRIVATE_INCLUDE_DIRS
+ ${XREATE_INCLUDE_DIRS}
+ ${COCO_PATH}
+ ${JEAYESON_INCLUDE_PATH}
+ ${LLVM_INCLUDE_DIRS}
+ ${POTASSCO_INCLUDE_PATH}
+)
+
+add_library(${PROJECT_NAME} SHARED ${SOURCE_FILES} ${COCO_SOURCE_FILES} )
+target_link_libraries(${PROJECT_NAME} ${LIBCLASP_LIBS} ${CLANG_LIBS} tbb)
+
+
+target_include_directories(${PROJECT_NAME} INTERFACE
+ ${XREATE_INCLUDE_DIRS}
+ ${COCO_PATH}
+ ${JEAYESON_INCLUDE_PATH}
+ ${LLVM_INCLUDE_DIRS}
+ ${POTASSCO_INCLUDE_PATH}
+)
+
+get_directory_property(DEFINITIONS_ALL DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} COMPILE_DEFINITIONS)
+message("definitions all: " ${DEFINITIONS_ALL})
+target_compile_definitions(${PROJECT_NAME} INTERFACE ${DEFINITIONS_ALL})
+
+get_directory_property(COMPILATION_OPTIONS_ALL DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR} COMPILE_OPTIONS)
+message("compilations all: " ${COMPILATION_OPTIONS_ALL})
+target_compile_options(${PROJECT_NAME} INTERFACE ${COMPILATION_OPTIONS_ALL})
+
+SET_TARGET_PROPERTIES(${PROJECT_NAME} PROPERTIES INTERFACE_LINK_LIBRARIES
+ LLVM-${LLVM_VERSION}
+)
+
+#set (LINK_INTERFACE_LIBRARIES "")
+
+# FUNCTION(PREPEND var prefix)
+# SET(listVar "")
+# FOREACH(f ${ARGN})
+# LIST(APPEND listVar "${prefix}/${f}")
+# ENDFOREACH(f)
+# SET(${var} "${listVar}" PARENT_SCOPE)
+# ENDFUNCTION(PREPEND)
+
+#set(COTIRE_UNITY_SOURCE_MAXIMUM_NUMBER_OF_INCLUDES "-j4")
+#cotire(xreate)
+
+
+# MACRO (ADD_PCH_RULE _header_filename _src_list)
+# SET(_gch_filename "${_header_filename}.gch")
+# LIST(APPEND ${_src_list} ${_gch_filename})
+# SET (_args ${CMAKE_CXX_FLAGS})
+# LIST(APPEND _args -c ${_header_filename} -o ${_gch_filename})
+# GET_DIRECTORY_PROPERTY(DIRINC INCLUDE_DIRECTORIES)
+# foreach (_inc ${DIRINC})
+# LIST(APPEND _args "-I" ${_inc})
+# endforeach(_inc ${DIRINC})
+# SEPARATE_ARGUMENTS(_args)
+# add_custom_command(OUTPUT ${_gch_filename}
+# COMMAND rm -f ${_gch_filename}
+# COMMAND ${CMAKE_CXX_COMPILER} ${CMAKE_CXX_COMPILER_ARG1} ${_args}
+# DEPENDS ${_header_filename})
+# ENDMACRO(ADD_PCH_RULE _header_filename _src_list)
+
+# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/ast.h SOURCE_FILES)
+# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/llvmlayer.h SOURCE_FILES)
+# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/clasplayer.h SOURCE_FILES)
+# ADD_PCH_RULE (${CMAKE_HOME_DIRECTORY}/src/pass/abstractpass.h SOURCE_FILES)
diff --git a/cpp/tests/CMakeLists.txt b/cpp/tests/CMakeLists.txt
new file mode 100644
index 0000000..c729026
--- /dev/null
+++ b/cpp/tests/CMakeLists.txt
@@ -0,0 +1,19 @@
+cmake_minimum_required(VERSION 2.8.11)
+project(xreate-tests)
+
+find_package(GTest REQUIRED)
+INCLUDE_DIRECTORIES(${GTEST_INCLUDE_DIRS})
+INCLUDE_DIRECTORIES("/usr/include/libxml2")
+INCLUDE_DIRECTORIES($<TARGET_PROPERTY:xreate,INCLUDE_DIRECTORIES>)
+
+
+
+# TESTS
+#=========================
+aux_source_directory(. TEST_FILES)
+add_executable(${PROJECT_NAME} ${TEST_FILES})
+target_link_libraries(${PROJECT_NAME} xreate ${GTEST_LIBRARIES} pthread xml2 gcov)
+
+add_custom_target (coverage
+ COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/src/code-coverage.sh
+ WORKING_DIRECTORY ${CMAKE_BINARY_DIR})
diff --git a/cpp/tests/context.cpp b/cpp/tests/context.cpp
index 77f2d8a..56655b3 100644
--- a/cpp/tests/context.cpp
+++ b/cpp/tests/context.cpp
@@ -1,267 +1,325 @@
/*
* frame-context.cpp
*
* Created on: Dec 3, 2015
* Author: pgess
*/
#include "passmanager.h"
#include "query/context.h"
#include "gtest/gtest.h"
#include <iostream>
+#include <boost/scoped_ptr.hpp>
using namespace xreate;
-TEST(Context, frame_Context1){
- PassManager* man = PassManager::prepareForCode(
- " import raw (\"core/control-context.lp\")\n"
- " testC = function::int {\n"
- " context:: testC.\n"
-
- " 0\n"
- " }\n"
-
- " testA = function:: int {\n"
- " context:: testA; test.\n"
-
- " testC()\n"
- " }\n"
-
- " testB = function:: int {\n"
- " context:: testB; test.\n"
-
- " testC()\n"
- " }\n"
- );
-
- ContextQuery* query = (ContextQuery*) man->clasp->registerQuery(new ContextQuery(), QueryId::ContextQuery);
- man->runWithoutCompilation();
-
- CodeScope* scopeTestC = man->root->findFunction("testC")->getEntryScope();
- const ContextDomain& context = query->getContext(man->clasp->pack(scopeTestC));
-
- int contextSize = context.size();
- EXPECT_EQ(2, contextSize);
+//TEST(Context, frame_Context1){
+// PassManager* man = PassManager::prepareForCode(
+// " import raw (\"core/control-context.lp\")\n"
+// " testC = function::int {\n"
+// " context:: testC.\n"
+//
+// " 0\n"
+// " }\n"
+//
+// " testA = function:: int {\n"
+// " context:: testA; test.\n"
+//
+// " testC()\n"
+// " }\n"
+//
+// " testB = function:: int {\n"
+// " context:: testB; test.\n"
+//
+// " testC()\n"
+// " }\n"
+// );
+//
+// ContextQuery* query = (ContextQuery*) man->clasp->registerQuery(new ContextQuery(), QueryId::ContextQuery);
+// man->runWithoutCompilation();
+//
+// CodeScope* scopeTestC = man->root->findFunction("testC")->getEntryScope();
+// const ContextDomain& context = query->getContext(man->clasp->pack(scopeTestC));
+//
+// int contextSize = context.size();
+// EXPECT_EQ(2, contextSize);
+//}
+//
+//TEST(Context, full_ContextBasedFunctionSpecialization){
+//
+// PassManager* man = PassManager::prepareForCode(
+// " case context::toMillimeters {\n"
+// " convert = function(source:: num)::num {\n"
+// " 10 * source \n"
+// " }\n"
+// " }\n"
+//
+// " case context::toInches {\n"
+// " convert = function(source:: num)::num {\n"
+// " 2 * source \n"
+// " }\n"
+// " }\n"
+//
+// "test = function(vrnt:: int)::int; entry {\n"
+// " switch(vrnt):: int\n"
+// " case 0 {\n"
+// " context:: toMillimeters.\n"
+// " convert(1)\n"
+// " }\n"
+// "\n"
+// " case 1 {\n"
+// " context:: toInches.\n"
+// " convert(1)\n"
+// " }\n"
+// " case default {0}\n"
+// " }" );
+//
+// int (*main)(int) = (int (*)(int)) man->run();
+// ASSERT_EQ(10, main(0));
+// ASSERT_EQ(2, main(1));
+//}
+//
+//TEST(Context, full_LoopContext){
+//
+// PassManager* man = PassManager::prepareForCode("case context:: a {\n"
+// " print = function:: string {\n"
+// " \"a\"\n"
+// " }}\n"
+// "\n"
+// " case context:: b {\n"
+// " print = function:: string {\n"
+// " \"b\"\n"
+// " }}\n"
+// "\n"
+// " case context:: c {\n"
+// " print = function:: string {\n"
+// " \"c\"\n"
+// " }}\n"
+// "\n"
+// " case context:: d {\n"
+// " print = function:: string {\n"
+// " \"d\"\n"
+// " }}\n"
+// "\n"
+// " start = function(command::int)::string; entry {\n"
+// " switch (command) :: string \n"
+// " case 0 {\n"
+// " context:: print(a); print(b); print(d).\n"
+// "\n"
+// " loop context (\"print\") {\n"
+// " print()\n"
+// " }\n"
+// " }\n"
+// "\n"
+// " case default {\n"
+// " context:: print(c).\n"
+// " loop context (\"print\") {\n"
+// " print()\n"
+// " }\n"
+// " }\n"
+// " }");
+//
+//
+// char* (*main)(int) =(char* (*)(int)) man->run();
+// ASSERT_STREQ("c", main(1));
+// ASSERT_STREQ("a", main(0));
+//}
+//
+//TEST(Context, full_RuleContext){
+// /*
+// "rule context:: childs(Child)\n"
+// " case artefact(Item)\n"
+// " {\n"
+// " artefact_depends(Item, Child)\n"
+// " }";
+// */
+//
+// PassManager* man = PassManager::prepareForCode(
+// " case context:: toMilli {\n"
+// " convert = function(length::int)::int{\n"
+// " 10 * length\n"
+// " }\n"
+// " }\n"
+// "\n"
+// " case context:: toCenti {\n"
+// " convert = function(length::int)::int{\n"
+// " length\n"
+// " }\n"
+// " }\n"
+// "\n"
+// " main=function::int; entry {\n"
+// " context:: output(milli).\n"
+// "\n"
+// " rule context::toMilli\n"
+// " case output(milli) {true}\n"
+// "\n"
+// " convert(1)\n"
+// " }" );
+// man->clasp->addRawScript("true.");
+//
+// int (*entry)() = (int (*)()) man->run();
+// ASSERT_EQ(10, entry());
+//}
+//
+//TEST(Context, full_InheritedRuleContext){
+// PassManager* man = PassManager::prepareForCode(
+// " import raw (\"core/control-context.lp\") \n"
+//
+// " case context:: toMilli {\n"
+// " convert = function(length::int)::int{\n"
+// " 10 * length\n"
+// " }\n"
+// " }\n"
+//
+// " case context:: toCenti {\n"
+// " convert = function(length::int)::int{\ n"
+// " length\n"
+// " }\n"
+// " }\n"
+// "\n"
+//
+// "main = function(comm:: num)::num; entry{\n"
+// " rule context::X case output(X) {true}\n"
+// "\n"
+// " switch (comm)::num \n"
+// " case 0 {\n"
+// " context:: output(toMilli).\n"
+// " convert(1)\n"
+// " }\n"
+// " case default {\n"
+// " context:: output(toCenti).\n"
+// " convert(1)\n"
+// " }\n"
+// " }");
+//
+// man->clasp->addRawScript("true.");
+// int (*entry)(int) = (int (*)(int)) man->run();
+// ASSERT_EQ(10, entry(0));
+// ASSERT_EQ(1, entry(1));
+//}
+//
+//
+//
+//TEST(Context, full_LateContext){
+// PassManager* man = PassManager::prepareForCode(
+// "import raw (\"core/control-context.lp\")\n"
+//
+// " convert = function(length:: num)::num{\n"
+// " 0\n"
+// " }\n"
+//
+// "case context:: milli {\n"
+// " convert = function(length:: num)::num{\n"
+// " 1000 * length\n"
+// " }\n"
+// "}\n"
+// "\n"
+// "case context:: centi {\n"
+// " convert = function(length:: num)::num{\n"
+// " 100 * length\n"
+// " }\n"
+// "}\n"
+// "\n"
+// "calculate = function(length:: num)::num {\n"
+// " convert(length)\n"
+// "}\n"
+// "\n"
+// "main = function(com:: num):: num; entry {\n"
+// " switch (com):: num \n"
+// " case 0 {\n"
+// " context:: milli.\n"
+// " calculate(1)\n"
+// " }\n"
+// "\n"
+// " case default{\n"
+// " context:: centi. \n"
+// " calculate(1)\n"
+// " }\n"
+// "}");
+//
+// man->runWithoutCompilation();
+// ContextQuery* queryContext = reinterpret_cast<ContextQuery*>(man->clasp->getQuery(QueryId::ContextQuery));
+// Expression exprSwitch = man->root->findFunction("main")->__entry->__body;
+// CodeScope* blockDefault = man->root->findFunction("main")->__entry->__body.operands[1].blocks.front();
+// ScopePacked blockDefaultId = man->clasp->pack(blockDefault);
+// const ContextDomain& domDefault = queryContext->getContext(blockDefaultId);
+// ASSERT_EQ(1, domDefault.count(Expression(Atom<Identifier_t>("centi"))));
+//
+// std::list<ManagedFnPtr> variants = man->root->getFunctionVariants("convert");
+// for (ManagedFnPtr f: variants){
+// const Expression guard = f->guardContext;
+// bool result = (guard.getValueString() == "centi" || guard.getValueString() == "milli" || !guard.isValid());
+// ASSERT_TRUE(result);
+// }
+//
+// const FunctionContextDemand& demMain = queryContext->getFunctionDemand("main");
+// ASSERT_EQ(0, demMain.size());
+//
+// const FunctionContextDemand& demCalculate = queryContext->getFunctionDemand("calculate");
+// ASSERT_EQ(1, demCalculate.size());
+// ASSERT_EQ(1, demCalculate.right.count("convert"));
+//
+// int (*entry)(int) = (int (*)(int)) man->run();
+// ASSERT_EQ(1000, entry(0));
+// ASSERT_EQ(100, entry(1));
+//}
+
+TEST(Context, pathDependent_context){
+ std::string program =
+R"CODE(case context:: convert(milli, meters) {
+ convert = function(length:: num) :: num {
+ 1000 * length
+ }
}
-TEST(Context, full_ContextBasedFunctionSpecialization){
-
- PassManager* man = PassManager::prepareForCode(
- " case context::toMillimeters {\n"
- " convert = function(source:: num)::num {\n"
- " 10 * source \n"
- " }\n"
- " }\n"
-
- " case context::toInches {\n"
- " convert = function(source:: num)::num {\n"
- " 2 * source \n"
- " }\n"
- " }\n"
-
- "test = function(vrnt:: int)::int; entry {\n"
- " switch(vrnt):: int\n"
- " case 0 {\n"
- " context:: toMillimeters.\n"
- " convert(1)\n"
- " }\n"
- "\n"
- " case 1 {\n"
- " context:: toInches.\n"
- " convert(1)\n"
- " }\n"
- " case default {0}\n"
- " }" );
-
- int (*main)(int) = (int (*)(int)) man->run();
- ASSERT_EQ(10, main(0));
- ASSERT_EQ(2, main(1));
+case context:: convert(centi, meters) {
+ convert = function(length:: num) :: num {
+ 100 * length
+ }
}
-TEST(Context, full_LoopContext){
-
- PassManager* man = PassManager::prepareForCode("case context:: a {\n"
- " print = function:: string {\n"
- " \"a\"\n"
- " }}\n"
- "\n"
- " case context:: b {\n"
- " print = function:: string {\n"
- " \"b\"\n"
- " }}\n"
- "\n"
- " case context:: c {\n"
- " print = function:: string {\n"
- " \"c\"\n"
- " }}\n"
- "\n"
- " case context:: d {\n"
- " print = function:: string {\n"
- " \"d\"\n"
- " }}\n"
- "\n"
- " start = function(command::int)::string; entry {\n"
- " switch (command) :: string \n"
- " case 0 {\n"
- " context:: print(a); print(b); print(d).\n"
- "\n"
- " loop context (\"print\") {\n"
- " print()\n"
- " }\n"
- " }\n"
- "\n"
- " case default {\n"
- " context:: print(c).\n"
- " loop context (\"print\") {\n"
- " print()\n"
- " }\n"
- " }\n"
- " }");
-
-
- char* (*main)(int) =(char* (*)(int)) man->run();
- ASSERT_STREQ("c", main(1));
- ASSERT_STREQ("a", main(0));
+case context:: convert(centi, kilo) {
+ convert = function(length:: num) :: num {
+ 100000 * length
+ }
}
-TEST(Context, full_RuleContext){
- /*
- "rule context:: childs(Child)\n"
- " case artefact(Item)\n"
- " {\n"
- " artefact_depends(Item, Child)\n"
- " }";
- */
-
- PassManager* man = PassManager::prepareForCode(
- " case context:: toMilli {\n"
- " convert = function(length::int)::int{\n"
- " 10 * length\n"
- " }\n"
- " }\n"
- "\n"
- " case context:: toCenti {\n"
- " convert = function(length::int)::int{\n"
- " length\n"
- " }\n"
- " }\n"
- "\n"
- " main=function::int; entry {\n"
- " context:: output(milli).\n"
- "\n"
- " rule context::toMilli\n"
- " case output(milli) {true}\n"
- "\n"
- " convert(1)\n"
- " }" );
- man->clasp->addRawScript("true.");
-
- int (*entry)() = (int (*)()) man->run();
- ASSERT_EQ(10, entry());
+case context:: convert(milli, kilo) {
+ convert = function(length:: num) :: num {
+ 1000000 * length
+ }
}
-TEST(Context, full_InheritedRuleContext){
- PassManager* man = PassManager::prepareForCode(
- " import raw (\"core/control-context.lp\") \n"
-
- " case context:: toMilli {\n"
- " convert = function(length::int)::int{\n"
- " 10 * length\n"
- " }\n"
- " }\n"
+test1_fromCentiToKilo = function(a::num)::num {
+ context:: input(centi).
- " case context:: toCenti {\n"
- " convert = function(length::int)::int{\n"
- " length\n"
- " }\n"
- " }\n"
- "\n"
-
- "main = function(comm:: num)::num; entry{\n"
- " rule context::X case output(X) {true}\n"
- "\n"
- " switch (comm)::num \n"
- " case 0 {\n"
- " context:: output(toMilli).\n"
- " convert(1)\n"
- " }\n"
- " case default {\n"
- " context:: output(toCenti).\n"
- " convert(1)\n"
- " }\n"
- " }");
-
- man->clasp->addRawScript("true.");
- int (*entry)(int) = (int (*)(int)) man->run();
- ASSERT_EQ(10, entry(0));
- ASSERT_EQ(1, entry(1));
+ toKilo(a)
}
-TEST(Context, full_LateContext){
- PassManager* man = PassManager::prepareForCode(
- "import raw (\"core/control-context.lp\")\n"
-
- " convert = function(length:: num)::num{\n"
- " 0\n"
- " }\n"
+test2_fromMilliToMeters = function(a::num)::num{
+ context:: input(milli).
- "case context:: milli {\n"
- " convert = function(length:: num)::num{\n"
- " 1000 * length\n"
- " }\n"
- "}\n"
- "\n"
- "case context:: centi {\n"
- " convert = function(length:: num)::num{\n"
- " 100 * length\n"
- " }\n"
- "}\n"
- "\n"
- "calculate = function(length:: num)::num {\n"
- " convert(length)\n"
- "}\n"
- "\n"
- "main = function(com:: num):: num; entry {\n"
- " switch (com):: num \n"
- " case 0 {\n"
- " context:: milli.\n"
- " calculate(1)\n"
- " }\n"
- "\n"
- " case default{\n"
- " context:: centi. \n"
- " calculate(1)\n"
- " }\n"
- "}");
-
- man->runWithoutCompilation();
- ContextQuery* queryContext = reinterpret_cast<ContextQuery*>(man->clasp->getQuery(QueryId::ContextQuery));
- Expression exprSwitch = man->root->findFunction("main")->__entry->__body;
- CodeScope* blockDefault = man->root->findFunction("main")->__entry->__body.operands[1].blocks.front();
- ScopePacked blockDefaultId = man->clasp->pack(blockDefault);
- const ContextDomain& domDefault = queryContext->getContext(blockDefaultId);
- ASSERT_EQ(1, domDefault.count(Expression(Atom<Identifier_t>("centi"))));
-
- std::list<ManagedFnPtr> variants = man->root->getFunctionVariants("convert");
- for (ManagedFnPtr f: variants){
- const Expression guard = f->guardContext;
- bool result = (guard.getValueString() == "centi" || guard.getValueString() == "milli" || !guard.isValid());
- ASSERT_TRUE(result);
- }
-
- const FunctionContextDemand& demMain = queryContext->getFunctionDemand("main");
- ASSERT_EQ(0, demMain.size());
-
- const FunctionContextDemand& demCalculate = queryContext->getFunctionDemand("calculate");
- ASSERT_EQ(1, demCalculate.size());
- ASSERT_EQ(1, demCalculate.right.count("convert"));
+ toMeters(a)
+}
- int (*entry)(int) = (int (*)(int)) man->run();
- ASSERT_EQ(1000, entry(0));
- ASSERT_EQ(100, entry(1));
+toMeters = function(a::num)::num {
+ rule context :: convert(X, meters) case input(X) {true}
+
+ convert(a)
}
+toKilo = function(a::num)::num {
+ rule context :: convert(X, kilo) case input(X) {true}
+
+ convert(a)
+})CODE";
+
+
+
+ boost::scoped_ptr<PassManager> man(PassManager::prepareForCode(move(program)));
+ man->runWithoutCompilation();
+}
diff --git a/documentation-tools/RemarkupParser/CMakeLists.txt b/documentation-tools/RemarkupParser/CMakeLists.txt
new file mode 100644
index 0000000..d2f6ef6
--- /dev/null
+++ b/documentation-tools/RemarkupParser/CMakeLists.txt
@@ -0,0 +1,23 @@
+project(RemarkupParser)
+aux_source_directory(cpp/ SRC_LIST)
+
+set(SRC_LIST ${SRC_LIST}
+ gen-cpp/PhabricatorParserPrivate.cpp
+ gen-cpp/remarkup_constants.cpp
+ gen-cpp/remarkup_types.cpp
+)
+
+include_directories(cpp/ gen-cpp/)
+set(REMARKUP_PARSER_INCLUDE_DIRS
+ ${CMAKE_CURRENT_SOURCE_DIR}/cpp/
+ ${CMAKE_CURRENT_SOURCE_DIR}/gen-cpp/
+
+CACHE INTERNAL "REMARKUP_PARSER: Include Directories" FORCE
+)
+
+
+
+add_library(${PROJECT_NAME} SHARED ${SRC_LIST})
+target_link_libraries(${PROJECT_NAME} thrift-1.0.0-dev Qt5::Core)
+
+set(CMAKE_BUILD_TYPE Debug)
diff --git a/documentation-tools/RemarkupParser/cpp/remarkupparser.cpp b/documentation-tools/RemarkupParser/cpp/remarkupparser.cpp
new file mode 100644
index 0000000..5926147
--- /dev/null
+++ b/documentation-tools/RemarkupParser/cpp/remarkupparser.cpp
@@ -0,0 +1,96 @@
+#include "remarkupparser.h"
+#include "PhabricatorParserPrivate.h"
+#include <thrift/protocol/TBinaryProtocol.h>
+#include <thrift/transport/TSocket.h>
+#include <thrift/transport/TTransportUtils.h>
+#include <QThread>
+#include <QDebug>
+#include <sys/types.h>
+#include <sys/stat.h>
+#include <fcntl.h>
+#include <QMetaType>
+#include <stdio.h>
+#include <QFile>
+
+using namespace std;
+using namespace std;
+using namespace apache::thrift;
+using namespace apache::thrift::protocol;
+using namespace apache::thrift::transport;
+
+const char* SOCKET_NAME = tempnam(0,0);
+const string SERVER_INDEX = "/private/prg/code/xreate/documentation-tools/RemarkupParser/php/index.php";
+const string LOCK_FILENAME = "/var/lock/phabricator-parser";
+
+void
+RemarkupParser::startPhabricatorParserServer(){
+ //acquire lock:
+// if (-1 == open( LOCK_FILENAME.c_str(), O_RDWR | O_CREAT | O_EXCL, 0666 )){
+// qDebug()<<"Lock acquiring failure";
+// return;
+// }
+
+ phpServerId = fork();
+ if(phpServerId < 0 )
+ {
+ qDebug()<<"fork failure";
+ return;
+ }
+
+ //child
+ if(phpServerId == 0)
+ {
+ execl("/usr/bin/php", "php", SERVER_INDEX.c_str(), SOCKET_NAME, NULL);
+ _exit(0);
+ }
+
+ qDebug()<<"successfully forked";
+}
+
+RemarkupParser::RemarkupParser(){
+ startPhabricatorParserServer();
+}
+
+bool
+RemarkupParser::connect(){
+ qDebug()<<"RemarkupParser::connect";
+ if (!QFile::exists(QString::fromLatin1(SOCKET_NAME))) return false;
+
+ boost::shared_ptr<TTransport> socket(new TSocket(SOCKET_NAME));
+ boost::shared_ptr<TTransport> transport(new TBufferedTransport(socket));
+ boost::shared_ptr<TProtocol> protocol(new TBinaryProtocol(transport));
+
+ client = new PhabricatorParserPrivateClient (protocol);
+ transport->open();
+
+ return true;
+}
+
+Blocks
+RemarkupParser::parse(const QString& text)
+{
+ qDebug()<<"RemarkupParser:: parsing";
+ std::vector<Block> output;
+
+ if (!client){
+ qDebug() << "Parser does not connected";
+ return output;
+ }
+
+ try {
+ client->parse(output, text.toStdString());
+ } catch (TException& tx) {
+ qDebug() << "RemarkupParser exception: " << tx.what();
+ }
+
+ return output;
+}
+
+RemarkupParser::~RemarkupParser(){
+ delete client;
+ // remove(LOCK_FILENAME.c_str());
+
+ qDebug()<<"RemarkupParser::kill PHP";
+ kill(phpServerId, SIGKILL);
+ remove(SOCKET_NAME);
+}
diff --git a/documentation-tools/RemarkupParser/cpp/remarkupparser.h b/documentation-tools/RemarkupParser/cpp/remarkupparser.h
new file mode 100644
index 0000000..87b15f8
--- /dev/null
+++ b/documentation-tools/RemarkupParser/cpp/remarkupparser.h
@@ -0,0 +1,26 @@
+#ifndef PHABRICATORPARSER_H
+#define PHABRICATORPARSER_H
+
+#include "remarkup_types.h"
+#include <QString>
+
+class PhabricatorParserPrivateClient;
+
+class RemarkupParser
+{
+public:
+ RemarkupParser();
+ ~RemarkupParser();
+
+ bool connect();
+ bool isConnected() const {return client != NULL;}
+ Blocks parse(const QString& text);
+
+private:
+ __pid_t phpServerId;
+
+ PhabricatorParserPrivateClient* client = NULL;
+ void startPhabricatorParserServer();
+};
+
+#endif // PHABRICATORPARSER_H
diff --git a/documentation-tools/RemarkupParser/php/RemarkupBlocksParser.php b/documentation-tools/RemarkupParser/php/RemarkupBlocksParser.php
new file mode 100644
index 0000000..9315b80
--- /dev/null
+++ b/documentation-tools/RemarkupParser/php/RemarkupBlocksParser.php
@@ -0,0 +1,71 @@
+<?php
+class RemarkupBlocksParser implements PhabricatorParserIf{
+
+ private $engine = null;
+
+ public function parse($text) {
+ $blocks = $this->engine->splitTextIntoBlocks($text);
+
+ $output = array();
+ foreach($blocks as $block) {
+ $block['rule'] = get_class($block['rule']);
+ $output[] = new Block($block);
+ }
+ return $output;
+ }
+
+
+ public function __construct() {
+ $this->engine = $this->initialize();
+ $this->engine->setMode(PhutilRemarkupEngine::MODE_TEXT);
+ }
+
+ private function initialize() {
+ $engine = new PhutilRemarkupEngine();
+ $engine->setConfig('uri.prefix', 'http://www.example.com/');
+
+ $engine->setConfig(
+ 'uri.allowed-protocols',
+ array(
+ 'http' => true,
+ 'mailto' => true,
+ 'tel' => true,
+ ));
+
+ $rules = array();
+ $rules[] = new PhutilRemarkupEscapeRemarkupRule();
+ $rules[] = new PhutilRemarkupMonospaceRule();
+ $rules[] = new PhutilRemarkupDocumentLinkRule();
+ $rules[] = new PhutilRemarkupHyperlinkRule();
+ $rules[] = new PhutilRemarkupBoldRule();
+ $rules[] = new PhutilRemarkupItalicRule();
+ $rules[] = new PhutilRemarkupDelRule();
+ $rules[] = new PhutilRemarkupUnderlineRule();
+ $rules[] = new PhutilRemarkupHighlightRule();
+
+ $blocks = array();
+ $blocks[] = new PhutilRemarkupQuotesBlockRule();
+ $blocks[] = new PhutilRemarkupReplyBlockRule();
+ $blocks[] = new PhutilRemarkupHeaderBlockRule();
+ $blocks[] = new PhutilRemarkupHorizontalRuleBlockRule();
+ $blocks[] = new PhutilRemarkupCodeBlockRule();
+ $blocks[] = new PhutilRemarkupLiteralBlockRule();
+ $blocks[] = new PhutilRemarkupNoteBlockRule();
+ $blocks[] = new PhutilRemarkupTableBlockRule();
+ $blocks[] = new PhutilRemarkupSimpleTableBlockRule();
+ $blocks[] = new PhutilRemarkupDefaultBlockRule();
+ $blocks[] = new PhutilRemarkupListBlockRule();
+ $blocks[] = new PhutilRemarkupInterpreterBlockRule();
+
+ foreach ($blocks as $block) {
+ if (!($block instanceof PhutilRemarkupCodeBlockRule)) {
+ $block->setMarkupRules($rules);
+ }
+ }
+
+ $engine->setBlockRules($blocks);
+
+ return $engine;
+ }
+
+}
diff --git a/documentation-tools/RemarkupParser/php/index.php b/documentation-tools/RemarkupParser/php/index.php
new file mode 100644
index 0000000..05f26b3
--- /dev/null
+++ b/documentation-tools/RemarkupParser/php/index.php
@@ -0,0 +1,193 @@
+<?php
+require_once '/opt/libphutil/src/object/Phobject.php';
+require_once '/opt/libphutil/src/moduleutils/core.php';
+require_once '/opt/libphutil/src/moduleutils/PhutilBootloader.php';
+require_once '/opt/libphutil/src/moduleutils/PhutilBootloaderException.php';
+require_once '/opt/libphutil/src/moduleutils/PhutilLibraryConflictException.php';
+require_once ("/opt/libphutil/src/utils/utils.php");
+require_once ("/opt/libphutil/src/xsprintf/csprintf.php");
+require_once ("/opt/libphutil/src/future/exec/execx.php");
+require_once ("/opt/libphutil/src/xsprintf/hgsprintf.php");
+require_once ("/opt/libphutil/src/markup/render.php");
+require_once ("/opt/libphutil/src/xsprintf/jsprintf.php");
+require_once ("/opt/libphutil/src/xsprintf/ldapsprintf.php");
+require_once ("/opt/libphutil/src/error/phlog.php");
+require_once ("/opt/libphutil/src/internationalization/pht.php");
+require_once ("/opt/libphutil/src/console/format.php");
+require_once ("/opt/libphutil/src/utils/viewutils.php");
+require_once ("/opt/libphutil/src/moduleutils/moduleutils.php");
+require_once ("/opt/libphutil/src/utils/utf8.php");
+require_once ("/opt/libphutil/src/xsprintf/pregsprintf.php");
+require_once ("/opt/libphutil/src/xsprintf/qsprintf.php");
+require_once ("/opt/libphutil/src/xsprintf/queryfx.php");
+require_once ("/opt/libphutil/src/xsprintf/tsprintf.php");
+require_once ("/opt/libphutil/src/xsprintf/urisprintf.php");
+require_once ("/opt/libphutil/src/parser/xhpast/parser_nodes.php");
+require_once ("/opt/libphutil/src/parser/xhpast/parser_tokens.php");
+require_once ("/opt/libphutil/src/xsprintf/xsprintf.php");
+require_once ("/opt/libphutil/src/filesystem/Filesystem.php");
+require_once ("/opt/libphutil/src/object/Phobject.php");
+require_once ("/opt/libphutil/src/filesystem/FileFinder.php");
+require_once ("/opt/libphutil/src/markup/PhutilMarkupEngine.php");
+require_once ("/opt/libphutil/src/markup/engine/PhutilRemarkupEngine.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupEscapeRemarkupRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupMonospaceRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupDocumentLinkRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupHyperlinkRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupBoldRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupItalicRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupDelRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupUnderlineRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupHighlightRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupQuotesBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupReplyBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupHorizontalRuleBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupCodeBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupLiteralBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupTableBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupSimpleTableBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupDefaultBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupListBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupInterpreterBlockRule.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/PhutilRemarkupBlockStorage.php");
+require_once ("/opt/libphutil/src/markup/PhutilSafeHTML.php");
+require_once ("/opt/libphutil/src/internationalization/PhutilTranslator.php");
+require_once ("/opt/libphutil/src/parser/PhutilSimpleOptions.php");
+require_once ("/opt/libphutil/src/lexer/PhutilLexer.php");
+require_once ("/opt/libphutil/src/lexer/PhutilSimpleOptionsLexer.php");
+require_once ("/opt/libphutil/src/parser/PhutilLanguageGuesser.php");
+require_once ("/opt/libphutil/src/markup/syntax/engine/PhutilSyntaxHighlighterEngine.php");
+require_once ("/opt/libphutil/src/markup/syntax/engine/PhutilDefaultSyntaxHighlighterEngine.php");
+require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilDefaultSyntaxHighlighter.php");
+require_once ("/opt/libphutil/src/future/Future.php");
+require_once ("/opt/libphutil/src/future/ImmediateFuture.php");
+require_once ("/opt/libphutil/src/parser/PhutilURI.php");
+require_once ("/opt/libphutil/src/symbols/PhutilSymbolLoader.php");
+require_once ("/opt/libphutil/src/symbols/PhutilClassMapQuery.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupBlockInterpreter.php");
+require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupTestInterpreterRule.php");
+require_once ("/opt/libphutil/src/parser/xhpast/bin/PhutilXHPASTBinary.php");
+require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilSyntaxHighlighter.php");
+require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilLexerSyntaxHighlighter.php");
+require_once ("/opt/libphutil/src/lexer/PhutilPHPFragmentLexer.php");
+
+require_once '/opt/thrift/lib/php/lib/Thrift/ClassLoader/ThriftClassLoader.php';
+use Thrift\ClassLoader\ThriftClassLoader;
+use Thrift\Protocol\TBinaryProtocol;
+use Thrift\Transport\TBufferedTransport;
+
+$loader = new ThriftClassLoader();
+$loader->registerNamespace('Thrift', '/opt/thrift/lib/php/lib');
+$loader->register();
+
+$GEN_DIR = dirname ( __FILE__ ).'/../gen-php';
+set_include_path(get_include_path() . PATH_SEPARATOR . $GEN_DIR);
+require_once 'Types.php';
+require_once 'PhabricatorParserPrivate.php';
+//require_once 'RemarkupBlocksParser.php';
+
+class RemarkupBlocksParser implements PhabricatorParserPrivateIf{
+
+ private $engine = null;
+
+ public function parse($text) {
+ echo "[PHP] RemarkupBlocksParser:: parsing text \n";
+
+ $blocks = $this->engine->splitTextIntoBlocks($text);
+
+ $output = array();
+ foreach($blocks as $block) {
+ $additionalData = "";
+
+ if ($block['rule'] instanceof PhutilRemarkupNoteBlockRule){
+ $additionalData = $block['rule']->additionalData;
+ }
+
+ $block['rule'] = get_class($block['rule']) . $additionalData;
+ $output[] = new Block($block);
+ }
+
+ return $output;
+ }
+
+
+ public function __construct() {
+ $this->engine = $this->initialize();
+ $this->engine->setMode(PhutilRemarkupEngine::MODE_TEXT);
+ }
+
+ private function initialize() {
+ $engine = new PhutilRemarkupEngine();
+ $engine->setConfig('uri.prefix', 'http://www.example.com/');
+
+ $engine->setConfig(
+ 'uri.allowed-protocols',
+ array(
+ 'http' => true,
+ 'mailto' => true,
+ 'tel' => true,
+ ));
+
+ $rules = array();
+ $rules[] = new PhutilRemarkupEscapeRemarkupRule();
+ $rules[] = new PhutilRemarkupMonospaceRule();
+ $rules[] = new PhutilRemarkupDocumentLinkRule();
+ $rules[] = new PhutilRemarkupHyperlinkRule();
+ $rules[] = new PhutilRemarkupBoldRule();
+ $rules[] = new PhutilRemarkupItalicRule();
+ $rules[] = new PhutilRemarkupDelRule();
+ $rules[] = new PhutilRemarkupUnderlineRule();
+ $rules[] = new PhutilRemarkupHighlightRule();
+
+ $blocks = array();
+ $blocks[] = new PhutilRemarkupQuotesBlockRule();
+ $blocks[] = new PhutilRemarkupReplyBlockRule();
+ $blocks[] = new PhutilRemarkupHeaderBlockRule();
+ $blocks[] = new PhutilRemarkupHorizontalRuleBlockRule();
+ $blocks[] = new PhutilRemarkupCodeBlockRule();
+ $blocks[] = new PhutilRemarkupLiteralBlockRule();
+ $blocks[] = new PhutilRemarkupNoteBlockRule();
+ $blocks[] = new PhutilRemarkupTableBlockRule();
+ $blocks[] = new PhutilRemarkupSimpleTableBlockRule();
+ $blocks[] = new PhutilRemarkupDefaultBlockRule();
+ $blocks[] = new PhutilRemarkupListBlockRule();
+ $blocks[] = new PhutilRemarkupInterpreterBlockRule();
+
+ foreach ($blocks as $block) {
+ if (!($block instanceof PhutilRemarkupCodeBlockRule)) {
+ $block->setMarkupRules($rules);
+ }
+ }
+
+ $engine->setBlockRules($blocks);
+
+ return $engine;
+ }
+
+}
+
+$handler = new RemarkupBlocksParser();
+$processor = new PhabricatorParserPrivateProcessor($handler);
+
+$socketName = $argv[1];
+echo "[PHP] socketName: ". $socketName . "\n";
+//$transport = new TBufferedTransport(new \Thrift\Transport\TSocketServer("unix://".$socketName));
+$transport = new \Thrift\Transport\TSocketServer("unix://".$socketName);
+$protocol = new TBinaryProtocol($transport, true, true);
+
+echo "[PHP] RemarkupBlocksParser:: loading.. \n";
+
+$transport->open();
+
+ for (;;) {
+ $processor->process($protocol, $protocol);
+ }
+
+$transport->close();
+
+
+
diff --git a/documentation-tools/remarkup/php/thrift-example.php b/documentation-tools/RemarkupParser/php/thrift-example.php
similarity index 100%
rename from documentation-tools/remarkup/php/thrift-example.php
rename to documentation-tools/RemarkupParser/php/thrift-example.php
diff --git a/documentation-tools/RemarkupParser/remarkup.thrift b/documentation-tools/RemarkupParser/remarkup.thrift
new file mode 100644
index 0000000..22ea2c2
--- /dev/null
+++ b/documentation-tools/RemarkupParser/remarkup.thrift
@@ -0,0 +1,11 @@
+struct Block {
+ 1: i32 start,
+ 2: i32 num_lines,
+ 3: string rule
+}
+
+typedef list<Block> Blocks
+
+service PhabricatorParserPrivate {
+ Blocks parse(1: string text)
+}
diff --git a/documentation-tools/remarkup/shared.thrift b/documentation-tools/RemarkupParser/shared.thrift
similarity index 100%
rename from documentation-tools/remarkup/shared.thrift
rename to documentation-tools/RemarkupParser/shared.thrift
diff --git a/documentation-tools/RemarkupParser/test/main.cpp b/documentation-tools/RemarkupParser/test/main.cpp
new file mode 100644
index 0000000..5361187
--- /dev/null
+++ b/documentation-tools/RemarkupParser/test/main.cpp
@@ -0,0 +1,46 @@
+#include <iostream>
+
+#include <thrift/protocol/TBinaryProtocol.h>
+#include <thrift/transport/TSocket.h>
+#include <thrift/transport/TTransportUtils.h>
+
+#include "gen-cpp/PhabricatorParserPrivate.h"
+
+using namespace std;
+using namespace apache::thrift;
+using namespace apache::thrift::protocol;
+using namespace apache::thrift::transport;
+
+int main() {
+ boost::shared_ptr<TTransport> socket(new TSocket("/tmp/phabricator-parser"));
+ boost::shared_ptr<TTransport> transport(new TBufferedTransport(socket));
+ boost::shared_ptr<TProtocol> protocol(new TBinaryProtocol(transport));
+ PhabricatorParserPrivateClient client(protocol);
+
+ string text =
+"= Large Header = \n \
+ \n\
+== Smaller Header == \n\
+ \n\
+## This is a Header As Well\n\
+ \n\
+Also a Large Header \n\
+=================== \n\
+ \n\
+Also a Smaller Header \n\
+---------------------";
+ try {
+ transport->open();
+
+
+ for (int i=0; i<answer.size(); ++i){
+ Block b = answer[i];
+ cout << b.start << " " << b.num_lines << " " << b.rule << endl;
+ }
+
+ cout << answer.size();
+
+ } catch (TException& tx) {
+ cout << "ERROR: " << tx.what() << endl;
+ }
+}
diff --git a/documentation-tools/remarkup/tutorial.thrift b/documentation-tools/RemarkupParser/tutorial.thrift
similarity index 100%
rename from documentation-tools/remarkup/tutorial.thrift
rename to documentation-tools/RemarkupParser/tutorial.thrift
diff --git a/documentation-tools/conduit-client-example.cpp b/documentation-tools/conduit-client-example.cpp
new file mode 100644
index 0000000..04bae68
--- /dev/null
+++ b/documentation-tools/conduit-client-example.cpp
@@ -0,0 +1,16 @@
+#include <iostream>
+#include "phrictionclient.h"
+#include <QCoreApplication>
+
+using namespace std;
+
+int main(int argc, char** argv)
+{
+ QCoreApplication a(argc, argv);
+
+ PhrictionClient c("http://192.168.11.130:80/api", "user", "api-ko4x6j4hujleutuxlntdqj37wfjj", "bydxwfn7pey6ack274dilywrgakkhjpmwrgieedqxlnxpocwovisz3ysgndzagtmft33e2tx3xyatwklzbrrzhyyonibim7nwbnpqo36fmer7hjvv5mmmammxf3fybqn52dgsmczrj44e4l4r774ajyiydmcxughsttgonydjsgf3b2rmvtunjtez4zbtcqmwrhjl3aijv22jefvymu2oz3tzu3zlv4lk226riq5utudndd6ae4ixzqmzyo6pqr");
+ c.create();
+
+ return a.exec();
+}
+
diff --git a/documentation-tools/conduit-client/CMakeLists.txt b/documentation-tools/conduit-client/CMakeLists.txt
new file mode 100644
index 0000000..ac9fd1d
--- /dev/null
+++ b/documentation-tools/conduit-client/CMakeLists.txt
@@ -0,0 +1,16 @@
+project(conduitclient)
+cmake_minimum_required(VERSION 2.8)
+
+aux_source_directory(. SRC_LIST)
+set(CMAKE_BUILD_TYPE Debug)
+
+add_library(${PROJECT_NAME} SHARED ${SRC_LIST})
+target_link_libraries(${PROJECT_NAME} Qt5::Core Qt5::Network)
+
+set(CONDUIT_CLIENT_INCLUDE_PATH
+ ${CMAKE_CURRENT_SOURCE_DIR}/
+
+ CACHE INTERNAL "Conduit client: Include Directories" FORCE
+)
+
+message("CONDUIT_CLIENT_INCLUDE_PATH:" ${CONDUIT_CLIENT_INCLUDE_PATH})
diff --git a/documentation-tools/conduit-client/conduitclient.cpp b/documentation-tools/conduit-client/conduitclient.cpp
new file mode 100644
index 0000000..0b88e83
--- /dev/null
+++ b/documentation-tools/conduit-client/conduitclient.cpp
@@ -0,0 +1,104 @@
+#include "conduitclient.h"
+#include "QtNetwork/QNetworkAccessManager"
+#include <QtNetwork/QNetworkRequest>
+#include <QUrlQuery>
+#include <QJsonDocument>
+#include <ctime>
+#include <QEventLoop>
+#include <QEventLoopLocker>
+#include <QtNetwork/QNetworkReply>
+
+QEventLoopLocker* lockerWaitingResponce = 0;
+
+ConduitClient::~ConduitClient(){}
+
+ConduitClient::ConduitClient(const QString& uri, const QString& user, const QString& token, const QString& sertificate)
+ : QObject(nullptr), m_URI(uri), m_user(user), m_conduitToken(token), m_Sertificate(sertificate), m_manager(new QNetworkAccessManager()), m_loopWaitingResponce(new QEventLoop())
+{
+ connect(m_manager, SIGNAL(finished(QNetworkReply*)),
+ this, SLOT(replyReceived(QNetworkReply*)));
+}
+
+bool
+ConduitClient::createSession()
+{
+ const QString token = QString("%1").arg(time(0));
+ const QString signatureRaw = token + m_Sertificate;
+ QByteArray signatureHash = QCryptographicHash::hash(signatureRaw.toLatin1(), QCryptographicHash::Sha1);
+ QString signatureHashLatin = QString::fromLatin1(signatureHash.toHex());
+ QJsonObject query;
+ query["authToken"] = token;
+ query["authSignature"] =signatureHashLatin;
+ query["client"] = "juffed-client";
+ query["clientVersion"] = "1";
+ query["user"] = m_user;
+
+ QJsonObject reply = requestRaw("conduit.connect", query);
+
+ //Example: "result\":{\"connectionID\":17,\"sessionKey\":\"l6jih3u7kxz3i624tewq2fvvmvmjiwzzp5ddicnk\"
+ if (reply.contains("result")){
+ QJsonObject sessionData = reply.find("result").value().toObject();
+
+ if (sessionData.empty()) return false;
+ m_SessionKey = sessionData["sessionKey"].toString();
+ m_ConnectionID = sessionData["connectionID"].toInt();;
+ return true;
+ }
+
+ return false;
+}
+
+QJsonObject
+ConduitClient::request(const QString& method, const QJsonObject& json){
+ if (m_SessionKey.isNull()) {
+ if (!createSession()){
+ return QJsonObject();
+ }
+ }
+
+ QJsonObject metaQuery;
+ metaQuery["sessionKey"] = m_SessionKey;
+ metaQuery["connectionID"] = m_ConnectionID;
+ metaQuery["token"] = m_conduitToken;
+
+ QJsonObject query(json);
+ query["__conduit__"] = metaQuery;
+
+ return requestRaw(method, query);
+}
+
+void
+ConduitClient::replyReceived(QNetworkReply* reply){
+ m_serverReply = reply;
+
+ delete lockerWaitingResponce;
+ lockerWaitingResponce = 0;
+}
+
+QJsonObject
+ConduitClient::requestRaw(const QString& method, const QJsonObject& json)
+{
+ QUrl server(m_URI + "/" + method);
+ QUrlQuery query;
+
+ query.addQueryItem("output", "json");
+ query.addQueryItem("__conduit__", "true");
+ query.addQueryItem("params", QJsonDocument(json).toJson(QJsonDocument::Compact));
+
+ server.setQuery(query);
+
+ qDebug()<<server;
+ QNetworkRequest request(server);
+ m_manager->get(request);
+ lockerWaitingResponce = new QEventLoopLocker(m_loopWaitingResponce);
+ m_loopWaitingResponce->exec();
+
+ QString replyJson(m_serverReply->readAll());
+ qDebug() << replyJson;
+ delete m_serverReply;
+
+ return QJsonDocument::fromJson(replyJson.toUtf8()).object();
+}
+
+//void
+//ConduitClient::req
diff --git a/documentation-tools/conduit-client/conduitclient.h b/documentation-tools/conduit-client/conduitclient.h
new file mode 100644
index 0000000..bbdc1dc
--- /dev/null
+++ b/documentation-tools/conduit-client/conduitclient.h
@@ -0,0 +1,48 @@
+#ifndef CONDUITCLIENT_H
+#define CONDUITCLIENT_H
+
+#include <QtCore/QObject>
+#include <QString>
+#include <QJsonObject>
+
+class QNetworkAccessManager;
+class QNetworkReply;
+class QEventLoop;
+
+class ConduitClient: public QObject {
+ Q_OBJECT
+
+public:
+ ConduitClient(const QString& uri, const QString& user, const QString& token, const QString& sertificate);
+ ~ConduitClient();
+ QJsonObject request(const QString& method, const QJsonObject& json);
+ bool createSession();
+
+public slots:
+ void replyReceived(QNetworkReply*);
+
+private:
+ QString m_URI;
+ QString m_user;
+ QString m_conduitToken;
+ QString m_SessionKey;
+ int m_ConnectionID;
+ QString m_Sertificate;
+
+ QNetworkAccessManager* m_manager;
+ QNetworkReply* m_serverReply;
+ QEventLoop* m_loopWaitingResponce;
+
+ QJsonObject requestRaw(const QString& method, const QJsonObject& json);
+
+
+// private int? ;
+// ClientName
+// ClientVersion
+// User
+// Certificate
+
+
+};
+
+#endif // CONDUITCLIENT_H
diff --git a/documentation-tools/conduit-client/phrictionclient.cpp b/documentation-tools/conduit-client/phrictionclient.cpp
new file mode 100644
index 0000000..635228a
--- /dev/null
+++ b/documentation-tools/conduit-client/phrictionclient.cpp
@@ -0,0 +1,85 @@
+#include "phrictionclient.h"
+#include "conduitclient.h"
+
+PhrictionClient::PhrictionClient(const QString& uri, const QString& user, const QString& token, const QString& sertificate)
+ : transport (new ConduitClient(uri, user, token, sertificate))
+{
+}
+
+bool
+PhrictionClient::createDocument(QString slug, QString title, QString content, QString description){
+ //try to create
+ QJsonObject request{
+ {"slug", slug}
+ ,{"title", title}
+ ,{"content", content}
+ ,{"description", description}
+ };
+
+ QJsonObject answer= transport->request("phriction.create", request);
+ QJsonValue error = answer.value("error_code");
+ if (!error.isUndefined() && error.isNull()){
+ return true;
+ }
+
+ return false;
+}
+
+bool
+PhrictionClient::updateDocument(QString slug, QString title, QString content, QString description){
+ //try to create
+ QJsonObject request{
+ {"slug", slug}
+ ,{"title", title}
+ ,{"content", content}
+ ,{"description", description}
+ };
+
+ QJsonObject answer= transport->request("phriction.edit", request);
+ QJsonValue error = answer.value("error_code");
+ if (!error.isUndefined() && error.isNull()){
+ return true;
+ }
+
+ return false;
+}
+
+bool PhrictionClient::existsDocument(const QString& slug){
+ QJsonObject request{
+ {"slug", slug}
+ };
+
+ QJsonObject answer= transport->request("phriction.info", request);
+ QJsonValue status = answer.value("result").toObject().value("status");
+
+ if (status.toString() == "exists"){
+ return true;
+ }
+
+ return false;
+}
+
+bool
+PhrictionClient::publishDocument(QString slug, QString title, QString content, QString description){
+ if (existsDocument(slug)){
+ return updateDocument(slug, title, content, description);
+ }
+
+ QStringList slugParts = slug.split('/');
+ slugParts.removeLast();
+
+ QString slugParent;
+ for (QString part: slugParts){
+ slugParent += part + "/";
+ if (! existsDocument(slugParent)){
+ createDocument(slugParent, part, " ", "");
+ }
+ }
+
+ return createDocument(slug, title, content, description);
+}
+
+
+PhrictionClient::~PhrictionClient(){
+ delete transport;
+}
diff --git a/documentation-tools/conduit-client/phrictionclient.h b/documentation-tools/conduit-client/phrictionclient.h
new file mode 100644
index 0000000..3c6c24e
--- /dev/null
+++ b/documentation-tools/conduit-client/phrictionclient.h
@@ -0,0 +1,23 @@
+#ifndef PHRICTIONCLIENT_H
+#define PHRICTIONCLIENT_H
+#include <QString>
+
+class ConduitClient;
+
+class PhrictionClient
+{
+public:
+ PhrictionClient(const QString& uri, const QString& user, const QString& token, const QString& sertificate);
+ ~PhrictionClient();
+
+
+ bool publishDocument(QString slug, QString title, QString content, QString description);
+private:
+ ConduitClient* transport;
+
+ bool createDocument(QString slug, QString title, QString content, QString description);
+ bool updateDocument(QString slug, QString title, QString content, QString description);
+ bool existsDocument(const QString& slug);
+};
+
+#endif // PHRICTIONCLIENT_H
diff --git a/documentation-tools/juffed-lexers/qscilexerremarkup.cpp b/documentation-tools/juffed-lexers/qscilexerremarkup.cpp
new file mode 100644
index 0000000..1e3c3b5
--- /dev/null
+++ b/documentation-tools/juffed-lexers/qscilexerremarkup.cpp
@@ -0,0 +1,171 @@
+#include "qscilexerremarkup.h"
+#include <qsciscintilla.h>
+#include <QRegularExpression>
+
+enum
+ {
+ Default = 0,
+ Hidden,
+ Bold,
+ Italic,
+ Del,
+ DocumentLink,
+ EscapeRemarkup,
+ Hightlight,
+ HyperLink,
+
+ Monospace,
+ Underline
+ };
+
+QRegularExpression patternBold("\\*\\*(.+?)\\*\\*");
+QRegularExpression patternItalic("(?<!:)//(.+?)//");
+
+const char*
+QsciLexerRemarkup::language() const
+{
+ return "Remarkup";
+}
+
+QString
+QsciLexerRemarkup::description(int style) const
+{
+ switch(style){
+ case Default:
+ return "Default";
+ case Bold:
+ return "Bold";
+ case Italic:
+ return "Italic";
+ case Hidden:
+ return "Hidden";
+ }
+
+ return QString();
+}
+
+QColor
+QsciLexerRemarkup::defaultColor(int style) const
+{
+ switch(style){
+ case Default:
+ break;
+
+ case Bold:
+ return Qt::red;
+ }
+
+ return QsciLexer::defaultColor();
+}
+
+QFont
+QsciLexerRemarkup::defaultFont(int style) const
+{
+// QFont f = QsciLexer::defaultFont();
+
+// switch(style){
+// case Default:
+// break;
+
+// case Bold:
+// f.setBold(true);
+// f.setCapitalization(QFont::AllUppercase);
+// break;
+// }
+
+// return f;
+ return QsciLexer::defaultFont();
+}
+
+QColor
+QsciLexerRemarkup::defaultPaper(int style) const
+{
+ return QsciLexer::defaultPaper();
+}
+
+QsciStyle
+QsciLexerRemarkup::getStyle(int style)
+{
+ return QsciStyle(style, description(style), defaultColor(style),
+ defaultPaper(style), defaultFont(style));
+}
+
+QsciLexerRemarkup::QsciLexerRemarkup(QObject * parent)
+ :QsciLexerCustom(parent)
+{
+
+}
+
+QsciLexerRemarkup::~QsciLexerRemarkup()
+{}
+
+void
+QsciLexerRemarkup::styleText(int start, int end){
+ if (!editor()) return;
+ if (start == end) return;
+ QsciScintilla* e = editor();
+
+ //reset styles:
+ startStyling(start);
+ setStyling(end-start, Default);
+
+ char * chars = new char[end - start + 1];
+ e->SendScintilla(QsciScintilla::SCI_GETTEXTRANGE, start, end, chars);
+ QString source = QString::fromUtf8(chars);
+ delete [] chars;
+
+ runPhutilRemarkupBoldRule(source,start);
+ runPhutilRemarkupItalicRule(source,start);
+}
+
+void
+QsciLexerRemarkup::runPhutilRemarkupBoldRule(const QString& source, int index){
+ int pos =0;
+ while(true) {
+ QRegularExpressionMatch result = patternBold.match(source, pos);
+ if (!result.hasMatch()){break;}
+
+ //hide:
+ startStyling(index + result.capturedStart(0));
+ setStyling(result.capturedLength(0), Hidden);
+
+ QString r = result.captured(1);
+ startStyling(index + result.capturedStart(1));
+ setStyling(result.capturedLength(1), Bold);
+ pos = result.capturedEnd(1) + 1;
+ }
+}
+
+void
+QsciLexerRemarkup::runPhutilRemarkupItalicRule(const QString& source, int index){
+ int pos =0;
+ while(true) {
+ QRegularExpressionMatch result = patternItalic.match(source, pos);
+ if (!result.hasMatch()){break;}
+
+ //hide:
+ startStyling(index + result.capturedStart(0));
+ setStyling(result.capturedLength(0), Hidden);
+
+ startStyling(index + result.capturedStart(1));
+ setStyling(result.capturedLength(1), Italic);
+ pos = result.capturedEnd(1) + 1;
+ }
+}
+
+//"PhutilRemarkupBoldRule.php: **bold**
+//PhutilRemarkupItalicRule.php: //italic//
+
+/*
+"PhutilRemarkupDelRule.php"
+"PhutilRemarkupDocumentLinkRule.php"
+"PhutilRemarkupEscapeRemarkupRule.php"
+"PhutilRemarkupHighlightRule.php"
+"PhutilRemarkupHyperlinkRule.php"
+
+"PhutilRemarkupLinebreaksRule.php"
+"PhutilRemarkupMonospaceRule.php"
+"PhutilRemarkupRule.php"
+"PhutilRemarkupUnderlineRule.php"
+*/
+
diff --git a/documentation-tools/juffed-lexers/qscilexerremarkup.h b/documentation-tools/juffed-lexers/qscilexerremarkup.h
new file mode 100644
index 0000000..597319d
--- /dev/null
+++ b/documentation-tools/juffed-lexers/qscilexerremarkup.h
@@ -0,0 +1,29 @@
+#ifndef QSCILEXERREMARKUP_H
+#define QSCILEXERREMARKUP_H
+
+#include <qscilexercustom.h>
+#include <qscistyle.h>
+
+class QsciLexerRemarkup : public QsciLexerCustom
+{
+ Q_OBJECT
+
+public:
+ virtual QColor defaultColor(int style) const;
+ virtual QFont defaultFont(int style) const;
+ virtual QColor defaultPaper(int style) const;
+ QsciStyle getStyle(int style);
+
+ QsciLexerRemarkup(QObject * parent = 0);
+ ~QsciLexerRemarkup();
+
+ void styleText(int start, int end);
+ QString description(int style) const;
+ const char* language() const;
+
+private:
+ void runPhutilRemarkupBoldRule(const QString& source, int index);
+ void runPhutilRemarkupItalicRule(const QString& source, int index);
+};
+
+#endif // QSCILEXERREMARKUP_H
diff --git a/documentation-tools/juffed-lexers/remarkup.xml b/documentation-tools/juffed-lexers/remarkup.xml
new file mode 100644
index 0000000..b47b7ab
--- /dev/null
+++ b/documentation-tools/juffed-lexers/remarkup.xml
@@ -0,0 +1,19 @@
+<?xml version="1.0" encoding="UTF-8" ?>
+<!DOCTYPE JuffScheme>
+<JuffScheme>
+<scheme name="Remarkup">
+ <style name="Bold" color="#000000" bold="true" capitalization="true" />
+ <style name="Italic" color="#000000" italic="true" />
+ <style name="Hidden" color="#B0B0B0" />
+</scheme>
+
+<blockrules>
+ <style name="PhutilRemarkupHeaderBlockRule" color="#fce0b0" marker="Background"/>
+ <style name="PhutilRemarkupNoteBlockRuleNOTE" color="#9797ff" marker="FullRectangle"/>
+ <style name="PhutilRemarkupNoteBlockRuleWARNING" color="#0000FF" marker="FullRectangle"/>
+ <style name="PhutilRemarkupNoteBlockRuleIMPORTANT" color="#0000FF" marker="FullRectangle"/>
+ <style name="PhutilRemarkupNoteBlockRuleSYNTAX" color="#f2f8ff" marker="Background"/>
+ <style name="PhutilRemarkupCodeBlockRule" color="#00FF00" marker="FullRectangle"/>
+
+</blockrules>
+</JuffScheme>
diff --git a/documentation-tools/juffed-plugins/fm/CMakeLists.txt b/documentation-tools/juffed-plugins/fm/CMakeLists.txt
new file mode 100644
index 0000000..4e206d5
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/CMakeLists.txt
@@ -0,0 +1,15 @@
+# fm
+
+QT_ADD_RESOURCES(fm_RES fm.qrc)
+QT_WRAP_UI(fm_UI ManageDlg.ui)
+
+set(fm_SRC
+ ManageDlg.cpp
+ FMPlugin.cpp
+ TreeView.cpp
+ filesystemmodel.cpp
+ )
+
+add_library(fm SHARED ${fm_SRC} ${fm_RES} ${fm_UI})
+target_link_libraries(fm ${JUFFED_LIBRARY} ${JUFF_QT_IMPORTED_TARGETS})
+install(TARGETS fm DESTINATION ${JUFFED_PLUGINS_DIR})
diff --git a/documentation-tools/juffed-plugins/fm/COPYING b/documentation-tools/juffed-plugins/fm/COPYING
new file mode 100644
index 0000000..3931a89
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/COPYING
@@ -0,0 +1,281 @@
+
+ GNU GENERAL PUBLIC LICENSE
+ Version 2, June 1991
+
+ Copyright (C) 1989, 1991 Free Software Foundation, Inc.
+ 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA
+ Everyone is permitted to copy and distribute verbatim copies
+ of this license document, but changing it is not allowed.
+
+ Preamble
+
+ The licenses for most software are designed to take away your
+freedom to share and change it. By contrast, the GNU General Public
+License is intended to guarantee your freedom to share and change free
+software--to make sure the software is free for all its users. This
+General Public License applies to most of the Free Software
+Foundation's software and to any other program whose authors commit to
+using it. (Some other Free Software Foundation software is covered by
+the GNU Library General Public License instead.) You can apply it to
+your programs, too.
+
+ When we speak of free software, we are referring to freedom, not
+price. Our General Public Licenses are designed to make sure that you
+have the freedom to distribute copies of free software (and charge for
+this service if you wish), that you receive source code or can get it
+if you want it, that you can change the software or use pieces of it
+in new free programs; and that you know you can do these things.
+
+ To protect your rights, we need to make restrictions that forbid
+anyone to deny you these rights or to ask you to surrender the rights.
+These restrictions translate to certain responsibilities for you if you
+distribute copies of the software, or if you modify it.
+
+ For example, if you distribute copies of such a program, whether
+gratis or for a fee, you must give the recipients all the rights that
+you have. You must make sure that they, too, receive or can get the
+source code. And you must show them these terms so they know their
+rights.
+
+ We protect your rights with two steps: (1) copyright the software, and
+(2) offer you this license which gives you legal permission to copy,
+distribute and/or modify the software.
+
+ Also, for each author's protection and ours, we want to make certain
+that everyone understands that there is no warranty for this free
+software. If the software is modified by someone else and passed on, we
+want its recipients to know that what they have is not the original, so
+that any problems introduced by others will not reflect on the original
+authors' reputations.
+
+ Finally, any free program is threatened constantly by software
+patents. We wish to avoid the danger that redistributors of a free
+program will individually obtain patent licenses, in effect making the
+program proprietary. To prevent this, we have made it clear that any
+patent must be licensed for everyone's free use or not licensed at all.
+
+ The precise terms and conditions for copying, distribution and
+modification follow.
+
+ GNU GENERAL PUBLIC LICENSE
+ TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION
+
+ 0. This License applies to any program or other work which contains
+a notice placed by the copyright holder saying it may be distributed
+under the terms of this General Public License. The "Program", below,
+refers to any such program or work, and a "work based on the Program"
+means either the Program or any derivative work under copyright law:
+that is to say, a work containing the Program or a portion of it,
+either verbatim or with modifications and/or translated into another
+language. (Hereinafter, translation is included without limitation in
+the term "modification".) Each licensee is addressed as "you".
+
+Activities other than copying, distribution and modification are not
+covered by this License; they are outside its scope. The act of
+running the Program is not restricted, and the output from the Program
+is covered only if its contents constitute a work based on the
+Program (independent of having been made by running the Program).
+Whether that is true depends on what the Program does.
+
+ 1. You may copy and distribute verbatim copies of the Program's
+source code as you receive it, in any medium, provided that you
+conspicuously and appropriately publish on each copy an appropriate
+copyright notice and disclaimer of warranty; keep intact all the
+notices that refer to this License and to the absence of any warranty;
+and give any other recipients of the Program a copy of this License
+along with the Program.
+
+You may charge a fee for the physical act of transferring a copy, and
+you may at your option offer warranty protection in exchange for a fee.
+
+ 2. You may modify your copy or copies of the Program or any portion
+of it, thus forming a work based on the Program, and copy and
+distribute such modifications or work under the terms of Section 1
+above, provided that you also meet all of these conditions:
+
+ a) You must cause the modified files to carry prominent notices
+ stating that you changed the files and the date of any change.
+
+ b) You must cause any work that you distribute or publish, that in
+ whole or in part contains or is derived from the Program or any
+ part thereof, to be licensed as a whole at no charge to all third
+ parties under the terms of this License.
+
+ c) If the modified program normally reads commands interactively
+ when run, you must cause it, when started running for such
+ interactive use in the most ordinary way, to print or display an
+ announcement including an appropriate copyright notice and a
+ notice that there is no warranty (or else, saying that you provide
+ a warranty) and that users may redistribute the program under
+ these conditions, and telling the user how to view a copy of this
+ License. (Exception: if the Program itself is interactive but
+ does not normally print such an announcement, your work based on
+ the Program is not required to print an announcement.)
+
+These requirements apply to the modified work as a whole. If
+identifiable sections of that work are not derived from the Program,
+and can be reasonably considered independent and separate works in
+themselves, then this License, and its terms, do not apply to those
+sections when you distribute them as separate works. But when you
+distribute the same sections as part of a whole which is a work based
+on the Program, the distribution of the whole must be on the terms of
+this License, whose permissions for other licensees extend to the
+entire whole, and thus to each and every part regardless of who wrote it.
+
+Thus, it is not the intent of this section to claim rights or contest
+your rights to work written entirely by you; rather, the intent is to
+exercise the right to control the distribution of derivative or
+collective works based on the Program.
+
+In addition, mere aggregation of another work not based on the Program
+with the Program (or with a work based on the Program) on a volume of
+a storage or distribution medium does not bring the other work under
+the scope of this License.
+
+ 3. You may copy and distribute the Program (or a work based on it,
+under Section 2) in object code or executable form under the terms of
+Sections 1 and 2 above provided that you also do one of the following:
+
+ a) Accompany it with the complete corresponding machine-readable
+ source code, which must be distributed under the terms of Sections
+ 1 and 2 above on a medium customarily used for software interchange; or,
+
+ b) Accompany it with a written offer, valid for at least three
+ years, to give any third party, for a charge no more than your
+ cost of physically performing source distribution, a complete
+ machine-readable copy of the corresponding source code, to be
+ distributed under the terms of Sections 1 and 2 above on a medium
+ customarily used for software interchange; or,
+
+ c) Accompany it with the information you received as to the offer
+ to distribute corresponding source code. (This alternative is
+ allowed only for noncommercial distribution and only if you
+ received the program in object code or executable form with such
+ an offer, in accord with Subsection b above.)
+
+The source code for a work means the preferred form of the work for
+making modifications to it. For an executable work, complete source
+code means all the source code for all modules it contains, plus any
+associated interface definition files, plus the scripts used to
+control compilation and installation of the executable. However, as a
+special exception, the source code distributed need not include
+anything that is normally distributed (in either source or binary
+form) with the major components (compiler, kernel, and so on) of the
+operating system on which the executable runs, unless that component
+itself accompanies the executable.
+
+If distribution of executable or object code is made by offering
+access to copy from a designated place, then offering equivalent
+access to copy the source code from the same place counts as
+distribution of the source code, even though third parties are not
+compelled to copy the source along with the object code.
+
+ 4. You may not copy, modify, sublicense, or distribute the Program
+except as expressly provided under this License. Any attempt
+otherwise to copy, modify, sublicense or distribute the Program is
+void, and will automatically terminate your rights under this License.
+However, parties who have received copies, or rights, from you under
+this License will not have their licenses terminated so long as such
+parties remain in full compliance.
+
+ 5. You are not required to accept this License, since you have not
+signed it. However, nothing else grants you permission to modify or
+distribute the Program or its derivative works. These actions are
+prohibited by law if you do not accept this License. Therefore, by
+modifying or distributing the Program (or any work based on the
+Program), you indicate your acceptance of this License to do so, and
+all its terms and conditions for copying, distributing or modifying
+the Program or works based on it.
+
+ 6. Each time you redistribute the Program (or any work based on the
+Program), the recipient automatically receives a license from the
+original licensor to copy, distribute or modify the Program subject to
+these terms and conditions. You may not impose any further
+restrictions on the recipients' exercise of the rights granted herein.
+You are not responsible for enforcing compliance by third parties to
+this License.
+
+ 7. If, as a consequence of a court judgment or allegation of patent
+infringement or for any other reason (not limited to patent issues),
+conditions are imposed on you (whether by court order, agreement or
+otherwise) that contradict the conditions of this License, they do not
+excuse you from the conditions of this License. If you cannot
+distribute so as to satisfy simultaneously your obligations under this
+License and any other pertinent obligations, then as a consequence you
+may not distribute the Program at all. For example, if a patent
+license would not permit royalty-free redistribution of the Program by
+all those who receive copies directly or indirectly through you, then
+the only way you could satisfy both it and this License would be to
+refrain entirely from distribution of the Program.
+
+If any portion of this section is held invalid or unenforceable under
+any particular circumstance, the balance of the section is intended to
+apply and the section as a whole is intended to apply in other
+circumstances.
+
+It is not the purpose of this section to induce you to infringe any
+patents or other property right claims or to contest validity of any
+such claims; this section has the sole purpose of protecting the
+integrity of the free software distribution system, which is
+implemented by public license practices. Many people have made
+generous contributions to the wide range of software distributed
+through that system in reliance on consistent application of that
+system; it is up to the author/donor to decide if he or she is willing
+to distribute software through any other system and a licensee cannot
+impose that choice.
+
+This section is intended to make thoroughly clear what is believed to
+be a consequence of the rest of this License.
+
+ 8. If the distribution and/or use of the Program is restricted in
+certain countries either by patents or by copyrighted interfaces, the
+original copyright holder who places the Program under this License
+may add an explicit geographical distribution limitation excluding
+those countries, so that distribution is permitted only in or among
+countries not thus excluded. In such case, this License incorporates
+the limitation as if written in the body of this License.
+
+ 9. The Free Software Foundation may publish revised and/or new versions
+of the General Public License from time to time. Such new versions will
+be similar in spirit to the present version, but may differ in detail to
+address new problems or concerns.
+
+Each version is given a distinguishing version number. If the Program
+specifies a version number of this License which applies to it and "any
+later version", you have the option of following the terms and conditions
+either of that version or of any later version published by the Free
+Software Foundation. If the Program does not specify a version number of
+this License, you may choose any version ever published by the Free Software
+Foundation.
+
+ 10. If you wish to incorporate parts of the Program into other free
+programs whose distribution conditions are different, write to the author
+to ask for permission. For software which is copyrighted by the Free
+Software Foundation, write to the Free Software Foundation; we sometimes
+make exceptions for this. Our decision will be guided by the two goals
+of preserving the free status of all derivatives of our free software and
+of promoting the sharing and reuse of software generally.
+
+ NO WARRANTY
+
+ 11. BECAUSE THE PROGRAM IS LICENSED FREE OF CHARGE, THERE IS NO WARRANTY
+FOR THE PROGRAM, TO THE EXTENT PERMITTED BY APPLICABLE LAW. EXCEPT WHEN
+OTHERWISE STATED IN WRITING THE COPYRIGHT HOLDERS AND/OR OTHER PARTIES
+PROVIDE THE PROGRAM "AS IS" WITHOUT WARRANTY OF ANY KIND, EITHER EXPRESSED
+OR IMPLIED, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF
+MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE. THE ENTIRE RISK AS
+TO THE QUALITY AND PERFORMANCE OF THE PROGRAM IS WITH YOU. SHOULD THE
+PROGRAM PROVE DEFECTIVE, YOU ASSUME THE COST OF ALL NECESSARY SERVICING,
+REPAIR OR CORRECTION.
+
+ 12. IN NO EVENT UNLESS REQUIRED BY APPLICABLE LAW OR AGREED TO IN WRITING
+WILL ANY COPYRIGHT HOLDER, OR ANY OTHER PARTY WHO MAY MODIFY AND/OR
+REDISTRIBUTE THE PROGRAM AS PERMITTED ABOVE, BE LIABLE TO YOU FOR DAMAGES,
+INCLUDING ANY GENERAL, SPECIAL, INCIDENTAL OR CONSEQUENTIAL DAMAGES ARISING
+OUT OF THE USE OR INABILITY TO USE THE PROGRAM (INCLUDING BUT NOT LIMITED
+TO LOSS OF DATA OR DATA BEING RENDERED INACCURATE OR LOSSES SUSTAINED BY
+YOU OR THIRD PARTIES OR A FAILURE OF THE PROGRAM TO OPERATE WITH ANY OTHER
+PROGRAMS), EVEN IF SUCH HOLDER OR OTHER PARTY HAS BEEN ADVISED OF THE
+POSSIBILITY OF SUCH DAMAGES.
+
+ END OF TERMS AND CONDITIONS
diff --git a/documentation-tools/juffed-plugins/fm/FMPlugin.cpp b/documentation-tools/juffed-plugins/fm/FMPlugin.cpp
new file mode 100644
index 0000000..e9304ab
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/FMPlugin.cpp
@@ -0,0 +1,288 @@
+//TASK read project path Settings::instance()->setValue(plugin->name(), key, value);
+//TASK provide decorations(icons, etc) - model role (see QVariant QFileSystemModel::data, Qt::DecorationRole)
+
+#include "FMPlugin.h"
+
+#include <QCompleter>
+#include <QHeaderView>
+#include <QInputDialog>
+#include <QLineEdit>
+#include <QMessageBox>
+#include <QMenu>
+#include <QAction>
+#include <QToolBar>
+#include <QHBoxLayout>
+#include <QVBoxLayout>
+#include <QCheckBox>
+
+#include <Log.h>
+#include <MainSettings.h>
+#include <PluginSettings.h>
+#include <EditorSettings.h>
+
+#include "ManageDlg.h"
+#include "TreeView.h"
+#include "filesystemmodel.h"
+
+FMPlugin::FMPlugin() : QObject(), JuffPlugin() {
+ sortColumn = PluginSettings::getInt(this, "sortColumn", 0);
+
+ w_ = new QWidget();
+ w_->setWindowTitle(tr("Project"));
+
+ model_ = new FileSystemModel;
+
+ model_->setRootPath("/");
+ model_->setNameFilters(QStringList{"*.remarkup"});
+ model_->setNameFilterDisables(false);
+
+ // filter |= QDir::Hidden;
+ // model_->setFilter(filter);
+
+ tree_ = new TreeView(this, model_);
+ tree_->setModel(model_);
+ tree_->setDragDropMode(QAbstractItemView::DragOnly);
+ tree_->setAllColumnsShowFocus(true);
+ tree_->sortByColumn(sortColumn, Qt::AscendingOrder);
+ tree_->setSortingEnabled(true);
+ tree_->setSelectionMode(QAbstractItemView::SingleSelection);
+
+ //drag & drop
+ tree_->setDragEnabled(true);
+ tree_->viewport()->setAcceptDrops(true);
+ tree_->setDropIndicatorShown(true);
+ tree_->setDragDropMode(QAbstractItemView::InternalMove);
+ model_->setReadOnly(false);
+ tree_->setEditTriggers(QTreeView::NoEditTriggers);
+ connect(model_, &FileSystemModel::fileMoved, this, &FMPlugin::onFileMoved);
+ connect(model_, &QFileSystemModel::fileRenamed, this, &FMPlugin::onFileRenamed);
+
+ tree_->header()->resizeSection(0, 180);
+ tree_->header()->resizeSection(1, 80);
+ tree_->header()->resizeSection(2, 60);
+ tree_->header()->resizeSection(3, 180);
+ tree_->initMenu();
+
+ tree_->setRootIsDecorated(true);
+ tree_->setItemsExpandable(true);
+
+//TOOLBAR:
+ // QToolBar * toolBar = new QToolBar("File Browser Tools", w_);
+ //backBtn_ = toolBar->addAction(QIcon::fromTheme("go-previous", QIcon(":icon_back")), tr("Go Back"), this, SLOT(back()));
+
+ /*
+ if ( Settings::instance()->valueExists(plugin->name(), key) )
+ return Settings::instance()->stringValue(plugin->name(), key);
+ */
+
+ pathEd_ = new QLineEdit("/private/prg/code/xreate/documentation/");
+ pathEd_->setCompleter(new QCompleter(model_));
+ connect(pathEd_, SIGNAL(returnPressed()), SLOT(onNewProjectRoot()));
+
+ QVBoxLayout* vBox = new QVBoxLayout();
+ vBox->setMargin(0);
+ vBox->setSpacing(2);
+
+// vBox->addWidget(toolBar);
+ vBox->addWidget(pathEd_);
+ vBox->addWidget(tree_);
+
+ w_->setLayout(vBox);
+
+ connect(tree_, SIGNAL(doubleClicked(const QModelIndex&)), SLOT(itemDoubleClicked(const QModelIndex&)));
+ connect(&fsWatcher_, SIGNAL(directoryChanged(const QString&)), SLOT(onDirChanged(const QString&)));
+
+ cd("/private/prg/code/xreate/documentation/");
+
+ applySettings();
+}
+
+FMPlugin::~FMPlugin() {
+ w_->deleteLater();
+}
+
+QString FMPlugin::name() const {
+ return "Project";
+}
+
+QString FMPlugin::title() const {
+ return tr("Project");
+}
+
+QString FMPlugin::targetEngine() const {
+ return "all";
+}
+
+QString FMPlugin::description() const {
+ return "List of project files";
+}
+
+QWidgetList FMPlugin::dockList() const {
+ QWidgetList wList;
+ wList << w_;
+ return wList;
+}
+
+bool FMPlugin::dockVisible(QWidget* w) const {
+ if ( w == w_ ) {
+ return true;
+ }
+ else {
+ return JuffPlugin::dockVisible(w);
+ }
+}
+
+void FMPlugin::cd(const QString& path) {
+ if ( QFileInfo(path).isDir() ) {
+ // remove currently watching paths
+ QStringList dirs = fsWatcher_.directories();
+ if ( !dirs.isEmpty() ) {
+ fsWatcher_.removePaths(dirs);
+ }
+
+ tree_->setRootIndex(model_->index(path));
+ pathEd_->setText(path);
+ pathEd_->setToolTip(path);
+
+ // add new current path for watching
+ fsWatcher_.addPath(path);
+ }
+}
+
+void FMPlugin::itemDoubleClicked(const QModelIndex& index) {
+ QString path = model_->filePath(index);
+ if ( QFileInfo(path).isDir() ) {
+// cd(path);
+ }
+ else {
+ api()->openDoc(path);
+ }
+}
+
+void FMPlugin::onDirChanged(const QString&) {
+}
+
+/*
+void FMPlugin::curFileDir() {
+ Juff::Document* doc = api()->currentDocument();
+ if ( !doc->isNull() && !doc->isNoname() ) {
+ cd(QFileInfo(doc->fileName()).absolutePath());
+ }
+}
+*/
+
+
+/*
+void FMPlugin::newDir() {
+ QString newDirName = QInputDialog::getText(tree_, tr("New directory"), tr("Directory name"));
+ if (newDirName.isEmpty())
+ return;
+
+ QDir curDir(model_->filePath(tree_->rootIndex()));
+ if ( !curDir.mkdir(newDirName) ) {
+ QMessageBox::warning(tree_, tr("Warning"),
+ tr("Couldn't create a dir named '%1'").arg(newDirName));
+ }
+}
+*/
+
+
+void FMPlugin::onNewProjectRoot() {
+ if ( QFileInfo(pathEd_->text()).isDir() )
+ cd(pathEd_->text());
+ else
+ pathEd_->setText(model_->filePath(tree_->rootIndex()));
+}
+
+void FMPlugin::onDocSaved(const QString& fileName) {
+ Q_UNUSED(fileName);
+}
+
+void FMPlugin::applySettings() {
+// QPalette plt = tree_->palette();
+// plt.setColor(QPalette::Base, EditorSettings::get(EditorSettings::DefaultBgColor));
+// plt.setColor(QPalette::Text, EditorSettings::get(EditorSettings::DefaultFontColor));
+//// plt.setColor(QPalette::Highlight, EditorSettings::get(EditorSettings::SelectionBgColor));
+//// plt.setColor(QPalette::HighlightedText, TextDocSettings::selectionTextColor());
+// tree_->setPalette(plt);
+// pathEd_->setPalette(plt);
+
+// PluginSettings::set(this, "ShowAsTree", showAsTree);
+// tree_->setRootIsDecorated(showAsTree);
+// tree_->setItemsExpandable(showAsTree);
+
+// QDir::Filters filter = QDir::AllDirs | QDir::AllEntries | QDir::NoDotAndDotDot;
+// if (showHidden)
+// filter |= QDir::Hidden;
+// model_->setFilter(filter);
+}
+
+QWidget * FMPlugin::settingsPage() const
+{
+// QWidget * cfg = new QWidget();
+// QVBoxLayout * l = new QVBoxLayout();
+// cfg->setLayout(l);
+
+// QCheckBox * treeCheckBox = new QCheckBox(tr("Show Directory Structure in a Tree"), cfg);
+// treeCheckBox->setChecked(showAsTree);
+// connect(treeCheckBox, SIGNAL(toggled(bool)), this, SLOT(treeCheckBox_toggled(bool)));
+
+// QCheckBox *showHiddenBox = new QCheckBox(tr("Show Hidden Files and Directories"), cfg);
+// showHiddenBox->setChecked(showHidden);
+// connect(showHiddenBox, SIGNAL(toggled(bool)), this, SLOT(showHiddenBox_toggled(bool)));
+
+// l->addWidget(treeCheckBox);
+// l->addWidget(showHiddenBox);
+// l->addItem(new QSpacerItem(1, 1, QSizePolicy::Minimum, QSizePolicy::MinimumExpanding));
+// return cfg;
+
+ return 0;
+}
+
+void
+FMPlugin::onFileMoved(const QString &path, const QString &pathOld){
+ QStringList docs = api()->docList();
+ const QFileInfo& fileInfo(path);
+
+ if ( fileInfo.isDir()){
+ //move section
+ for(QString& file: docs) {
+ if (file.left(pathOld.length()).compare(pathOld) == 0){
+ //file affected
+ Juff::Document* docUpdated = api()->document(file);
+
+ const QString& fileNewName = file.replace(0, pathOld.length(), path);
+ docUpdated->setFileName(fileNewName);
+ }
+ }
+ } {
+ //move topic
+ for(const QString& file: docs) {
+ if (file.length() == pathOld.length() && file.compare(pathOld) == 0){
+ Juff::Document* docUpdated = api()->document(file);
+ docUpdated->setFileName(path);
+ break;
+ }
+ }
+ }
+}
+
+void
+FMPlugin::onFileRenamed(const QString &path, const QString &oldName, const QString &newName){
+ const QString& fileOldName = path + QDir::separator() + oldName;
+ const QString& fileNewName = path + QDir::separator() + newName;
+
+ QStringList docs = api()->docList();
+
+ for(const QString& file: docs) {
+ if (file.length() == fileOldName.length() && file.compare(fileOldName) == 0){
+ Juff::Document* docUpdated = api()->document(file);
+ docUpdated->setFileName(fileNewName);
+ break;
+ }
+ }
+}
+
+#if QT_VERSION < 0x050000
+Q_EXPORT_PLUGIN2(fm, FMPlugin)
+#endif
diff --git a/documentation-tools/juffed-plugins/fm/FMPlugin.h b/documentation-tools/juffed-plugins/fm/FMPlugin.h
new file mode 100644
index 0000000..373fec1
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/FMPlugin.h
@@ -0,0 +1,83 @@
+#ifndef _FM_PLUGIN_H_
+#define _FM_PLUGIN_H_
+
+class QLineEdit;
+class QAction;
+class TreeView;
+
+#include <QDebug>
+#include <QtCore/QObject>
+#include <QtCore/QStack>
+#include <QFileSystemModel>
+
+#include <JuffPlugin.h>
+
+class FileSystemModel;
+
+class FMPlugin : public QObject, public JuffPlugin {
+Q_OBJECT
+#if QT_VERSION >= 0x050000
+Q_PLUGIN_METADATA(IID JuffPlugin_iid)
+#endif
+Q_INTERFACES(JuffPlugin)
+public:
+ FMPlugin();
+ virtual ~FMPlugin();
+
+ // info
+ virtual QString name() const;
+ virtual QString title() const;
+ virtual QString description() const;
+ virtual QString targetEngine() const;
+
+ // controls
+ virtual QWidgetList dockList() const;
+ virtual bool dockVisible(QWidget*) const;
+
+ virtual void applySettings();
+ QWidget * settingsPage() const;
+
+protected:
+ virtual void onDocSaved(const QString&);
+
+protected slots:
+ void itemDoubleClicked(const QModelIndex&);
+ void home();
+ void up();
+ void back();
+ void curFileDir();
+ void favorites();
+ void newDir();
+ void addToFavorites();
+ void goToFavorite();
+ void manageFavorites();
+ void onNewProjectRoot();
+ void treeCheckBox_toggled(bool value);
+ void showHiddenBox_toggled(bool value);
+
+ void onDirChanged(const QString&);
+ void onFileMoved(const QString &path, const QString &pathOld);
+ void onFileRenamed(const QString &path, const QString &oldName, const QString &newName);
+
+private:
+ void cd(const QString&);
+ void initFavoritesMenu();
+
+ bool showAsTree;
+ bool showHidden;
+ int sortColumn;
+
+ QWidget* w_;
+ TreeView* tree_;
+ FileSystemModel* model_;
+ QLineEdit* pathEd_;
+ QAction* backBtn_;
+ QStack<QString> history_;
+ QStringList favorites_;
+ QMenu* favoritesMenu_;
+ QAction* addToFavoritesAct_;
+ QAction* manageFavoritesAct_;
+ QFileSystemWatcher fsWatcher_;
+};
+
+#endif
diff --git a/documentation-tools/juffed-plugins/fm/INSTALL b/documentation-tools/juffed-plugins/fm/INSTALL
new file mode 100644
index 0000000..cde010b
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/INSTALL
@@ -0,0 +1,7 @@
+Compilation:
+ $ qmake
+ $ make
+
+Installation:
+ $ mkdir ~/.config/juff/plugins
+ $ cp ../libfm.so ~/.config/juffed/plugins/
diff --git a/documentation-tools/juffed-plugins/fm/ManageDlg.cpp b/documentation-tools/juffed-plugins/fm/ManageDlg.cpp
new file mode 100644
index 0000000..511ee0f
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/ManageDlg.cpp
@@ -0,0 +1,30 @@
+#include "ManageDlg.h"
+
+#include <JuffPlugin.h>
+#include <PluginSettings.h>
+
+ManageDlg::ManageDlg(QWidget* parent, JuffPlugin* plugin) : QDialog(parent), plugin_(plugin) {
+ ui.setupUi(this);
+ QString str = PluginSettings::getString(plugin, "favorites");
+ if ( !str.isEmpty() )
+ favorites_ = str.split(";");
+
+ ui.favoritesList->addItems(favorites_);
+
+ connect(ui.deleteBtn, SIGNAL(clicked()), SLOT(deleteItem()));
+ connect(ui.closeBtn, SIGNAL(clicked()), SLOT(close()));
+}
+
+void ManageDlg::deleteItem() {
+ QListWidgetItem* item = ui.favoritesList->currentItem();
+ if ( NULL != item ) {
+ QString text = item->text();
+ favorites_.removeAll(text);
+ delete item;
+ }
+}
+
+void ManageDlg::close() {
+ PluginSettings::set(plugin_, "favorites", favorites_.join(";"));
+ accept();
+}
diff --git a/documentation-tools/juffed-plugins/fm/ManageDlg.h b/documentation-tools/juffed-plugins/fm/ManageDlg.h
new file mode 100644
index 0000000..83abdae
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/ManageDlg.h
@@ -0,0 +1,23 @@
+#ifndef __MANAGE_DLG_H__
+#define __MANAGE_DLG_H__
+
+#include "ui_ManageDlg.h"
+
+class JuffPlugin;
+
+class ManageDlg : public QDialog {
+Q_OBJECT
+public:
+ ManageDlg(QWidget*, JuffPlugin*);
+
+private slots:
+ void deleteItem();
+ void close();
+
+private:
+ Ui::ManageDlg ui;
+ QStringList favorites_;
+ JuffPlugin* plugin_;
+};
+
+#endif /* __MANAGE_DLG_H__ */
diff --git a/documentation-tools/juffed-plugins/fm/ManageDlg.ui b/documentation-tools/juffed-plugins/fm/ManageDlg.ui
new file mode 100644
index 0000000..07ad188
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/ManageDlg.ui
@@ -0,0 +1,51 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<ui version="4.0">
+ <class>ManageDlg</class>
+ <widget class="QDialog" name="ManageDlg">
+ <property name="geometry">
+ <rect>
+ <x>0</x>
+ <y>0</y>
+ <width>619</width>
+ <height>215</height>
+ </rect>
+ </property>
+ <property name="windowTitle">
+ <string>Manage favorites</string>
+ </property>
+ <layout class="QGridLayout" name="gridLayout">
+ <item row="0" column="0" rowspan="3">
+ <widget class="QListWidget" name="favoritesList"/>
+ </item>
+ <item row="0" column="1">
+ <widget class="QPushButton" name="deleteBtn">
+ <property name="text">
+ <string>Delete</string>
+ </property>
+ </widget>
+ </item>
+ <item row="1" column="1">
+ <spacer name="verticalSpacer">
+ <property name="orientation">
+ <enum>Qt::Vertical</enum>
+ </property>
+ <property name="sizeHint" stdset="0">
+ <size>
+ <width>20</width>
+ <height>138</height>
+ </size>
+ </property>
+ </spacer>
+ </item>
+ <item row="2" column="1">
+ <widget class="QPushButton" name="closeBtn">
+ <property name="text">
+ <string>Close</string>
+ </property>
+ </widget>
+ </item>
+ </layout>
+ </widget>
+ <resources/>
+ <connections/>
+</ui>
diff --git a/documentation-tools/juffed-plugins/fm/README b/documentation-tools/juffed-plugins/fm/README
new file mode 100644
index 0000000..ccf93b3
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/README
@@ -0,0 +1,12 @@
+FMPlugin
+Simple file manager
+
+Allows to navigate through your file system
+and open files without using a file open dialog.
+
+License:
+ GPLv2 (see COPYING for details).
+
+Homepage:
+ http://code.google.com/p/juffed-plugins/
+
diff --git a/documentation-tools/juffed-plugins/fm/TreeView.cpp b/documentation-tools/juffed-plugins/fm/TreeView.cpp
new file mode 100644
index 0000000..b6f9830
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/TreeView.cpp
@@ -0,0 +1,104 @@
+#include <QDebug>
+
+#include "TreeView.h"
+
+#include <QInputDialog>
+#include <QMessageBox>
+#include <QKeyEvent>
+#include <QHeaderView>
+#include <QMenu>
+#include <QFileSystemModel>
+
+#include <PluginSettings.h>
+#include <QFileSystemModel>
+
+TreeView::TreeView(JuffPlugin* plugin, QFileSystemModel* model, QWidget* parent)
+: QTreeView(parent)
+, plugin_(plugin)
+, __model(model)
+{
+ header()->installEventFilter(this);
+}
+
+void TreeView::initMenu() {
+ __menuSection = new QMenu(this);
+ __menuTopic = new QMenu(this);
+
+ __actionRename = __menuSection->addAction("Rename", this, SLOT(rename()));
+ __menuSection->addAction("Add section", this, SLOT(addChildSection()));
+ __menuSection->addAction("Add topic", this, SLOT(addChildTopic()));
+
+ __menuTopic->addAction("Rename", this, SLOT(rename()));
+
+}
+
+void TreeView::mousePressEvent(QMouseEvent * e){
+ __actionRename->setDisabled(false);
+
+ if (e->button() == Qt::RightButton){
+ QModelIndex currentIndex = this->indexAt(e->pos());
+ if (currentIndex.isValid()){
+ setCurrentIndex(currentIndex);
+
+ QString path = __model->filePath(currentIndex);
+ if ( QFileInfo(path).isDir() ) {
+ __menuSection->exec(QCursor::pos());
+ }
+ else {
+ __menuTopic->exec(QCursor::pos());
+ }
+ return;
+ }
+
+ setCurrentIndex(rootIndex());
+ __actionRename->setDisabled(true);
+ __menuSection->exec(QCursor::pos());
+ }
+
+ QTreeView::mousePressEvent(e);
+}
+
+void TreeView::rename() {
+ this->edit(currentIndex());
+// QFileSystemModel* dirModel = qobject_cast<QFileSystemModel*>(model());
+// if ( 0 != dirModel ) {
+// QFileInfo fi = dirModel->fileInfo(currentIndex());
+// QString newFileName = QInputDialog::getText(this, tr("Rename"), tr("File name"),
+// QLineEdit::Normal, fi.baseName());
+// if (newFileName.isEmpty())
+// return;
+
+// QString oldFileName = fi.absoluteFilePath();
+// QFile file(oldFileName);
+
+// QDir::setCurrent(fi.absolutePath());
+// if ( !file.rename(newFileName + ".remarkup") ) {
+// QMessageBox::warning(this, tr("Warning"),
+// tr("Rename failed: file '%1' already exists").arg(newFileName));
+// } else {
+// emit fileMoved(fi.absoluteDir().absolutePath() + QDir::separator() + newFileName + ".remarkup", oldFileName);
+// }
+// }
+}
+
+void TreeView::addChildTopic(){
+ QString newTopicName = QInputDialog::getText(this, tr("New topic"), tr("Topic name"));
+ if (newTopicName.isEmpty())
+ return;
+
+ QFile file(__model->filePath(currentIndex()) + '/' + newTopicName + ".remarkup");
+ file.open(QIODevice::WriteOnly);
+ file.close();
+}
+
+void TreeView::addChildSection(){
+ QString newDirName = QInputDialog::getText(this, tr("New section"), tr("Section name"));
+ if (newDirName.isEmpty())
+ return;
+
+ QDir curDir(__model->filePath(currentIndex()));
+ if ( !curDir.mkdir(newDirName) ) {
+ QMessageBox::warning(this, tr("Warning"),
+ tr("Couldn't create a section named '%1'").arg(newDirName));
+ }
+}
diff --git a/documentation-tools/juffed-plugins/fm/TreeView.h b/documentation-tools/juffed-plugins/fm/TreeView.h
new file mode 100644
index 0000000..cc7f6f0
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/TreeView.h
@@ -0,0 +1,42 @@
+#ifndef _FM_TREE_VIEW_H_
+#define _FM_TREE_VIEW_H_
+
+class JuffPlugin;
+class QMenu;
+
+#include <QTreeView>
+
+QT_BEGIN_NAMESPACE
+ class QFileSystemModel;
+QT_END_NAMESPACE
+
+class TreeView : public QTreeView {
+Q_OBJECT
+public:
+ TreeView(JuffPlugin*, QFileSystemModel* model, QWidget* parent = 0);
+ void initMenu();
+
+public slots:
+ void rename();
+ void addChildSection();
+ void addChildTopic();
+
+signals:
+ void fileMoved(const QString& newFileName, const QString& oldFileName);
+
+protected:
+ virtual void mousePressEvent(QMouseEvent * event) override;
+
+private:
+ JuffPlugin* plugin_;
+ QMenu* __menuSection;
+ QMenu* __menuTopic;
+ QAction* __actionRename;
+
+ QFileSystemModel* __model;
+
+
+
+};
+
+#endif
diff --git a/documentation-tools/juffed-plugins/fm/bookmark-new.png b/documentation-tools/juffed-plugins/fm/bookmark-new.png
new file mode 100644
index 0000000..6cf6443
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/bookmark-new.png differ
diff --git a/documentation-tools/juffed-plugins/fm/filesystemmodel.cpp b/documentation-tools/juffed-plugins/fm/filesystemmodel.cpp
new file mode 100644
index 0000000..35b9663
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/filesystemmodel.cpp
@@ -0,0 +1,25 @@
+#include "filesystemmodel.h"
+
+#include <QMimeData>
+#include <QUrl>
+
+FileSystemModel::FileSystemModel(): QFileSystemModel() {}
+
+bool FileSystemModel::dropMimeData(const QMimeData *data
+ , Qt::DropAction action
+ ,int row
+ , int column
+ , const QModelIndex &parent) {
+
+ bool success = QFileSystemModel::dropMimeData(data, action, row, column, parent);
+
+ if (success){
+ QString pathTo = filePath(parent) + QDir::separator();
+ QString pathOld = data->urls().begin()->toLocalFile();
+ QString pathNew = pathTo + QFileInfo(pathOld).fileName();
+
+ emit fileMoved(pathNew, pathOld);
+ }
+
+ return success;
+}
diff --git a/documentation-tools/juffed-plugins/fm/filesystemmodel.h b/documentation-tools/juffed-plugins/fm/filesystemmodel.h
new file mode 100644
index 0000000..9168067
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/filesystemmodel.h
@@ -0,0 +1,16 @@
+#ifndef FILESYSTEMMODEL_H
+#define FILESYSTEMMODEL_H
+#include <QFileSystemModel>
+
+class FileSystemModel: public QFileSystemModel {
+ Q_OBJECT
+
+ public:
+ FileSystemModel();
+ bool dropMimeData(const QMimeData *data, Qt::DropAction action, int row, int column, const QModelIndex &parent) Q_DECL_OVERRIDE;
+
+signals:
+ void fileMoved(const QString& path, const QString& pathOld);
+
+};
+#endif // FILESYSTEMMODEL_H
diff --git a/documentation-tools/juffed-plugins/fm/fm.pro b/documentation-tools/juffed-plugins/fm/fm.pro
new file mode 100644
index 0000000..2a32944
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/fm.pro
@@ -0,0 +1,8 @@
+include(../juffed-plugins.pri)
+
+TARGET = fm
+
+HEADERS = FMPlugin.h TreeView.h ManageDlg.h
+SOURCES = FMPlugin.cpp TreeView.cpp ManageDlg.cpp
+FORMS = ManageDlg.ui
+RESOURCES = fm.qrc
diff --git a/documentation-tools/juffed-plugins/fm/fm.qrc b/documentation-tools/juffed-plugins/fm/fm.qrc
new file mode 100644
index 0000000..4ebc4af
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/fm.qrc
@@ -0,0 +1,10 @@
+<RCC>
+ <qresource>
+ <file alias="icon_back">go-previous.png</file>
+ <file alias="icon_up">go-up.png</file>
+ <file alias="icon_home">go-home.png</file>
+ <file alias="icon_current">go-current.png</file>
+ <file alias="icon_bookmarks">bookmark-new.png</file>
+ <file alias="icon_new_dir">folder-new.png</file>
+ </qresource>
+</RCC>
diff --git a/documentation-tools/juffed-plugins/fm/folder-new.png b/documentation-tools/juffed-plugins/fm/folder-new.png
new file mode 100644
index 0000000..628f4d5
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/folder-new.png differ
diff --git a/documentation-tools/juffed-plugins/fm/go-current.png b/documentation-tools/juffed-plugins/fm/go-current.png
new file mode 100644
index 0000000..5fe9df0
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/go-current.png differ
diff --git a/documentation-tools/juffed-plugins/fm/go-home.png b/documentation-tools/juffed-plugins/fm/go-home.png
new file mode 100644
index 0000000..a46fb22
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/go-home.png differ
diff --git a/documentation-tools/juffed-plugins/fm/go-previous.png b/documentation-tools/juffed-plugins/fm/go-previous.png
new file mode 100644
index 0000000..659cd90
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/go-previous.png differ
diff --git a/documentation-tools/juffed-plugins/fm/go-up.png b/documentation-tools/juffed-plugins/fm/go-up.png
new file mode 100644
index 0000000..fa9a7d7
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/go-up.png differ
diff --git a/documentation-tools/juffed-plugins/fm/install.sh b/documentation-tools/juffed-plugins/fm/install.sh
new file mode 100755
index 0000000..c57a938
--- /dev/null
+++ b/documentation-tools/juffed-plugins/fm/install.sh
@@ -0,0 +1,3 @@
+#!/bin/sh
+
+cp ../libfm.so ~/.config/juff/plugins/
diff --git a/documentation-tools/juffed-plugins/fm/view-refresh.png b/documentation-tools/juffed-plugins/fm/view-refresh.png
new file mode 100644
index 0000000..3fd71d6
Binary files /dev/null and b/documentation-tools/juffed-plugins/fm/view-refresh.png differ
diff --git a/documentation-tools/juffed-plugins/phapublisher/CMakeLists.txt b/documentation-tools/juffed-plugins/phapublisher/CMakeLists.txt
new file mode 100644
index 0000000..13f762d
--- /dev/null
+++ b/documentation-tools/juffed-plugins/phapublisher/CMakeLists.txt
@@ -0,0 +1,14 @@
+set(SRC
+ phapublisher.cpp
+)
+
+qt5_wrap_ui(UI
+ publishersettings.ui
+)
+
+INCLUDE_DIRECTORIES(${CONDUIT_CLIENT_INCLUDE_PATH})
+add_library(phapublisher SHARED ${SRC} ${UI})
+target_link_libraries(phapublisher conduitclient ${JUFFED_LIBRARY} ${JUFF_QT_IMPORTED_TARGETS})
+install(TARGETS phapublisher DESTINATION ${JUFFED_PLUGINS_DIR})
+
+message("UI files: " ${UI})
diff --git a/documentation-tools/juffed-plugins/phapublisher/phapublisher.cpp b/documentation-tools/juffed-plugins/phapublisher/phapublisher.cpp
new file mode 100644
index 0000000..5d9f80e
--- /dev/null
+++ b/documentation-tools/juffed-plugins/phapublisher/phapublisher.cpp
@@ -0,0 +1,112 @@
+//TEST check publishing with intermdeiate non existent categories
+
+#include "phapublisher.h"
+#include "phrictionclient.h"
+#include "ui_publishersettings.h"
+#include "PluginSettings.h"
+#include <QMessageBox>
+#include <QFileInfo>
+class UiSettings : public QWidget, public Ui::Form{};
+
+void
+PhaPublisher::publishCurrentPage(){
+ Juff::Document* doc = api()->currentDocument();
+ QString filename = doc->fileName();
+ if (!filename.startsWith(cfgRoot)){
+ QMessageBox::critical(doc, "Phabricator Publisher", "Current document located not in scope of a project", QMessageBox::Ok);
+ }
+
+ QFileInfo finfo(filename);
+
+ QString slug;
+ if (finfo.baseName()=="index"){
+ slug = filename.mid(cfgRoot.length(), filename.length()-cfgRoot.length()-finfo.fileName().length()-1);
+ } else {
+ slug = filename.mid(cfgRoot.length(), filename.length()-cfgRoot.length()-finfo.completeSuffix().length()-1);
+ }
+
+ QString title = finfo.baseName();
+ QString content;
+ doc->getText(content);
+
+ if (transport->publishDocument(slug, title, content, "")){
+ QMessageBox::information(nullptr, "Phabricator publisher", "Page published successfully");
+ }
+}
+
+PhaPublisher::PhaPublisher() : QObject(), JuffPlugin()
+ ,uiSettings(new UiSettings()), transport(0)
+{
+ cfgUrl = PluginSettings::getString(this, "url", "http://localhost/");
+ cfgUser = PluginSettings::getString(this, "user", "pgess");
+ cfgCertificate = PluginSettings::getString(this, "certificate", "");
+ cfgToken = PluginSettings::getString(this, "token", "");
+ cfgRoot = PluginSettings::getString(this, "root", "~");
+
+ uiSettings->setupUi(uiSettings);
+ uiSettings->uiUrl->setText(cfgUrl);
+ uiSettings->uiUser->setText(cfgUser);
+ uiSettings->uiSertificate->setPlainText(cfgCertificate);
+ uiSettings->uiToken->setText(cfgToken);
+ uiSettings->uiRoot->setText(cfgRoot);
+}
+
+QWidget*
+PhaPublisher::settingsPage() const{
+ return uiSettings;
+}
+
+void PhaPublisher::applySettings(){
+ if (!uiSettings) return;
+
+ cfgUrl = uiSettings->uiUrl->text();
+ cfgUser = uiSettings->uiUser->text();
+ cfgCertificate = uiSettings->uiSertificate->toPlainText();
+ cfgToken = uiSettings->uiToken->text();
+ cfgRoot = uiSettings->uiRoot->text();
+
+ PluginSettings::set(this, "url", cfgUrl);
+ PluginSettings::set(this, "user", cfgUser);
+ PluginSettings::set(this, "certificate", cfgCertificate);
+ PluginSettings::set(this, "token", cfgToken);
+ PluginSettings::set(this, "root", cfgRoot);
+
+ delete transport;
+ transport = new PhrictionClient(cfgUrl, cfgUser, cfgToken, cfgCertificate);
+}
+
+void
+PhaPublisher::init(){
+ //transport = new PhrictionClient();
+}
+
+QAction* createAction(const QString& title, const QKeySequence& key, const QObject* this_, const char* slot) {
+ QAction* act = new QAction(title, 0);
+ this_->connect(act, SIGNAL(triggered()), this_, slot);
+ act->setShortcut(key);
+ return act;
+}
+
+// info
+QString PhaPublisher::name()const {return "publisher";}
+QString PhaPublisher::title() const {return "Phabricator publisher";}
+QString PhaPublisher::description() const {return "Phabricator publisher";}
+QString PhaPublisher::targetEngine() const{ return "all";}
+
+
+Juff::ActionList
+PhaPublisher::mainMenuActions(Juff::MenuID id) const {
+ Juff::ActionList list;
+
+ if ( id == Juff::MenuEdit ) {
+ list << createAction(tr("publish"), QKeySequence("Ctrl+Shift+B"), this, SLOT(publishCurrentPage()));
+ }
+ return list;
+}
+
+PhaPublisher::~PhaPublisher(){
+ delete transport;
+ //delete uiSettings;
+ uiSettings->deleteLater();
+}
+
diff --git a/documentation-tools/juffed-plugins/phapublisher/phapublisher.h b/documentation-tools/juffed-plugins/phapublisher/phapublisher.h
new file mode 100644
index 0000000..efad109
--- /dev/null
+++ b/documentation-tools/juffed-plugins/phapublisher/phapublisher.h
@@ -0,0 +1,45 @@
+#ifndef PHAPUBLISHER_H
+#define PHAPUBLISHER_H
+#include <QtCore/QObject>
+#include "JuffPlugin.h"
+
+class PhrictionClient;
+class UiSettings;
+
+class PhaPublisher : public QObject, public JuffPlugin
+{
+ Q_OBJECT
+ Q_PLUGIN_METADATA(IID JuffPlugin_iid)
+ Q_INTERFACES(JuffPlugin)
+
+public:
+ explicit PhaPublisher();
+ ~PhaPublisher();
+ void init();
+
+ // info
+ virtual QString name() const;
+ virtual QString title() const;
+ virtual QString description() const;
+ virtual QString targetEngine() const;
+ Juff::ActionList mainMenuActions(Juff::MenuID id) const;
+ virtual QWidget* settingsPage() const override;
+ virtual void applySettings();
+signals:
+
+public slots:
+ void publishCurrentPage();
+
+
+private:
+ PhrictionClient* transport;
+ UiSettings* uiSettings;
+
+ QString cfgUrl;
+ QString cfgUser;
+ QString cfgCertificate;
+ QString cfgToken;
+ QString cfgRoot;
+};
+
+#endif // PHAPUBLISHER_H
diff --git a/documentation-tools/juffed-plugins/phapublisher/publishersettings.ui b/documentation-tools/juffed-plugins/phapublisher/publishersettings.ui
new file mode 100644
index 0000000..bf1ed5b
--- /dev/null
+++ b/documentation-tools/juffed-plugins/phapublisher/publishersettings.ui
@@ -0,0 +1,138 @@
+<?xml version="1.0" encoding="UTF-8"?>
+<ui version="4.0">
+ <class>Form</class>
+ <widget class="QWidget" name="Form">
+ <property name="geometry">
+ <rect>
+ <x>0</x>
+ <y>0</y>
+ <width>598</width>
+ <height>451</height>
+ </rect>
+ </property>
+ <property name="windowTitle">
+ <string>Form</string>
+ </property>
+ <layout class="QVBoxLayout" name="verticalLayout">
+ <item>
+ <widget class="QGroupBox" name="groupBox_2">
+ <property name="title">
+ <string>Publisher settings</string>
+ </property>
+ <layout class="QFormLayout" name="formLayout">
+ <item row="0" column="0">
+ <widget class="QLabel" name="label_5">
+ <property name="text">
+ <string>Documentation root:</string>
+ </property>
+ </widget>
+ </item>
+ <item row="0" column="1">
+ <widget class="QLineEdit" name="uiRoot"/>
+ </item>
+ <item row="1" column="1">
+ <widget class="QPushButton" name="btnChooseFile">
+ <property name="text">
+ <string>Choose</string>
+ </property>
+ <property name="flat">
+ <bool>false</bool>
+ </property>
+ </widget>
+ </item>
+ </layout>
+ </widget>
+ </item>
+ <item>
+ <widget class="QGroupBox" name="groupBox">
+ <property name="title">
+ <string>Phabricator settings</string>
+ </property>
+ <layout class="QFormLayout" name="formLayout_2">
+ <property name="topMargin">
+ <number>0</number>
+ </property>
+ <item row="0" column="0">
+ <widget class="QLabel" name="label">
+ <property name="sizePolicy">
+ <sizepolicy hsizetype="Preferred" vsizetype="Preferred">
+ <horstretch>0</horstretch>
+ <verstretch>0</verstretch>
+ </sizepolicy>
+ </property>
+ <property name="text">
+ <string>Url:</string>
+ </property>
+ <property name="alignment">
+ <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
+ </property>
+ </widget>
+ </item>
+ <item row="0" column="1">
+ <widget class="QLineEdit" name="uiUrl"/>
+ </item>
+ <item row="1" column="0">
+ <widget class="QLabel" name="label_2">
+ <property name="sizePolicy">
+ <sizepolicy hsizetype="MinimumExpanding" vsizetype="Expanding">
+ <horstretch>0</horstretch>
+ <verstretch>0</verstretch>
+ </sizepolicy>
+ </property>
+ <property name="text">
+ <string>User:</string>
+ </property>
+ <property name="alignment">
+ <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
+ </property>
+ </widget>
+ </item>
+ <item row="1" column="1">
+ <widget class="QLineEdit" name="uiUser"/>
+ </item>
+ <item row="2" column="0">
+ <widget class="QLabel" name="label_3">
+ <property name="sizePolicy">
+ <sizepolicy hsizetype="MinimumExpanding" vsizetype="Expanding">
+ <horstretch>0</horstretch>
+ <verstretch>0</verstretch>
+ </sizepolicy>
+ </property>
+ <property name="text">
+ <string>Token:</string>
+ </property>
+ <property name="alignment">
+ <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
+ </property>
+ </widget>
+ </item>
+ <item row="2" column="1">
+ <widget class="QLineEdit" name="uiToken"/>
+ </item>
+ <item row="3" column="0">
+ <widget class="QLabel" name="label_4">
+ <property name="sizePolicy">
+ <sizepolicy hsizetype="MinimumExpanding" vsizetype="Expanding">
+ <horstretch>0</horstretch>
+ <verstretch>0</verstretch>
+ </sizepolicy>
+ </property>
+ <property name="text">
+ <string>Certificate:</string>
+ </property>
+ <property name="alignment">
+ <set>Qt::AlignRight|Qt::AlignTrailing|Qt::AlignVCenter</set>
+ </property>
+ </widget>
+ </item>
+ <item row="3" column="1">
+ <widget class="QPlainTextEdit" name="uiSertificate"/>
+ </item>
+ </layout>
+ </widget>
+ </item>
+ </layout>
+ </widget>
+ <resources/>
+ <connections/>
+</ui>
diff --git a/documentation-tools/juffed-plugins/remarkupplugin/CMakeLists.txt b/documentation-tools/juffed-plugins/remarkupplugin/CMakeLists.txt
new file mode 100644
index 0000000..0d3ebb5
--- /dev/null
+++ b/documentation-tools/juffed-plugins/remarkupplugin/CMakeLists.txt
@@ -0,0 +1,11 @@
+set(SRC
+ remarkupplugin.cpp
+ remarkupparserthread.cpp
+)
+
+message ( STATUS "Ramarkup dirs: " ${REMARKUP_PARSER_INCLUDE_DIRS})
+
+INCLUDE_DIRECTORIES(${REMARKUP_PARSER_INCLUDE_DIRS})
+add_library(remarkupplugin SHARED ${SRC})
+target_link_libraries(remarkupplugin RemarkupParser ${JUFFED_LIBRARY} ${JUFF_QT_IMPORTED_TARGETS})
+install(TARGETS remarkupplugin DESTINATION ${JUFFED_PLUGINS_DIR})
diff --git a/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.cpp b/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.cpp
new file mode 100644
index 0000000..dc575df
--- /dev/null
+++ b/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.cpp
@@ -0,0 +1,85 @@
+#include "remarkupparserthread.h"
+#include "JuffScintilla.h"
+#include <QMutexLocker>
+#include <QDebug>
+
+//TASK queue texts of different docs for parsing. disconnect OnActivated event
+void
+RemarkupParserThread::run(){
+ exec();
+}
+
+void
+RemarkupParserThread::parseText(){
+ qDebug() << "RemarkupParserThread::run";
+
+ if (! parser->isConnected()){
+ if (!parser->connect()){
+ qDebug()<<"Can't connect.. sleep";
+ emit timerRestart();
+ return;
+ }
+ }
+
+ Juff::JuffScintilla* doc = getDoc();
+ if (doc == nullptr) return;
+
+ QString text = doc->text();
+ if (text.isEmpty()) return;
+
+ Blocks output = parser->parse(text);
+
+// for (int i=0; i<output.size(); ++i){
+// const Block& block = output.at(i);
+// qDebug() << QString("%1: [%2 - %3]").arg(QString::fromStdString(block.rule)).arg(block.start).arg(block.start+block.num_lines -1);
+// }
+
+ qDebug() << "RemarkupParserThread::text parsed. Sleeping..";
+ Juff::JuffScintilla* recheckDoc = getDoc();
+
+ //if doc is changed discard parsed tree
+ if (recheckDoc !=doc) return;
+
+ emit updateDoc(output, doc);
+}
+
+RemarkupParserThread::RemarkupParserThread(QObject *parent)
+ : QThread(parent)
+{
+ qRegisterMetaType <Blocks> ( "Blocks");
+
+ timerLagging.setInterval(800);
+ timerLagging.setSingleShot(true);
+ connect(&timerLagging, SIGNAL(timeout()), this, SLOT(parseText()));
+ connect(this, SIGNAL(timerRestart()), &timerLagging, SLOT(start()));
+ connect(this, SIGNAL(timerStop()), &timerLagging, SLOT(stop()));
+
+ parser = new RemarkupParser();
+}
+
+RemarkupParserThread::~RemarkupParserThread(){
+ delete parser;
+}
+
+Juff::JuffScintilla*
+RemarkupParserThread::getDoc(){
+ QMutexLocker lock(&docLock);
+
+ return this->doc;
+}
+
+void
+RemarkupParserThread::processDoc(Juff::JuffScintilla* doc){
+ QMutexLocker lock(&docLock);
+ this->doc = doc;
+
+ emit timerRestart();
+}
+
+void RemarkupParserThread::stop(){
+ QMutexLocker lock(&docLock);
+ this->doc = nullptr;
+
+ emit timerStop();
+}
+
diff --git a/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.h b/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.h
new file mode 100644
index 0000000..833dbd7
--- /dev/null
+++ b/documentation-tools/juffed-plugins/remarkupplugin/remarkupparserthread.h
@@ -0,0 +1,43 @@
+#ifndef REMARKUPPARSERTHREAD_H
+#define REMARKUPPARSERTHREAD_H
+#include "remarkupparser.h"
+
+#include <QObject>
+#include <QMutex>
+#include <QTimer>
+#include <QThread>
+
+namespace Juff {
+ class JuffScintilla;
+};
+
+
+class RemarkupParserThread : public QThread
+{
+ Q_OBJECT
+public:
+ explicit RemarkupParserThread(QObject *parent = 0);
+ ~RemarkupParserThread();
+ void processDoc(Juff::JuffScintilla* doc);
+ void stop();
+
+signals:
+ void updateDoc(const Blocks& blocks, Juff::JuffScintilla* doc);
+ void timerRestart();
+ void timerStop();
+
+public slots:
+ void parseText();
+
+private:
+ //RemarkupParserThread* __instance;
+ QMutex docLock;
+ QTimer timerLagging;
+ Juff::JuffScintilla* doc;
+ RemarkupParser* parser;
+
+ Juff::JuffScintilla* getDoc();
+ void run();
+};
+
+#endif // REMARKUPPARSERTHREAD_H
diff --git a/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.cpp b/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.cpp
new file mode 100644
index 0000000..8ef8c70
--- /dev/null
+++ b/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.cpp
@@ -0,0 +1,281 @@
+#include "remarkupplugin.h"
+#include "remarkupparserthread.h"
+
+#include "SciDoc.h"
+#include "JuffScintilla.h"
+#include <QDebug>
+#include <QMap>
+#include <QDomDocument>
+#include "AppInfo.h"
+#include "LexerStorage.h"
+
+using namespace Juff;
+
+//"PhutilRemarkupDefaultBlockRule",
+
+//"PhutilRemarkupHeaderBlockRule",
+//"PhutilRemarkupListBlockRule",
+//"PhutilRemarkupNoteBlockRule",
+
+//"PhutilRemarkupHorizontalRuleBlockRule",
+//"PhutilRemarkupReplyBlockRule",
+//"PhutilRemarkupCodeBlockRule",
+//"PhutilRemarkupLiteralBlockRule",
+//"PhutilRemarkupTableBlockRule",
+//"PhutilRemarkupSimpleTableBlockRule",
+
+//"PhutilRemarkupInterpreterBlockRule"
+
+class BlocksHighlighter{
+
+public:
+ BlocksHighlighter(){
+ }
+
+ QDomDocument loadScheme(){
+ QDomDocument doc("JuffScheme");
+ QString fileName = QString("remarkup.xml");
+ fileName = AppInfo::configDirPath() + "/hlschemes/" + fileName;
+
+ QFile file(fileName);
+ if ( !file.open(QIODevice::ReadOnly) ) {
+ qDebug()<<(QString("[BlocksScheme] No scheme found in '%1'. Using the standard one.").arg(fileName));
+ return doc;
+ }
+
+ QString err;
+ int errLine, errCol;
+ if ( !doc.setContent(&file, &err, &errLine, &errCol) ) {
+ qDebug() << QString("[BlocksScheme] File %1: XML reading error: '%2', line %3, column %4")
+ .arg(fileName).arg(err).arg(errLine).arg(errCol);
+ }
+
+ file.close();
+ return doc;
+ }
+
+ void init(){
+ const QDomDocument docScheme = loadScheme();
+
+ QDomElement root = docScheme.documentElement().firstChildElement("blockrules");
+
+// while (!root.isNull() && root.tagName().compare("blockrules")){
+// root = root.nextSiblingElement();
+// }
+
+
+ if (root.isNull()){
+ qDebug()<<"[BlocksScheme] not found blocks section";
+ return;
+ }
+
+ int countRules=0;
+ QDomElement styleNode = root.firstChildElement("style");
+ while (!styleNode.isNull()){
+ QString rule = styleNode.attribute("name", "");
+ if (!rule.isEmpty()) {
+ QColor colorBg = QVariant(styleNode.attribute("color", "#FFFFFF")).value<QColor>();
+ QString markerType = styleNode.attribute("marker", "Background");
+
+ QsciScintilla::MarkerSymbol marker = QsciScintilla::Background;
+
+ if (markerType == "FullRectangle"){
+ marker = QsciScintilla::FullRectangle;
+
+ } else if (markerType == "LeftRectangle") {
+ marker = QsciScintilla::LeftRectangle;
+
+ } else if (markerType == "Underline") {
+ marker = QsciScintilla::Underline;
+ }
+
+ rules.insert(rule, countRules);
+ Style style;
+ style.color = colorBg;
+ style.marker = marker;
+ markerStyles.append(style);
+
+ ++countRules;
+ }
+
+ styleNode = styleNode.nextSiblingElement("style");
+ }
+ }
+
+ void defineMarkersForDoc(JuffScintilla* doc){
+ doc->setMarginMarkerMask(0, 32767);
+
+ for (int i=0; i<markerStyles.size(); ++i){
+ doc->markerDefine(markerStyles[i].marker, markersCustomOffset + i);
+ doc->setMarkerBackgroundColor(markerStyles[i].color, markersCustomOffset + i);
+ }
+ }
+
+ void removeMarkers(JuffScintilla* doc){
+ for (int i=0; i<markerStyles.size(); ++i){
+ doc->markerDeleteAll(i + markersCustomOffset);
+ }
+ }
+
+ void highlightDoc(JuffScintilla* doc, const std::vector<Block>& blocks){
+ removeMarkers(doc);
+
+ for (size_t i=0; i<blocks.size(); ++i){
+ const Block& block = blocks.at(i);
+
+ QString debug__rulename = QString::fromStdString(block.rule);
+ int ruleId = rules.value(QString::fromStdString(block.rule), -1);
+
+ //unknown rule
+ if (ruleId == -1) continue;
+
+ int markerForRule = ruleId + markersCustomOffset;
+
+ for (int lineNo = block.start; lineNo<block.start + block.num_lines; ++lineNo){
+ doc->markerAdd(lineNo, markerForRule);
+ }
+ }
+ }
+
+private:
+
+ struct Style {
+ QColor color;
+ QsciScintilla::MarkerSymbol marker;
+ };
+
+ QVector<Style> markerStyles;
+ QMap<QString, int> rules;
+ const int markersCustomOffset = 3;
+} highlighterRemarkupBlocks;
+
+QAction* createAction(const QString& title, const QKeySequence& key, const QObject* this_, const char* slot) {
+ QAction* act = new QAction(title, 0);
+ this_->connect(act, SIGNAL(triggered()), this_, slot);
+ act->setShortcut(key);
+ return act;
+}
+
+RemarkupPlugin::RemarkupPlugin(): QObject(), JuffPlugin()
+{
+ threadRemarkupParser = new RemarkupParserThread();
+
+ connect(threadRemarkupParser, SIGNAL(updateDoc(Blocks, Juff::JuffScintilla*)), this, SLOT(onTextParsed(Blocks, Juff::JuffScintilla*)));
+}
+
+RemarkupPlugin::~RemarkupPlugin(){
+ threadRemarkupParser->exit();
+ threadRemarkupParser->wait();
+ delete threadRemarkupParser;
+}
+
+// info
+QString RemarkupPlugin::name()const {return "Remarkup";}
+QString RemarkupPlugin::title() const {return "Remarkup";}
+QString RemarkupPlugin::description() const {return "Remarkup";}
+QString RemarkupPlugin::targetEngine() const{
+ return "all";
+}
+
+void
+RemarkupPlugin::init(){
+
+ connect(api(), SIGNAL(docOpened(Juff::Document*, Juff::PanelIndex)), SLOT(onDocOpened(Juff::Document*, Juff::PanelIndex)));
+ connect(api(), SIGNAL(docTextChanged(Juff::Document*)), this, SLOT(refresh(Juff::Document*)));
+ connect(api(), SIGNAL(docActivated(Juff::Document*)), this, SLOT(refresh(Juff::Document*)));
+ connect(api(), SIGNAL(docSyntaxChanged(Juff::Document*,QString)), this, SLOT(onDocSyntaxChanged(Juff::Document*,QString)));
+ connect(api(), SIGNAL(docClosed(Juff::Document*)), this, SLOT(onDocClosed(Juff::Document*)));
+ connect(api(), SIGNAL(docRenamed(Juff::Document*,QString)), this, SLOT(onDocRenamed(Juff::Document*,QString)));
+
+
+ highlighterRemarkupBlocks.init();
+ threadRemarkupParser->start();
+
+// connect(api(), SIGNAL(docActivated(Juff::Document*)), this, SLOT(onDocActivated(Juff::Document*)));
+
+
+
+}
+
+void
+RemarkupPlugin::onDocOpened(Juff::Document* doc, Juff::PanelIndex panel){
+ onDocSyntaxChanged(doc, doc->syntax());
+}
+
+void
+RemarkupPlugin::onDocClosed(Juff::Document* doc){
+ if (doc->isNull()) {return;}
+ if (doc->syntax().compare("Remarkup") !=0){return;}
+
+ threadRemarkupParser->stop();
+}
+
+
+void
+RemarkupPlugin::onDocSyntaxChanged(Juff::Document* doc, QString syntax){
+ if (syntax.compare("Remarkup") !=0){return;}
+
+ SciDoc* __doc = qobject_cast<SciDoc*> (doc);
+ JuffScintilla* scintilla = __doc->getCurrentEditor();
+ if (!scintilla) return;
+
+ highlighterRemarkupBlocks.defineMarkersForDoc(scintilla);
+}
+
+void
+RemarkupPlugin::refresh(Juff::Document* doc){
+ if (doc->isNull()) {return;}
+ if (doc->syntax().compare("Remarkup") !=0){return;}
+
+ SciDoc* __doc = qobject_cast<SciDoc*> (api()->currentDocument());
+ JuffScintilla* scintilla = __doc->getCurrentEditor();
+
+ threadRemarkupParser->processDoc(scintilla);
+}
+
+void
+RemarkupPlugin::onTextParsed(const Blocks& blocks, Juff::JuffScintilla* doc){
+ highlighterRemarkupBlocks.highlightDoc(doc, blocks);
+
+}
+
+Juff::ActionList
+RemarkupPlugin::mainMenuActions(Juff::MenuID id) const {
+ Juff::ActionList list;
+
+ if ( id == Juff::MenuEdit ) {
+ list << createAction(tr("Bold"), QKeySequence("Ctrl+B"), this, SLOT(setSelectionBold()));
+ list << createAction(tr("Italic"), QKeySequence("Ctrl+I"), this, SLOT(setSelectionItalic()));
+ }
+ return list;
+}
+
+void
+RemarkupPlugin::setSelectionItalic(){
+ Juff::Document* doc = this->api()->currentDocument();
+ if (doc->isNull()) {return;}
+ if (doc->syntax().compare("Remarkup") !=0){return;}
+
+ QString textInput;
+ doc->getSelectedText(textInput);
+
+ QString textItalic = QString("//%1//").arg(textInput);
+ doc->replaceSelectedText(textItalic);
+}
+
+void
+RemarkupPlugin::setSelectionBold(){
+ Juff::Document* doc = this->api()->currentDocument();
+ if (doc->isNull()) {return;}
+ if (doc->syntax().compare("Remarkup") !=0){return;}
+
+ QString textInput;
+ doc->getSelectedText(textInput);
+
+ QString textBold = QString("**%1**").arg(textInput);
+ doc->replaceSelectedText(textBold);
+}
+
+
+
+
diff --git a/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.h b/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.h
new file mode 100644
index 0000000..698b53c
--- /dev/null
+++ b/documentation-tools/juffed-plugins/remarkupplugin/remarkupplugin.h
@@ -0,0 +1,48 @@
+#ifndef REMARKUPPLUGIN_H
+#define REMARKUPPLUGIN_H
+
+#include <QtCore/QObject>
+#include "JuffPlugin.h"
+#include "remarkupparser.h"
+
+class RemarkupParserThread;
+
+namespace Juff {
+ class JuffScintilla;
+}
+
+class RemarkupPlugin: public QObject, public JuffPlugin{
+ Q_OBJECT
+ Q_PLUGIN_METADATA(IID JuffPlugin_iid)
+ Q_INTERFACES(JuffPlugin)
+
+public:
+ RemarkupPlugin();
+ ~RemarkupPlugin();
+ void init();
+
+ // info
+ virtual QString name() const;
+ virtual QString title() const;
+ virtual QString description() const;
+ virtual QString targetEngine() const;
+ Juff::ActionList mainMenuActions(Juff::MenuID id) const;
+
+signals:
+
+
+public slots:
+ void refresh(Juff::Document* doc);
+ void onDocSyntaxChanged(Juff::Document* doc, QString syntax);
+ void onTextParsed(const Blocks& blocks, Juff::JuffScintilla* doc);
+ void onDocOpened(Juff::Document* doc, Juff::PanelIndex panel);
+ void onDocClosed(Juff::Document* doc);
+
+ void setSelectionBold();
+ void setSelectionItalic();
+
+private:
+ RemarkupParserThread* threadRemarkupParser;
+};
+
+#endif // REMARKUPPLUGIN_H
diff --git a/documentation-tools/remarkup/php/index.php b/documentation-tools/remarkup/php/index.php
deleted file mode 100644
index 2cd89b7..0000000
--- a/documentation-tools/remarkup/php/index.php
+++ /dev/null
@@ -1,347 +0,0 @@
-<?php
-//require_once '/opt/libphutil/src/object/Phobject.php';
-//require_once '/opt/libphutil/src/moduleutils/core.php';
-//require_once '/opt/libphutil/src/moduleutils/PhutilBootloader.php';
-//require_once '/opt/libphutil/src/moduleutils/PhutilBootloaderException.php';
-//require_once '/opt/libphutil/src/moduleutils/PhutilLibraryConflictException.php';
-//
-////require "/opt/libphutil/src/__phutil_library_init__.php";
-//
-//require_once ("/opt/libphutil/src/utils/utils.php");
-//require_once ("/opt/libphutil/src/xsprintf/csprintf.php");
-//require_once ("/opt/libphutil/src/future/exec/execx.php");
-//require_once ("/opt/libphutil/src/xsprintf/hgsprintf.php");
-//require_once ("/opt/libphutil/src/markup/render.php");
-//require_once ("/opt/libphutil/src/xsprintf/jsprintf.php");
-//require_once ("/opt/libphutil/src/xsprintf/ldapsprintf.php");
-//require_once ("/opt/libphutil/src/error/phlog.php");
-//require_once ("/opt/libphutil/src/internationalization/pht.php");
-//require_once ("/opt/libphutil/src/console/format.php");
-//require_once ("/opt/libphutil/src/utils/viewutils.php");
-//require_once ("/opt/libphutil/src/moduleutils/moduleutils.php");
-//require_once ("/opt/libphutil/src/utils/utf8.php");
-//require_once ("/opt/libphutil/src/xsprintf/pregsprintf.php");
-//require_once ("/opt/libphutil/src/xsprintf/qsprintf.php");
-//require_once ("/opt/libphutil/src/xsprintf/queryfx.php");
-//require_once ("/opt/libphutil/src/xsprintf/tsprintf.php");
-//require_once ("/opt/libphutil/src/xsprintf/urisprintf.php");
-//require_once ("/opt/libphutil/src/parser/xhpast/parser_nodes.php");
-//require_once ("/opt/libphutil/src/parser/xhpast/parser_tokens.php");
-//require_once ("/opt/libphutil/src/xsprintf/xsprintf.php");
-//require_once ("/opt/libphutil/src/filesystem/Filesystem.php");
-//require_once ("/opt/libphutil/src/object/Phobject.php");
-//require_once ("/opt/libphutil/src/filesystem/FileFinder.php");
-//require_once ("/opt/libphutil/src/markup/PhutilMarkupEngine.php");
-//require_once ("/opt/libphutil/src/markup/engine/PhutilRemarkupEngine.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupEscapeRemarkupRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupMonospaceRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupDocumentLinkRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupHyperlinkRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupBoldRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupItalicRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupDelRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupUnderlineRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/markuprule/PhutilRemarkupHighlightRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupQuotesBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupReplyBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupHorizontalRuleBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupCodeBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupLiteralBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupTableBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupSimpleTableBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupDefaultBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupListBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupInterpreterBlockRule.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/PhutilRemarkupBlockStorage.php");
-//require_once ("/opt/libphutil/src/markup/PhutilSafeHTML.php");
-//require_once ("/opt/libphutil/src/internationalization/PhutilTranslator.php");
-//require_once ("/opt/libphutil/src/parser/PhutilSimpleOptions.php");
-//require_once ("/opt/libphutil/src/lexer/PhutilLexer.php");
-//require_once ("/opt/libphutil/src/lexer/PhutilSimpleOptionsLexer.php");
-//require_once ("/opt/libphutil/src/parser/PhutilLanguageGuesser.php");
-//require_once ("/opt/libphutil/src/markup/syntax/engine/PhutilSyntaxHighlighterEngine.php");
-//require_once ("/opt/libphutil/src/markup/syntax/engine/PhutilDefaultSyntaxHighlighterEngine.php");
-//require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilDefaultSyntaxHighlighter.php");
-//require_once ("/opt/libphutil/src/future/Future.php");
-//require_once ("/opt/libphutil/src/future/ImmediateFuture.php");
-//require_once ("/opt/libphutil/src/parser/PhutilURI.php");
-//require_once ("/opt/libphutil/src/symbols/PhutilSymbolLoader.php");
-//require_once ("/opt/libphutil/src/symbols/PhutilClassMapQuery.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupBlockInterpreter.php");
-//require_once ("/opt/libphutil/src/markup/engine/remarkup/blockrule/PhutilRemarkupTestInterpreterRule.php");
-//require_once ("/opt/libphutil/src/parser/xhpast/bin/PhutilXHPASTBinary.php");
-//require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilSyntaxHighlighter.php");
-//require_once ("/opt/libphutil/src/markup/syntax/highlighter/PhutilLexerSyntaxHighlighter.php");
-//require_once ("/opt/libphutil/src/lexer/PhutilPHPFragmentLexer.php");
-//
-//
-//
-//final class PhutilRemarkupEngineTestCase {
-//
-// public function testEngine() {
-// $root = '/opt/libphutil/src/markup/engine/__tests__/remarkup/';
-//
-// $engine = $this->buildNewTestEngine();
-// $actual_output = $engine->markupText("**duck quack**");
-// echo $actual_output;
-//
-//
-//// foreach (Filesystem::listDirectory($root, $hidden = false) as $file) {
-//// echo $root.$file;
-//// $contents = Filesystem::readFile($root.$file);
-////
-//// $result = "\n" . $this->markupText($root.$file);
-//// echo $result;
-//// }
-//
-// $result ="";
-// return $result;
-// }
-//
-// private function markupText($markup_file) {
-// $contents = Filesystem::readFile($markup_file);
-// $file = basename($markup_file);
-//
-// $parts = explode("\n~~~~~~~~~~\n", $contents);
-//// $this->assertEqual(3, count($parts), $markup_file);
-//
-// list($input_remarkup, $expected_output, $expected_text) = $parts;
-//
-// $engine = $this->buildNewTestEngine();
-//
-// switch ($file) {
-// case 'raw-escape.txt':
-//
-// // NOTE: Here, we want to test PhutilRemarkupEscapeRemarkupRule and
-// // PhutilRemarkupBlockStorage, which are triggered by "\1". In the
-// // test, "~" is used as a placeholder for "\1" since it's hard to type
-// // "\1".
-//
-// $input_remarkup = str_replace('~', "\1", $input_remarkup);
-// $expected_output = str_replace('~', "\1", $expected_output);
-// $expected_text = str_replace('~', "\1", $expected_text);
-// break;
-// case 'toc.txt':
-// $engine->setConfig('header.generate-toc', true);
-// break;
-// }
-//
-// $actual_output = (string)$engine->markupText($input_remarkup);
-//
-// switch ($file) {
-// case 'toc.txt':
-// $table_of_contents =
-// PhutilRemarkupHeaderBlockRule::renderTableOfContents($engine);
-// $actual_output = $table_of_contents."\n\n".$actual_output;
-// break;
-// }
-//
-//// $this->assertEqual(
-//// $expected_output,
-//// $actual_output,
-//// pht("Failed to markup HTML in file '%s'.", $file));
-//
-// $engine->setMode(PhutilRemarkupEngine::MODE_TEXT);
-// $actual_output = (string)$engine->markupText($input_remarkup);
-//
-// return $actual_output;
-//// $this->assertEqual(
-//// $expected_text,
-//// $actual_output,
-//// pht("Failed to markup text in file '%s'.", $file));
-// }
-//
-// private function buildNewTestEngine() {
-// echo "preprocess0";
-// $engine = new PhutilRemarkupEngine();
-// $engine->setConfig('uri.prefix', 'http://www.example.com/');
-//
-// $engine->setConfig(
-// 'uri.allowed-protocols',
-// array(
-// 'http' => true,
-// 'mailto' => true,
-// 'tel' => true,
-// ));
-//
-// echo "preprocess01";
-// $rules = array();
-// $rules[] = new PhutilRemarkupEscapeRemarkupRule();
-// $rules[] = new PhutilRemarkupMonospaceRule();
-// $rules[] = new PhutilRemarkupDocumentLinkRule();
-// $rules[] = new PhutilRemarkupHyperlinkRule();
-// $rules[] = new PhutilRemarkupBoldRule();
-// $rules[] = new PhutilRemarkupItalicRule();
-// $rules[] = new PhutilRemarkupDelRule();
-// $rules[] = new PhutilRemarkupUnderlineRule();
-// $rules[] = new PhutilRemarkupHighlightRule();
-//
-// echo "preprocess02";
-// $blocks = array();
-// $blocks[] = new PhutilRemarkupQuotesBlockRule();
-// $blocks[] = new PhutilRemarkupReplyBlockRule();
-// $blocks[] = new PhutilRemarkupHeaderBlockRule();
-// $blocks[] = new PhutilRemarkupHorizontalRuleBlockRule();
-// $blocks[] = new PhutilRemarkupCodeBlockRule();
-// $blocks[] = new PhutilRemarkupLiteralBlockRule();
-// $blocks[] = new PhutilRemarkupNoteBlockRule();
-// $blocks[] = new PhutilRemarkupTableBlockRule();
-// $blocks[] = new PhutilRemarkupSimpleTableBlockRule();
-// $blocks[] = new PhutilRemarkupDefaultBlockRule();
-// $blocks[] = new PhutilRemarkupListBlockRule();
-// $blocks[] = new PhutilRemarkupInterpreterBlockRule();
-//
-// echo "preprocess03";
-// foreach ($blocks as $block) {
-// if (!($block instanceof PhutilRemarkupCodeBlockRule)) {
-// $block->setMarkupRules($rules);
-// }
-// }
-//
-// echo "preprocess04";
-// $engine->setBlockRules($blocks);
-//
-// return $engine;
-// }
-//
-//}
-//
-//echo "aaaa";
-//$test = new PhutilRemarkupEngineTestCase();
-//$output = $test->testEngine();
-
-namespace tutorial\php;
-
-error_reporting(E_ALL);
-
-require_once '/opt/thrift/lib/php/lib/Thrift/ClassLoader/ThriftClassLoader.php';
-
-use Thrift\ClassLoader\ThriftClassLoader;
-
-$GEN_DIR = '/private/prg/code/tests/php-cpp-integration-test/thrift/gen-php';
-
-$loader = new ThriftClassLoader();
-$loader->registerNamespace('Thrift', '/opt/thrift/lib/php/lib');
-$loader->registerDefinition('shared', $GEN_DIR);
-$loader->registerDefinition('tutorial', $GEN_DIR);
-$loader->register();
-
-/*
- * Licensed to the Apache Software Foundation (ASF) under one
- * or more contributor license agreements. See the NOTICE file
- * distributed with this work for additional information
- * regarding copyright ownership. The ASF licenses this file
- * to you under the Apache License, Version 2.0 (the
- * "License"); you may not use this file except in compliance
- * with the License. You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing,
- * software distributed under the License is distributed on an
- * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
- * KIND, either express or implied. See the License for the
- * specific language governing permissions and limitations
- * under the License.
- */
-
-/*
- * This is not a stand-alone server. It should be run as a normal
- * php web script (like through Apache's mod_php) or as a cgi script
- * (like with the included runserver.py). You can connect to it with
- * THttpClient in any language that supports it. The PHP tutorial client
- * will work if you pass it the argument "--http".
- */
-
-if (php_sapi_name() == 'cli') {
- ini_set("display_errors", "stderr");
-}
-
-use Thrift\Protocol\TBinaryProtocol;
-use Thrift\Transport\TPhpStream;
-use Thrift\Transport\TBufferedTransport;
-
-class CalculatorHandler implements \tutorial\CalculatorIf {
- protected $log = array();
-
- public function ping() {
- error_log("ping()");
- }
-
- public function add($num1, $num2) {
- error_log("add({$num1}, {$num2})");
- return $num1 + $num2;
- }
-
- public function calculate($logid, \tutorial\Work $w) {
- error_log("calculate({$logid}, {{$w->op}, {$w->num1}, {$w->num2}})");
- switch ($w->op) {
- case \tutorial\Operation::ADD:
- $val = $w->num1 + $w->num2;
- break;
- case \tutorial\Operation::SUBTRACT:
- $val = $w->num1 - $w->num2;
- break;
- case \tutorial\Operation::MULTIPLY:
- $val = $w->num1 * $w->num2;
- break;
- case \tutorial\Operation::DIVIDE:
- if ($w->num2 == 0) {
- $io = new \tutorial\InvalidOperation();
- $io->whatOp = $w->op;
- $io->why = "Cannot divide by 0";
- throw $io;
- }
- $val = $w->num1 / $w->num2;
- break;
- default:
- $io = new \tutorial\InvalidOperation();
- $io->whatOp = $w->op;
- $io->why = "Invalid Operation";
- throw $io;
- }
-
- $log = new \shared\SharedStruct();
- $log->key = $logid;
- $log->value = (string)$val;
- $this->log[$logid] = $log;
-
- return $val;
- }
-
- public function getStruct($key) {
- error_log("getStruct({$key})");
- // This actually doesn't work because the PHP interpreter is
- // restarted for every request.
- //return $this->log[$key];
- return new \shared\SharedStruct(array("key" => $key, "value" => "PHP is stateless!"));
- }
-
- public function zip() {
- error_log("zip()");
- }
-
-};
-
-header('Content-Type', 'application/x-thrift');
-if (php_sapi_name() == 'cli') {
- echo "\r\n";
-}
-
-$handler = new CalculatorHandler();
-$processor = new \tutorial\CalculatorProcessor($handler);
-
-//boost::shared_ptr<TTransport> socket(new TSocket("localhost", 9090));
-// boost::shared_ptr<TTransport> transport(new TBufferedTransport());
-$transport = new TBufferedTransport(new \Thrift\Transport\TSocketServer('unix:///tmp/mysock3'));
-$protocol = new TBinaryProtocol($transport, true, true);
-
-$transport->open();
-for (;;) {
- $processor->process($protocol, $protocol);
-}
-$transport->close();
\ No newline at end of file
diff --git a/documentation-tools/thrift/php/TSocketServer.php b/documentation-tools/thrift/php/TSocketServer.php
index 715b52d..d1c34da 100644
--- a/documentation-tools/thrift/php/TSocketServer.php
+++ b/documentation-tools/thrift/php/TSocketServer.php
@@ -1,66 +1,65 @@
<?php
namespace Thrift\Transport;
use Thrift\Exception\TException;
use Thrift\Factory\TStringFuncFactory;
class TSocketServer extends TTransport {
private $address = null;
private $client = null;
+ private $server = null;
public function __construct($address) {
- unlink($address);
$this->address = $address;
}
public function open()
{
- $server = stream_socket_server($this->address, $errno, $errorMessage);
- if ($server === false) {
+ $this->server = stream_socket_server($this->address, $errno, $errorMessage);
+ if ($this->server === false) {
throw new TException("Could not bind to socket: $errorMessage");
}
- $this->client = @stream_socket_accept($server, 3600);
+ $this->client = @stream_socket_accept($this->server, 3600);
if (!$this->client){
throw new TException("Could not accept connection");
}
}
public function close()
{
fclose($this->client);
}
public function isOpen()
{
return !is_null($client);
}
public function read($len)
{
- $data = @fread($this->client, $len);
- if ($data === FALSE || $data === '') {
- throw new TException('TPhpStream: Could not read '.$len.' bytes');
- }
+ do {
+ $data = @fread($this->client, $len);
+ } while ($data === FALSE || $data === '');
return $data;
}
public function write($buf)
{
while (TStringFuncFactory::create()->strlen($buf) > 0) {
$got = @fwrite($this->client, $buf);
if ($got === 0 || $got === FALSE) {
throw new TException('TPhpStream: Could not write '.TStringFuncFactory::create()->strlen($buf).' bytes');
}
$buf = TStringFuncFactory::create()->substr($buf, $got);
}
}
public function flush()
{
@fflush($this->client);
}
}
diff --git a/documentation/Articles/declarative_patterns.remarkup b/documentation/Articles/declarative_patterns.remarkup
new file mode 100644
index 0000000..fbe712a
--- /dev/null
+++ b/documentation/Articles/declarative_patterns.remarkup
@@ -0,0 +1,36 @@
+==Overview==
+
+In addition to a classic object-oriented common patterns and practices there are
+several areas and problems not very easy and conveniently expressed by object oriented approach.
+
+Few of so called declarative patterns are presented below :
+
+* **Dependency resolution**
+
+ Starting at some level of complexity every software project usually faces need of managing dependencies
+ of some kind like package dependencies, components and versions dependency.
+ Dependencies are easily represented by declarative means and moreover
+ usually do not change in runtime. Thus dependency management is good candidate for static or comile-time resolution. S ee [[articles/declarative_patterns#dependency-resolution|details]].
+
+* **Demand and supply**
+
+ A big amount of a problems could be expressed in terms of supply and demand of interacting components.
+ See [[articles/declarative_patterns#demand-and-supply|details]].
+
+* **Share or joint use**:
+
+ Examples: first/last use determination, share use planning and allocation(memory, etc).See for example [[concepts/usage| resources usage]].
+
+===Dependency resolution===
+
+
+
+fdfdfdfd
+
+
+===Demand and supply===
+
+dssdsds
+
+===Share or joint use===
+bdfvfsl;
\ No newline at end of file
diff --git a/documentation/Articles/gestalts.remarkup b/documentation/Articles/gestalts.remarkup
new file mode 100644
index 0000000..338b33f
--- /dev/null
+++ b/documentation/Articles/gestalts.remarkup
@@ -0,0 +1,12 @@
+==Computational gestalts==
+There are number of commonly accepted concepts that are redefined or reevaluated in xreate.
+
+* Interface is comprised of common subset of features supported by all concrete implementations.
+ See [[concepts/containers]]
+
+* Entity's annotations(like function, objects annotations) are exposed to and could be intospected by client code in other words by caller.
+ Most usefull is exposing of caller annotations to a callee code.
+
+ *Lazy/eager execution opposition.
+ It's possible to determine which sites of using resource should treat it as lazy resource or eager otherwise.
+ See [[concepts/usage]] for details.
\ No newline at end of file
diff --git a/documentation/Aspects/Optimization/index.remarkup b/documentation/Aspects/Optimization/index.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Aspects/Security and safety/access_control.remarkup b/documentation/Aspects/Security and safety/access_control.remarkup
new file mode 100644
index 0000000..5f163ef
--- /dev/null
+++ b/documentation/Aspects/Security and safety/access_control.remarkup
@@ -0,0 +1 @@
+access rights/usage control
\ No newline at end of file
diff --git a/documentation/Aspects/Security and safety/bugs_and_review_management.remarkup b/documentation/Aspects/Security and safety/bugs_and_review_management.remarkup
new file mode 100644
index 0000000..a3fb1d1
--- /dev/null
+++ b/documentation/Aspects/Security and safety/bugs_and_review_management.remarkup
@@ -0,0 +1,3 @@
+* review triggers based on conditions
+* review after related code is changed
+* enable/disabled parts of code depening on external libs bugs status fetched from bugtracker, etc
\ No newline at end of file
diff --git a/documentation/Aspects/Security and safety/component_guaranties.remarkup b/documentation/Aspects/Security and safety/component_guaranties.remarkup
new file mode 100644
index 0000000..7088aa8
--- /dev/null
+++ b/documentation/Aspects/Security and safety/component_guaranties.remarkup
@@ -0,0 +1,2 @@
+components guaranties/invariants(expressed among other things by unit-tests).
+Additionally, comply with whole infrastructure specifications&requirements(like a only permitted opened ports, etc)
\ No newline at end of file
diff --git a/documentation/Aspects/Security and safety/index.remarkup b/documentation/Aspects/Security and safety/index.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Aspects/Security and safety/krivyakin.remarkup b/documentation/Aspects/Security and safety/krivyakin.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Aspects/Security and safety/metrics.remarkup b/documentation/Aspects/Security and safety/metrics.remarkup
new file mode 100644
index 0000000..4eb1570
--- /dev/null
+++ b/documentation/Aspects/Security and safety/metrics.remarkup
@@ -0,0 +1 @@
+software metrics based on frame's scores
\ No newline at end of file
diff --git a/documentation/Aspects/Security and safety/pointers_safety.remarkup b/documentation/Aspects/Security and safety/pointers_safety.remarkup
new file mode 100644
index 0000000..7d659c0
--- /dev/null
+++ b/documentation/Aspects/Security and safety/pointers_safety.remarkup
@@ -0,0 +1 @@
+safety checks against invalid values (like validptr, etc)
\ No newline at end of file
diff --git a/documentation/Aspects/Security and safety/unit-tests.remarkup b/documentation/Aspects/Security and safety/unit-tests.remarkup
new file mode 100644
index 0000000..ae3cf58
--- /dev/null
+++ b/documentation/Aspects/Security and safety/unit-tests.remarkup
@@ -0,0 +1 @@
+Each unit-test expresses code constraints/invariants(states, etc) in declarative form. Unit-tests could be checked statically as well as dynamically.
\ No newline at end of file
diff --git a/documentation/Aspects/Virtualization/virtualization.remarkup b/documentation/Aspects/Virtualization/virtualization.remarkup
new file mode 100644
index 0000000..fbfd527
--- /dev/null
+++ b/documentation/Aspects/Virtualization/virtualization.remarkup
@@ -0,0 +1 @@
+virtualization(override external resources invokes, dependency injection)
\ No newline at end of file
diff --git a/documentation/Aspects/refactoring.remarkup b/documentation/Aspects/refactoring.remarkup
new file mode 100644
index 0000000..6b1531f
--- /dev/null
+++ b/documentation/Aspects/refactoring.remarkup
@@ -0,0 +1 @@
+Refactoring - ability to quiclky transform and apply software to different tasks
\ No newline at end of file
diff --git a/documentation/Concepts/adhocs.remarkup b/documentation/Concepts/adhocs.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Concepts/cache.remarkup b/documentation/Concepts/cache.remarkup
new file mode 100644
index 0000000..d7984e0
--- /dev/null
+++ b/documentation/Concepts/cache.remarkup
@@ -0,0 +1,7 @@
+* enclosed(parent-child) cache storage
+
+* sequential flow cache storage
+
+* cache implemented as set of cache invariant annotations, like:
+//sin = function (x:: num)::num; cache(ResultDependsOnArgumentsOnly)//
+
diff --git a/documentation/Concepts/computation_monads.remarkup b/documentation/Concepts/computation_monads.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Concepts/containers.remarkup b/documentation/Concepts/containers.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Concepts/context.remarkup b/documentation/Concepts/context.remarkup
new file mode 100644
index 0000000..ccbd334
--- /dev/null
+++ b/documentation/Concepts/context.remarkup
@@ -0,0 +1,6 @@
+===Static context===
+
+===Dynamic context===
+
+===Path dependent dynamic context===
+
diff --git a/documentation/Concepts/macroses.remarkup b/documentation/Concepts/macroses.remarkup
new file mode 100644
index 0000000..3433aa4
--- /dev/null
+++ b/documentation/Concepts/macroses.remarkup
@@ -0,0 +1,4 @@
+==Macroses==
+
+Annotation(marks) in order to execute code specific rewriting procedure.
+For example for functions marked as "List" the overloaded version should be added to handle containers
\ No newline at end of file
diff --git a/documentation/Concepts/pointers_management.remarkup b/documentation/Concepts/pointers_management.remarkup
new file mode 100644
index 0000000..cbb4c31
--- /dev/null
+++ b/documentation/Concepts/pointers_management.remarkup
@@ -0,0 +1,7 @@
+Pointers management implementations:
+
+* scope bounded poinetrs
+* parent bounded pointers
+* use-aware pointers
+* last usage detection
+* validptr
\ No newline at end of file
diff --git a/documentation/Concepts/strings.remarkup b/documentation/Concepts/strings.remarkup
new file mode 100644
index 0000000..df0a078
--- /dev/null
+++ b/documentation/Concepts/strings.remarkup
@@ -0,0 +1,5 @@
+String implementations:
+
+* zero-ended char*
+* size-aware string
+* on-the-fly
\ No newline at end of file
diff --git a/documentation/Concepts/usage.remarkup b/documentation/Concepts/usage.remarkup
new file mode 100644
index 0000000..67f3a6b
--- /dev/null
+++ b/documentation/Concepts/usage.remarkup
@@ -0,0 +1,10 @@
+Analyzing CFG to find first/last usage of resources:
+
+**Usecases:**
+* resource using in loop: acquire resource right before loop, and freeing right after that.
+* group of probable first/last using: every site in the group should treat resource as lazy(with safety check),
+ and eager otherwise
+
+**Examples:**
+* appending to a file
+* memory management
\ No newline at end of file
diff --git a/documentation/Concepts/views.remarkup b/documentation/Concepts/views.remarkup
new file mode 100644
index 0000000..c0fa073
--- /dev/null
+++ b/documentation/Concepts/views.remarkup
@@ -0,0 +1,5 @@
+Views: holds related data in consistent way while data changes
+
+==Methods==
+* "delta(difference)" method for supporting view in consistent way
+
diff --git a/documentation/Concepts/wells_time_machine.remarkup b/documentation/Concepts/wells_time_machine.remarkup
new file mode 100644
index 0000000..c0c0c8c
--- /dev/null
+++ b/documentation/Concepts/wells_time_machine.remarkup
@@ -0,0 +1,2 @@
+* Feature: Run computation on relevant occasion in the future
+* Past: Run specific computation prior the moment they needed
\ No newline at end of file
diff --git a/documentation/Frames/index.remarkup b/documentation/Frames/index.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Internal/documentation-style.remarkup b/documentation/Internal/documentation-style.remarkup
new file mode 100644
index 0000000..7e46602
--- /dev/null
+++ b/documentation/Internal/documentation-style.remarkup
@@ -0,0 +1,3 @@
+multi word designations or names should be lowercase and minus separated
+
+Example: ##very-long-function-name##
diff --git a/documentation/Internal/remarkup-examples.remarkup b/documentation/Internal/remarkup-examples.remarkup
new file mode 100644
index 0000000..559f040
--- /dev/null
+++ b/documentation/Internal/remarkup-examples.remarkup
@@ -0,0 +1,71 @@
+
+Header
+===
+
+sdsds
+
+==One header==
+
+
+fdfdfd
+
+Another header here
+====
+
+
+> Quoted Text
+
+Header
+--------
+
+
+
+- milk
+- eggs
+- bread
+
+* duck
+* duck
+* goose
+
+
+
+ x: code
+
+**TOPIC 1**
+
+
+|| xyz
+COUNTEREXAMPLE
+ function f() {
+ global $$variable_variable;
+ }
+
+
+
+
+## Overview
+Xreate is a language for write safe and efficient programs.
+#Getting started
+
+(NOTE): Best practices in proton pack operation include not crossing the streams.
+
+(WARNING): Crossing the streams can result in total protonic reversal!
+
+(IMPORTANT): Don't cross the streams!
+
+
+
+
+#Syntax
+
+- [X] Preheat oven to 450 degrees.
+- [ ] Zest 35 lemons.
+
+##Functions
+ : The *function declaration* defines a function with the parameters.
+
+###Syntax
+function
+
+This formula here \\(y=a\sum_{k=1}^{\infty}\frac{1}{k^{2}}\\) is a test.
\ No newline at end of file
diff --git a/documentation/Syntax/Annotations/index.remarkup b/documentation/Syntax/Annotations/index.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/Interfaces/adhoc.remarkup b/documentation/Syntax/Interfaces/adhoc.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/Interfaces/cfa.remarkup b/documentation/Syntax/Interfaces/cfa.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/Interfaces/dfa.remarkup b/documentation/Syntax/Interfaces/dfa.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/Interfaces/extern-c.remarkup b/documentation/Syntax/Interfaces/extern-c.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/Interfaces/index.remarkup b/documentation/Syntax/Interfaces/index.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/Syntax/control-flow-statements.remarkup b/documentation/Syntax/control-flow-statements.remarkup
new file mode 100644
index 0000000..4901695
--- /dev/null
+++ b/documentation/Syntax/control-flow-statements.remarkup
@@ -0,0 +1,132 @@
+==Control flow statements==
+
+These statements control execution flow of a program.
+
+=== If-Else ===
+The If-Else statement executes a //main-expression// branch if a specified condition is true.
+If the condition is false, //alternative-expression// branch will be executed.
+
+SYNTAX: if (//condition//):: //type//
+{
+ [//declaration//. [ //declaration//.]... ]
+ //main-expression//
+} else {
+ [//declaration//. [ //declaration//.]... ]
+ //alternative-expression//
+}
+
+//condition//
+Bool type expression(see [[types]]) in order to decide which branch of IF-ELSE statement should be executed.
+
+//main expression//
+Executed if condition yelds ##true##
+
+//alternative expression//
+Executed if condition yelds ##false##
+
+Example:
+ x = if (y<0) :: num
+ {-1} else {1}
+
+===Switch===
+Executes one of the given branches depending on condition.
+
+SYNTAX: switch (//condition// :: //type-of-condition//)
+[case //variant// {
+ [declaration. [ declaration.]... ]
+ //transformation-expression//
+}]...
+
+//condition//
+Expression to determing which case branch should be executed
+
+//type-of-condition//
+Type of condition result
+
+//variant//
+Expression to compare against //condition//.
+If the //condition// evaluation result corresponds to the //variant// then a relevant branch is executed.
+Special label **default** is allowed and designates default branch executed if no other branches are selected.
+
+//transformation-expression//
+Determines result of a switch statement if relevant branch is selected based on //condition//.
+
+Example:
+ x = 2:: num.
+
+ y = switch (x:: string)
+ case 0 {"zero"}
+ case 1 {"one"}
+ case 2 {"two"}
+ case default {"unknown"}
+
+
+
+===Map loop===
+Iterates over input collection and stores result in another collection.
+Every input collection's value transformed using //transformation-expression//.
+
+SYNTAX: loop map ( //input-collection// -> //alias// :: //type-of-element//) :: //type-of-collection//
+{
+ [declaration. [ declaration.]... ]
+ //transformation-expression//
+}
+
+//input-collection//
+Variable or expression to designate collection for transformation
+
+//alias//
+Variable name to designate current collection element in scope of inner code block
+
+//type-of-element//
+Type of collection's element
+
+//type-of-collection//
+Type of resulting transformed collection
+
+//transformation-expression//
+Transformation applied to an every collection element
+
+Example:
+ input = [1..10].
+ twice = loop map (input -> x:: num):: [num]
+ { 2 * x }
+
+===Fold loop===
+Iterates over input collection in order to accumulate result by applying //transformation-expression// to each element and intermediate accumulator.
+
+SYNTAX: loop fold (//input-collection// -> //alias// :: //type-of-element//, //accumulator//->//accumulator-alias//) :: //type-of-accumulator//
+{
+ [declaration. [ declaration.]... ]
+ //transformation-expression//
+}
+
+//input-collection//
+Variable or expression to designate collection
+
+//alias//
+Variable name to designate current collection element in scope of inner code block
+
+//type-of-element//
+Type of collection's element
+
+//accumulator//
+Expression to evaluate initial value of accumulator
+
+//accumulator-alias//
+Variable name to designate current value of accumulator in scope of inner code block
+
+//type-of-accumulator//
+Type of folding result
+
+//transformation-expression//
+Expression applied to an every collection's element and current state of accumulator in order to
+calculate accumulator value for next step of iteration.
+
+This is an example:
+
+ numbers = [1..10] :: [num].
+ sumOfNumbers = loop fold(numbers->x; 0->result)
+ {result + x}
+
+
diff --git a/documentation/Syntax/functions.remarkup b/documentation/Syntax/functions.remarkup
new file mode 100644
index 0000000..48af284
--- /dev/null
+++ b/documentation/Syntax/functions.remarkup
@@ -0,0 +1,18 @@
+==Functions==
+
+Xreate allows to define functions with a list of zero or more parameters and sequence of a statements as a body.
+
+SYNTAX: //name// = function [ (param:: type [, param:: type]... ) ] :: type; [; tag-expression ...]
+{
+ [declaration. [ declaration.] ]
+ result-expression
+}
+
+NOTE: There is alse special function syntax. See [[Syntax/prefunctions]]
+
+This is an example of a simple function:
+ sum = function(a:: num, b:: num):: num {
+ a + b
+ }
+
+
\ No newline at end of file
diff --git a/documentation/Syntax/index.remarkup b/documentation/Syntax/index.remarkup
new file mode 100644
index 0000000..6e2d7e1
--- /dev/null
+++ b/documentation/Syntax/index.remarkup
@@ -0,0 +1 @@
+This section devoted to a syntax description
\ No newline at end of file
diff --git a/documentation/Syntax/packages.remarkup b/documentation/Syntax/packages.remarkup
new file mode 100644
index 0000000..46b5187
--- /dev/null
+++ b/documentation/Syntax/packages.remarkup
@@ -0,0 +1 @@
+package dependencies
\ No newline at end of file
diff --git a/documentation/Syntax/types.remarkup b/documentation/Syntax/types.remarkup
new file mode 100644
index 0000000..e69de29
diff --git a/documentation/index.remarkup b/documentation/index.remarkup
new file mode 100644
index 0000000..a58a0cb
--- /dev/null
+++ b/documentation/index.remarkup
@@ -0,0 +1,13 @@
+## Overview
+
+Xreate is a language for write safe and efficient programs.
+
+## Getting started
+
+### Functions
+ :
+
+#### Syntax
+
+
+This formula here \\(y=a\sum_{k=1}^{\infty}\frac{1}{k^{2}}\\) is a test.
diff --git a/scripts/metatests/context-latecontext1.lp b/scripts/metatests/context-latecontext1.lp
new file mode 100644
index 0000000..536cdd3
--- /dev/null
+++ b/scripts/metatests/context-latecontext1.lp
@@ -0,0 +1,26 @@
+#include "control-context.lp".
+
+%body
+function(a;b;c;d).
+scope(0..4).
+cfa_parent(0, function(a)).
+cfa_parent(1, function(b)).
+cfa_parent(2, function(c)).
+cfa_parent(3, function(d)).
+cfa_parent(4, function(d)).
+
+cfa_call(0, c).
+cfa_call(1, c).
+cfa_call(2, d).
+
+bind_scope(0, a).
+bind_scope(1, b).
+
+cfa_function_specializations(d, a).
+cfa_function_specializations(d, b).
+
+#show bind_scope_decision/3.
+%#show bind_scope_demand/2.
+#show bind_function_demand/2.
+%#show cfa_call/2.
+%#show bind_function_demand/2.
\ No newline at end of file
diff --git a/scripts/metatests/context-latecontext-ContextLoopResolution-direct.lp b/scripts/metatests/context-latecontext2.lp
similarity index 77%
rename from scripts/metatests/context-latecontext-ContextLoopResolution-direct.lp
rename to scripts/metatests/context-latecontext2.lp
index a60cfbd..b8c9195 100644
--- a/scripts/metatests/context-latecontext-ContextLoopResolution-direct.lp
+++ b/scripts/metatests/context-latecontext2.lp
@@ -1,68 +1,67 @@
%#include "control-context.lp".
function(a;b;c).
scope(0..2).
%general
cfa_parent(0, function(a)).
cfa_parent(1, function(b)).
cfa_parent(2, function(c)).
cfa_call(0, c).
cfa_call(1, c).
%additional
keys(key0; key1).
data(key0, x).
data(key0, y).
data(key1, m).
data(key1, n).
bind_scope(0, keys(key0), strong).
bind_scope(1, keys(key1), strong).
%bind_scope(0, a, strong).
%bind_scope(0, b, strong).
%bind_scope(1, a, strong).
%bind_scope(1, c, strong).
bind_scope(2, set(X), Options):- bind_scope(2, keys(Key), Options), data(Key, X).
%nested scope propagation:
bind_scope(Scope, Context, Info) :- bind_scope(ScopeParent, Context, Info), cfa_parent(Scope, scope(ScopeParent)).
%strong inter-function propagation:
-bind_scope(Scope, Context,strong) :- bind_scope(ScopeParent, Context, strong): cfa_call(ScopeParent, FnCur
-rent); cfa_parent(Scope, function(FnCurrent)); cfa_call(_, FnCurrent); bind_scope(_, Context, strong); scope(Scope).
+bind_scope(Scope, Context,strong) :- bind_scope(ScopeParent, Context, strong): cfa_call(ScopeParent, FnCurrent); cfa_parent(Scope, function(FnCurrent)); cfa_call(_, FnCurrent); bind_scope(_, Context, strong); scope(Scope).
%weak inter-function propagation
bind_scope(Scope, Context, weak(ScopeParent)):- not bind_scope(Scope, Context, strong), bind_scope(ScopeParent, Context, strong), cfa_call(ScopeParent, FnCurrent), cfa_parent(Scope, function(FnCurrent)).
-bind_scope(Scope, Context, weak(Id)):- not bind_scope(Scope, Context, strong), bind_scope(ScopeParent, Context, weak(Id)), cfa_call(ScopeParent, FnCurrent), cfa_parent(Scope, function(FnCurrent)).
+bind_scope(Scope, Context, weak(Id)):- nbind_scope(Scope, Context, strong), bind_scope(ScopeParent, Context, weak(Id)), cfa_call(ScopeParent, FnCurrent), cfa_parent(Scope, function(FnCurrent)).
%make decisions
-bind_scope_decision(Scope,z loop(Subject), Scope):- cfa_contextloop(Scope, Subject), demand_dependency(loop(Subject), X), bind_scope(Scope, X, strong).
+bind_scope_decision(Scope, loop(Subject), Scope):- cfa_contextloop(Scope, Subject), demand_dependency(loop(Subject), X), bind_scope(Scope, X, strong).
bind_scope_decision(Scope, loop(Subject), Scope):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, loop(Subject)), demand_dependency(loop(Subject), X), bind_scope(Scope, X, strong).
%demand
-% bind_scope_demand(Scope, loop(Subject)):- cfa_contextloop(Sccghjcope, Subject), not bind_scope_decision(Scope, loop(Subject), _).
+% bind_scope_demand(Scope, loop(Subject)):- cfa_contextloop(Scope, Subject), not bind_scope_decision(Scope, loop(Subject), _).
%nested scope demand propagation
bind_scope_demand(Scope, Subject):- bind_scope_demand(ScopeChild, Subject), cfa_parent(ScopeChild, scope(Scope)).
bind_function_demand(Fn, Subject):- bind_scope_demand(Scope, Subject), cfa_parent(Scope, function(Fn)).
%inter-function propagation demand
bind_scope_demand(Scope, Subject):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, Subject), not bind_scope_decision(Scope, Subject, _).
-
+
% GENERATED DECLARATIONS FOR DECISION DEPENDENCIES
% -- cfa_contextloop(Scope, $set) => demand_dependency(loop(set), X) :- bind_scope($ScopeId, X = $set(_), weak(_)).
demand_dependency(loop(set), X) :- bind_scope(2, X, weak(_)), X = set(_).
% -- rule($Conditon, %Result) => demand_dependency(loop(X), $Condition) :- demand_dependency(loop(X), $Result)
demand_dependency(loop(X), keys(Key)) :- demand_dependency(loop(X), set(Y)), bind_scope(_, keys(Key), _).
cfa_contextloop(2, set).
#show bind_scope/3.
diff --git a/scripts/metatests/context-latecontext3.lp b/scripts/metatests/context-latecontext3.lp
new file mode 100644
index 0000000..ca81a77
--- /dev/null
+++ b/scripts/metatests/context-latecontext3.lp
@@ -0,0 +1,48 @@
+ %% SCHEMA:
+ %% specialization(Fn, Scope) - problem of what specialization of Fn should be picked up in Scope.
+ %% resolution_dependency(Resolution, Dependency) - Dependency is necessary prerequisite for choosing Resolution.
+ %%
+
+ %nested scope propagation:
+bind_scope(Scope, Context, Info) :- bind_scope(ScopeParent, Context, Info), cfa_parent(Scope, scope(ScopeParent)).
+
+ %strong/uniform inter-function propagation:
+bind_scope(Scope, Context,Info) :- bind_scope(ScopeParent, Context, Info): cfa_call(ScopeParent, FnCurrent); cfa_parent(Scope, function(FnCurrent)); cfa_call(_, FnCurrent); bind_scope(_, Context, Info); scope(Scope).
+
+ %weak inter-function propagation
+bind_scope(Scope, Context, weak(ScopeParent)):- not bind_scope(Scope, Context, strong), bind_scope(ScopeParent, Context, strong), cfa_call(ScopeParent, FnCurrent), cfa_parent(Scope, function(FnCurrent)).
+bind_scope(Scope, Context, weak(ScopeParent, Info)):- Info<>strong, not bind_scope(Scope, Context, Info), bind_scope(ScopeParent, Context, Info), cfa_call(ScopeParent, FnCurrent), cfa_parent(Scope, function(FnCurrent)).
+
+
+ %make decisions
+%%%bind_scope_decision(Scope, loop(Subject), Scope):- cfa_contextloop(Scope, Subject), demand_dependency(loop(Subject), X), bind_scope(Scope, X, strong).*
+%%%bind_scope_decision(Scope, loop(Subject), Scope):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, loop(Subject)), demand_dependency(loop(Subject), X), bind_scope(Scope, X, strong).
+
+ %on-site decision
+% ASSERT: ON-SITE DECISION SHOULD BE FIRST CLASS (checks at least one specialization exists)
+bind_scope_decision(Scope, Subject, Resolution):- bind_scope(Scope, Resolution, strong), Subject = specialization(Fn, Scope), cfa_call(Scope, Fn), cfa_function_specializations(Fn, Resolution).
+bind_scope_decision(ScopeSource, Subject, Resolution):- bind_scope(Scope, Resolution, weak(ScopeSource)), Subject = specialization(Fn, Scope), cfa_call(Scope, Fn), cfa_function_specializations(Fn, Resolution).
+bind_scope_decision(ScopeSource, Subject, resolution_dependency(Resolution, Dependency)):- bind_scope(Scope, Resolution, weak(ScopeSource, Dependency)), Subject = specialization(Fn, Scope), cfa_call(Scope, Fn), cfa_function_specializations(Fn, Resolution).
+
+ %dependent decisions
+bind_scope_demand(Scope, dependency(Subject, Scope)) :- bind_scope_decision(Scope, Subject, resolution_dependency(_, Dependency)).
+
+bind_scope_decision(ScopeSource, dependency(Subject, Scope), Dependency) :- Dependency = weak(ScopeSource), bind_scope_demand(Scope, dependency(Subject, Scope)), scope_dependencies(dependency(Subject, Scope), Dependency).
+bind_scope_demand(ScopeSource, dependency(Subject, ScopeSource)) :- Dependency = weak(ScopeSource, DependencyTail), bind_scope_demand(Scope, dependency(Subject, Scope)), scope_dependencies(dependency(Subject, Scope), Dependency).
+
+ %dependent decision helpers:
+scope_dependencies(dependency(Subject, Scope), Dependency) :- bind_scope_decision(Scope, Subject, resolution_dependency(_, Dependency)).
+scope_dependencies(dependency(Subject, ScopeSource), DependencyTail) :- Dependency = weak(ScopeSource, DependencyTail), bind_scope_demand(Scope, dependency(Subject, Scope)), scope_dependencies(dependency(Subject, Scope), Dependency).
+
+ %on-site demand
+% ASSERT: ON-SITE DEMAND SHOULD BE dependent OF on-site decision (check there are no specializations AT ALL)
+%%%bind_scope_demand(Scope, Subject):- cfa_contextloop(Scope, Subject), not bind_scope_decision(Scope, loop(Subject), _).
+bind_scope_demand(Scope, Subject):- Subject = specialization(FnCallee, Scope), cfa_call(Scope, FnCallee), cfa_function_specializations(FnCallee, _), not bind_scope_decision(Scope, Subject, _).
+
+ %nested scope demand propagation
+%ASSERT: NO DECISION CHECKS. because decisions linked to a leaf(function execution sites) scopes
+bind_scope_demand(Scope, Subject):- bind_scope_demand(ScopeChild, Subject), cfa_parent(ScopeChild, scope(Scope)).
+bind_function_demand(Fn, Subject):- bind_scope_demand(Scope, Subject), cfa_parent(Scope, function(Fn)).
+
+ %inter-function propagation demand
+bind_scope_demand(Scope, Subject):- cfa_call(Scope, FnChild), bind_function_demand(FnChild, Subject), not bind_scope_decision(Scope, Subject, _).
diff --git a/scripts/metatests/tests1-context-latecontext3.lp b/scripts/metatests/tests1-context-latecontext3.lp
new file mode 100644
index 0000000..28c84be
--- /dev/null
+++ b/scripts/metatests/tests1-context-latecontext3.lp
@@ -0,0 +1,39 @@
+#include "context-latecontext3.lp".
+function(a;b;r0; r1; sink; fnOut).
+scope(0..5).
+
+ %CFA
+cfa_parent(0, function(a)).
+cfa_parent(1, function(b)).
+cfa_parent(2, function(r0)).
+cfa_parent(3, function(r1)).
+cfa_parent(4, function(sink)).
+cfa_parent(5, function(fnOut)).
+
+cfa_function_specializations(fnOut, keys(keyA0)).
+cfa_function_specializations(fnOut, keys(keyA1)).
+cfa_function_specializations(fnOut, keys(keyB0)).
+cfa_function_specializations(fnOut, keys(keyB1)).
+
+
+cfa_call(0, r0).
+cfa_call(0, r1).
+cfa_call(1, r0).
+cfa_call(1, r1).
+cfa_call(2, sink).
+cfa_call(3, sink).
+cfa_call(4, fnOut).
+
+ %context rules
+ % r0: keyA->keyA0, keyB->keyB0
+ % r1: keyA->keyA1, keyB->keyB1
+bind_scope(2, keys(keyA0), Info) :- bind_scope(2, keys(keyA), Info).
+bind_scope(2, keys(keyB0), Info) :- bind_scope(2, keys(keyB), Info).
+bind_scope(3, keys(keyA1), Info) :- bind_scope(3, keys(keyA), Info).
+bind_scope(3, keys(keyB1), Info) :- bind_scope(3, keys(keyB), Info).
+
+ %initial context
+keys(keyA; keyB; keyA0; keyA1; keyB0; keyB1).
+
+bind_scope(0, keys(keyA), strong).
+bind_scope(1, keys(keyB), strong).
diff --git a/tools/execution-server/CMakeLists.txt b/tools/execution-server/CMakeLists.txt
new file mode 100644
index 0000000..ba241d1
--- /dev/null
+++ b/tools/execution-server/CMakeLists.txt
@@ -0,0 +1,43 @@
+cmake_minimum_required(VERSION 2.8.11)
+project(execution-server)
+
+add_compile_options(-std=c++14)
+
+
+
+# THRIFT
+#===================================
+add_custom_command(OUTPUT THRIFT_WRAPPER_OUTPUT
+ COMMAND ${CMAKE_CURRENT_SOURCE_DIR}/gen-thrift-wrapper.sh
+ DEPENDS "${CMAKE_CURRENT_SOURCE_DIR}/xreate_communication.thrift"
+ WORKING_DIRECTORY ${CMAKE_CURRENT_SOURCE_DIR}
+)
+
+message("THRIFT DEFINITION: " ${CMAKE_CURRENT_SOURCE_DIR}/xreate_communication.thrift)
+
+set(SOURCES_THRIFT_WRAPPER
+ ${CMAKE_CURRENT_SOURCE_DIR}/generated-cpp/xreate_communication_constants.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/generated-cpp/xreate_communication_types.cpp
+ ${CMAKE_CURRENT_SOURCE_DIR}/generated-cpp/AbstractXreateServer.cpp
+)
+
+include_directories(
+ ${CMAKE_CURRENT_SOURCE_DIR}/.
+ ${CMAKE_CURRENT_SOURCE_DIR}/generated-cpp/
+)
+
+
+
+# TESTS
+#===================================
+add_subdirectory(tests)
+
+
+
+# EXECUTION SERVER
+#===================================
+aux_source_directory(src SOURCE)
+include_directories(cpp/ )
+
+add_executable(${PROJECT_NAME} ${SOURCE} ${SOURCES_THRIFT_WRAPPER})
+target_link_libraries(${PROJECT_NAME} xreate thrift-1.0.0-dev)
\ No newline at end of file
diff --git a/tools/execution-server/gen-thrift-wrapper.sh b/tools/execution-server/gen-thrift-wrapper.sh
new file mode 100755
index 0000000..ac5c8ea
--- /dev/null
+++ b/tools/execution-server/gen-thrift-wrapper.sh
@@ -0,0 +1,12 @@
+THRIFT_COMPILER=/opt/thrift/compiler/cpp/thrift
+FRONTEND_PATH=/private/prg/code/xreate/tools/phabricator/xreate-frontend/resources
+SERVER_PATH=/private/prg/code/xreate/tools/execution-server
+
+
+rm -r $SERVER_PATH/generated-cpp
+mkdir $SERVER_PATH/generated-cpp
+$THRIFT_COMPILER -out $SERVER_PATH/generated-cpp -gen cpp:moveable_types $SERVER_PATH/xreate_communication.thrift
+
+rm -r $FRONTEND_PATH/generated-js
+mkdir $FRONTEND_PATH/generated-js
+$THRIFT_COMPILER -out $FRONTEND_PATH/generated-js -gen js:jquery $SERVER_PATH/xreate_communication.thrift
diff --git a/tools/execution-server/src/XreateServer.cpp b/tools/execution-server/src/XreateServer.cpp
new file mode 100644
index 0000000..ac4953f
--- /dev/null
+++ b/tools/execution-server/src/XreateServer.cpp
@@ -0,0 +1,105 @@
+/*
+ * File: XreateServer.cpp
+ * Author: pgess
+ *
+ * Created on March 29, 2016, 11:43 AM
+ */
+
+#include "XreateServer.h"
+#include "Parser.h"
+
+#include <iostream>
+#include <thrift/transport/TSocket.h>
+#include <boost/format.hpp>
+#include <boost/algorithm/string/join.hpp>
+
+using namespace std;
+using namespace apache::thrift::transport;
+
+class RecordedParserErrors: public Errors{
+public:
+ RecordedParserErrors(): Errors(), output(L"-- line %d col %d: %ls\n") {}
+
+ virtual void SynErr(int line, int col, int n) {
+ wchar_t* errorMessage = convertErrorCode(n);
+ wstring message ((output % line % col % errorMessage).str());
+ log.push_back(string(message.begin(), message.end()));
+
+ coco_string_delete(errorMessage);
+ count++;
+ }
+
+ virtual void Error(int line, int col, const wchar_t *s){
+ wstring message ((output % line % col % s).str());
+ log.push_back(string(message.begin(),message.end()));
+
+ count++;
+ }
+
+ virtual void Warning(int line, int col, const wchar_t *s){
+ wstring message ((output % line % col % s).str());
+ log.push_back(string(message.begin(),message.end()));
+ }
+
+ virtual void Warning(const wchar_t *s){
+ wstring message(s);
+
+ log.push_back(string(message.begin(),message.end()));
+ }
+
+ const std::list<std::string> getLog(){
+ return log;
+ }
+
+
+private:
+ std::list<std::string> log;
+ boost::wformat output;
+};
+
+XreateServer::XreateServer() {}
+
+void
+XreateServer::ping(std::string& _return, const std::string& text){
+ cout << "ping: "<<text << endl;
+
+ _return = text;
+}
+
+void
+XreateServer::checkSyntax(SyntaxCheckResult& _return, const std::string& program){
+ Scanner scanner(reinterpret_cast<const unsigned char*>(program.c_str()), program.size());
+ Parser parser(&scanner);
+ delete parser.errors;
+ RecordedParserErrors* errors = new RecordedParserErrors();
+ parser.errors = errors;
+ parser.Parse();
+
+ SyntaxCheckResult result;
+ result.success = parser.errors->count == 0;
+ auto log = errors->getLog();
+ result.output = boost::algorithm::join(errors->getLog(), "");
+
+ _return = result;
+}
+
+
+XreateServerFactory::~XreateServerFactory() {}
+
+AbstractXreateServerIf*
+XreateServerFactory::getHandler(const ::apache::thrift::TConnectionInfo& connInfo)
+{
+ boost::shared_ptr<TSocket> sock = boost::dynamic_pointer_cast<TSocket>(connInfo.transport);
+ cout << "Incoming connection\n";
+ cout << "\tSocketInfo: " << sock->getSocketInfo() << "\n";
+ cout << "\tPeerHost: " << sock->getPeerHost() << "\n";
+ cout << "\tPeerAddress: " << sock->getPeerAddress() << "\n";
+ cout << "\tPeerPort: " << sock->getPeerPort() << "\n";
+ return new XreateServer();
+}
+
+
+void
+XreateServerFactory::releaseHandler(AbstractXreateServerIf* handler) {
+ delete handler;
+}
\ No newline at end of file
diff --git a/tools/execution-server/src/XreateServer.h b/tools/execution-server/src/XreateServer.h
new file mode 100644
index 0000000..d75ba3c
--- /dev/null
+++ b/tools/execution-server/src/XreateServer.h
@@ -0,0 +1,34 @@
+/*
+ * To change this license header, choose License Headers in Project Properties.
+ * To change this template file, choose Tools | Templates
+ * and open the template in the editor.
+ */
+
+/*
+ * File: XreateServer.h
+ * Author: pgess
+ *
+ * Created on March 29, 2016, 11:43 AM
+ */
+
+#ifndef XREATESERVER_H
+#define XREATESERVER_H
+
+#include "AbstractXreateServer.h"
+
+class XreateServer: public AbstractXreateServerIf {
+ public:
+ XreateServer();
+ virtual void ping(std::string& _return, const std::string& text);
+ virtual void checkSyntax(SyntaxCheckResult& _return, const std::string& program);
+};
+
+class XreateServerFactory : virtual public AbstractXreateServerIfFactory {
+ public:
+ virtual ~XreateServerFactory();
+ virtual AbstractXreateServerIf* getHandler(const ::apache::thrift::TConnectionInfo& connInfo);
+ virtual void releaseHandler(AbstractXreateServerIf* handler);
+};
+
+#endif /* XREATESERVER_H */
+
diff --git a/tools/execution-server/src/main.cpp b/tools/execution-server/src/main.cpp
new file mode 100644
index 0000000..e9c8cf9
--- /dev/null
+++ b/tools/execution-server/src/main.cpp
@@ -0,0 +1,62 @@
+/*
+ * File: main.cpp
+ * Author: pgess
+ *
+ * Created on March 28, 2016, 3:59 PM
+ */
+
+#include "XreateServer.h"
+
+#include <thrift/concurrency/ThreadManager.h>
+#include <thrift/concurrency/PlatformThreadFactory.h>
+#include <thrift/protocol/TJSONProtocol.h>
+#include <thrift/server/TThreadPoolServer.h>
+#include <thrift/server/TThreadedServer.h>
+#include <thrift/transport/TServerSocket.h>
+#include <thrift/transport/TSocket.h>
+#include <thrift/transport/THttpServer.h>
+#include <thrift/transport/TTransportUtils.h>
+#include <thrift/TToString.h>
+
+#include <boost/make_shared.hpp>
+
+#include <iostream>
+#include <stdexcept>
+#include <sstream>
+
+using namespace std;
+using namespace apache::thrift;
+using namespace apache::thrift::concurrency;
+using namespace apache::thrift::protocol;
+using namespace apache::thrift::transport;
+using namespace apache::thrift::server;
+
+int main() {
+ TThreadedServer server(
+
+ //Processor:
+ boost::make_shared<AbstractXreateServerProcessorFactory>(boost::make_shared<XreateServerFactory>()),
+
+ //serverSocket:
+ boost::shared_ptr<TServerSocket>(new TServerSocket(9090)),
+
+ //transportFactory
+ boost::shared_ptr<TTransportFactory>(new THttpServerTransportFactory()),
+
+ //protocolFactory
+ boost::shared_ptr<TProtocolFactory>(new TJSONProtocolFactory())
+ );
+
+ /*
+ * new TThreadPoolServer(,
+ serverSocket,
+ transportFactory,
+ ,
+ threadManager)
+ */
+
+ cout << "Starting the server..." << endl;
+ server.serve();
+ cout << "Done." << endl;
+ return 0;
+}
\ No newline at end of file
diff --git a/tools/execution-server/tests/CMakeLists.txt b/tools/execution-server/tests/CMakeLists.txt
new file mode 100644
index 0000000..6e37616
--- /dev/null
+++ b/tools/execution-server/tests/CMakeLists.txt
@@ -0,0 +1,14 @@
+cmake_minimum_required(VERSION 2.8.11)
+project(execution-server-tests)
+add_compile_options(-D_THRIFT_USING_CLANG_LIBCXX=1)
+
+find_package(GTest REQUIRED)
+INCLUDE_DIRECTORIES(${GTEST_INCLUDE_DIRS})
+
+INCLUDE_DIRECTORIES(
+ .
+)
+
+aux_source_directory(. SOURCE)
+add_executable(${PROJECT_NAME} ${SOURCE} ${SOURCES_THRIFT_WRAPPER})
+target_link_libraries(${PROJECT_NAME} thrift-1.0.0-dev ${GTEST_BOTH_LIBRARIES} pthread)
diff --git a/tools/execution-server/tests/main.cpp b/tools/execution-server/tests/main.cpp
new file mode 100644
index 0000000..f49dd85
--- /dev/null
+++ b/tools/execution-server/tests/main.cpp
@@ -0,0 +1,102 @@
+#include "gtest/gtest.h"
+
+#include "AbstractXreateServer.h"
+
+#include <thrift/protocol/TJSONProtocol.h>
+#include <thrift/transport/TSocket.h>
+#include <thrift/transport/TTransportUtils.h>
+
+#include <iostream>
+
+using namespace std;
+
+class RemoteConnection {
+public:
+ AbstractXreateServerClient* server;
+
+ RemoteConnection(){
+ __socket.reset(new apache::thrift::transport::TSocket("localhost", 9090));
+ __transport.reset(new apache::thrift::transport::TBufferedTransport(__socket));
+ __protocol.reset(new apache::thrift::protocol::TJSONProtocol(__transport));
+
+ server = new AbstractXreateServerClient(__protocol);
+ __transport->open();
+ }
+
+ ~RemoteConnection(){
+ delete server;
+ __transport->close();
+ }
+
+
+private:
+ boost::shared_ptr<apache::thrift::transport::TTransport> __socket;
+ boost::shared_ptr<apache::thrift::transport::TTransport> __transport;
+ boost::shared_ptr<apache::thrift::protocol::TProtocol> __protocol;
+};
+
+
+TEST(Common, ping1){
+ std::string responce;
+
+ try {
+ RemoteConnection connection;
+
+ connection.server->ping(responce, "test1");
+ std::cout << "ping: " << responce << std::endl;
+
+ } catch (apache::thrift::TException& tx) {
+ cout << "ERROR: " << tx.what() << endl;
+ }
+
+ ASSERT_STREQ("test1", responce.c_str());
+}
+
+TEST(Common, syntaxcheckSuccess1){
+ SyntaxCheckResult r;
+
+ try {
+ RemoteConnection connection;
+
+ const string program =
+ "a = function:: num; entry \
+ {}";
+
+ connection.server->checkSyntax(r, program);
+
+ r.printTo(cout);cout << endl;
+
+ } catch(apache::thrift::TException& tx){
+ cout << "ERROR: " << tx.what() << endl;
+ }
+
+ ASSERT_TRUE(r.success);
+}
+
+TEST(Common, syntaxcheckFail1){
+ SyntaxCheckResult r;
+
+ try {
+ RemoteConnection connection;
+
+ const string program =
+ "a = function:: num; entry \
+ {x+y.}";
+
+ connection.server->checkSyntax(r, program);
+
+ r.printTo(cout); cout << endl;
+
+ } catch(apache::thrift::TException& tx){
+ cout << "ERROR: " << tx.what() << endl;
+ }
+
+ ASSERT_TRUE(r.output.size() > 0);
+ ASSERT_FALSE(r.success);
+}
+
+int main(int argc, char **argv) {
+ testing::GTEST_FLAG(color) = "yes";
+ testing::InitGoogleTest(&argc, argv);
+ return RUN_ALL_TESTS();
+}
\ No newline at end of file
diff --git a/tools/execution-server/xreate_communication.thrift b/tools/execution-server/xreate_communication.thrift
new file mode 100644
index 0000000..2e3e4ec
--- /dev/null
+++ b/tools/execution-server/xreate_communication.thrift
@@ -0,0 +1,9 @@
+struct SyntaxCheckResult {
+ 1: bool success,
+ 2: string output
+}
+
+service AbstractXreateServer {
+ string ping (1: string text),
+ SyntaxCheckResult checkSyntax(1: string program)
+}
diff --git a/tools/phabricator/install-xreate-frontend.sh b/tools/phabricator/install-xreate-frontend.sh
new file mode 100755
index 0000000..09d3496
--- /dev/null
+++ b/tools/phabricator/install-xreate-frontend.sh
@@ -0,0 +1,60 @@
+#!/bin/bash
+XREATE_FRONTEND=/opt/bitnami/apps/xreate-frontend
+PHABRICATOR_DIR=/opt/bitnami/apps/phabricator/htdocs/
+PHABRICATOR_CONTROL_SCRIPT=/opt/bitnami/php/scripts/ctl.sh
+
+SOURCES=(
+ 'XreateFrontendApplication.php' 'XreateFrontendController.php'
+)
+
+SOURCES_JS=('xreate_communication_types.js' 'AbstractXreateServer.js')
+
+FILE_AbstractXreateServer=$PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/xreate-interface.js
+
+HEADER_AbstractXreateServer="/**
+ * @requires thrift
+ * @provides xreate-interface
+ */"
+
+HEADER_jquery="/**
+ * @provides jquery
+ * @do-not-minify
+ */
+"
+HEADER_thrift="/**
+ * @requires jquery
+ * @provides thrift
+ */"
+
+echo "Install files\n"
+ for FILE in "${SOURCES[@]}"
+ do
+ rm "$PHABRICATOR_DIR/src/extensions/$FILE"
+ ln -s "$XREATE_FRONTEND/$FILE" "$PHABRICATOR_DIR/src/extensions/$FILE"
+ done
+
+echo "Install resources"
+ rm -r "$PHABRICATOR_DIR/webroot/rsrc/xreate-frontend"
+ mkdir -p "$PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js"
+ echo "$HEADER_thrift" > $PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/thrift.js
+ cat $XREATE_FRONTEND/resources/vendors/thrift/thrift.js >> $PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/thrift.js
+
+ echo "$HEADER_jquery" > $PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/jquery-2.2.2.min.js
+ cat $XREATE_FRONTEND/resources/vendors/jquery/jquery-2.2.2.min.js >> $PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/jquery-2.2.2.min.js
+
+ echo "$HEADER_AbstractXreateServer" > $FILE_AbstractXreateServer
+ for FILE in "${SOURCES_JS[@]}"
+ do
+ cat $XREATE_FRONTEND/resources/generated-js/$FILE >> $FILE_AbstractXreateServer
+ done
+
+ cp $XREATE_FRONTEND/resources/js/* $PHABRICATOR_DIR/webroot/rsrc/xreate-frontend/js/
+
+echo "Update celeriiy map"
+ $PHABRICATOR_DIR/bin/celerity map
+
+echo "Restart phabricator"
+ $PHABRICATOR_CONTROL_SCRIPT stop
+ $PHABRICATOR_CONTROL_SCRIPT start
+
+echo "Note: dodnt forget to setup ssh tunnel to allow ajax requests!"
diff --git a/tools/phabricator/patches/all/libphutil/1-blockrule-syntax.patch b/tools/phabricator/patches/all/libphutil/1-blockrule-syntax.patch
new file mode 100644
index 0000000..097f863
--- /dev/null
+++ b/tools/phabricator/patches/all/libphutil/1-blockrule-syntax.patch
@@ -0,0 +1,42 @@
+diff --git a/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php b/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php
+index e80f3e1..4c33173 100644
+--- a/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php
++++ b/src/markup/engine/remarkup/blockrule/PhutilRemarkupNoteBlockRule.php
+@@ -2,10 +2,13 @@
+
+ final class PhutilRemarkupNoteBlockRule extends PhutilRemarkupBlockRule {
+
++ public $additionalData = "";
++
+ public function getMatchingLineCount(array $lines, $cursor) {
+ $num_lines = 0;
+
+- if (preg_match($this->getRegEx(), $lines[$cursor])) {
++ $matches = array();
++ if (preg_match($this->getRegEx(), $lines[$cursor], $matches)) {
+ $num_lines++;
+ $cursor++;
+
+@@ -19,6 +22,14 @@ final class PhutilRemarkupNoteBlockRule extends PhutilRemarkupBlockRule {
+ }
+ }
+
++ //extract type of Note
++ if ($num_lines>0)
++ if (idx($matches, 'showword')) {
++ $this->additionalData = $matches['showword'];
++ } else {
++ $this->additionalData = $matches['hideword'];
++ }
++
+ return $num_lines;
+ }
+
+@@ -104,6 +115,7 @@ final class PhutilRemarkupNoteBlockRule extends PhutilRemarkupBlockRule {
+ 'NOTE',
+ 'IMPORTANT',
+ 'WARNING',
++ 'SYNTAX'
+ );
+
+ foreach ($words as $k => $word) {
\ No newline at end of file
diff --git a/tools/phabricator/patches/all/phabricator/1-remarkup-linebreaks-disabled.patch b/tools/phabricator/patches/all/phabricator/1-remarkup-linebreaks-disabled.patch
new file mode 100644
index 0000000..59e15b2
--- /dev/null
+++ b/tools/phabricator/patches/all/phabricator/1-remarkup-linebreaks-disabled.patch
@@ -0,0 +1,10 @@
+--- a/src/infrastructure/markup/PhabricatorMarkupEngine.php
++++ b/src/infrastructure/markup/PhabricatorMarkupEngine.php
+@@ -454,7 +454,7 @@ final class PhabricatorMarkupEngine extends Phobject {
+ 'uri.full' => false,
+ 'syntax-highlighter.engine' => PhabricatorEnv::getEnvConfig(
+ 'syntax-highlighter.engine'),
+- 'preserve-linebreaks' => true,
++ 'preserve-linebreaks' => false,
+ );
+ }
diff --git a/tools/phabricator/patches/local/libphutil/1-blockrules.patch b/tools/phabricator/patches/local/libphutil/1-blockrules.patch
new file mode 100644
index 0000000..dfe0d73
--- /dev/null
+++ b/tools/phabricator/patches/local/libphutil/1-blockrules.patch
@@ -0,0 +1,65 @@
+diff --git a/src/markup/engine/PhutilRemarkupEngine.php b/src/markup/engine/PhutilRemarkupEngine.php
+index d883154..9a44c2f 100644
+--- a/src/markup/engine/PhutilRemarkupEngine.php
++++ b/src/markup/engine/PhutilRemarkupEngine.php
+@@ -123,6 +123,7 @@ final class PhutilRemarkupEngine extends PhutilMarkupEngine {
+
+ $blocks = $this->splitTextIntoBlocks($text);
+
++
+ $output = array();
+ foreach ($blocks as $block) {
+ $output[] = $this->markupBlock($block);
+@@ -133,15 +134,16 @@ final class PhutilRemarkupEngine extends PhutilMarkupEngine {
+ $this->storage = null;
+ $metadata = $this->metadata;
+
+-
+ return array(
+ 'output' => $output,
+ 'storage' => $map,
+ 'metadata' => $metadata,
+ );
++
++
+ }
+
+- private function splitTextIntoBlocks($text, $depth = 0) {
++ public function splitTextIntoBlocks($text, $depth = 0) {
+ // Apply basic block and paragraph normalization to the text. NOTE: We don't
+ // strip trailing whitespace because it is semantic in some contexts,
+ // notably inlined diffs that the author intends to show as a code block.
+@@ -164,7 +166,7 @@ final class PhutilRemarkupEngine extends PhutilMarkupEngine {
+ $curr_block = array(
+ 'start' => $cursor,
+ 'num_lines' => $num_lines,
+- 'rule' => $block_rule,
++ 'rule' => clone $block_rule,
+ 'is_empty' => self::isEmptyBlock($text, $cursor, $num_lines),
+ 'children' => array(),
+ );
+diff --git a/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php b/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php
+index cbcb143..6e91488 100644
+--- a/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php
++++ b/src/markup/engine/remarkup/blockrule/PhutilRemarkupHeaderBlockRule.php
+@@ -16,13 +16,13 @@ final class PhutilRemarkupHeaderBlockRule extends PhutilRemarkupBlockRule {
+ }
+ }
+
+- if ($num_lines) {
+- $cursor++;
+- while (isset($lines[$cursor]) && !strlen(trim($lines[$cursor]))) {
+- $num_lines++;
+- $cursor++;
+- }
+- }
++// if ($num_lines) {
++// $cursor++;
++// while (isset($lines[$cursor]) && !strlen(trim($lines[$cursor]))) {
++// $num_lines++;
++// $cursor++;
++// }
++// }
+
+ return $num_lines;
+ }
\ No newline at end of file
diff --git a/tools/phabricator/upgrade-bitnami-phabricator.sh b/tools/phabricator/upgrade-bitnami-phabricator.sh
new file mode 100644
index 0000000..d302f87
--- /dev/null
+++ b/tools/phabricator/upgrade-bitnami-phabricator.sh
@@ -0,0 +1,28 @@
+CONTROL_PHPFPM_SCRIPT=/opt/bitnami/php/scripts/ctl.sh
+PHABRICATOR_PATH=/opt/bitnami/apps/phabricator
+
+CONTROL_PHABRICATOR_SCRIPT=$PHABRICATOR_PATH/scripts/ctl.sh
+
+ echo "Stop phabricator"
+ $CONTROL_PHABRICATOR_SCRIPT stop
+ $CONTROL_PHPFPM_SCRIPT stop
+
+ echo "Upgrade arcanist"
+ cd $PHABRICATOR_PATH/arcanist
+ git pull origin stable --rebase
+
+ echo "Upgrade libphutil"
+ cd $PHABRICATOR_PATH/libphutil
+ git pull origin stable --rebase
+
+ echo "Upgrade htdocs"
+ cd $PHABRICATOR_PATH/htdocs
+ git pull origin stable --rebase
+
+ echo "Upgrade storage"
+ $PHABRICATOR_PATH/htdocs/bin/storage upgrade
+
+
+ echo "Start phabricator"
+ $CONTROL_PHABRICATOR_SCRIPT start
+ $CONTROL_PHPFPM_SCRIPT start
diff --git a/tools/phabricator/xreate-frontend/XreateFrontendApplication.php b/tools/phabricator/xreate-frontend/XreateFrontendApplication.php
new file mode 100644
index 0000000..ba8e085
--- /dev/null
+++ b/tools/phabricator/xreate-frontend/XreateFrontendApplication.php
@@ -0,0 +1,46 @@
+<?php
+
+final class XreateFrontendApplication extends PhabricatorApplication {
+ public function getIcon() {
+ return 'fa-magnet';
+ }
+
+ public function getTitleGlyph() {
+ return "\xE2\x8F\x9A";
+ }
+
+ public function getFlavorText() {
+ return pht('Xreate frontend');
+ }
+
+ public function getApplicationGroup() {
+ return self::GROUP_DEVELOPER;
+ }
+
+ public function isPrototype() {
+ return false;
+ }
+
+ public function getApplicationOrder() {
+ return 0.110;
+ }
+
+ public function getName() {
+ return 'Xreate Frontend';
+ }
+
+ public function getShortDescription() {
+ return 'Executes xreate scripts';
+ }
+
+ public function getBaseURI() {
+ return '/xreate/';
+ }
+
+ public function getRoutes() {
+ return array(
+ '/xreate/' => 'XreateFrontendController'
+ );
+ }
+
+}
diff --git a/tools/phabricator/xreate-frontend/XreateFrontendController.php b/tools/phabricator/xreate-frontend/XreateFrontendController.php
new file mode 100644
index 0000000..5e78cd9
--- /dev/null
+++ b/tools/phabricator/xreate-frontend/XreateFrontendController.php
@@ -0,0 +1,40 @@
+<?php
+
+class XreateFrontendController extends PhabricatorController{
+ public function handleRequest(AphrontRequest $request) {
+ $view = $this->newPage();
+
+ require_celerity_resource('xreate-syntax-check');
+
+ $actions = new PHUIButtonBarView();
+ $button = id(new PHUIButtonView())
+ ->setTag('a')
+ ->setID("check-syntax")
+ ->setColor(PHUIButtonView::SIMPLE)
+ ->setTitle('Check syntax')
+ ->setText('Check syntax');
+
+ $actions->addButton($button);
+
+ $view->appendChild(phutil_tag(
+ 'div',
+ array(),
+ array(
+ $actions,
+ phutil_tag('br'),
+ phutil_tag('textarea', array('id'=>'program')),
+ phutil_tag('textarea', array('id'=>'log'))
+ )));
+
+ return $view;
+ }
+}
+
+/*
+ * return hsprintf(
+ '%s%s%s',
+ parent::getHead(),
+ $font_css,
+ $response->renderSingleResource('javelin-magical-init', 'phabricator'));
+ *
+ * */
\ No newline at end of file
diff --git a/tools/phabricator/xreate-frontend/resources/js/xreate-client-syntax-check.js b/tools/phabricator/xreate-frontend/resources/js/xreate-client-syntax-check.js
new file mode 100644
index 0000000..03b4453
--- /dev/null
+++ b/tools/phabricator/xreate-frontend/resources/js/xreate-client-syntax-check.js
@@ -0,0 +1,21 @@
+/**
+ * @requires xreate-interface
+ * @provides xreate-syntax-check
+ * @do-not-minify
+ */
+function checkSyntax(input, callbackResponce) {
+ var transport = new Thrift.Transport("http://localhost:9090");
+ var protocol = new Thrift.Protocol(transport);
+ var client = new AbstractXreateServerClient(protocol);
+
+ client.checkSyntax(input, callbackResponce);
+}
+
+document.addEventListener('DOMContentLoaded', function(){
+ $("#check-syntax").click(function(){
+ var program = $("#program").val();
+ checkSyntax(program, function(responce){
+ $("#log").val(responce.output);
+ });
+ })
+}, false);
diff --git a/tools/phabricator/xreate-frontend/resources/vendors/jquery/jquery-2.2.2.min.js b/tools/phabricator/xreate-frontend/resources/vendors/jquery/jquery-2.2.2.min.js
new file mode 100644
index 0000000..3b7a7f2
--- /dev/null
+++ b/tools/phabricator/xreate-frontend/resources/vendors/jquery/jquery-2.2.2.min.js
@@ -0,0 +1,4 @@
+/*! jQuery v2.2.2 | (c) jQuery Foundation | jquery.org/license */
+!function(a,b){"object"==typeof module&&"object"==typeof module.exports?module.exports=a.document?b(a,!0):function(a){if(!a.document)throw new Error("jQuery requires a window with a document");return b(a)}:b(a)}("undefined"!=typeof window?window:this,function(a,b){var c=[],d=a.document,e=c.slice,f=c.concat,g=c.push,h=c.indexOf,i={},j=i.toString,k=i.hasOwnProperty,l={},m="2.2.2",n=function(a,b){return new n.fn.init(a,b)},o=/^[\s\uFEFF\xA0]+|[\s\uFEFF\xA0]+$/g,p=/^-ms-/,q=/-([\da-z])/gi,r=function(a,b){return b.toUpperCase()};n.fn=n.prototype={jquery:m,constructor:n,selector:"",length:0,toArray:function(){return e.call(this)},get:function(a){return null!=a?0>a?this[a+this.length]:this[a]:e.call(this)},pushStack:function(a){var b=n.merge(this.constructor(),a);return b.prevObject=this,b.context=this.context,b},each:function(a){return n.each(this,a)},map:function(a){return this.pushStack(n.map(this,function(b,c){return a.call(b,c,b)}))},slice:function(){return this.pushStack(e.apply(this,arguments))},first:function(){return this.eq(0)},last:function(){return this.eq(-1)},eq:function(a){var b=this.length,c=+a+(0>a?b:0);return this.pushStack(c>=0&&b>c?[this[c]]:[])},end:function(){return this.prevObject||this.constructor()},push:g,sort:c.sort,splice:c.splice},n.extend=n.fn.extend=function(){var a,b,c,d,e,f,g=arguments[0]||{},h=1,i=arguments.length,j=!1;for("boolean"==typeof g&&(j=g,g=arguments[h]||{},h++),"object"==typeof g||n.isFunction(g)||(g={}),h===i&&(g=this,h--);i>h;h++)if(null!=(a=arguments[h]))for(b in a)c=g[b],d=a[b],g!==d&&(j&&d&&(n.isPlainObject(d)||(e=n.isArray(d)))?(e?(e=!1,f=c&&n.isArray(c)?c:[]):f=c&&n.isPlainObject(c)?c:{},g[b]=n.extend(j,f,d)):void 0!==d&&(g[b]=d));return g},n.extend({expando:"jQuery"+(m+Math.random()).replace(/\D/g,""),isReady:!0,error:function(a){throw new Error(a)},noop:function(){},isFunction:function(a){return"function"===n.type(a)},isArray:Array.isArray,isWindow:function(a){return null!=a&&a===a.window},isNumeric:function(a){var b=a&&a.toString();return!n.isArray(a)&&b-parseFloat(b)+1>=0},isPlainObject:function(a){var b;if("object"!==n.type(a)||a.nodeType||n.isWindow(a))return!1;if(a.constructor&&!k.call(a,"constructor")&&!k.call(a.constructor.prototype||{},"isPrototypeOf"))return!1;for(b in a);return void 0===b||k.call(a,b)},isEmptyObject:function(a){var b;for(b in a)return!1;return!0},type:function(a){return null==a?a+"":"object"==typeof a||"function"==typeof a?i[j.call(a)]||"object":typeof a},globalEval:function(a){var b,c=eval;a=n.trim(a),a&&(1===a.indexOf("use strict")?(b=d.createElement("script"),b.text=a,d.head.appendChild(b).parentNode.removeChild(b)):c(a))},camelCase:function(a){return a.replace(p,"ms-").replace(q,r)},nodeName:function(a,b){return a.nodeName&&a.nodeName.toLowerCase()===b.toLowerCase()},each:function(a,b){var c,d=0;if(s(a)){for(c=a.length;c>d;d++)if(b.call(a[d],d,a[d])===!1)break}else for(d in a)if(b.call(a[d],d,a[d])===!1)break;return a},trim:function(a){return null==a?"":(a+"").replace(o,"")},makeArray:function(a,b){var c=b||[];return null!=a&&(s(Object(a))?n.merge(c,"string"==typeof a?[a]:a):g.call(c,a)),c},inArray:function(a,b,c){return null==b?-1:h.call(b,a,c)},merge:function(a,b){for(var c=+b.length,d=0,e=a.length;c>d;d++)a[e++]=b[d];return a.length=e,a},grep:function(a,b,c){for(var d,e=[],f=0,g=a.length,h=!c;g>f;f++)d=!b(a[f],f),d!==h&&e.push(a[f]);return e},map:function(a,b,c){var d,e,g=0,h=[];if(s(a))for(d=a.length;d>g;g++)e=b(a[g],g,c),null!=e&&h.push(e);else for(g in a)e=b(a[g],g,c),null!=e&&h.push(e);return f.apply([],h)},guid:1,proxy:function(a,b){var c,d,f;return"string"==typeof b&&(c=a[b],b=a,a=c),n.isFunction(a)?(d=e.call(arguments,2),f=function(){return a.apply(b||this,d.concat(e.call(arguments)))},f.guid=a.guid=a.guid||n.guid++,f):void 0},now:Date.now,support:l}),"function"==typeof Symbol&&(n.fn[Symbol.iterator]=c[Symbol.iterator]),n.each("Boolean Number String Function Array Date RegExp Object Error Symbol".split(" "),function(a,b){i["[object "+b+"]"]=b.toLowerCase()});function s(a){var b=!!a&&"length"in a&&a.length,c=n.type(a);return"function"===c||n.isWindow(a)?!1:"array"===c||0===b||"number"==typeof b&&b>0&&b-1 in a}var t=function(a){var b,c,d,e,f,g,h,i,j,k,l,m,n,o,p,q,r,s,t,u="sizzle"+1*new Date,v=a.document,w=0,x=0,y=ga(),z=ga(),A=ga(),B=function(a,b){return a===b&&(l=!0),0},C=1<<31,D={}.hasOwnProperty,E=[],F=E.pop,G=E.push,H=E.push,I=E.slice,J=function(a,b){for(var c=0,d=a.length;d>c;c++)if(a[c]===b)return c;return-1},K="checked|selected|async|autofocus|autoplay|controls|defer|disabled|hidden|ismap|loop|multiple|open|readonly|required|scoped",L="[\\x20\\t\\r\\n\\f]",M="(?:\\\\.|[\\w-]|[^\\x00-\\xa0])+",N="\\["+L+"*("+M+")(?:"+L+"*([*^$|!~]?=)"+L+"*(?:'((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\"|("+M+"))|)"+L+"*\\]",O=":("+M+")(?:\\((('((?:\\\\.|[^\\\\'])*)'|\"((?:\\\\.|[^\\\\\"])*)\")|((?:\\\\.|[^\\\\()[\\]]|"+N+")*)|.*)\\)|)",P=new RegExp(L+"+","g"),Q=new RegExp("^"+L+"+|((?:^|[^\\\\])(?:\\\\.)*)"+L+"+$","g"),R=new RegExp("^"+L+"*,"+L+"*"),S=new RegExp("^"+L+"*([>+~]|"+L+")"+L+"*"),T=new RegExp("="+L+"*([^\\]'\"]*?)"+L+"*\\]","g"),U=new RegExp(O),V=new RegExp("^"+M+"$"),W={ID:new RegExp("^#("+M+")"),CLASS:new RegExp("^\\.("+M+")"),TAG:new RegExp("^("+M+"|[*])"),ATTR:new RegExp("^"+N),PSEUDO:new RegExp("^"+O),CHILD:new RegExp("^:(only|first|last|nth|nth-last)-(child|of-type)(?:\\("+L+"*(even|odd|(([+-]|)(\\d*)n|)"+L+"*(?:([+-]|)"+L+"*(\\d+)|))"+L+"*\\)|)","i"),bool:new RegExp("^(?:"+K+")$","i"),needsContext:new RegExp("^"+L+"*[>+~]|:(even|odd|eq|gt|lt|nth|first|last)(?:\\("+L+"*((?:-\\d)?\\d*)"+L+"*\\)|)(?=[^-]|$)","i")},X=/^(?:input|select|textarea|button)$/i,Y=/^h\d$/i,Z=/^[^{]+\{\s*\[native \w/,$=/^(?:#([\w-]+)|(\w+)|\.([\w-]+))$/,_=/[+~]/,aa=/'|\\/g,ba=new RegExp("\\\\([\\da-f]{1,6}"+L+"?|("+L+")|.)","ig"),ca=function(a,b,c){var d="0x"+b-65536;return d!==d||c?b:0>d?String.fromCharCode(d+65536):String.fromCharCode(d>>10|55296,1023&d|56320)},da=function(){m()};try{H.apply(E=I.call(v.childNodes),v.childNodes),E[v.childNodes.length].nodeType}catch(ea){H={apply:E.length?function(a,b){G.apply(a,I.call(b))}:function(a,b){var c=a.length,d=0;while(a[c++]=b[d++]);a.length=c-1}}}function fa(a,b,d,e){var f,h,j,k,l,o,r,s,w=b&&b.ownerDocument,x=b?b.nodeType:9;if(d=d||[],"string"!=typeof a||!a||1!==x&&9!==x&&11!==x)return d;if(!e&&((b?b.ownerDocument||b:v)!==n&&m(b),b=b||n,p)){if(11!==x&&(o=$.exec(a)))if(f=o[1]){if(9===x){if(!(j=b.getElementById(f)))return d;if(j.id===f)return d.push(j),d}else if(w&&(j=w.getElementById(f))&&t(b,j)&&j.id===f)return d.push(j),d}else{if(o[2])return H.apply(d,b.getElementsByTagName(a)),d;if((f=o[3])&&c.getElementsByClassName&&b.getElementsByClassName)return H.apply(d,b.getElementsByClassName(f)),d}if(c.qsa&&!A[a+" "]&&(!q||!q.test(a))){if(1!==x)w=b,s=a;else if("object"!==b.nodeName.toLowerCase()){(k=b.getAttribute("id"))?k=k.replace(aa,"\\$&"):b.setAttribute("id",k=u),r=g(a),h=r.length,l=V.test(k)?"#"+k:"[id='"+k+"']";while(h--)r[h]=l+" "+qa(r[h]);s=r.join(","),w=_.test(a)&&oa(b.parentNode)||b}if(s)try{return H.apply(d,w.querySelectorAll(s)),d}catch(y){}finally{k===u&&b.removeAttribute("id")}}}return i(a.replace(Q,"$1"),b,d,e)}function ga(){var a=[];function b(c,e){return a.push(c+" ")>d.cacheLength&&delete b[a.shift()],b[c+" "]=e}return b}function ha(a){return a[u]=!0,a}function ia(a){var b=n.createElement("div");try{return!!a(b)}catch(c){return!1}finally{b.parentNode&&b.parentNode.removeChild(b),b=null}}function ja(a,b){var c=a.split("|"),e=c.length;while(e--)d.attrHandle[c[e]]=b}function ka(a,b){var c=b&&a,d=c&&1===a.nodeType&&1===b.nodeType&&(~b.sourceIndex||C)-(~a.sourceIndex||C);if(d)return d;if(c)while(c=c.nextSibling)if(c===b)return-1;return a?1:-1}function la(a){return function(b){var c=b.nodeName.toLowerCase();return"input"===c&&b.type===a}}function ma(a){return function(b){var c=b.nodeName.toLowerCase();return("input"===c||"button"===c)&&b.type===a}}function na(a){return ha(function(b){return b=+b,ha(function(c,d){var e,f=a([],c.length,b),g=f.length;while(g--)c[e=f[g]]&&(c[e]=!(d[e]=c[e]))})})}function oa(a){return a&&"undefined"!=typeof a.getElementsByTagName&&a}c=fa.support={},f=fa.isXML=function(a){var b=a&&(a.ownerDocument||a).documentElement;return b?"HTML"!==b.nodeName:!1},m=fa.setDocument=function(a){var b,e,g=a?a.ownerDocument||a:v;return g!==n&&9===g.nodeType&&g.documentElement?(n=g,o=n.documentElement,p=!f(n),(e=n.defaultView)&&e.top!==e&&(e.addEventListener?e.addEventListener("unload",da,!1):e.attachEvent&&e.attachEvent("onunload",da)),c.attributes=ia(function(a){return a.className="i",!a.getAttribute("className")}),c.getElementsByTagName=ia(function(a){return a.appendChild(n.createComment("")),!a.getElementsByTagName("*").length}),c.getElementsByClassName=Z.test(n.getElementsByClassName),c.getById=ia(function(a){return o.appendChild(a).id=u,!n.getElementsByName||!n.getElementsByName(u).length}),c.getById?(d.find.ID=function(a,b){if("undefined"!=typeof b.getElementById&&p){var c=b.getElementById(a);return c?[c]:[]}},d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){return a.getAttribute("id")===b}}):(delete d.find.ID,d.filter.ID=function(a){var b=a.replace(ba,ca);return function(a){var c="undefined"!=typeof a.getAttributeNode&&a.getAttributeNode("id");return c&&c.value===b}}),d.find.TAG=c.getElementsByTagName?function(a,b){return"undefined"!=typeof b.getElementsByTagName?b.getElementsByTagName(a):c.qsa?b.querySelectorAll(a):void 0}:function(a,b){var c,d=[],e=0,f=b.getElementsByTagName(a);if("*"===a){while(c=f[e++])1===c.nodeType&&d.push(c);return d}return f},d.find.CLASS=c.getElementsByClassName&&function(a,b){return"undefined"!=typeof b.getElementsByClassName&&p?b.getElementsByClassName(a):void 0},r=[],q=[],(c.qsa=Z.test(n.querySelectorAll))&&(ia(function(a){o.appendChild(a).innerHTML="<a id='"+u+"'></a><select id='"+u+"-\r\\' msallowcapture=''><option selected=''></option></select>",a.querySelectorAll("[msallowcapture^='']").length&&q.push("[*^$]="+L+"*(?:''|\"\")"),a.querySelectorAll("[selected]").length||q.push("\\["+L+"*(?:value|"+K+")"),a.querySelectorAll("[id~="+u+"-]").length||q.push("~="),a.querySelectorAll(":checked").length||q.push(":checked"),a.querySelectorAll("a#"+u+"+*").length||q.push(".#.+[+~]")}),ia(function(a){var b=n.createElement("input");b.setAttribute("type","hidden"),a.appendChild(b).setAttribute("name","D"),a.querySelectorAll("[name=d]").length&&q.push("name"+L+"*[*^$|!~]?="),a.querySelectorAll(":enabled").length||q.push(":enabled",":disabled"),a.querySelectorAll("*,:x"),q.push(",.*:")})),(c.matchesSelector=Z.test(s=o.matches||o.webkitMatchesSelector||o.mozMatchesSelector||o.oMatchesSelector||o.msMatchesSelector))&&ia(function(a){c.disconnectedMatch=s.call(a,"div"),s.call(a,"[s!='']:x"),r.push("!=",O)}),q=q.length&&new RegExp(q.join("|")),r=r.length&&new RegExp(r.join("|")),b=Z.test(o.compareDocumentPosition),t=b||Z.test(o.contains)?function(a,b){var c=9===a.nodeType?a.documentElement:a,d=b&&b.parentNode;return a===d||!(!d||1!==d.nodeType||!(c.contains?c.contains(d):a.compareDocumentPosition&&16&a.compareDocumentPosition(d)))}:function(a,b){if(b)while(b=b.parentNode)if(b===a)return!0;return!1},B=b?function(a,b){if(a===b)return l=!0,0;var d=!a.compareDocumentPosition-!b.compareDocumentPosition;return d?d:(d=(a.ownerDocument||a)===(b.ownerDocument||b)?a.compareDocumentPosition(b):1,1&d||!c.sortDetached&&b.compareDocumentPosition(a)===d?a===n||a.ownerDocument===v&&t(v,a)?-1:b===n||b.ownerDocument===v&&t(v,b)?1:k?J(k,a)-J(k,b):0:4&d?-1:1)}:function(a,b){if(a===b)return l=!0,0;var c,d=0,e=a.parentNode,f=b.parentNode,g=[a],h=[b];if(!e||!f)return a===n?-1:b===n?1:e?-1:f?1:k?J(k,a)-J(k,b):0;if(e===f)return ka(a,b);c=a;while(c=c.parentNode)g.unshift(c);c=b;while(c=c.parentNode)h.unshift(c);while(g[d]===h[d])d++;return d?ka(g[d],h[d]):g[d]===v?-1:h[d]===v?1:0},n):n},fa.matches=function(a,b){return fa(a,null,null,b)},fa.matchesSelector=function(a,b){if((a.ownerDocument||a)!==n&&m(a),b=b.replace(T,"='$1']"),c.matchesSelector&&p&&!A[b+" "]&&(!r||!r.test(b))&&(!q||!q.test(b)))try{var d=s.call(a,b);if(d||c.disconnectedMatch||a.document&&11!==a.document.nodeType)return d}catch(e){}return fa(b,n,null,[a]).length>0},fa.contains=function(a,b){return(a.ownerDocument||a)!==n&&m(a),t(a,b)},fa.attr=function(a,b){(a.ownerDocument||a)!==n&&m(a);var e=d.attrHandle[b.toLowerCase()],f=e&&D.call(d.attrHandle,b.toLowerCase())?e(a,b,!p):void 0;return void 0!==f?f:c.attributes||!p?a.getAttribute(b):(f=a.getAttributeNode(b))&&f.specified?f.value:null},fa.error=function(a){throw new Error("Syntax error, unrecognized expression: "+a)},fa.uniqueSort=function(a){var b,d=[],e=0,f=0;if(l=!c.detectDuplicates,k=!c.sortStable&&a.slice(0),a.sort(B),l){while(b=a[f++])b===a[f]&&(e=d.push(f));while(e--)a.splice(d[e],1)}return k=null,a},e=fa.getText=function(a){var b,c="",d=0,f=a.nodeType;if(f){if(1===f||9===f||11===f){if("string"==typeof a.textContent)return a.textContent;for(a=a.firstChild;a;a=a.nextSibling)c+=e(a)}else if(3===f||4===f)return a.nodeValue}else while(b=a[d++])c+=e(b);return c},d=fa.selectors={cacheLength:50,createPseudo:ha,match:W,attrHandle:{},find:{},relative:{">":{dir:"parentNode",first:!0}," ":{dir:"parentNode"},"+":{dir:"previousSibling",first:!0},"~":{dir:"previousSibling"}},preFilter:{ATTR:function(a){return a[1]=a[1].replace(ba,ca),a[3]=(a[3]||a[4]||a[5]||"").replace(ba,ca),"~="===a[2]&&(a[3]=" "+a[3]+" "),a.slice(0,4)},CHILD:function(a){return a[1]=a[1].toLowerCase(),"nth"===a[1].slice(0,3)?(a[3]||fa.error(a[0]),a[4]=+(a[4]?a[5]+(a[6]||1):2*("even"===a[3]||"odd"===a[3])),a[5]=+(a[7]+a[8]||"odd"===a[3])):a[3]&&fa.error(a[0]),a},PSEUDO:function(a){var b,c=!a[6]&&a[2];return W.CHILD.test(a[0])?null:(a[3]?a[2]=a[4]||a[5]||"":c&&U.test(c)&&(b=g(c,!0))&&(b=c.indexOf(")",c.length-b)-c.length)&&(a[0]=a[0].slice(0,b),a[2]=c.slice(0,b)),a.slice(0,3))}},filter:{TAG:function(a){var b=a.replace(ba,ca).toLowerCase();return"*"===a?function(){return!0}:function(a){return a.nodeName&&a.nodeName.toLowerCase()===b}},CLASS:function(a){var b=y[a+" "];return b||(b=new RegExp("(^|"+L+")"+a+"("+L+"|$)"))&&y(a,function(a){return b.test("string"==typeof a.className&&a.className||"undefined"!=typeof a.getAttribute&&a.getAttribute("class")||"")})},ATTR:function(a,b,c){return function(d){var e=fa.attr(d,a);return null==e?"!="===b:b?(e+="","="===b?e===c:"!="===b?e!==c:"^="===b?c&&0===e.indexOf(c):"*="===b?c&&e.indexOf(c)>-1:"$="===b?c&&e.slice(-c.length)===c:"~="===b?(" "+e.replace(P," ")+" ").indexOf(c)>-1:"|="===b?e===c||e.slice(0,c.length+1)===c+"-":!1):!0}},CHILD:function(a,b,c,d,e){var f="nth"!==a.slice(0,3),g="last"!==a.slice(-4),h="of-type"===b;return 1===d&&0===e?function(a){return!!a.parentNode}:function(b,c,i){var j,k,l,m,n,o,p=f!==g?"nextSibling":"previousSibling",q=b.parentNode,r=h&&b.nodeName.toLowerCase(),s=!i&&!h,t=!1;if(q){if(f){while(p){m=b;while(m=m[p])if(h?m.nodeName.toLowerCase()===r:1===m.nodeType)return!1;o=p="only"===a&&!o&&"nextSibling"}return!0}if(o=[g?q.firstChild:q.lastChild],g&&s){m=q,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n&&j[2],m=n&&q.childNodes[n];while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if(1===m.nodeType&&++t&&m===b){k[a]=[w,n,t];break}}else if(s&&(m=b,l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),j=k[a]||[],n=j[0]===w&&j[1],t=n),t===!1)while(m=++n&&m&&m[p]||(t=n=0)||o.pop())if((h?m.nodeName.toLowerCase()===r:1===m.nodeType)&&++t&&(s&&(l=m[u]||(m[u]={}),k=l[m.uniqueID]||(l[m.uniqueID]={}),k[a]=[w,t]),m===b))break;return t-=e,t===d||t%d===0&&t/d>=0}}},PSEUDO:function(a,b){var c,e=d.pseudos[a]||d.setFilters[a.toLowerCase()]||fa.error("unsupported pseudo: "+a);return e[u]?e(b):e.length>1?(c=[a,a,"",b],d.setFilters.hasOwnProperty(a.toLowerCase())?ha(function(a,c){var d,f=e(a,b),g=f.length;while(g--)d=J(a,f[g]),a[d]=!(c[d]=f[g])}):function(a){return e(a,0,c)}):e}},pseudos:{not:ha(function(a){var b=[],c=[],d=h(a.replace(Q,"$1"));return d[u]?ha(function(a,b,c,e){var f,g=d(a,null,e,[]),h=a.length;while(h--)(f=g[h])&&(a[h]=!(b[h]=f))}):function(a,e,f){return b[0]=a,d(b,null,f,c),b[0]=null,!c.pop()}}),has:ha(function(a){return function(b){return fa(a,b).length>0}}),contains:ha(function(a){return a=a.replace(ba,ca),function(b){return(b.textContent||b.innerText||e(b)).indexOf(a)>-1}}),lang:ha(function(a){return V.test(a||"")||fa.error("unsupported lang: "+a),a=a.replace(ba,ca).toLowerCase(),function(b){var c;do if(c=p?b.lang:b.getAttribute("xml:lang")||b.getAttribute("lang"))return c=c.toLowerCase(),c===a||0===c.indexOf(a+"-");while((b=b.parentNode)&&1===b.nodeType);return!1}}),target:function(b){var c=a.location&&a.location.hash;return c&&c.slice(1)===b.id},root:function(a){return a===o},focus:function(a){return a===n.activeElement&&(!n.hasFocus||n.hasFocus())&&!!(a.type||a.href||~a.tabIndex)},enabled:function(a){return a.disabled===!1},disabled:function(a){return a.disabled===!0},checked:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&!!a.checked||"option"===b&&!!a.selected},selected:function(a){return a.parentNode&&a.parentNode.selectedIndex,a.selected===!0},empty:function(a){for(a=a.firstChild;a;a=a.nextSibling)if(a.nodeType<6)return!1;return!0},parent:function(a){return!d.pseudos.empty(a)},header:function(a){return Y.test(a.nodeName)},input:function(a){return X.test(a.nodeName)},button:function(a){var b=a.nodeName.toLowerCase();return"input"===b&&"button"===a.type||"button"===b},text:function(a){var b;return"input"===a.nodeName.toLowerCase()&&"text"===a.type&&(null==(b=a.getAttribute("type"))||"text"===b.toLowerCase())},first:na(function(){return[0]}),last:na(function(a,b){return[b-1]}),eq:na(function(a,b,c){return[0>c?c+b:c]}),even:na(function(a,b){for(var c=0;b>c;c+=2)a.push(c);return a}),odd:na(function(a,b){for(var c=1;b>c;c+=2)a.push(c);return a}),lt:na(function(a,b,c){for(var d=0>c?c+b:c;--d>=0;)a.push(d);return a}),gt:na(function(a,b,c){for(var d=0>c?c+b:c;++d<b;)a.push(d);return a})}},d.pseudos.nth=d.pseudos.eq;for(b in{radio:!0,checkbox:!0,file:!0,password:!0,image:!0})d.pseudos[b]=la(b);for(b in{submit:!0,reset:!0})d.pseudos[b]=ma(b);function pa(){}pa.prototype=d.filters=d.pseudos,d.setFilters=new pa,g=fa.tokenize=function(a,b){var c,e,f,g,h,i,j,k=z[a+" "];if(k)return b?0:k.slice(0);h=a,i=[],j=d.preFilter;while(h){c&&!(e=R.exec(h))||(e&&(h=h.slice(e[0].length)||h),i.push(f=[])),c=!1,(e=S.exec(h))&&(c=e.shift(),f.push({value:c,type:e[0].replace(Q," ")}),h=h.slice(c.length));for(g in d.filter)!(e=W[g].exec(h))||j[g]&&!(e=j[g](e))||(c=e.shift(),f.push({value:c,type:g,matches:e}),h=h.slice(c.length));if(!c)break}return b?h.length:h?fa.error(a):z(a,i).slice(0)};function qa(a){for(var b=0,c=a.length,d="";c>b;b++)d+=a[b].value;return d}function ra(a,b,c){var d=b.dir,e=c&&"parentNode"===d,f=x++;return b.first?function(b,c,f){while(b=b[d])if(1===b.nodeType||e)return a(b,c,f)}:function(b,c,g){var h,i,j,k=[w,f];if(g){while(b=b[d])if((1===b.nodeType||e)&&a(b,c,g))return!0}else while(b=b[d])if(1===b.nodeType||e){if(j=b[u]||(b[u]={}),i=j[b.uniqueID]||(j[b.uniqueID]={}),(h=i[d])&&h[0]===w&&h[1]===f)return k[2]=h[2];if(i[d]=k,k[2]=a(b,c,g))return!0}}}function sa(a){return a.length>1?function(b,c,d){var e=a.length;while(e--)if(!a[e](b,c,d))return!1;return!0}:a[0]}function ta(a,b,c){for(var d=0,e=b.length;e>d;d++)fa(a,b[d],c);return c}function ua(a,b,c,d,e){for(var f,g=[],h=0,i=a.length,j=null!=b;i>h;h++)(f=a[h])&&(c&&!c(f,d,e)||(g.push(f),j&&b.push(h)));return g}function va(a,b,c,d,e,f){return d&&!d[u]&&(d=va(d)),e&&!e[u]&&(e=va(e,f)),ha(function(f,g,h,i){var j,k,l,m=[],n=[],o=g.length,p=f||ta(b||"*",h.nodeType?[h]:h,[]),q=!a||!f&&b?p:ua(p,m,a,h,i),r=c?e||(f?a:o||d)?[]:g:q;if(c&&c(q,r,h,i),d){j=ua(r,n),d(j,[],h,i),k=j.length;while(k--)(l=j[k])&&(r[n[k]]=!(q[n[k]]=l))}if(f){if(e||a){if(e){j=[],k=r.length;while(k--)(l=r[k])&&j.push(q[k]=l);e(null,r=[],j,i)}k=r.length;while(k--)(l=r[k])&&(j=e?J(f,l):m[k])>-1&&(f[j]=!(g[j]=l))}}else r=ua(r===g?r.splice(o,r.length):r),e?e(null,g,r,i):H.apply(g,r)})}function wa(a){for(var b,c,e,f=a.length,g=d.relative[a[0].type],h=g||d.relative[" "],i=g?1:0,k=ra(function(a){return a===b},h,!0),l=ra(function(a){return J(b,a)>-1},h,!0),m=[function(a,c,d){var e=!g&&(d||c!==j)||((b=c).nodeType?k(a,c,d):l(a,c,d));return b=null,e}];f>i;i++)if(c=d.relative[a[i].type])m=[ra(sa(m),c)];else{if(c=d.filter[a[i].type].apply(null,a[i].matches),c[u]){for(e=++i;f>e;e++)if(d.relative[a[e].type])break;return va(i>1&&sa(m),i>1&&qa(a.slice(0,i-1).concat({value:" "===a[i-2].type?"*":""})).replace(Q,"$1"),c,e>i&&wa(a.slice(i,e)),f>e&&wa(a=a.slice(e)),f>e&&qa(a))}m.push(c)}return sa(m)}function xa(a,b){var c=b.length>0,e=a.length>0,f=function(f,g,h,i,k){var l,o,q,r=0,s="0",t=f&&[],u=[],v=j,x=f||e&&d.find.TAG("*",k),y=w+=null==v?1:Math.random()||.1,z=x.length;for(k&&(j=g===n||g||k);s!==z&&null!=(l=x[s]);s++){if(e&&l){o=0,g||l.ownerDocument===n||(m(l),h=!p);while(q=a[o++])if(q(l,g||n,h)){i.push(l);break}k&&(w=y)}c&&((l=!q&&l)&&r--,f&&t.push(l))}if(r+=s,c&&s!==r){o=0;while(q=b[o++])q(t,u,g,h);if(f){if(r>0)while(s--)t[s]||u[s]||(u[s]=F.call(i));u=ua(u)}H.apply(i,u),k&&!f&&u.length>0&&r+b.length>1&&fa.uniqueSort(i)}return k&&(w=y,j=v),t};return c?ha(f):f}return h=fa.compile=function(a,b){var c,d=[],e=[],f=A[a+" "];if(!f){b||(b=g(a)),c=b.length;while(c--)f=wa(b[c]),f[u]?d.push(f):e.push(f);f=A(a,xa(e,d)),f.selector=a}return f},i=fa.select=function(a,b,e,f){var i,j,k,l,m,n="function"==typeof a&&a,o=!f&&g(a=n.selector||a);if(e=e||[],1===o.length){if(j=o[0]=o[0].slice(0),j.length>2&&"ID"===(k=j[0]).type&&c.getById&&9===b.nodeType&&p&&d.relative[j[1].type]){if(b=(d.find.ID(k.matches[0].replace(ba,ca),b)||[])[0],!b)return e;n&&(b=b.parentNode),a=a.slice(j.shift().value.length)}i=W.needsContext.test(a)?0:j.length;while(i--){if(k=j[i],d.relative[l=k.type])break;if((m=d.find[l])&&(f=m(k.matches[0].replace(ba,ca),_.test(j[0].type)&&oa(b.parentNode)||b))){if(j.splice(i,1),a=f.length&&qa(j),!a)return H.apply(e,f),e;break}}}return(n||h(a,o))(f,b,!p,e,!b||_.test(a)&&oa(b.parentNode)||b),e},c.sortStable=u.split("").sort(B).join("")===u,c.detectDuplicates=!!l,m(),c.sortDetached=ia(function(a){return 1&a.compareDocumentPosition(n.createElement("div"))}),ia(function(a){return a.innerHTML="<a href='#'></a>","#"===a.firstChild.getAttribute("href")})||ja("type|href|height|width",function(a,b,c){return c?void 0:a.getAttribute(b,"type"===b.toLowerCase()?1:2)}),c.attributes&&ia(function(a){return a.innerHTML="<input/>",a.firstChild.setAttribute("value",""),""===a.firstChild.getAttribute("value")})||ja("value",function(a,b,c){return c||"input"!==a.nodeName.toLowerCase()?void 0:a.defaultValue}),ia(function(a){return null==a.getAttribute("disabled")})||ja(K,function(a,b,c){var d;return c?void 0:a[b]===!0?b.toLowerCase():(d=a.getAttributeNode(b))&&d.specified?d.value:null}),fa}(a);n.find=t,n.expr=t.selectors,n.expr[":"]=n.expr.pseudos,n.uniqueSort=n.unique=t.uniqueSort,n.text=t.getText,n.isXMLDoc=t.isXML,n.contains=t.contains;var u=function(a,b,c){var d=[],e=void 0!==c;while((a=a[b])&&9!==a.nodeType)if(1===a.nodeType){if(e&&n(a).is(c))break;d.push(a)}return d},v=function(a,b){for(var c=[];a;a=a.nextSibling)1===a.nodeType&&a!==b&&c.push(a);return c},w=n.expr.match.needsContext,x=/^<([\w-]+)\s*\/?>(?:<\/\1>|)$/,y=/^.[^:#\[\.,]*$/;function z(a,b,c){if(n.isFunction(b))return n.grep(a,function(a,d){return!!b.call(a,d,a)!==c});if(b.nodeType)return n.grep(a,function(a){return a===b!==c});if("string"==typeof b){if(y.test(b))return n.filter(b,a,c);b=n.filter(b,a)}return n.grep(a,function(a){return h.call(b,a)>-1!==c})}n.filter=function(a,b,c){var d=b[0];return c&&(a=":not("+a+")"),1===b.length&&1===d.nodeType?n.find.matchesSelector(d,a)?[d]:[]:n.find.matches(a,n.grep(b,function(a){return 1===a.nodeType}))},n.fn.extend({find:function(a){var b,c=this.length,d=[],e=this;if("string"!=typeof a)return this.pushStack(n(a).filter(function(){for(b=0;c>b;b++)if(n.contains(e[b],this))return!0}));for(b=0;c>b;b++)n.find(a,e[b],d);return d=this.pushStack(c>1?n.unique(d):d),d.selector=this.selector?this.selector+" "+a:a,d},filter:function(a){return this.pushStack(z(this,a||[],!1))},not:function(a){return this.pushStack(z(this,a||[],!0))},is:function(a){return!!z(this,"string"==typeof a&&w.test(a)?n(a):a||[],!1).length}});var A,B=/^(?:\s*(<[\w\W]+>)[^>]*|#([\w-]*))$/,C=n.fn.init=function(a,b,c){var e,f;if(!a)return this;if(c=c||A,"string"==typeof a){if(e="<"===a[0]&&">"===a[a.length-1]&&a.length>=3?[null,a,null]:B.exec(a),!e||!e[1]&&b)return!b||b.jquery?(b||c).find(a):this.constructor(b).find(a);if(e[1]){if(b=b instanceof n?b[0]:b,n.merge(this,n.parseHTML(e[1],b&&b.nodeType?b.ownerDocument||b:d,!0)),x.test(e[1])&&n.isPlainObject(b))for(e in b)n.isFunction(this[e])?this[e](b[e]):this.attr(e,b[e]);return this}return f=d.getElementById(e[2]),f&&f.parentNode&&(this.length=1,this[0]=f),this.context=d,this.selector=a,this}return a.nodeType?(this.context=this[0]=a,this.length=1,this):n.isFunction(a)?void 0!==c.ready?c.ready(a):a(n):(void 0!==a.selector&&(this.selector=a.selector,this.context=a.context),n.makeArray(a,this))};C.prototype=n.fn,A=n(d);var D=/^(?:parents|prev(?:Until|All))/,E={children:!0,contents:!0,next:!0,prev:!0};n.fn.extend({has:function(a){var b=n(a,this),c=b.length;return this.filter(function(){for(var a=0;c>a;a++)if(n.contains(this,b[a]))return!0})},closest:function(a,b){for(var c,d=0,e=this.length,f=[],g=w.test(a)||"string"!=typeof a?n(a,b||this.context):0;e>d;d++)for(c=this[d];c&&c!==b;c=c.parentNode)if(c.nodeType<11&&(g?g.index(c)>-1:1===c.nodeType&&n.find.matchesSelector(c,a))){f.push(c);break}return this.pushStack(f.length>1?n.uniqueSort(f):f)},index:function(a){return a?"string"==typeof a?h.call(n(a),this[0]):h.call(this,a.jquery?a[0]:a):this[0]&&this[0].parentNode?this.first().prevAll().length:-1},add:function(a,b){return this.pushStack(n.uniqueSort(n.merge(this.get(),n(a,b))))},addBack:function(a){return this.add(null==a?this.prevObject:this.prevObject.filter(a))}});function F(a,b){while((a=a[b])&&1!==a.nodeType);return a}n.each({parent:function(a){var b=a.parentNode;return b&&11!==b.nodeType?b:null},parents:function(a){return u(a,"parentNode")},parentsUntil:function(a,b,c){return u(a,"parentNode",c)},next:function(a){return F(a,"nextSibling")},prev:function(a){return F(a,"previousSibling")},nextAll:function(a){return u(a,"nextSibling")},prevAll:function(a){return u(a,"previousSibling")},nextUntil:function(a,b,c){return u(a,"nextSibling",c)},prevUntil:function(a,b,c){return u(a,"previousSibling",c)},siblings:function(a){return v((a.parentNode||{}).firstChild,a)},children:function(a){return v(a.firstChild)},contents:function(a){return a.contentDocument||n.merge([],a.childNodes)}},function(a,b){n.fn[a]=function(c,d){var e=n.map(this,b,c);return"Until"!==a.slice(-5)&&(d=c),d&&"string"==typeof d&&(e=n.filter(d,e)),this.length>1&&(E[a]||n.uniqueSort(e),D.test(a)&&e.reverse()),this.pushStack(e)}});var G=/\S+/g;function H(a){var b={};return n.each(a.match(G)||[],function(a,c){b[c]=!0}),b}n.Callbacks=function(a){a="string"==typeof a?H(a):n.extend({},a);var b,c,d,e,f=[],g=[],h=-1,i=function(){for(e=a.once,d=b=!0;g.length;h=-1){c=g.shift();while(++h<f.length)f[h].apply(c[0],c[1])===!1&&a.stopOnFalse&&(h=f.length,c=!1)}a.memory||(c=!1),b=!1,e&&(f=c?[]:"")},j={add:function(){return f&&(c&&!b&&(h=f.length-1,g.push(c)),function d(b){n.each(b,function(b,c){n.isFunction(c)?a.unique&&j.has(c)||f.push(c):c&&c.length&&"string"!==n.type(c)&&d(c)})}(arguments),c&&!b&&i()),this},remove:function(){return n.each(arguments,function(a,b){var c;while((c=n.inArray(b,f,c))>-1)f.splice(c,1),h>=c&&h--}),this},has:function(a){return a?n.inArray(a,f)>-1:f.length>0},empty:function(){return f&&(f=[]),this},disable:function(){return e=g=[],f=c="",this},disabled:function(){return!f},lock:function(){return e=g=[],c||(f=c=""),this},locked:function(){return!!e},fireWith:function(a,c){return e||(c=c||[],c=[a,c.slice?c.slice():c],g.push(c),b||i()),this},fire:function(){return j.fireWith(this,arguments),this},fired:function(){return!!d}};return j},n.extend({Deferred:function(a){var b=[["resolve","done",n.Callbacks("once memory"),"resolved"],["reject","fail",n.Callbacks("once memory"),"rejected"],["notify","progress",n.Callbacks("memory")]],c="pending",d={state:function(){return c},always:function(){return e.done(arguments).fail(arguments),this},then:function(){var a=arguments;return n.Deferred(function(c){n.each(b,function(b,f){var g=n.isFunction(a[b])&&a[b];e[f[1]](function(){var a=g&&g.apply(this,arguments);a&&n.isFunction(a.promise)?a.promise().progress(c.notify).done(c.resolve).fail(c.reject):c[f[0]+"With"](this===d?c.promise():this,g?[a]:arguments)})}),a=null}).promise()},promise:function(a){return null!=a?n.extend(a,d):d}},e={};return d.pipe=d.then,n.each(b,function(a,f){var g=f[2],h=f[3];d[f[1]]=g.add,h&&g.add(function(){c=h},b[1^a][2].disable,b[2][2].lock),e[f[0]]=function(){return e[f[0]+"With"](this===e?d:this,arguments),this},e[f[0]+"With"]=g.fireWith}),d.promise(e),a&&a.call(e,e),e},when:function(a){var b=0,c=e.call(arguments),d=c.length,f=1!==d||a&&n.isFunction(a.promise)?d:0,g=1===f?a:n.Deferred(),h=function(a,b,c){return function(d){b[a]=this,c[a]=arguments.length>1?e.call(arguments):d,c===i?g.notifyWith(b,c):--f||g.resolveWith(b,c)}},i,j,k;if(d>1)for(i=new Array(d),j=new Array(d),k=new Array(d);d>b;b++)c[b]&&n.isFunction(c[b].promise)?c[b].promise().progress(h(b,j,i)).done(h(b,k,c)).fail(g.reject):--f;return f||g.resolveWith(k,c),g.promise()}});var I;n.fn.ready=function(a){return n.ready.promise().done(a),this},n.extend({isReady:!1,readyWait:1,holdReady:function(a){a?n.readyWait++:n.ready(!0)},ready:function(a){(a===!0?--n.readyWait:n.isReady)||(n.isReady=!0,a!==!0&&--n.readyWait>0||(I.resolveWith(d,[n]),n.fn.triggerHandler&&(n(d).triggerHandler("ready"),n(d).off("ready"))))}});function J(){d.removeEventListener("DOMContentLoaded",J),a.removeEventListener("load",J),n.ready()}n.ready.promise=function(b){return I||(I=n.Deferred(),"complete"===d.readyState||"loading"!==d.readyState&&!d.documentElement.doScroll?a.setTimeout(n.ready):(d.addEventListener("DOMContentLoaded",J),a.addEventListener("load",J))),I.promise(b)},n.ready.promise();var K=function(a,b,c,d,e,f,g){var h=0,i=a.length,j=null==c;if("object"===n.type(c)){e=!0;for(h in c)K(a,b,h,c[h],!0,f,g)}else if(void 0!==d&&(e=!0,n.isFunction(d)||(g=!0),j&&(g?(b.call(a,d),b=null):(j=b,b=function(a,b,c){return j.call(n(a),c)})),b))for(;i>h;h++)b(a[h],c,g?d:d.call(a[h],h,b(a[h],c)));return e?a:j?b.call(a):i?b(a[0],c):f},L=function(a){return 1===a.nodeType||9===a.nodeType||!+a.nodeType};function M(){this.expando=n.expando+M.uid++}M.uid=1,M.prototype={register:function(a,b){var c=b||{};return a.nodeType?a[this.expando]=c:Object.defineProperty(a,this.expando,{value:c,writable:!0,configurable:!0}),a[this.expando]},cache:function(a){if(!L(a))return{};var b=a[this.expando];return b||(b={},L(a)&&(a.nodeType?a[this.expando]=b:Object.defineProperty(a,this.expando,{value:b,configurable:!0}))),b},set:function(a,b,c){var d,e=this.cache(a);if("string"==typeof b)e[b]=c;else for(d in b)e[d]=b[d];return e},get:function(a,b){return void 0===b?this.cache(a):a[this.expando]&&a[this.expando][b]},access:function(a,b,c){var d;return void 0===b||b&&"string"==typeof b&&void 0===c?(d=this.get(a,b),void 0!==d?d:this.get(a,n.camelCase(b))):(this.set(a,b,c),void 0!==c?c:b)},remove:function(a,b){var c,d,e,f=a[this.expando];if(void 0!==f){if(void 0===b)this.register(a);else{n.isArray(b)?d=b.concat(b.map(n.camelCase)):(e=n.camelCase(b),b in f?d=[b,e]:(d=e,d=d in f?[d]:d.match(G)||[])),c=d.length;while(c--)delete f[d[c]]}(void 0===b||n.isEmptyObject(f))&&(a.nodeType?a[this.expando]=void 0:delete a[this.expando])}},hasData:function(a){var b=a[this.expando];return void 0!==b&&!n.isEmptyObject(b)}};var N=new M,O=new M,P=/^(?:\{[\w\W]*\}|\[[\w\W]*\])$/,Q=/[A-Z]/g;function R(a,b,c){var d;if(void 0===c&&1===a.nodeType)if(d="data-"+b.replace(Q,"-$&").toLowerCase(),c=a.getAttribute(d),"string"==typeof c){try{c="true"===c?!0:"false"===c?!1:"null"===c?null:+c+""===c?+c:P.test(c)?n.parseJSON(c):c;
+}catch(e){}O.set(a,b,c)}else c=void 0;return c}n.extend({hasData:function(a){return O.hasData(a)||N.hasData(a)},data:function(a,b,c){return O.access(a,b,c)},removeData:function(a,b){O.remove(a,b)},_data:function(a,b,c){return N.access(a,b,c)},_removeData:function(a,b){N.remove(a,b)}}),n.fn.extend({data:function(a,b){var c,d,e,f=this[0],g=f&&f.attributes;if(void 0===a){if(this.length&&(e=O.get(f),1===f.nodeType&&!N.get(f,"hasDataAttrs"))){c=g.length;while(c--)g[c]&&(d=g[c].name,0===d.indexOf("data-")&&(d=n.camelCase(d.slice(5)),R(f,d,e[d])));N.set(f,"hasDataAttrs",!0)}return e}return"object"==typeof a?this.each(function(){O.set(this,a)}):K(this,function(b){var c,d;if(f&&void 0===b){if(c=O.get(f,a)||O.get(f,a.replace(Q,"-$&").toLowerCase()),void 0!==c)return c;if(d=n.camelCase(a),c=O.get(f,d),void 0!==c)return c;if(c=R(f,d,void 0),void 0!==c)return c}else d=n.camelCase(a),this.each(function(){var c=O.get(this,d);O.set(this,d,b),a.indexOf("-")>-1&&void 0!==c&&O.set(this,a,b)})},null,b,arguments.length>1,null,!0)},removeData:function(a){return this.each(function(){O.remove(this,a)})}}),n.extend({queue:function(a,b,c){var d;return a?(b=(b||"fx")+"queue",d=N.get(a,b),c&&(!d||n.isArray(c)?d=N.access(a,b,n.makeArray(c)):d.push(c)),d||[]):void 0},dequeue:function(a,b){b=b||"fx";var c=n.queue(a,b),d=c.length,e=c.shift(),f=n._queueHooks(a,b),g=function(){n.dequeue(a,b)};"inprogress"===e&&(e=c.shift(),d--),e&&("fx"===b&&c.unshift("inprogress"),delete f.stop,e.call(a,g,f)),!d&&f&&f.empty.fire()},_queueHooks:function(a,b){var c=b+"queueHooks";return N.get(a,c)||N.access(a,c,{empty:n.Callbacks("once memory").add(function(){N.remove(a,[b+"queue",c])})})}}),n.fn.extend({queue:function(a,b){var c=2;return"string"!=typeof a&&(b=a,a="fx",c--),arguments.length<c?n.queue(this[0],a):void 0===b?this:this.each(function(){var c=n.queue(this,a,b);n._queueHooks(this,a),"fx"===a&&"inprogress"!==c[0]&&n.dequeue(this,a)})},dequeue:function(a){return this.each(function(){n.dequeue(this,a)})},clearQueue:function(a){return this.queue(a||"fx",[])},promise:function(a,b){var c,d=1,e=n.Deferred(),f=this,g=this.length,h=function(){--d||e.resolveWith(f,[f])};"string"!=typeof a&&(b=a,a=void 0),a=a||"fx";while(g--)c=N.get(f[g],a+"queueHooks"),c&&c.empty&&(d++,c.empty.add(h));return h(),e.promise(b)}});var S=/[+-]?(?:\d*\.|)\d+(?:[eE][+-]?\d+|)/.source,T=new RegExp("^(?:([+-])=|)("+S+")([a-z%]*)$","i"),U=["Top","Right","Bottom","Left"],V=function(a,b){return a=b||a,"none"===n.css(a,"display")||!n.contains(a.ownerDocument,a)};function W(a,b,c,d){var e,f=1,g=20,h=d?function(){return d.cur()}:function(){return n.css(a,b,"")},i=h(),j=c&&c[3]||(n.cssNumber[b]?"":"px"),k=(n.cssNumber[b]||"px"!==j&&+i)&&T.exec(n.css(a,b));if(k&&k[3]!==j){j=j||k[3],c=c||[],k=+i||1;do f=f||".5",k/=f,n.style(a,b,k+j);while(f!==(f=h()/i)&&1!==f&&--g)}return c&&(k=+k||+i||0,e=c[1]?k+(c[1]+1)*c[2]:+c[2],d&&(d.unit=j,d.start=k,d.end=e)),e}var X=/^(?:checkbox|radio)$/i,Y=/<([\w:-]+)/,Z=/^$|\/(?:java|ecma)script/i,$={option:[1,"<select multiple='multiple'>","</select>"],thead:[1,"<table>","</table>"],col:[2,"<table><colgroup>","</colgroup></table>"],tr:[2,"<table><tbody>","</tbody></table>"],td:[3,"<table><tbody><tr>","</tr></tbody></table>"],_default:[0,"",""]};$.optgroup=$.option,$.tbody=$.tfoot=$.colgroup=$.caption=$.thead,$.th=$.td;function _(a,b){var c="undefined"!=typeof a.getElementsByTagName?a.getElementsByTagName(b||"*"):"undefined"!=typeof a.querySelectorAll?a.querySelectorAll(b||"*"):[];return void 0===b||b&&n.nodeName(a,b)?n.merge([a],c):c}function aa(a,b){for(var c=0,d=a.length;d>c;c++)N.set(a[c],"globalEval",!b||N.get(b[c],"globalEval"))}var ba=/<|&#?\w+;/;function ca(a,b,c,d,e){for(var f,g,h,i,j,k,l=b.createDocumentFragment(),m=[],o=0,p=a.length;p>o;o++)if(f=a[o],f||0===f)if("object"===n.type(f))n.merge(m,f.nodeType?[f]:f);else if(ba.test(f)){g=g||l.appendChild(b.createElement("div")),h=(Y.exec(f)||["",""])[1].toLowerCase(),i=$[h]||$._default,g.innerHTML=i[1]+n.htmlPrefilter(f)+i[2],k=i[0];while(k--)g=g.lastChild;n.merge(m,g.childNodes),g=l.firstChild,g.textContent=""}else m.push(b.createTextNode(f));l.textContent="",o=0;while(f=m[o++])if(d&&n.inArray(f,d)>-1)e&&e.push(f);else if(j=n.contains(f.ownerDocument,f),g=_(l.appendChild(f),"script"),j&&aa(g),c){k=0;while(f=g[k++])Z.test(f.type||"")&&c.push(f)}return l}!function(){var a=d.createDocumentFragment(),b=a.appendChild(d.createElement("div")),c=d.createElement("input");c.setAttribute("type","radio"),c.setAttribute("checked","checked"),c.setAttribute("name","t"),b.appendChild(c),l.checkClone=b.cloneNode(!0).cloneNode(!0).lastChild.checked,b.innerHTML="<textarea>x</textarea>",l.noCloneChecked=!!b.cloneNode(!0).lastChild.defaultValue}();var da=/^key/,ea=/^(?:mouse|pointer|contextmenu|drag|drop)|click/,fa=/^([^.]*)(?:\.(.+)|)/;function ga(){return!0}function ha(){return!1}function ia(){try{return d.activeElement}catch(a){}}function ja(a,b,c,d,e,f){var g,h;if("object"==typeof b){"string"!=typeof c&&(d=d||c,c=void 0);for(h in b)ja(a,h,c,d,b[h],f);return a}if(null==d&&null==e?(e=c,d=c=void 0):null==e&&("string"==typeof c?(e=d,d=void 0):(e=d,d=c,c=void 0)),e===!1)e=ha;else if(!e)return a;return 1===f&&(g=e,e=function(a){return n().off(a),g.apply(this,arguments)},e.guid=g.guid||(g.guid=n.guid++)),a.each(function(){n.event.add(this,b,e,d,c)})}n.event={global:{},add:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=N.get(a);if(r){c.handler&&(f=c,c=f.handler,e=f.selector),c.guid||(c.guid=n.guid++),(i=r.events)||(i=r.events={}),(g=r.handle)||(g=r.handle=function(b){return"undefined"!=typeof n&&n.event.triggered!==b.type?n.event.dispatch.apply(a,arguments):void 0}),b=(b||"").match(G)||[""],j=b.length;while(j--)h=fa.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o&&(l=n.event.special[o]||{},o=(e?l.delegateType:l.bindType)||o,l=n.event.special[o]||{},k=n.extend({type:o,origType:q,data:d,handler:c,guid:c.guid,selector:e,needsContext:e&&n.expr.match.needsContext.test(e),namespace:p.join(".")},f),(m=i[o])||(m=i[o]=[],m.delegateCount=0,l.setup&&l.setup.call(a,d,p,g)!==!1||a.addEventListener&&a.addEventListener(o,g)),l.add&&(l.add.call(a,k),k.handler.guid||(k.handler.guid=c.guid)),e?m.splice(m.delegateCount++,0,k):m.push(k),n.event.global[o]=!0)}},remove:function(a,b,c,d,e){var f,g,h,i,j,k,l,m,o,p,q,r=N.hasData(a)&&N.get(a);if(r&&(i=r.events)){b=(b||"").match(G)||[""],j=b.length;while(j--)if(h=fa.exec(b[j])||[],o=q=h[1],p=(h[2]||"").split(".").sort(),o){l=n.event.special[o]||{},o=(d?l.delegateType:l.bindType)||o,m=i[o]||[],h=h[2]&&new RegExp("(^|\\.)"+p.join("\\.(?:.*\\.|)")+"(\\.|$)"),g=f=m.length;while(f--)k=m[f],!e&&q!==k.origType||c&&c.guid!==k.guid||h&&!h.test(k.namespace)||d&&d!==k.selector&&("**"!==d||!k.selector)||(m.splice(f,1),k.selector&&m.delegateCount--,l.remove&&l.remove.call(a,k));g&&!m.length&&(l.teardown&&l.teardown.call(a,p,r.handle)!==!1||n.removeEvent(a,o,r.handle),delete i[o])}else for(o in i)n.event.remove(a,o+b[j],c,d,!0);n.isEmptyObject(i)&&N.remove(a,"handle events")}},dispatch:function(a){a=n.event.fix(a);var b,c,d,f,g,h=[],i=e.call(arguments),j=(N.get(this,"events")||{})[a.type]||[],k=n.event.special[a.type]||{};if(i[0]=a,a.delegateTarget=this,!k.preDispatch||k.preDispatch.call(this,a)!==!1){h=n.event.handlers.call(this,a,j),b=0;while((f=h[b++])&&!a.isPropagationStopped()){a.currentTarget=f.elem,c=0;while((g=f.handlers[c++])&&!a.isImmediatePropagationStopped())a.rnamespace&&!a.rnamespace.test(g.namespace)||(a.handleObj=g,a.data=g.data,d=((n.event.special[g.origType]||{}).handle||g.handler).apply(f.elem,i),void 0!==d&&(a.result=d)===!1&&(a.preventDefault(),a.stopPropagation()))}return k.postDispatch&&k.postDispatch.call(this,a),a.result}},handlers:function(a,b){var c,d,e,f,g=[],h=b.delegateCount,i=a.target;if(h&&i.nodeType&&("click"!==a.type||isNaN(a.button)||a.button<1))for(;i!==this;i=i.parentNode||this)if(1===i.nodeType&&(i.disabled!==!0||"click"!==a.type)){for(d=[],c=0;h>c;c++)f=b[c],e=f.selector+" ",void 0===d[e]&&(d[e]=f.needsContext?n(e,this).index(i)>-1:n.find(e,this,null,[i]).length),d[e]&&d.push(f);d.length&&g.push({elem:i,handlers:d})}return h<b.length&&g.push({elem:this,handlers:b.slice(h)}),g},props:"altKey bubbles cancelable ctrlKey currentTarget detail eventPhase metaKey relatedTarget shiftKey target timeStamp view which".split(" "),fixHooks:{},keyHooks:{props:"char charCode key keyCode".split(" "),filter:function(a,b){return null==a.which&&(a.which=null!=b.charCode?b.charCode:b.keyCode),a}},mouseHooks:{props:"button buttons clientX clientY offsetX offsetY pageX pageY screenX screenY toElement".split(" "),filter:function(a,b){var c,e,f,g=b.button;return null==a.pageX&&null!=b.clientX&&(c=a.target.ownerDocument||d,e=c.documentElement,f=c.body,a.pageX=b.clientX+(e&&e.scrollLeft||f&&f.scrollLeft||0)-(e&&e.clientLeft||f&&f.clientLeft||0),a.pageY=b.clientY+(e&&e.scrollTop||f&&f.scrollTop||0)-(e&&e.clientTop||f&&f.clientTop||0)),a.which||void 0===g||(a.which=1&g?1:2&g?3:4&g?2:0),a}},fix:function(a){if(a[n.expando])return a;var b,c,e,f=a.type,g=a,h=this.fixHooks[f];h||(this.fixHooks[f]=h=ea.test(f)?this.mouseHooks:da.test(f)?this.keyHooks:{}),e=h.props?this.props.concat(h.props):this.props,a=new n.Event(g),b=e.length;while(b--)c=e[b],a[c]=g[c];return a.target||(a.target=d),3===a.target.nodeType&&(a.target=a.target.parentNode),h.filter?h.filter(a,g):a},special:{load:{noBubble:!0},focus:{trigger:function(){return this!==ia()&&this.focus?(this.focus(),!1):void 0},delegateType:"focusin"},blur:{trigger:function(){return this===ia()&&this.blur?(this.blur(),!1):void 0},delegateType:"focusout"},click:{trigger:function(){return"checkbox"===this.type&&this.click&&n.nodeName(this,"input")?(this.click(),!1):void 0},_default:function(a){return n.nodeName(a.target,"a")}},beforeunload:{postDispatch:function(a){void 0!==a.result&&a.originalEvent&&(a.originalEvent.returnValue=a.result)}}}},n.removeEvent=function(a,b,c){a.removeEventListener&&a.removeEventListener(b,c)},n.Event=function(a,b){return this instanceof n.Event?(a&&a.type?(this.originalEvent=a,this.type=a.type,this.isDefaultPrevented=a.defaultPrevented||void 0===a.defaultPrevented&&a.returnValue===!1?ga:ha):this.type=a,b&&n.extend(this,b),this.timeStamp=a&&a.timeStamp||n.now(),void(this[n.expando]=!0)):new n.Event(a,b)},n.Event.prototype={constructor:n.Event,isDefaultPrevented:ha,isPropagationStopped:ha,isImmediatePropagationStopped:ha,preventDefault:function(){var a=this.originalEvent;this.isDefaultPrevented=ga,a&&a.preventDefault()},stopPropagation:function(){var a=this.originalEvent;this.isPropagationStopped=ga,a&&a.stopPropagation()},stopImmediatePropagation:function(){var a=this.originalEvent;this.isImmediatePropagationStopped=ga,a&&a.stopImmediatePropagation(),this.stopPropagation()}},n.each({mouseenter:"mouseover",mouseleave:"mouseout",pointerenter:"pointerover",pointerleave:"pointerout"},function(a,b){n.event.special[a]={delegateType:b,bindType:b,handle:function(a){var c,d=this,e=a.relatedTarget,f=a.handleObj;return e&&(e===d||n.contains(d,e))||(a.type=f.origType,c=f.handler.apply(this,arguments),a.type=b),c}}}),n.fn.extend({on:function(a,b,c,d){return ja(this,a,b,c,d)},one:function(a,b,c,d){return ja(this,a,b,c,d,1)},off:function(a,b,c){var d,e;if(a&&a.preventDefault&&a.handleObj)return d=a.handleObj,n(a.delegateTarget).off(d.namespace?d.origType+"."+d.namespace:d.origType,d.selector,d.handler),this;if("object"==typeof a){for(e in a)this.off(e,b,a[e]);return this}return b!==!1&&"function"!=typeof b||(c=b,b=void 0),c===!1&&(c=ha),this.each(function(){n.event.remove(this,a,c,b)})}});var ka=/<(?!area|br|col|embed|hr|img|input|link|meta|param)(([\w:-]+)[^>]*)\/>/gi,la=/<script|<style|<link/i,ma=/checked\s*(?:[^=]|=\s*.checked.)/i,na=/^true\/(.*)/,oa=/^\s*<!(?:\[CDATA\[|--)|(?:\]\]|--)>\s*$/g;function pa(a,b){return n.nodeName(a,"table")&&n.nodeName(11!==b.nodeType?b:b.firstChild,"tr")?a.getElementsByTagName("tbody")[0]||a.appendChild(a.ownerDocument.createElement("tbody")):a}function qa(a){return a.type=(null!==a.getAttribute("type"))+"/"+a.type,a}function ra(a){var b=na.exec(a.type);return b?a.type=b[1]:a.removeAttribute("type"),a}function sa(a,b){var c,d,e,f,g,h,i,j;if(1===b.nodeType){if(N.hasData(a)&&(f=N.access(a),g=N.set(b,f),j=f.events)){delete g.handle,g.events={};for(e in j)for(c=0,d=j[e].length;d>c;c++)n.event.add(b,e,j[e][c])}O.hasData(a)&&(h=O.access(a),i=n.extend({},h),O.set(b,i))}}function ta(a,b){var c=b.nodeName.toLowerCase();"input"===c&&X.test(a.type)?b.checked=a.checked:"input"!==c&&"textarea"!==c||(b.defaultValue=a.defaultValue)}function ua(a,b,c,d){b=f.apply([],b);var e,g,h,i,j,k,m=0,o=a.length,p=o-1,q=b[0],r=n.isFunction(q);if(r||o>1&&"string"==typeof q&&!l.checkClone&&ma.test(q))return a.each(function(e){var f=a.eq(e);r&&(b[0]=q.call(this,e,f.html())),ua(f,b,c,d)});if(o&&(e=ca(b,a[0].ownerDocument,!1,a,d),g=e.firstChild,1===e.childNodes.length&&(e=g),g||d)){for(h=n.map(_(e,"script"),qa),i=h.length;o>m;m++)j=e,m!==p&&(j=n.clone(j,!0,!0),i&&n.merge(h,_(j,"script"))),c.call(a[m],j,m);if(i)for(k=h[h.length-1].ownerDocument,n.map(h,ra),m=0;i>m;m++)j=h[m],Z.test(j.type||"")&&!N.access(j,"globalEval")&&n.contains(k,j)&&(j.src?n._evalUrl&&n._evalUrl(j.src):n.globalEval(j.textContent.replace(oa,"")))}return a}function va(a,b,c){for(var d,e=b?n.filter(b,a):a,f=0;null!=(d=e[f]);f++)c||1!==d.nodeType||n.cleanData(_(d)),d.parentNode&&(c&&n.contains(d.ownerDocument,d)&&aa(_(d,"script")),d.parentNode.removeChild(d));return a}n.extend({htmlPrefilter:function(a){return a.replace(ka,"<$1></$2>")},clone:function(a,b,c){var d,e,f,g,h=a.cloneNode(!0),i=n.contains(a.ownerDocument,a);if(!(l.noCloneChecked||1!==a.nodeType&&11!==a.nodeType||n.isXMLDoc(a)))for(g=_(h),f=_(a),d=0,e=f.length;e>d;d++)ta(f[d],g[d]);if(b)if(c)for(f=f||_(a),g=g||_(h),d=0,e=f.length;e>d;d++)sa(f[d],g[d]);else sa(a,h);return g=_(h,"script"),g.length>0&&aa(g,!i&&_(a,"script")),h},cleanData:function(a){for(var b,c,d,e=n.event.special,f=0;void 0!==(c=a[f]);f++)if(L(c)){if(b=c[N.expando]){if(b.events)for(d in b.events)e[d]?n.event.remove(c,d):n.removeEvent(c,d,b.handle);c[N.expando]=void 0}c[O.expando]&&(c[O.expando]=void 0)}}}),n.fn.extend({domManip:ua,detach:function(a){return va(this,a,!0)},remove:function(a){return va(this,a)},text:function(a){return K(this,function(a){return void 0===a?n.text(this):this.empty().each(function(){1!==this.nodeType&&11!==this.nodeType&&9!==this.nodeType||(this.textContent=a)})},null,a,arguments.length)},append:function(){return ua(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=pa(this,a);b.appendChild(a)}})},prepend:function(){return ua(this,arguments,function(a){if(1===this.nodeType||11===this.nodeType||9===this.nodeType){var b=pa(this,a);b.insertBefore(a,b.firstChild)}})},before:function(){return ua(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this)})},after:function(){return ua(this,arguments,function(a){this.parentNode&&this.parentNode.insertBefore(a,this.nextSibling)})},empty:function(){for(var a,b=0;null!=(a=this[b]);b++)1===a.nodeType&&(n.cleanData(_(a,!1)),a.textContent="");return this},clone:function(a,b){return a=null==a?!1:a,b=null==b?a:b,this.map(function(){return n.clone(this,a,b)})},html:function(a){return K(this,function(a){var b=this[0]||{},c=0,d=this.length;if(void 0===a&&1===b.nodeType)return b.innerHTML;if("string"==typeof a&&!la.test(a)&&!$[(Y.exec(a)||["",""])[1].toLowerCase()]){a=n.htmlPrefilter(a);try{for(;d>c;c++)b=this[c]||{},1===b.nodeType&&(n.cleanData(_(b,!1)),b.innerHTML=a);b=0}catch(e){}}b&&this.empty().append(a)},null,a,arguments.length)},replaceWith:function(){var a=[];return ua(this,arguments,function(b){var c=this.parentNode;n.inArray(this,a)<0&&(n.cleanData(_(this)),c&&c.replaceChild(b,this))},a)}}),n.each({appendTo:"append",prependTo:"prepend",insertBefore:"before",insertAfter:"after",replaceAll:"replaceWith"},function(a,b){n.fn[a]=function(a){for(var c,d=[],e=n(a),f=e.length-1,h=0;f>=h;h++)c=h===f?this:this.clone(!0),n(e[h])[b](c),g.apply(d,c.get());return this.pushStack(d)}});var wa,xa={HTML:"block",BODY:"block"};function ya(a,b){var c=n(b.createElement(a)).appendTo(b.body),d=n.css(c[0],"display");return c.detach(),d}function za(a){var b=d,c=xa[a];return c||(c=ya(a,b),"none"!==c&&c||(wa=(wa||n("<iframe frameborder='0' width='0' height='0'/>")).appendTo(b.documentElement),b=wa[0].contentDocument,b.write(),b.close(),c=ya(a,b),wa.detach()),xa[a]=c),c}var Aa=/^margin/,Ba=new RegExp("^("+S+")(?!px)[a-z%]+$","i"),Ca=function(b){var c=b.ownerDocument.defaultView;return c&&c.opener||(c=a),c.getComputedStyle(b)},Da=function(a,b,c,d){var e,f,g={};for(f in b)g[f]=a.style[f],a.style[f]=b[f];e=c.apply(a,d||[]);for(f in b)a.style[f]=g[f];return e},Ea=d.documentElement;!function(){var b,c,e,f,g=d.createElement("div"),h=d.createElement("div");if(h.style){h.style.backgroundClip="content-box",h.cloneNode(!0).style.backgroundClip="",l.clearCloneStyle="content-box"===h.style.backgroundClip,g.style.cssText="border:0;width:8px;height:0;top:0;left:-9999px;padding:0;margin-top:1px;position:absolute",g.appendChild(h);function i(){h.style.cssText="-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box;position:relative;display:block;margin:auto;border:1px;padding:1px;top:1%;width:50%",h.innerHTML="",Ea.appendChild(g);var d=a.getComputedStyle(h);b="1%"!==d.top,f="2px"===d.marginLeft,c="4px"===d.width,h.style.marginRight="50%",e="4px"===d.marginRight,Ea.removeChild(g)}n.extend(l,{pixelPosition:function(){return i(),b},boxSizingReliable:function(){return null==c&&i(),c},pixelMarginRight:function(){return null==c&&i(),e},reliableMarginLeft:function(){return null==c&&i(),f},reliableMarginRight:function(){var b,c=h.appendChild(d.createElement("div"));return c.style.cssText=h.style.cssText="-webkit-box-sizing:content-box;box-sizing:content-box;display:block;margin:0;border:0;padding:0",c.style.marginRight=c.style.width="0",h.style.width="1px",Ea.appendChild(g),b=!parseFloat(a.getComputedStyle(c).marginRight),Ea.removeChild(g),h.removeChild(c),b}})}}();function Fa(a,b,c){var d,e,f,g,h=a.style;return c=c||Ca(a),g=c?c.getPropertyValue(b)||c[b]:void 0,""!==g&&void 0!==g||n.contains(a.ownerDocument,a)||(g=n.style(a,b)),c&&!l.pixelMarginRight()&&Ba.test(g)&&Aa.test(b)&&(d=h.width,e=h.minWidth,f=h.maxWidth,h.minWidth=h.maxWidth=h.width=g,g=c.width,h.width=d,h.minWidth=e,h.maxWidth=f),void 0!==g?g+"":g}function Ga(a,b){return{get:function(){return a()?void delete this.get:(this.get=b).apply(this,arguments)}}}var Ha=/^(none|table(?!-c[ea]).+)/,Ia={position:"absolute",visibility:"hidden",display:"block"},Ja={letterSpacing:"0",fontWeight:"400"},Ka=["Webkit","O","Moz","ms"],La=d.createElement("div").style;function Ma(a){if(a in La)return a;var b=a[0].toUpperCase()+a.slice(1),c=Ka.length;while(c--)if(a=Ka[c]+b,a in La)return a}function Na(a,b,c){var d=T.exec(b);return d?Math.max(0,d[2]-(c||0))+(d[3]||"px"):b}function Oa(a,b,c,d,e){for(var f=c===(d?"border":"content")?4:"width"===b?1:0,g=0;4>f;f+=2)"margin"===c&&(g+=n.css(a,c+U[f],!0,e)),d?("content"===c&&(g-=n.css(a,"padding"+U[f],!0,e)),"margin"!==c&&(g-=n.css(a,"border"+U[f]+"Width",!0,e))):(g+=n.css(a,"padding"+U[f],!0,e),"padding"!==c&&(g+=n.css(a,"border"+U[f]+"Width",!0,e)));return g}function Pa(b,c,e){var f=!0,g="width"===c?b.offsetWidth:b.offsetHeight,h=Ca(b),i="border-box"===n.css(b,"boxSizing",!1,h);if(d.msFullscreenElement&&a.top!==a&&b.getClientRects().length&&(g=Math.round(100*b.getBoundingClientRect()[c])),0>=g||null==g){if(g=Fa(b,c,h),(0>g||null==g)&&(g=b.style[c]),Ba.test(g))return g;f=i&&(l.boxSizingReliable()||g===b.style[c]),g=parseFloat(g)||0}return g+Oa(b,c,e||(i?"border":"content"),f,h)+"px"}function Qa(a,b){for(var c,d,e,f=[],g=0,h=a.length;h>g;g++)d=a[g],d.style&&(f[g]=N.get(d,"olddisplay"),c=d.style.display,b?(f[g]||"none"!==c||(d.style.display=""),""===d.style.display&&V(d)&&(f[g]=N.access(d,"olddisplay",za(d.nodeName)))):(e=V(d),"none"===c&&e||N.set(d,"olddisplay",e?c:n.css(d,"display"))));for(g=0;h>g;g++)d=a[g],d.style&&(b&&"none"!==d.style.display&&""!==d.style.display||(d.style.display=b?f[g]||"":"none"));return a}n.extend({cssHooks:{opacity:{get:function(a,b){if(b){var c=Fa(a,"opacity");return""===c?"1":c}}}},cssNumber:{animationIterationCount:!0,columnCount:!0,fillOpacity:!0,flexGrow:!0,flexShrink:!0,fontWeight:!0,lineHeight:!0,opacity:!0,order:!0,orphans:!0,widows:!0,zIndex:!0,zoom:!0},cssProps:{"float":"cssFloat"},style:function(a,b,c,d){if(a&&3!==a.nodeType&&8!==a.nodeType&&a.style){var e,f,g,h=n.camelCase(b),i=a.style;return b=n.cssProps[h]||(n.cssProps[h]=Ma(h)||h),g=n.cssHooks[b]||n.cssHooks[h],void 0===c?g&&"get"in g&&void 0!==(e=g.get(a,!1,d))?e:i[b]:(f=typeof c,"string"===f&&(e=T.exec(c))&&e[1]&&(c=W(a,b,e),f="number"),null!=c&&c===c&&("number"===f&&(c+=e&&e[3]||(n.cssNumber[h]?"":"px")),l.clearCloneStyle||""!==c||0!==b.indexOf("background")||(i[b]="inherit"),g&&"set"in g&&void 0===(c=g.set(a,c,d))||(i[b]=c)),void 0)}},css:function(a,b,c,d){var e,f,g,h=n.camelCase(b);return b=n.cssProps[h]||(n.cssProps[h]=Ma(h)||h),g=n.cssHooks[b]||n.cssHooks[h],g&&"get"in g&&(e=g.get(a,!0,c)),void 0===e&&(e=Fa(a,b,d)),"normal"===e&&b in Ja&&(e=Ja[b]),""===c||c?(f=parseFloat(e),c===!0||isFinite(f)?f||0:e):e}}),n.each(["height","width"],function(a,b){n.cssHooks[b]={get:function(a,c,d){return c?Ha.test(n.css(a,"display"))&&0===a.offsetWidth?Da(a,Ia,function(){return Pa(a,b,d)}):Pa(a,b,d):void 0},set:function(a,c,d){var e,f=d&&Ca(a),g=d&&Oa(a,b,d,"border-box"===n.css(a,"boxSizing",!1,f),f);return g&&(e=T.exec(c))&&"px"!==(e[3]||"px")&&(a.style[b]=c,c=n.css(a,b)),Na(a,c,g)}}}),n.cssHooks.marginLeft=Ga(l.reliableMarginLeft,function(a,b){return b?(parseFloat(Fa(a,"marginLeft"))||a.getBoundingClientRect().left-Da(a,{marginLeft:0},function(){return a.getBoundingClientRect().left}))+"px":void 0}),n.cssHooks.marginRight=Ga(l.reliableMarginRight,function(a,b){return b?Da(a,{display:"inline-block"},Fa,[a,"marginRight"]):void 0}),n.each({margin:"",padding:"",border:"Width"},function(a,b){n.cssHooks[a+b]={expand:function(c){for(var d=0,e={},f="string"==typeof c?c.split(" "):[c];4>d;d++)e[a+U[d]+b]=f[d]||f[d-2]||f[0];return e}},Aa.test(a)||(n.cssHooks[a+b].set=Na)}),n.fn.extend({css:function(a,b){return K(this,function(a,b,c){var d,e,f={},g=0;if(n.isArray(b)){for(d=Ca(a),e=b.length;e>g;g++)f[b[g]]=n.css(a,b[g],!1,d);return f}return void 0!==c?n.style(a,b,c):n.css(a,b)},a,b,arguments.length>1)},show:function(){return Qa(this,!0)},hide:function(){return Qa(this)},toggle:function(a){return"boolean"==typeof a?a?this.show():this.hide():this.each(function(){V(this)?n(this).show():n(this).hide()})}});function Ra(a,b,c,d,e){return new Ra.prototype.init(a,b,c,d,e)}n.Tween=Ra,Ra.prototype={constructor:Ra,init:function(a,b,c,d,e,f){this.elem=a,this.prop=c,this.easing=e||n.easing._default,this.options=b,this.start=this.now=this.cur(),this.end=d,this.unit=f||(n.cssNumber[c]?"":"px")},cur:function(){var a=Ra.propHooks[this.prop];return a&&a.get?a.get(this):Ra.propHooks._default.get(this)},run:function(a){var b,c=Ra.propHooks[this.prop];return this.options.duration?this.pos=b=n.easing[this.easing](a,this.options.duration*a,0,1,this.options.duration):this.pos=b=a,this.now=(this.end-this.start)*b+this.start,this.options.step&&this.options.step.call(this.elem,this.now,this),c&&c.set?c.set(this):Ra.propHooks._default.set(this),this}},Ra.prototype.init.prototype=Ra.prototype,Ra.propHooks={_default:{get:function(a){var b;return 1!==a.elem.nodeType||null!=a.elem[a.prop]&&null==a.elem.style[a.prop]?a.elem[a.prop]:(b=n.css(a.elem,a.prop,""),b&&"auto"!==b?b:0)},set:function(a){n.fx.step[a.prop]?n.fx.step[a.prop](a):1!==a.elem.nodeType||null==a.elem.style[n.cssProps[a.prop]]&&!n.cssHooks[a.prop]?a.elem[a.prop]=a.now:n.style(a.elem,a.prop,a.now+a.unit)}}},Ra.propHooks.scrollTop=Ra.propHooks.scrollLeft={set:function(a){a.elem.nodeType&&a.elem.parentNode&&(a.elem[a.prop]=a.now)}},n.easing={linear:function(a){return a},swing:function(a){return.5-Math.cos(a*Math.PI)/2},_default:"swing"},n.fx=Ra.prototype.init,n.fx.step={};var Sa,Ta,Ua=/^(?:toggle|show|hide)$/,Va=/queueHooks$/;function Wa(){return a.setTimeout(function(){Sa=void 0}),Sa=n.now()}function Xa(a,b){var c,d=0,e={height:a};for(b=b?1:0;4>d;d+=2-b)c=U[d],e["margin"+c]=e["padding"+c]=a;return b&&(e.opacity=e.width=a),e}function Ya(a,b,c){for(var d,e=(_a.tweeners[b]||[]).concat(_a.tweeners["*"]),f=0,g=e.length;g>f;f++)if(d=e[f].call(c,b,a))return d}function Za(a,b,c){var d,e,f,g,h,i,j,k,l=this,m={},o=a.style,p=a.nodeType&&V(a),q=N.get(a,"fxshow");c.queue||(h=n._queueHooks(a,"fx"),null==h.unqueued&&(h.unqueued=0,i=h.empty.fire,h.empty.fire=function(){h.unqueued||i()}),h.unqueued++,l.always(function(){l.always(function(){h.unqueued--,n.queue(a,"fx").length||h.empty.fire()})})),1===a.nodeType&&("height"in b||"width"in b)&&(c.overflow=[o.overflow,o.overflowX,o.overflowY],j=n.css(a,"display"),k="none"===j?N.get(a,"olddisplay")||za(a.nodeName):j,"inline"===k&&"none"===n.css(a,"float")&&(o.display="inline-block")),c.overflow&&(o.overflow="hidden",l.always(function(){o.overflow=c.overflow[0],o.overflowX=c.overflow[1],o.overflowY=c.overflow[2]}));for(d in b)if(e=b[d],Ua.exec(e)){if(delete b[d],f=f||"toggle"===e,e===(p?"hide":"show")){if("show"!==e||!q||void 0===q[d])continue;p=!0}m[d]=q&&q[d]||n.style(a,d)}else j=void 0;if(n.isEmptyObject(m))"inline"===("none"===j?za(a.nodeName):j)&&(o.display=j);else{q?"hidden"in q&&(p=q.hidden):q=N.access(a,"fxshow",{}),f&&(q.hidden=!p),p?n(a).show():l.done(function(){n(a).hide()}),l.done(function(){var b;N.remove(a,"fxshow");for(b in m)n.style(a,b,m[b])});for(d in m)g=Ya(p?q[d]:0,d,l),d in q||(q[d]=g.start,p&&(g.end=g.start,g.start="width"===d||"height"===d?1:0))}}function $a(a,b){var c,d,e,f,g;for(c in a)if(d=n.camelCase(c),e=b[d],f=a[c],n.isArray(f)&&(e=f[1],f=a[c]=f[0]),c!==d&&(a[d]=f,delete a[c]),g=n.cssHooks[d],g&&"expand"in g){f=g.expand(f),delete a[d];for(c in f)c in a||(a[c]=f[c],b[c]=e)}else b[d]=e}function _a(a,b,c){var d,e,f=0,g=_a.prefilters.length,h=n.Deferred().always(function(){delete i.elem}),i=function(){if(e)return!1;for(var b=Sa||Wa(),c=Math.max(0,j.startTime+j.duration-b),d=c/j.duration||0,f=1-d,g=0,i=j.tweens.length;i>g;g++)j.tweens[g].run(f);return h.notifyWith(a,[j,f,c]),1>f&&i?c:(h.resolveWith(a,[j]),!1)},j=h.promise({elem:a,props:n.extend({},b),opts:n.extend(!0,{specialEasing:{},easing:n.easing._default},c),originalProperties:b,originalOptions:c,startTime:Sa||Wa(),duration:c.duration,tweens:[],createTween:function(b,c){var d=n.Tween(a,j.opts,b,c,j.opts.specialEasing[b]||j.opts.easing);return j.tweens.push(d),d},stop:function(b){var c=0,d=b?j.tweens.length:0;if(e)return this;for(e=!0;d>c;c++)j.tweens[c].run(1);return b?(h.notifyWith(a,[j,1,0]),h.resolveWith(a,[j,b])):h.rejectWith(a,[j,b]),this}}),k=j.props;for($a(k,j.opts.specialEasing);g>f;f++)if(d=_a.prefilters[f].call(j,a,k,j.opts))return n.isFunction(d.stop)&&(n._queueHooks(j.elem,j.opts.queue).stop=n.proxy(d.stop,d)),d;return n.map(k,Ya,j),n.isFunction(j.opts.start)&&j.opts.start.call(a,j),n.fx.timer(n.extend(i,{elem:a,anim:j,queue:j.opts.queue})),j.progress(j.opts.progress).done(j.opts.done,j.opts.complete).fail(j.opts.fail).always(j.opts.always)}n.Animation=n.extend(_a,{tweeners:{"*":[function(a,b){var c=this.createTween(a,b);return W(c.elem,a,T.exec(b),c),c}]},tweener:function(a,b){n.isFunction(a)?(b=a,a=["*"]):a=a.match(G);for(var c,d=0,e=a.length;e>d;d++)c=a[d],_a.tweeners[c]=_a.tweeners[c]||[],_a.tweeners[c].unshift(b)},prefilters:[Za],prefilter:function(a,b){b?_a.prefilters.unshift(a):_a.prefilters.push(a)}}),n.speed=function(a,b,c){var d=a&&"object"==typeof a?n.extend({},a):{complete:c||!c&&b||n.isFunction(a)&&a,duration:a,easing:c&&b||b&&!n.isFunction(b)&&b};return d.duration=n.fx.off?0:"number"==typeof d.duration?d.duration:d.duration in n.fx.speeds?n.fx.speeds[d.duration]:n.fx.speeds._default,null!=d.queue&&d.queue!==!0||(d.queue="fx"),d.old=d.complete,d.complete=function(){n.isFunction(d.old)&&d.old.call(this),d.queue&&n.dequeue(this,d.queue)},d},n.fn.extend({fadeTo:function(a,b,c,d){return this.filter(V).css("opacity",0).show().end().animate({opacity:b},a,c,d)},animate:function(a,b,c,d){var e=n.isEmptyObject(a),f=n.speed(b,c,d),g=function(){var b=_a(this,n.extend({},a),f);(e||N.get(this,"finish"))&&b.stop(!0)};return g.finish=g,e||f.queue===!1?this.each(g):this.queue(f.queue,g)},stop:function(a,b,c){var d=function(a){var b=a.stop;delete a.stop,b(c)};return"string"!=typeof a&&(c=b,b=a,a=void 0),b&&a!==!1&&this.queue(a||"fx",[]),this.each(function(){var b=!0,e=null!=a&&a+"queueHooks",f=n.timers,g=N.get(this);if(e)g[e]&&g[e].stop&&d(g[e]);else for(e in g)g[e]&&g[e].stop&&Va.test(e)&&d(g[e]);for(e=f.length;e--;)f[e].elem!==this||null!=a&&f[e].queue!==a||(f[e].anim.stop(c),b=!1,f.splice(e,1));!b&&c||n.dequeue(this,a)})},finish:function(a){return a!==!1&&(a=a||"fx"),this.each(function(){var b,c=N.get(this),d=c[a+"queue"],e=c[a+"queueHooks"],f=n.timers,g=d?d.length:0;for(c.finish=!0,n.queue(this,a,[]),e&&e.stop&&e.stop.call(this,!0),b=f.length;b--;)f[b].elem===this&&f[b].queue===a&&(f[b].anim.stop(!0),f.splice(b,1));for(b=0;g>b;b++)d[b]&&d[b].finish&&d[b].finish.call(this);delete c.finish})}}),n.each(["toggle","show","hide"],function(a,b){var c=n.fn[b];n.fn[b]=function(a,d,e){return null==a||"boolean"==typeof a?c.apply(this,arguments):this.animate(Xa(b,!0),a,d,e)}}),n.each({slideDown:Xa("show"),slideUp:Xa("hide"),slideToggle:Xa("toggle"),fadeIn:{opacity:"show"},fadeOut:{opacity:"hide"},fadeToggle:{opacity:"toggle"}},function(a,b){n.fn[a]=function(a,c,d){return this.animate(b,a,c,d)}}),n.timers=[],n.fx.tick=function(){var a,b=0,c=n.timers;for(Sa=n.now();b<c.length;b++)a=c[b],a()||c[b]!==a||c.splice(b--,1);c.length||n.fx.stop(),Sa=void 0},n.fx.timer=function(a){n.timers.push(a),a()?n.fx.start():n.timers.pop()},n.fx.interval=13,n.fx.start=function(){Ta||(Ta=a.setInterval(n.fx.tick,n.fx.interval))},n.fx.stop=function(){a.clearInterval(Ta),Ta=null},n.fx.speeds={slow:600,fast:200,_default:400},n.fn.delay=function(b,c){return b=n.fx?n.fx.speeds[b]||b:b,c=c||"fx",this.queue(c,function(c,d){var e=a.setTimeout(c,b);d.stop=function(){a.clearTimeout(e)}})},function(){var a=d.createElement("input"),b=d.createElement("select"),c=b.appendChild(d.createElement("option"));a.type="checkbox",l.checkOn=""!==a.value,l.optSelected=c.selected,b.disabled=!0,l.optDisabled=!c.disabled,a=d.createElement("input"),a.value="t",a.type="radio",l.radioValue="t"===a.value}();var ab,bb=n.expr.attrHandle;n.fn.extend({attr:function(a,b){return K(this,n.attr,a,b,arguments.length>1)},removeAttr:function(a){return this.each(function(){n.removeAttr(this,a)})}}),n.extend({attr:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return"undefined"==typeof a.getAttribute?n.prop(a,b,c):(1===f&&n.isXMLDoc(a)||(b=b.toLowerCase(),e=n.attrHooks[b]||(n.expr.match.bool.test(b)?ab:void 0)),void 0!==c?null===c?void n.removeAttr(a,b):e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:(a.setAttribute(b,c+""),c):e&&"get"in e&&null!==(d=e.get(a,b))?d:(d=n.find.attr(a,b),null==d?void 0:d))},attrHooks:{type:{set:function(a,b){if(!l.radioValue&&"radio"===b&&n.nodeName(a,"input")){var c=a.value;return a.setAttribute("type",b),c&&(a.value=c),b}}}},removeAttr:function(a,b){var c,d,e=0,f=b&&b.match(G);if(f&&1===a.nodeType)while(c=f[e++])d=n.propFix[c]||c,n.expr.match.bool.test(c)&&(a[d]=!1),a.removeAttribute(c)}}),ab={set:function(a,b,c){return b===!1?n.removeAttr(a,c):a.setAttribute(c,c),c}},n.each(n.expr.match.bool.source.match(/\w+/g),function(a,b){var c=bb[b]||n.find.attr;bb[b]=function(a,b,d){var e,f;return d||(f=bb[b],bb[b]=e,e=null!=c(a,b,d)?b.toLowerCase():null,bb[b]=f),e}});var cb=/^(?:input|select|textarea|button)$/i,db=/^(?:a|area)$/i;n.fn.extend({prop:function(a,b){return K(this,n.prop,a,b,arguments.length>1)},removeProp:function(a){return this.each(function(){delete this[n.propFix[a]||a]})}}),n.extend({prop:function(a,b,c){var d,e,f=a.nodeType;if(3!==f&&8!==f&&2!==f)return 1===f&&n.isXMLDoc(a)||(b=n.propFix[b]||b,
+e=n.propHooks[b]),void 0!==c?e&&"set"in e&&void 0!==(d=e.set(a,c,b))?d:a[b]=c:e&&"get"in e&&null!==(d=e.get(a,b))?d:a[b]},propHooks:{tabIndex:{get:function(a){var b=n.find.attr(a,"tabindex");return b?parseInt(b,10):cb.test(a.nodeName)||db.test(a.nodeName)&&a.href?0:-1}}},propFix:{"for":"htmlFor","class":"className"}}),l.optSelected||(n.propHooks.selected={get:function(a){var b=a.parentNode;return b&&b.parentNode&&b.parentNode.selectedIndex,null},set:function(a){var b=a.parentNode;b&&(b.selectedIndex,b.parentNode&&b.parentNode.selectedIndex)}}),n.each(["tabIndex","readOnly","maxLength","cellSpacing","cellPadding","rowSpan","colSpan","useMap","frameBorder","contentEditable"],function(){n.propFix[this.toLowerCase()]=this});var eb=/[\t\r\n\f]/g;function fb(a){return a.getAttribute&&a.getAttribute("class")||""}n.fn.extend({addClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).addClass(a.call(this,b,fb(this)))});if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=fb(c),d=1===c.nodeType&&(" "+e+" ").replace(eb," ")){g=0;while(f=b[g++])d.indexOf(" "+f+" ")<0&&(d+=f+" ");h=n.trim(d),e!==h&&c.setAttribute("class",h)}}return this},removeClass:function(a){var b,c,d,e,f,g,h,i=0;if(n.isFunction(a))return this.each(function(b){n(this).removeClass(a.call(this,b,fb(this)))});if(!arguments.length)return this.attr("class","");if("string"==typeof a&&a){b=a.match(G)||[];while(c=this[i++])if(e=fb(c),d=1===c.nodeType&&(" "+e+" ").replace(eb," ")){g=0;while(f=b[g++])while(d.indexOf(" "+f+" ")>-1)d=d.replace(" "+f+" "," ");h=n.trim(d),e!==h&&c.setAttribute("class",h)}}return this},toggleClass:function(a,b){var c=typeof a;return"boolean"==typeof b&&"string"===c?b?this.addClass(a):this.removeClass(a):n.isFunction(a)?this.each(function(c){n(this).toggleClass(a.call(this,c,fb(this),b),b)}):this.each(function(){var b,d,e,f;if("string"===c){d=0,e=n(this),f=a.match(G)||[];while(b=f[d++])e.hasClass(b)?e.removeClass(b):e.addClass(b)}else void 0!==a&&"boolean"!==c||(b=fb(this),b&&N.set(this,"__className__",b),this.setAttribute&&this.setAttribute("class",b||a===!1?"":N.get(this,"__className__")||""))})},hasClass:function(a){var b,c,d=0;b=" "+a+" ";while(c=this[d++])if(1===c.nodeType&&(" "+fb(c)+" ").replace(eb," ").indexOf(b)>-1)return!0;return!1}});var gb=/\r/g,hb=/[\x20\t\r\n\f]+/g;n.fn.extend({val:function(a){var b,c,d,e=this[0];{if(arguments.length)return d=n.isFunction(a),this.each(function(c){var e;1===this.nodeType&&(e=d?a.call(this,c,n(this).val()):a,null==e?e="":"number"==typeof e?e+="":n.isArray(e)&&(e=n.map(e,function(a){return null==a?"":a+""})),b=n.valHooks[this.type]||n.valHooks[this.nodeName.toLowerCase()],b&&"set"in b&&void 0!==b.set(this,e,"value")||(this.value=e))});if(e)return b=n.valHooks[e.type]||n.valHooks[e.nodeName.toLowerCase()],b&&"get"in b&&void 0!==(c=b.get(e,"value"))?c:(c=e.value,"string"==typeof c?c.replace(gb,""):null==c?"":c)}}}),n.extend({valHooks:{option:{get:function(a){var b=n.find.attr(a,"value");return null!=b?b:n.trim(n.text(a)).replace(hb," ")}},select:{get:function(a){for(var b,c,d=a.options,e=a.selectedIndex,f="select-one"===a.type||0>e,g=f?null:[],h=f?e+1:d.length,i=0>e?h:f?e:0;h>i;i++)if(c=d[i],(c.selected||i===e)&&(l.optDisabled?!c.disabled:null===c.getAttribute("disabled"))&&(!c.parentNode.disabled||!n.nodeName(c.parentNode,"optgroup"))){if(b=n(c).val(),f)return b;g.push(b)}return g},set:function(a,b){var c,d,e=a.options,f=n.makeArray(b),g=e.length;while(g--)d=e[g],(d.selected=n.inArray(n.valHooks.option.get(d),f)>-1)&&(c=!0);return c||(a.selectedIndex=-1),f}}}}),n.each(["radio","checkbox"],function(){n.valHooks[this]={set:function(a,b){return n.isArray(b)?a.checked=n.inArray(n(a).val(),b)>-1:void 0}},l.checkOn||(n.valHooks[this].get=function(a){return null===a.getAttribute("value")?"on":a.value})});var ib=/^(?:focusinfocus|focusoutblur)$/;n.extend(n.event,{trigger:function(b,c,e,f){var g,h,i,j,l,m,o,p=[e||d],q=k.call(b,"type")?b.type:b,r=k.call(b,"namespace")?b.namespace.split("."):[];if(h=i=e=e||d,3!==e.nodeType&&8!==e.nodeType&&!ib.test(q+n.event.triggered)&&(q.indexOf(".")>-1&&(r=q.split("."),q=r.shift(),r.sort()),l=q.indexOf(":")<0&&"on"+q,b=b[n.expando]?b:new n.Event(q,"object"==typeof b&&b),b.isTrigger=f?2:3,b.namespace=r.join("."),b.rnamespace=b.namespace?new RegExp("(^|\\.)"+r.join("\\.(?:.*\\.|)")+"(\\.|$)"):null,b.result=void 0,b.target||(b.target=e),c=null==c?[b]:n.makeArray(c,[b]),o=n.event.special[q]||{},f||!o.trigger||o.trigger.apply(e,c)!==!1)){if(!f&&!o.noBubble&&!n.isWindow(e)){for(j=o.delegateType||q,ib.test(j+q)||(h=h.parentNode);h;h=h.parentNode)p.push(h),i=h;i===(e.ownerDocument||d)&&p.push(i.defaultView||i.parentWindow||a)}g=0;while((h=p[g++])&&!b.isPropagationStopped())b.type=g>1?j:o.bindType||q,m=(N.get(h,"events")||{})[b.type]&&N.get(h,"handle"),m&&m.apply(h,c),m=l&&h[l],m&&m.apply&&L(h)&&(b.result=m.apply(h,c),b.result===!1&&b.preventDefault());return b.type=q,f||b.isDefaultPrevented()||o._default&&o._default.apply(p.pop(),c)!==!1||!L(e)||l&&n.isFunction(e[q])&&!n.isWindow(e)&&(i=e[l],i&&(e[l]=null),n.event.triggered=q,e[q](),n.event.triggered=void 0,i&&(e[l]=i)),b.result}},simulate:function(a,b,c){var d=n.extend(new n.Event,c,{type:a,isSimulated:!0});n.event.trigger(d,null,b),d.isDefaultPrevented()&&c.preventDefault()}}),n.fn.extend({trigger:function(a,b){return this.each(function(){n.event.trigger(a,b,this)})},triggerHandler:function(a,b){var c=this[0];return c?n.event.trigger(a,b,c,!0):void 0}}),n.each("blur focus focusin focusout load resize scroll unload click dblclick mousedown mouseup mousemove mouseover mouseout mouseenter mouseleave change select submit keydown keypress keyup error contextmenu".split(" "),function(a,b){n.fn[b]=function(a,c){return arguments.length>0?this.on(b,null,a,c):this.trigger(b)}}),n.fn.extend({hover:function(a,b){return this.mouseenter(a).mouseleave(b||a)}}),l.focusin="onfocusin"in a,l.focusin||n.each({focus:"focusin",blur:"focusout"},function(a,b){var c=function(a){n.event.simulate(b,a.target,n.event.fix(a))};n.event.special[b]={setup:function(){var d=this.ownerDocument||this,e=N.access(d,b);e||d.addEventListener(a,c,!0),N.access(d,b,(e||0)+1)},teardown:function(){var d=this.ownerDocument||this,e=N.access(d,b)-1;e?N.access(d,b,e):(d.removeEventListener(a,c,!0),N.remove(d,b))}}});var jb=a.location,kb=n.now(),lb=/\?/;n.parseJSON=function(a){return JSON.parse(a+"")},n.parseXML=function(b){var c;if(!b||"string"!=typeof b)return null;try{c=(new a.DOMParser).parseFromString(b,"text/xml")}catch(d){c=void 0}return c&&!c.getElementsByTagName("parsererror").length||n.error("Invalid XML: "+b),c};var mb=/#.*$/,nb=/([?&])_=[^&]*/,ob=/^(.*?):[ \t]*([^\r\n]*)$/gm,pb=/^(?:about|app|app-storage|.+-extension|file|res|widget):$/,qb=/^(?:GET|HEAD)$/,rb=/^\/\//,sb={},tb={},ub="*/".concat("*"),vb=d.createElement("a");vb.href=jb.href;function wb(a){return function(b,c){"string"!=typeof b&&(c=b,b="*");var d,e=0,f=b.toLowerCase().match(G)||[];if(n.isFunction(c))while(d=f[e++])"+"===d[0]?(d=d.slice(1)||"*",(a[d]=a[d]||[]).unshift(c)):(a[d]=a[d]||[]).push(c)}}function xb(a,b,c,d){var e={},f=a===tb;function g(h){var i;return e[h]=!0,n.each(a[h]||[],function(a,h){var j=h(b,c,d);return"string"!=typeof j||f||e[j]?f?!(i=j):void 0:(b.dataTypes.unshift(j),g(j),!1)}),i}return g(b.dataTypes[0])||!e["*"]&&g("*")}function yb(a,b){var c,d,e=n.ajaxSettings.flatOptions||{};for(c in b)void 0!==b[c]&&((e[c]?a:d||(d={}))[c]=b[c]);return d&&n.extend(!0,a,d),a}function zb(a,b,c){var d,e,f,g,h=a.contents,i=a.dataTypes;while("*"===i[0])i.shift(),void 0===d&&(d=a.mimeType||b.getResponseHeader("Content-Type"));if(d)for(e in h)if(h[e]&&h[e].test(d)){i.unshift(e);break}if(i[0]in c)f=i[0];else{for(e in c){if(!i[0]||a.converters[e+" "+i[0]]){f=e;break}g||(g=e)}f=f||g}return f?(f!==i[0]&&i.unshift(f),c[f]):void 0}function Ab(a,b,c,d){var e,f,g,h,i,j={},k=a.dataTypes.slice();if(k[1])for(g in a.converters)j[g.toLowerCase()]=a.converters[g];f=k.shift();while(f)if(a.responseFields[f]&&(c[a.responseFields[f]]=b),!i&&d&&a.dataFilter&&(b=a.dataFilter(b,a.dataType)),i=f,f=k.shift())if("*"===f)f=i;else if("*"!==i&&i!==f){if(g=j[i+" "+f]||j["* "+f],!g)for(e in j)if(h=e.split(" "),h[1]===f&&(g=j[i+" "+h[0]]||j["* "+h[0]])){g===!0?g=j[e]:j[e]!==!0&&(f=h[0],k.unshift(h[1]));break}if(g!==!0)if(g&&a["throws"])b=g(b);else try{b=g(b)}catch(l){return{state:"parsererror",error:g?l:"No conversion from "+i+" to "+f}}}return{state:"success",data:b}}n.extend({active:0,lastModified:{},etag:{},ajaxSettings:{url:jb.href,type:"GET",isLocal:pb.test(jb.protocol),global:!0,processData:!0,async:!0,contentType:"application/x-www-form-urlencoded; charset=UTF-8",accepts:{"*":ub,text:"text/plain",html:"text/html",xml:"application/xml, text/xml",json:"application/json, text/javascript"},contents:{xml:/\bxml\b/,html:/\bhtml/,json:/\bjson\b/},responseFields:{xml:"responseXML",text:"responseText",json:"responseJSON"},converters:{"* text":String,"text html":!0,"text json":n.parseJSON,"text xml":n.parseXML},flatOptions:{url:!0,context:!0}},ajaxSetup:function(a,b){return b?yb(yb(a,n.ajaxSettings),b):yb(n.ajaxSettings,a)},ajaxPrefilter:wb(sb),ajaxTransport:wb(tb),ajax:function(b,c){"object"==typeof b&&(c=b,b=void 0),c=c||{};var e,f,g,h,i,j,k,l,m=n.ajaxSetup({},c),o=m.context||m,p=m.context&&(o.nodeType||o.jquery)?n(o):n.event,q=n.Deferred(),r=n.Callbacks("once memory"),s=m.statusCode||{},t={},u={},v=0,w="canceled",x={readyState:0,getResponseHeader:function(a){var b;if(2===v){if(!h){h={};while(b=ob.exec(g))h[b[1].toLowerCase()]=b[2]}b=h[a.toLowerCase()]}return null==b?null:b},getAllResponseHeaders:function(){return 2===v?g:null},setRequestHeader:function(a,b){var c=a.toLowerCase();return v||(a=u[c]=u[c]||a,t[a]=b),this},overrideMimeType:function(a){return v||(m.mimeType=a),this},statusCode:function(a){var b;if(a)if(2>v)for(b in a)s[b]=[s[b],a[b]];else x.always(a[x.status]);return this},abort:function(a){var b=a||w;return e&&e.abort(b),z(0,b),this}};if(q.promise(x).complete=r.add,x.success=x.done,x.error=x.fail,m.url=((b||m.url||jb.href)+"").replace(mb,"").replace(rb,jb.protocol+"//"),m.type=c.method||c.type||m.method||m.type,m.dataTypes=n.trim(m.dataType||"*").toLowerCase().match(G)||[""],null==m.crossDomain){j=d.createElement("a");try{j.href=m.url,j.href=j.href,m.crossDomain=vb.protocol+"//"+vb.host!=j.protocol+"//"+j.host}catch(y){m.crossDomain=!0}}if(m.data&&m.processData&&"string"!=typeof m.data&&(m.data=n.param(m.data,m.traditional)),xb(sb,m,c,x),2===v)return x;k=n.event&&m.global,k&&0===n.active++&&n.event.trigger("ajaxStart"),m.type=m.type.toUpperCase(),m.hasContent=!qb.test(m.type),f=m.url,m.hasContent||(m.data&&(f=m.url+=(lb.test(f)?"&":"?")+m.data,delete m.data),m.cache===!1&&(m.url=nb.test(f)?f.replace(nb,"$1_="+kb++):f+(lb.test(f)?"&":"?")+"_="+kb++)),m.ifModified&&(n.lastModified[f]&&x.setRequestHeader("If-Modified-Since",n.lastModified[f]),n.etag[f]&&x.setRequestHeader("If-None-Match",n.etag[f])),(m.data&&m.hasContent&&m.contentType!==!1||c.contentType)&&x.setRequestHeader("Content-Type",m.contentType),x.setRequestHeader("Accept",m.dataTypes[0]&&m.accepts[m.dataTypes[0]]?m.accepts[m.dataTypes[0]]+("*"!==m.dataTypes[0]?", "+ub+"; q=0.01":""):m.accepts["*"]);for(l in m.headers)x.setRequestHeader(l,m.headers[l]);if(m.beforeSend&&(m.beforeSend.call(o,x,m)===!1||2===v))return x.abort();w="abort";for(l in{success:1,error:1,complete:1})x[l](m[l]);if(e=xb(tb,m,c,x)){if(x.readyState=1,k&&p.trigger("ajaxSend",[x,m]),2===v)return x;m.async&&m.timeout>0&&(i=a.setTimeout(function(){x.abort("timeout")},m.timeout));try{v=1,e.send(t,z)}catch(y){if(!(2>v))throw y;z(-1,y)}}else z(-1,"No Transport");function z(b,c,d,h){var j,l,t,u,w,y=c;2!==v&&(v=2,i&&a.clearTimeout(i),e=void 0,g=h||"",x.readyState=b>0?4:0,j=b>=200&&300>b||304===b,d&&(u=zb(m,x,d)),u=Ab(m,u,x,j),j?(m.ifModified&&(w=x.getResponseHeader("Last-Modified"),w&&(n.lastModified[f]=w),w=x.getResponseHeader("etag"),w&&(n.etag[f]=w)),204===b||"HEAD"===m.type?y="nocontent":304===b?y="notmodified":(y=u.state,l=u.data,t=u.error,j=!t)):(t=y,!b&&y||(y="error",0>b&&(b=0))),x.status=b,x.statusText=(c||y)+"",j?q.resolveWith(o,[l,y,x]):q.rejectWith(o,[x,y,t]),x.statusCode(s),s=void 0,k&&p.trigger(j?"ajaxSuccess":"ajaxError",[x,m,j?l:t]),r.fireWith(o,[x,y]),k&&(p.trigger("ajaxComplete",[x,m]),--n.active||n.event.trigger("ajaxStop")))}return x},getJSON:function(a,b,c){return n.get(a,b,c,"json")},getScript:function(a,b){return n.get(a,void 0,b,"script")}}),n.each(["get","post"],function(a,b){n[b]=function(a,c,d,e){return n.isFunction(c)&&(e=e||d,d=c,c=void 0),n.ajax(n.extend({url:a,type:b,dataType:e,data:c,success:d},n.isPlainObject(a)&&a))}}),n._evalUrl=function(a){return n.ajax({url:a,type:"GET",dataType:"script",async:!1,global:!1,"throws":!0})},n.fn.extend({wrapAll:function(a){var b;return n.isFunction(a)?this.each(function(b){n(this).wrapAll(a.call(this,b))}):(this[0]&&(b=n(a,this[0].ownerDocument).eq(0).clone(!0),this[0].parentNode&&b.insertBefore(this[0]),b.map(function(){var a=this;while(a.firstElementChild)a=a.firstElementChild;return a}).append(this)),this)},wrapInner:function(a){return n.isFunction(a)?this.each(function(b){n(this).wrapInner(a.call(this,b))}):this.each(function(){var b=n(this),c=b.contents();c.length?c.wrapAll(a):b.append(a)})},wrap:function(a){var b=n.isFunction(a);return this.each(function(c){n(this).wrapAll(b?a.call(this,c):a)})},unwrap:function(){return this.parent().each(function(){n.nodeName(this,"body")||n(this).replaceWith(this.childNodes)}).end()}}),n.expr.filters.hidden=function(a){return!n.expr.filters.visible(a)},n.expr.filters.visible=function(a){return a.offsetWidth>0||a.offsetHeight>0||a.getClientRects().length>0};var Bb=/%20/g,Cb=/\[\]$/,Db=/\r?\n/g,Eb=/^(?:submit|button|image|reset|file)$/i,Fb=/^(?:input|select|textarea|keygen)/i;function Gb(a,b,c,d){var e;if(n.isArray(b))n.each(b,function(b,e){c||Cb.test(a)?d(a,e):Gb(a+"["+("object"==typeof e&&null!=e?b:"")+"]",e,c,d)});else if(c||"object"!==n.type(b))d(a,b);else for(e in b)Gb(a+"["+e+"]",b[e],c,d)}n.param=function(a,b){var c,d=[],e=function(a,b){b=n.isFunction(b)?b():null==b?"":b,d[d.length]=encodeURIComponent(a)+"="+encodeURIComponent(b)};if(void 0===b&&(b=n.ajaxSettings&&n.ajaxSettings.traditional),n.isArray(a)||a.jquery&&!n.isPlainObject(a))n.each(a,function(){e(this.name,this.value)});else for(c in a)Gb(c,a[c],b,e);return d.join("&").replace(Bb,"+")},n.fn.extend({serialize:function(){return n.param(this.serializeArray())},serializeArray:function(){return this.map(function(){var a=n.prop(this,"elements");return a?n.makeArray(a):this}).filter(function(){var a=this.type;return this.name&&!n(this).is(":disabled")&&Fb.test(this.nodeName)&&!Eb.test(a)&&(this.checked||!X.test(a))}).map(function(a,b){var c=n(this).val();return null==c?null:n.isArray(c)?n.map(c,function(a){return{name:b.name,value:a.replace(Db,"\r\n")}}):{name:b.name,value:c.replace(Db,"\r\n")}}).get()}}),n.ajaxSettings.xhr=function(){try{return new a.XMLHttpRequest}catch(b){}};var Hb={0:200,1223:204},Ib=n.ajaxSettings.xhr();l.cors=!!Ib&&"withCredentials"in Ib,l.ajax=Ib=!!Ib,n.ajaxTransport(function(b){var c,d;return l.cors||Ib&&!b.crossDomain?{send:function(e,f){var g,h=b.xhr();if(h.open(b.type,b.url,b.async,b.username,b.password),b.xhrFields)for(g in b.xhrFields)h[g]=b.xhrFields[g];b.mimeType&&h.overrideMimeType&&h.overrideMimeType(b.mimeType),b.crossDomain||e["X-Requested-With"]||(e["X-Requested-With"]="XMLHttpRequest");for(g in e)h.setRequestHeader(g,e[g]);c=function(a){return function(){c&&(c=d=h.onload=h.onerror=h.onabort=h.onreadystatechange=null,"abort"===a?h.abort():"error"===a?"number"!=typeof h.status?f(0,"error"):f(h.status,h.statusText):f(Hb[h.status]||h.status,h.statusText,"text"!==(h.responseType||"text")||"string"!=typeof h.responseText?{binary:h.response}:{text:h.responseText},h.getAllResponseHeaders()))}},h.onload=c(),d=h.onerror=c("error"),void 0!==h.onabort?h.onabort=d:h.onreadystatechange=function(){4===h.readyState&&a.setTimeout(function(){c&&d()})},c=c("abort");try{h.send(b.hasContent&&b.data||null)}catch(i){if(c)throw i}},abort:function(){c&&c()}}:void 0}),n.ajaxSetup({accepts:{script:"text/javascript, application/javascript, application/ecmascript, application/x-ecmascript"},contents:{script:/\b(?:java|ecma)script\b/},converters:{"text script":function(a){return n.globalEval(a),a}}}),n.ajaxPrefilter("script",function(a){void 0===a.cache&&(a.cache=!1),a.crossDomain&&(a.type="GET")}),n.ajaxTransport("script",function(a){if(a.crossDomain){var b,c;return{send:function(e,f){b=n("<script>").prop({charset:a.scriptCharset,src:a.url}).on("load error",c=function(a){b.remove(),c=null,a&&f("error"===a.type?404:200,a.type)}),d.head.appendChild(b[0])},abort:function(){c&&c()}}}});var Jb=[],Kb=/(=)\?(?=&|$)|\?\?/;n.ajaxSetup({jsonp:"callback",jsonpCallback:function(){var a=Jb.pop()||n.expando+"_"+kb++;return this[a]=!0,a}}),n.ajaxPrefilter("json jsonp",function(b,c,d){var e,f,g,h=b.jsonp!==!1&&(Kb.test(b.url)?"url":"string"==typeof b.data&&0===(b.contentType||"").indexOf("application/x-www-form-urlencoded")&&Kb.test(b.data)&&"data");return h||"jsonp"===b.dataTypes[0]?(e=b.jsonpCallback=n.isFunction(b.jsonpCallback)?b.jsonpCallback():b.jsonpCallback,h?b[h]=b[h].replace(Kb,"$1"+e):b.jsonp!==!1&&(b.url+=(lb.test(b.url)?"&":"?")+b.jsonp+"="+e),b.converters["script json"]=function(){return g||n.error(e+" was not called"),g[0]},b.dataTypes[0]="json",f=a[e],a[e]=function(){g=arguments},d.always(function(){void 0===f?n(a).removeProp(e):a[e]=f,b[e]&&(b.jsonpCallback=c.jsonpCallback,Jb.push(e)),g&&n.isFunction(f)&&f(g[0]),g=f=void 0}),"script"):void 0}),n.parseHTML=function(a,b,c){if(!a||"string"!=typeof a)return null;"boolean"==typeof b&&(c=b,b=!1),b=b||d;var e=x.exec(a),f=!c&&[];return e?[b.createElement(e[1])]:(e=ca([a],b,f),f&&f.length&&n(f).remove(),n.merge([],e.childNodes))};var Lb=n.fn.load;n.fn.load=function(a,b,c){if("string"!=typeof a&&Lb)return Lb.apply(this,arguments);var d,e,f,g=this,h=a.indexOf(" ");return h>-1&&(d=n.trim(a.slice(h)),a=a.slice(0,h)),n.isFunction(b)?(c=b,b=void 0):b&&"object"==typeof b&&(e="POST"),g.length>0&&n.ajax({url:a,type:e||"GET",dataType:"html",data:b}).done(function(a){f=arguments,g.html(d?n("<div>").append(n.parseHTML(a)).find(d):a)}).always(c&&function(a,b){g.each(function(){c.apply(g,f||[a.responseText,b,a])})}),this},n.each(["ajaxStart","ajaxStop","ajaxComplete","ajaxError","ajaxSuccess","ajaxSend"],function(a,b){n.fn[b]=function(a){return this.on(b,a)}}),n.expr.filters.animated=function(a){return n.grep(n.timers,function(b){return a===b.elem}).length};function Mb(a){return n.isWindow(a)?a:9===a.nodeType&&a.defaultView}n.offset={setOffset:function(a,b,c){var d,e,f,g,h,i,j,k=n.css(a,"position"),l=n(a),m={};"static"===k&&(a.style.position="relative"),h=l.offset(),f=n.css(a,"top"),i=n.css(a,"left"),j=("absolute"===k||"fixed"===k)&&(f+i).indexOf("auto")>-1,j?(d=l.position(),g=d.top,e=d.left):(g=parseFloat(f)||0,e=parseFloat(i)||0),n.isFunction(b)&&(b=b.call(a,c,n.extend({},h))),null!=b.top&&(m.top=b.top-h.top+g),null!=b.left&&(m.left=b.left-h.left+e),"using"in b?b.using.call(a,m):l.css(m)}},n.fn.extend({offset:function(a){if(arguments.length)return void 0===a?this:this.each(function(b){n.offset.setOffset(this,a,b)});var b,c,d=this[0],e={top:0,left:0},f=d&&d.ownerDocument;if(f)return b=f.documentElement,n.contains(b,d)?(e=d.getBoundingClientRect(),c=Mb(f),{top:e.top+c.pageYOffset-b.clientTop,left:e.left+c.pageXOffset-b.clientLeft}):e},position:function(){if(this[0]){var a,b,c=this[0],d={top:0,left:0};return"fixed"===n.css(c,"position")?b=c.getBoundingClientRect():(a=this.offsetParent(),b=this.offset(),n.nodeName(a[0],"html")||(d=a.offset()),d.top+=n.css(a[0],"borderTopWidth",!0),d.left+=n.css(a[0],"borderLeftWidth",!0)),{top:b.top-d.top-n.css(c,"marginTop",!0),left:b.left-d.left-n.css(c,"marginLeft",!0)}}},offsetParent:function(){return this.map(function(){var a=this.offsetParent;while(a&&"static"===n.css(a,"position"))a=a.offsetParent;return a||Ea})}}),n.each({scrollLeft:"pageXOffset",scrollTop:"pageYOffset"},function(a,b){var c="pageYOffset"===b;n.fn[a]=function(d){return K(this,function(a,d,e){var f=Mb(a);return void 0===e?f?f[b]:a[d]:void(f?f.scrollTo(c?f.pageXOffset:e,c?e:f.pageYOffset):a[d]=e)},a,d,arguments.length)}}),n.each(["top","left"],function(a,b){n.cssHooks[b]=Ga(l.pixelPosition,function(a,c){return c?(c=Fa(a,b),Ba.test(c)?n(a).position()[b]+"px":c):void 0})}),n.each({Height:"height",Width:"width"},function(a,b){n.each({padding:"inner"+a,content:b,"":"outer"+a},function(c,d){n.fn[d]=function(d,e){var f=arguments.length&&(c||"boolean"!=typeof d),g=c||(d===!0||e===!0?"margin":"border");return K(this,function(b,c,d){var e;return n.isWindow(b)?b.document.documentElement["client"+a]:9===b.nodeType?(e=b.documentElement,Math.max(b.body["scroll"+a],e["scroll"+a],b.body["offset"+a],e["offset"+a],e["client"+a])):void 0===d?n.css(b,c,g):n.style(b,c,d,g)},b,f?d:void 0,f,null)}})}),n.fn.extend({bind:function(a,b,c){return this.on(a,null,b,c)},unbind:function(a,b){return this.off(a,null,b)},delegate:function(a,b,c,d){return this.on(b,a,c,d)},undelegate:function(a,b,c){return 1===arguments.length?this.off(a,"**"):this.off(b,a||"**",c)},size:function(){return this.length}}),n.fn.andSelf=n.fn.addBack,"function"==typeof define&&define.amd&&define("jquery",[],function(){return n});var Nb=a.jQuery,Ob=a.$;return n.noConflict=function(b){return a.$===n&&(a.$=Ob),b&&a.jQuery===n&&(a.jQuery=Nb),n},b||(a.jQuery=a.$=n),n});
diff --git a/tools/phabricator/xreate-frontend/resources/vendors/thrift/thrift.js b/tools/phabricator/xreate-frontend/resources/vendors/thrift/thrift.js
new file mode 100644
index 0000000..c846162
--- /dev/null
+++ b/tools/phabricator/xreate-frontend/resources/vendors/thrift/thrift.js
@@ -0,0 +1,1549 @@
+/*
+ * Licensed to the Apache Software Foundation (ASF) under one
+ * or more contributor license agreements. See the NOTICE file
+ * distributed with this work for additional information
+ * regarding copyright ownership. The ASF licenses this file
+ * to you under the Apache License, Version 2.0 (the
+ * "License"); you may not use this file except in compliance
+ * with the License. You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing,
+ * software distributed under the License is distributed on an
+ * "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY
+ * KIND, either express or implied. See the License for the
+ * specific language governing permissions and limitations
+ * under the License.
+ */
+
+/*jshint evil:true*/
+
+/**
+ * The Thrift namespace houses the Apache Thrift JavaScript library
+ * elements providing JavaScript bindings for the Apache Thrift RPC
+ * system. End users will typically only directly make use of the
+ * Transport (TXHRTransport/TWebSocketTransport) and Protocol
+ * (TJSONPRotocol/TBinaryProtocol) constructors.
+ *
+ * Object methods beginning with a __ (e.g. __onOpen()) are internal
+ * and should not be called outside of the object's own methods.
+ *
+ * This library creates one global object: Thrift
+ * Code in this library must never create additional global identifiers,
+ * all features must be scoped within the Thrift namespace.
+ * @namespace
+ * @example
+ * var transport = new Thrift.Transport("http://localhost:8585");
+ * var protocol = new Thrift.Protocol(transport);
+ * var client = new MyThriftSvcClient(protocol);
+ * var result = client.MyMethod();
+ */
+var Thrift = {
+ /**
+ * Thrift JavaScript library version.
+ * @readonly
+ * @const {string} Version
+ * @memberof Thrift
+ */
+ Version: '1.0.0-dev',
+
+ /**
+ * Thrift IDL type string to Id mapping.
+ * @readonly
+ * @property {number} STOP - End of a set of fields.
+ * @property {number} VOID - No value (only legal for return types).
+ * @property {number} BOOL - True/False integer.
+ * @property {number} BYTE - Signed 8 bit integer.
+ * @property {number} I08 - Signed 8 bit integer.
+ * @property {number} DOUBLE - 64 bit IEEE 854 floating point.
+ * @property {number} I16 - Signed 16 bit integer.
+ * @property {number} I32 - Signed 32 bit integer.
+ * @property {number} I64 - Signed 64 bit integer.
+ * @property {number} STRING - Array of bytes representing a string of characters.
+ * @property {number} UTF7 - Array of bytes representing a string of UTF7 encoded characters.
+ * @property {number} STRUCT - A multifield type.
+ * @property {number} MAP - A collection type (map/associative-array/dictionary).
+ * @property {number} SET - A collection type (unordered and without repeated values).
+ * @property {number} LIST - A collection type (unordered).
+ * @property {number} UTF8 - Array of bytes representing a string of UTF8 encoded characters.
+ * @property {number} UTF16 - Array of bytes representing a string of UTF16 encoded characters.
+ */
+ Type: {
+ 'STOP' : 0,
+ 'VOID' : 1,
+ 'BOOL' : 2,
+ 'BYTE' : 3,
+ 'I08' : 3,
+ 'DOUBLE' : 4,
+ 'I16' : 6,
+ 'I32' : 8,
+ 'I64' : 10,
+ 'STRING' : 11,
+ 'UTF7' : 11,
+ 'STRUCT' : 12,
+ 'MAP' : 13,
+ 'SET' : 14,
+ 'LIST' : 15,
+ 'UTF8' : 16,
+ 'UTF16' : 17
+ },
+
+ /**
+ * Thrift RPC message type string to Id mapping.
+ * @readonly
+ * @property {number} CALL - RPC call sent from client to server.
+ * @property {number} REPLY - RPC call normal response from server to client.
+ * @property {number} EXCEPTION - RPC call exception response from server to client.
+ * @property {number} ONEWAY - Oneway RPC call from client to server with no response.
+ */
+ MessageType: {
+ 'CALL' : 1,
+ 'REPLY' : 2,
+ 'EXCEPTION' : 3,
+ 'ONEWAY' : 4
+ },
+
+ /**
+ * Utility function returning the count of an object's own properties.
+ * @param {object} obj - Object to test.
+ * @returns {number} number of object's own properties
+ */
+ objectLength: function(obj) {
+ var length = 0;
+ for (var k in obj) {
+ if (obj.hasOwnProperty(k)) {
+ length++;
+ }
+ }
+ return length;
+ },
+
+ /**
+ * Utility function to establish prototype inheritance.
+ * @see {@link http://javascript.crockford.com/prototypal.html|Prototypal Inheritance}
+ * @param {function} constructor - Contstructor function to set as derived.
+ * @param {function} superConstructor - Contstructor function to set as base.
+ * @param {string} [name] - Type name to set as name property in derived prototype.
+ */
+ inherits: function(constructor, superConstructor, name) {
+ function F() {}
+ F.prototype = superConstructor.prototype;
+ constructor.prototype = new F();
+ constructor.prototype.name = name || "";
+ }
+};
+
+/**
+ * Initializes a Thrift TException instance.
+ * @constructor
+ * @augments Error
+ * @param {string} message - The TException message (distinct from the Error message).
+ * @classdesc TException is the base class for all Thrift exceptions types.
+ */
+Thrift.TException = function(message) {
+ this.message = message;
+};
+Thrift.inherits(Thrift.TException, Error, 'TException');
+
+/**
+ * Returns the message set on the exception.
+ * @readonly
+ * @returns {string} exception message
+ */
+Thrift.TException.prototype.getMessage = function() {
+ return this.message;
+};
+
+/**
+ * Thrift Application Exception type string to Id mapping.
+ * @readonly
+ * @property {number} UNKNOWN - Unknown/undefined.
+ * @property {number} UNKNOWN_METHOD - Client attempted to call a method unknown to the server.
+ * @property {number} INVALID_MESSAGE_TYPE - Client passed an unknown/unsupported MessageType.
+ * @property {number} WRONG_METHOD_NAME - Unused.
+ * @property {number} BAD_SEQUENCE_ID - Unused in Thrift RPC, used to flag proprietary sequence number errors.
+ * @property {number} MISSING_RESULT - Raised by a server processor if a handler fails to supply the required return result.
+ * @property {number} INTERNAL_ERROR - Something bad happened.
+ * @property {number} PROTOCOL_ERROR - The protocol layer failed to serialize or deserialize data.
+ * @property {number} INVALID_TRANSFORM - Unused.
+ * @property {number} INVALID_PROTOCOL - The protocol (or version) is not supported.
+ * @property {number} UNSUPPORTED_CLIENT_TYPE - Unused.
+ */
+Thrift.TApplicationExceptionType = {
+ 'UNKNOWN' : 0,
+ 'UNKNOWN_METHOD' : 1,
+ 'INVALID_MESSAGE_TYPE' : 2,
+ 'WRONG_METHOD_NAME' : 3,
+ 'BAD_SEQUENCE_ID' : 4,
+ 'MISSING_RESULT' : 5,
+ 'INTERNAL_ERROR' : 6,
+ 'PROTOCOL_ERROR' : 7,
+ 'INVALID_TRANSFORM' : 8,
+ 'INVALID_PROTOCOL' : 9,
+ 'UNSUPPORTED_CLIENT_TYPE' : 10
+};
+
+/**
+ * Initializes a Thrift TApplicationException instance.
+ * @constructor
+ * @augments Thrift.TException
+ * @param {string} message - The TApplicationException message (distinct from the Error message).
+ * @param {Thrift.TApplicationExceptionType} [code] - The TApplicationExceptionType code.
+ * @classdesc TApplicationException is the exception class used to propagate exceptions from an RPC server back to a calling client.
+*/
+Thrift.TApplicationException = function(message, code) {
+ this.message = message;
+ this.code = typeof code === "number" ? code : 0;
+};
+Thrift.inherits(Thrift.TApplicationException, Thrift.TException, 'TApplicationException');
+
+/**
+ * Read a TApplicationException from the supplied protocol.
+ * @param {object} input - The input protocol to read from.
+ */
+Thrift.TApplicationException.prototype.read = function(input) {
+ while (1) {
+ var ret = input.readFieldBegin();
+
+ if (ret.ftype == Thrift.Type.STOP) {
+ break;
+ }
+
+ var fid = ret.fid;
+
+ switch (fid) {
+ case 1:
+ if (ret.ftype == Thrift.Type.STRING) {
+ ret = input.readString();
+ this.message = ret.value;
+ } else {
+ ret = input.skip(ret.ftype);
+ }
+ break;
+ case 2:
+ if (ret.ftype == Thrift.Type.I32) {
+ ret = input.readI32();
+ this.code = ret.value;
+ } else {
+ ret = input.skip(ret.ftype);
+ }
+ break;
+ default:
+ ret = input.skip(ret.ftype);
+ break;
+ }
+
+ input.readFieldEnd();
+ }
+
+ input.readStructEnd();
+};
+
+/**
+ * Wite a TApplicationException to the supplied protocol.
+ * @param {object} output - The output protocol to write to.
+ */
+Thrift.TApplicationException.prototype.write = function(output) {
+ output.writeStructBegin('TApplicationException');
+
+ if (this.message) {
+ output.writeFieldBegin('message', Thrift.Type.STRING, 1);
+ output.writeString(this.getMessage());
+ output.writeFieldEnd();
+ }
+
+ if (this.code) {
+ output.writeFieldBegin('type', Thrift.Type.I32, 2);
+ output.writeI32(this.code);
+ output.writeFieldEnd();
+ }
+
+ output.writeFieldStop();
+ output.writeStructEnd();
+};
+
+/**
+ * Returns the application exception code set on the exception.
+ * @readonly
+ * @returns {Thrift.TApplicationExceptionType} exception code
+ */
+Thrift.TApplicationException.prototype.getCode = function() {
+ return this.code;
+};
+
+Thrift.TProtocolExceptionType = {
+ UNKNOWN: 0,
+ INVALID_DATA: 1,
+ NEGATIVE_SIZE: 2,
+ SIZE_LIMIT: 3,
+ BAD_VERSION: 4,
+ NOT_IMPLEMENTED: 5,
+ DEPTH_LIMIT: 6
+};
+
+Thrift.TProtocolException = function TProtocolException(type, message) {
+ Error.call(this);
+ Error.captureStackTrace(this, this.constructor);
+ this.name = this.constructor.name;
+ this.type = type;
+ this.message = message;
+};
+Thrift.inherits(Thrift.TProtocolException, Thrift.TException, 'TProtocolException');
+
+/**
+ * Constructor Function for the XHR transport.
+ * If you do not specify a url then you must handle XHR operations on
+ * your own. This type can also be constructed using the Transport alias
+ * for backward compatibility.
+ * @constructor
+ * @param {string} [url] - The URL to connect to.
+ * @classdesc The Apache Thrift Transport layer performs byte level I/O
+ * between RPC clients and servers. The JavaScript TXHRTransport object
+ * uses Http[s]/XHR. Target servers must implement the http[s] transport
+ * (see: node.js example server_http.js).
+ * @example
+ * var transport = new Thrift.TXHRTransport("http://localhost:8585");
+ */
+Thrift.Transport = Thrift.TXHRTransport = function(url, options) {
+ this.url = url;
+ this.wpos = 0;
+ this.rpos = 0;
+ this.useCORS = (options && options.useCORS);
+ this.send_buf = '';
+ this.recv_buf = '';
+};
+
+Thrift.TXHRTransport.prototype = {
+ /**
+ * Gets the browser specific XmlHttpRequest Object.
+ * @returns {object} the browser XHR interface object
+ */
+ getXmlHttpRequestObject: function() {
+ try { return new XMLHttpRequest(); } catch (e1) { }
+ try { return new ActiveXObject('Msxml2.XMLHTTP'); } catch (e2) { }
+ try { return new ActiveXObject('Microsoft.XMLHTTP'); } catch (e3) { }
+
+ throw "Your browser doesn't support XHR.";
+ },
+
+ /**
+ * Sends the current XRH request if the transport was created with a URL
+ * and the async parameter is false. If the transport was not created with
+ * a URL, or the async parameter is True and no callback is provided, or
+ * the URL is an empty string, the current send buffer is returned.
+ * @param {object} async - If true the current send buffer is returned.
+ * @param {object} callback - Optional async completion callback
+ * @returns {undefined|string} Nothing or the current send buffer.
+ * @throws {string} If XHR fails.
+ */
+ flush: function(async, callback) {
+ var self = this;
+ if ((async && !callback) || this.url === undefined || this.url === '') {
+ return this.send_buf;
+ }
+
+ var xreq = this.getXmlHttpRequestObject();
+
+ if (xreq.overrideMimeType) {
+ xreq.overrideMimeType('application/vnd.apache.thrift.json; charset=utf-8');
+ }
+
+ if (callback) {
+ //Ignore XHR callbacks until the data arrives, then call the
+ // client's callback
+ xreq.onreadystatechange =
+ (function() {
+ var clientCallback = callback;
+ return function() {
+ if (this.readyState == 4 && this.status == 200) {
+ self.setRecvBuffer(this.responseText);
+ clientCallback();
+ }
+ };
+ }());
+
+ // detect net::ERR_CONNECTION_REFUSED and call the callback.
+ xreq.onerror =
+ (function() {
+ var clientCallback = callback;
+ return function() {
+ clientCallback();
+ };
+ }());
+
+ }
+
+ xreq.open('POST', this.url, !!async);
+
+ if (xreq.setRequestHeader) {
+ xreq.setRequestHeader('Accept', 'application/vnd.apache.thrift.json; charset=utf-8');
+ xreq.setRequestHeader('Content-Type', 'application/vnd.apache.thrift.json; charset=utf-8');
+ }
+
+ xreq.send(this.send_buf);
+ if (async && callback) {
+ return;
+ }
+
+ if (xreq.readyState != 4) {
+ throw 'encountered an unknown ajax ready state: ' + xreq.readyState;
+ }
+
+ if (xreq.status != 200) {
+ throw 'encountered a unknown request status: ' + xreq.status;
+ }
+
+ this.recv_buf = xreq.responseText;
+ this.recv_buf_sz = this.recv_buf.length;
+ this.wpos = this.recv_buf.length;
+ this.rpos = 0;
+ },
+
+ /**
+ * Creates a jQuery XHR object to be used for a Thrift server call.
+ * @param {object} client - The Thrift Service client object generated by the IDL compiler.
+ * @param {object} postData - The message to send to the server.
+ * @param {function} args - The original call arguments with the success call back at the end.
+ * @param {function} recv_method - The Thrift Service Client receive method for the call.
+ * @returns {object} A new jQuery XHR object.
+ * @throws {string} If the jQuery version is prior to 1.5 or if jQuery is not found.
+ */
+ jqRequest: function(client, postData, args, recv_method) {
+ if (typeof jQuery === 'undefined' ||
+ typeof jQuery.Deferred === 'undefined') {
+ throw 'Thrift.js requires jQuery 1.5+ to use asynchronous requests';
+ }
+
+ var thriftTransport = this;
+
+ var jqXHR = jQuery.ajax({
+ url: this.url,
+ data: postData,
+ type: 'POST',
+ cache: false,
+ contentType: 'application/vnd.apache.thrift.json; charset=utf-8',
+ dataType: 'text thrift',
+ converters: {
+ 'text thrift' : function(responseData) {
+ thriftTransport.setRecvBuffer(responseData);
+ var value = recv_method.call(client);
+ return value;
+ }
+ },
+ context: client,
+ success: jQuery.makeArray(args).pop()
+ });
+
+ return jqXHR;
+ },
+
+ /**
+ * Sets the buffer to provide the protocol when deserializing.
+ * @param {string} buf - The buffer to supply the protocol.
+ */
+ setRecvBuffer: function(buf) {
+ this.recv_buf = buf;
+ this.recv_buf_sz = this.recv_buf.length;
+ this.wpos = this.recv_buf.length;
+ this.rpos = 0;
+ },
+
+ /**
+ * Returns true if the transport is open, XHR always returns true.
+ * @readonly
+ * @returns {boolean} Always True.
+ */
+ isOpen: function() {
+ return true;
+ },
+
+ /**
+ * Opens the transport connection, with XHR this is a nop.
+ */
+ open: function() {},
+
+ /**
+ * Closes the transport connection, with XHR this is a nop.
+ */
+ close: function() {},
+
+ /**
+ * Returns the specified number of characters from the response
+ * buffer.
+ * @param {number} len - The number of characters to return.
+ * @returns {string} Characters sent by the server.
+ */
+ read: function(len) {
+ var avail = this.wpos - this.rpos;
+
+ if (avail === 0) {
+ return '';
+ }
+
+ var give = len;
+
+ if (avail < len) {
+ give = avail;
+ }
+
+ var ret = this.read_buf.substr(this.rpos, give);
+ this.rpos += give;
+
+ //clear buf when complete?
+ return ret;
+ },
+
+ /**
+ * Returns the entire response buffer.
+ * @returns {string} Characters sent by the server.
+ */
+ readAll: function() {
+ return this.recv_buf;
+ },
+
+ /**
+ * Sets the send buffer to buf.
+ * @param {string} buf - The buffer to send.
+ */
+ write: function(buf) {
+ this.send_buf = buf;
+ },
+
+ /**
+ * Returns the send buffer.
+ * @readonly
+ * @returns {string} The send buffer.
+ */
+ getSendBuffer: function() {
+ return this.send_buf;
+ }
+
+};
+
+
+/**
+ * Constructor Function for the WebSocket transport.
+ * @constructor
+ * @param {string} [url] - The URL to connect to.
+ * @classdesc The Apache Thrift Transport layer performs byte level I/O
+ * between RPC clients and servers. The JavaScript TWebSocketTransport object
+ * uses the WebSocket protocol. Target servers must implement WebSocket.
+ * (see: node.js example server_http.js).
+ * @example
+ * var transport = new Thrift.TWebSocketTransport("http://localhost:8585");
+ */
+Thrift.TWebSocketTransport = function(url) {
+ this.__reset(url);
+};
+
+Thrift.TWebSocketTransport.prototype = {
+ __reset: function(url) {
+ this.url = url; //Where to connect
+ this.socket = null; //The web socket
+ this.callbacks = []; //Pending callbacks
+ this.send_pending = []; //Buffers/Callback pairs waiting to be sent
+ this.send_buf = ''; //Outbound data, immutable until sent
+ this.recv_buf = ''; //Inbound data
+ this.rb_wpos = 0; //Network write position in receive buffer
+ this.rb_rpos = 0; //Client read position in receive buffer
+ },
+
+ /**
+ * Sends the current WS request and registers callback. The async
+ * parameter is ignored (WS flush is always async) and the callback
+ * function parameter is required.
+ * @param {object} async - Ignored.
+ * @param {object} callback - The client completion callback.
+ * @returns {undefined|string} Nothing (undefined)
+ */
+ flush: function(async, callback) {
+ var self = this;
+ if (this.isOpen()) {
+ //Send data and register a callback to invoke the client callback
+ this.socket.send(this.send_buf);
+ this.callbacks.push((function() {
+ var clientCallback = callback;
+ return function(msg) {
+ self.setRecvBuffer(msg);
+ clientCallback();
+ };
+ }()));
+ } else {
+ //Queue the send to go out __onOpen
+ this.send_pending.push({
+ buf: this.send_buf,
+ cb: callback
+ });
+ }
+ },
+
+ __onOpen: function() {
+ var self = this;
+ if (this.send_pending.length > 0) {
+ //If the user made calls before the connection was fully
+ //open, send them now
+ this.send_pending.forEach(function(elem) {
+ this.socket.send(elem.buf);
+ this.callbacks.push((function() {
+ var clientCallback = elem.cb;
+ return function(msg) {
+ self.setRecvBuffer(msg);
+ clientCallback();
+ };
+ }()));
+ });
+ this.send_pending = [];
+ }
+ },
+
+ __onClose: function(evt) {
+ this.__reset(this.url);
+ },
+
+ __onMessage: function(evt) {
+ if (this.callbacks.length) {
+ this.callbacks.shift()(evt.data);
+ }
+ },
+
+ __onError: function(evt) {
+ console.log("Thrift WebSocket Error: " + evt.toString());
+ this.socket.close();
+ },
+
+ /**
+ * Sets the buffer to use when receiving server responses.
+ * @param {string} buf - The buffer to receive server responses.
+ */
+ setRecvBuffer: function(buf) {
+ this.recv_buf = buf;
+ this.recv_buf_sz = this.recv_buf.length;
+ this.wpos = this.recv_buf.length;
+ this.rpos = 0;
+ },
+
+ /**
+ * Returns true if the transport is open
+ * @readonly
+ * @returns {boolean}
+ */
+ isOpen: function() {
+ return this.socket && this.socket.readyState == this.socket.OPEN;
+ },
+
+ /**
+ * Opens the transport connection
+ */
+ open: function() {
+ //If OPEN/CONNECTING/CLOSING ignore additional opens
+ if (this.socket && this.socket.readyState != this.socket.CLOSED) {
+ return;
+ }
+ //If there is no socket or the socket is closed:
+ this.socket = new WebSocket(this.url);
+ this.socket.onopen = this.__onOpen.bind(this);
+ this.socket.onmessage = this.__onMessage.bind(this);
+ this.socket.onerror = this.__onError.bind(this);
+ this.socket.onclose = this.__onClose.bind(this);
+ },
+
+ /**
+ * Closes the transport connection
+ */
+ close: function() {
+ this.socket.close();
+ },
+
+ /**
+ * Returns the specified number of characters from the response
+ * buffer.
+ * @param {number} len - The number of characters to return.
+ * @returns {string} Characters sent by the server.
+ */
+ read: function(len) {
+ var avail = this.wpos - this.rpos;
+
+ if (avail === 0) {
+ return '';
+ }
+
+ var give = len;
+
+ if (avail < len) {
+ give = avail;
+ }
+
+ var ret = this.read_buf.substr(this.rpos, give);
+ this.rpos += give;
+
+ //clear buf when complete?
+ return ret;
+ },
+
+ /**
+ * Returns the entire response buffer.
+ * @returns {string} Characters sent by the server.
+ */
+ readAll: function() {
+ return this.recv_buf;
+ },
+
+ /**
+ * Sets the send buffer to buf.
+ * @param {string} buf - The buffer to send.
+ */
+ write: function(buf) {
+ this.send_buf = buf;
+ },
+
+ /**
+ * Returns the send buffer.
+ * @readonly
+ * @returns {string} The send buffer.
+ */
+ getSendBuffer: function() {
+ return this.send_buf;
+ }
+
+};
+
+/**
+ * Initializes a Thrift JSON protocol instance.
+ * @constructor
+ * @param {Thrift.Transport} transport - The transport to serialize to/from.
+ * @classdesc Apache Thrift Protocols perform serialization which enables cross
+ * language RPC. The Protocol type is the JavaScript browser implementation
+ * of the Apache Thrift TJSONProtocol.
+ * @example
+ * var protocol = new Thrift.Protocol(transport);
+ */
+Thrift.TJSONProtocol = Thrift.Protocol = function(transport) {
+ this.tstack = [];
+ this.tpos = [];
+ this.transport = transport;
+};
+
+/**
+ * Thrift IDL type Id to string mapping.
+ * @readonly
+ * @see {@link Thrift.Type}
+ */
+Thrift.Protocol.Type = {};
+Thrift.Protocol.Type[Thrift.Type.BOOL] = '"tf"';
+Thrift.Protocol.Type[Thrift.Type.BYTE] = '"i8"';
+Thrift.Protocol.Type[Thrift.Type.I16] = '"i16"';
+Thrift.Protocol.Type[Thrift.Type.I32] = '"i32"';
+Thrift.Protocol.Type[Thrift.Type.I64] = '"i64"';
+Thrift.Protocol.Type[Thrift.Type.DOUBLE] = '"dbl"';
+Thrift.Protocol.Type[Thrift.Type.STRUCT] = '"rec"';
+Thrift.Protocol.Type[Thrift.Type.STRING] = '"str"';
+Thrift.Protocol.Type[Thrift.Type.MAP] = '"map"';
+Thrift.Protocol.Type[Thrift.Type.LIST] = '"lst"';
+Thrift.Protocol.Type[Thrift.Type.SET] = '"set"';
+
+/**
+ * Thrift IDL type string to Id mapping.
+ * @readonly
+ * @see {@link Thrift.Type}
+ */
+Thrift.Protocol.RType = {};
+Thrift.Protocol.RType.tf = Thrift.Type.BOOL;
+Thrift.Protocol.RType.i8 = Thrift.Type.BYTE;
+Thrift.Protocol.RType.i16 = Thrift.Type.I16;
+Thrift.Protocol.RType.i32 = Thrift.Type.I32;
+Thrift.Protocol.RType.i64 = Thrift.Type.I64;
+Thrift.Protocol.RType.dbl = Thrift.Type.DOUBLE;
+Thrift.Protocol.RType.rec = Thrift.Type.STRUCT;
+Thrift.Protocol.RType.str = Thrift.Type.STRING;
+Thrift.Protocol.RType.map = Thrift.Type.MAP;
+Thrift.Protocol.RType.lst = Thrift.Type.LIST;
+Thrift.Protocol.RType.set = Thrift.Type.SET;
+
+/**
+ * The TJSONProtocol version number.
+ * @readonly
+ * @const {number} Version
+ * @memberof Thrift.Protocol
+ */
+ Thrift.Protocol.Version = 1;
+
+Thrift.Protocol.prototype = {
+ /**
+ * Returns the underlying transport.
+ * @readonly
+ * @returns {Thrift.Transport} The underlying transport.
+ */
+ getTransport: function() {
+ return this.transport;
+ },
+
+ /**
+ * Serializes the beginning of a Thrift RPC message.
+ * @param {string} name - The service method to call.
+ * @param {Thrift.MessageType} messageType - The type of method call.
+ * @param {number} seqid - The sequence number of this call (always 0 in Apache Thrift).
+ */
+ writeMessageBegin: function(name, messageType, seqid) {
+ this.tstack = [];
+ this.tpos = [];
+
+ this.tstack.push([Thrift.Protocol.Version, '"' +
+ name + '"', messageType, seqid]);
+ },
+
+ /**
+ * Serializes the end of a Thrift RPC message.
+ */
+ writeMessageEnd: function() {
+ var obj = this.tstack.pop();
+
+ this.wobj = this.tstack.pop();
+ this.wobj.push(obj);
+
+ this.wbuf = '[' + this.wobj.join(',') + ']';
+
+ this.transport.write(this.wbuf);
+ },
+
+
+ /**
+ * Serializes the beginning of a struct.
+ * @param {string} name - The name of the struct.
+ */
+ writeStructBegin: function(name) {
+ this.tpos.push(this.tstack.length);
+ this.tstack.push({});
+ },
+
+ /**
+ * Serializes the end of a struct.
+ */
+ writeStructEnd: function() {
+
+ var p = this.tpos.pop();
+ var struct = this.tstack[p];
+ var str = '{';
+ var first = true;
+ for (var key in struct) {
+ if (first) {
+ first = false;
+ } else {
+ str += ',';
+ }
+
+ str += key + ':' + struct[key];
+ }
+
+ str += '}';
+ this.tstack[p] = str;
+ },
+
+ /**
+ * Serializes the beginning of a struct field.
+ * @param {string} name - The name of the field.
+ * @param {Thrift.Protocol.Type} fieldType - The data type of the field.
+ * @param {number} fieldId - The field's unique identifier.
+ */
+ writeFieldBegin: function(name, fieldType, fieldId) {
+ this.tpos.push(this.tstack.length);
+ this.tstack.push({ 'fieldId': '"' +
+ fieldId + '"', 'fieldType': Thrift.Protocol.Type[fieldType]
+ });
+
+ },
+
+ /**
+ * Serializes the end of a field.
+ */
+ writeFieldEnd: function() {
+ var value = this.tstack.pop();
+ var fieldInfo = this.tstack.pop();
+
+ this.tstack[this.tstack.length - 1][fieldInfo.fieldId] = '{' +
+ fieldInfo.fieldType + ':' + value + '}';
+ this.tpos.pop();
+ },
+
+ /**
+ * Serializes the end of the set of fields for a struct.
+ */
+ writeFieldStop: function() {
+ //na
+ },
+
+ /**
+ * Serializes the beginning of a map collection.
+ * @param {Thrift.Type} keyType - The data type of the key.
+ * @param {Thrift.Type} valType - The data type of the value.
+ * @param {number} [size] - The number of elements in the map (ignored).
+ */
+ writeMapBegin: function(keyType, valType, size) {
+ this.tpos.push(this.tstack.length);
+ this.tstack.push([Thrift.Protocol.Type[keyType],
+ Thrift.Protocol.Type[valType], 0]);
+ },
+
+ /**
+ * Serializes the end of a map.
+ */
+ writeMapEnd: function() {
+ var p = this.tpos.pop();
+
+ if (p == this.tstack.length) {
+ return;
+ }
+
+ if ((this.tstack.length - p - 1) % 2 !== 0) {
+ this.tstack.push('');
+ }
+
+ var size = (this.tstack.length - p - 1) / 2;
+
+ this.tstack[p][this.tstack[p].length - 1] = size;
+
+ var map = '}';
+ var first = true;
+ while (this.tstack.length > p + 1) {
+ var v = this.tstack.pop();
+ var k = this.tstack.pop();
+ if (first) {
+ first = false;
+ } else {
+ map = ',' + map;
+ }
+
+ if (! isNaN(k)) { k = '"' + k + '"'; } //json "keys" need to be strings
+ map = k + ':' + v + map;
+ }
+ map = '{' + map;
+
+ this.tstack[p].push(map);
+ this.tstack[p] = '[' + this.tstack[p].join(',') + ']';
+ },
+
+ /**
+ * Serializes the beginning of a list collection.
+ * @param {Thrift.Type} elemType - The data type of the elements.
+ * @param {number} size - The number of elements in the list.
+ */
+ writeListBegin: function(elemType, size) {
+ this.tpos.push(this.tstack.length);
+ this.tstack.push([Thrift.Protocol.Type[elemType], size]);
+ },
+
+ /**
+ * Serializes the end of a list.
+ */
+ writeListEnd: function() {
+ var p = this.tpos.pop();
+
+ while (this.tstack.length > p + 1) {
+ var tmpVal = this.tstack[p + 1];
+ this.tstack.splice(p + 1, 1);
+ this.tstack[p].push(tmpVal);
+ }
+
+ this.tstack[p] = '[' + this.tstack[p].join(',') + ']';
+ },
+
+ /**
+ * Serializes the beginning of a set collection.
+ * @param {Thrift.Type} elemType - The data type of the elements.
+ * @param {number} size - The number of elements in the list.
+ */
+ writeSetBegin: function(elemType, size) {
+ this.tpos.push(this.tstack.length);
+ this.tstack.push([Thrift.Protocol.Type[elemType], size]);
+ },
+
+ /**
+ * Serializes the end of a set.
+ */
+ writeSetEnd: function() {
+ var p = this.tpos.pop();
+
+ while (this.tstack.length > p + 1) {
+ var tmpVal = this.tstack[p + 1];
+ this.tstack.splice(p + 1, 1);
+ this.tstack[p].push(tmpVal);
+ }
+
+ this.tstack[p] = '[' + this.tstack[p].join(',') + ']';
+ },
+
+ /** Serializes a boolean */
+ writeBool: function(value) {
+ this.tstack.push(value ? 1 : 0);
+ },
+
+ /** Serializes a number */
+ writeByte: function(i8) {
+ this.tstack.push(i8);
+ },
+
+ /** Serializes a number */
+ writeI16: function(i16) {
+ this.tstack.push(i16);
+ },
+
+ /** Serializes a number */
+ writeI32: function(i32) {
+ this.tstack.push(i32);
+ },
+
+ /** Serializes a number */
+ writeI64: function(i64) {
+ this.tstack.push(i64);
+ },
+
+ /** Serializes a number */
+ writeDouble: function(dbl) {
+ this.tstack.push(dbl);
+ },
+
+ /** Serializes a string */
+ writeString: function(str) {
+ // We do not encode uri components for wire transfer:
+ if (str === null) {
+ this.tstack.push(null);
+ } else {
+ // concat may be slower than building a byte buffer
+ var escapedString = '';
+ for (var i = 0; i < str.length; i++) {
+ var ch = str.charAt(i); // a single double quote: "
+ if (ch === '\"') {
+ escapedString += '\\\"'; // write out as: \"
+ } else if (ch === '\\') { // a single backslash
+ escapedString += '\\\\'; // write out as double backslash
+ } else if (ch === '\b') { // a single backspace: invisible
+ escapedString += '\\b'; // write out as: \b"
+ } else if (ch === '\f') { // a single formfeed: invisible
+ escapedString += '\\f'; // write out as: \f"
+ } else if (ch === '\n') { // a single newline: invisible
+ escapedString += '\\n'; // write out as: \n"
+ } else if (ch === '\r') { // a single return: invisible
+ escapedString += '\\r'; // write out as: \r"
+ } else if (ch === '\t') { // a single tab: invisible
+ escapedString += '\\t'; // write out as: \t"
+ } else {
+ escapedString += ch; // Else it need not be escaped
+ }
+ }
+ this.tstack.push('"' + escapedString + '"');
+ }
+ },
+
+ /** Serializes a string */
+ writeBinary: function(binary) {
+ var str = '';
+ if (typeof binary == 'string') {
+ str = binary;
+ } else if (binary instanceof Uint8Array) {
+ var arr = binary;
+ for (var i = 0; i < arr.length; ++i) {
+ str += String.fromCharCode(arr[i]);
+ }
+ } else {
+ throw new TypeError('writeBinary only accepts String or Uint8Array.');
+ }
+ this.tstack.push('"' + btoa(str) + '"');
+ },
+
+ /**
+ @class
+ @name AnonReadMessageBeginReturn
+ @property {string} fname - The name of the service method.
+ @property {Thrift.MessageType} mtype - The type of message call.
+ @property {number} rseqid - The sequence number of the message (0 in Thrift RPC).
+ */
+ /**
+ * Deserializes the beginning of a message.
+ * @returns {AnonReadMessageBeginReturn}
+ */
+ readMessageBegin: function() {
+ this.rstack = [];
+ this.rpos = [];
+
+ if (typeof JSON !== 'undefined' && typeof JSON.parse === 'function') {
+ this.robj = JSON.parse(this.transport.readAll());
+ } else if (typeof jQuery !== 'undefined') {
+ this.robj = jQuery.parseJSON(this.transport.readAll());
+ } else {
+ this.robj = eval(this.transport.readAll());
+ }
+
+ var r = {};
+ var version = this.robj.shift();
+
+ if (version != Thrift.Protocol.Version) {
+ throw 'Wrong thrift protocol version: ' + version;
+ }
+
+ r.fname = this.robj.shift();
+ r.mtype = this.robj.shift();
+ r.rseqid = this.robj.shift();
+
+
+ //get to the main obj
+ this.rstack.push(this.robj.shift());
+
+ return r;
+ },
+
+ /** Deserializes the end of a message. */
+ readMessageEnd: function() {
+ },
+
+ /**
+ * Deserializes the beginning of a struct.
+ * @param {string} [name] - The name of the struct (ignored)
+ * @returns {object} - An object with an empty string fname property
+ */
+ readStructBegin: function(name) {
+ var r = {};
+ r.fname = '';
+
+ //incase this is an array of structs
+ if (this.rstack[this.rstack.length - 1] instanceof Array) {
+ this.rstack.push(this.rstack[this.rstack.length - 1].shift());
+ }
+
+ return r;
+ },
+
+ /** Deserializes the end of a struct. */
+ readStructEnd: function() {
+ if (this.rstack[this.rstack.length - 2] instanceof Array) {
+ this.rstack.pop();
+ }
+ },
+
+ /**
+ @class
+ @name AnonReadFieldBeginReturn
+ @property {string} fname - The name of the field (always '').
+ @property {Thrift.Type} ftype - The data type of the field.
+ @property {number} fid - The unique identifier of the field.
+ */
+ /**
+ * Deserializes the beginning of a field.
+ * @returns {AnonReadFieldBeginReturn}
+ */
+ readFieldBegin: function() {
+ var r = {};
+
+ var fid = -1;
+ var ftype = Thrift.Type.STOP;
+
+ //get a fieldId
+ for (var f in (this.rstack[this.rstack.length - 1])) {
+ if (f === null) {
+ continue;
+ }
+
+ fid = parseInt(f, 10);
+ this.rpos.push(this.rstack.length);
+
+ var field = this.rstack[this.rstack.length - 1][fid];
+
+ //remove so we don't see it again
+ delete this.rstack[this.rstack.length - 1][fid];
+
+ this.rstack.push(field);
+
+ break;
+ }
+
+ if (fid != -1) {
+
+ //should only be 1 of these but this is the only
+ //way to match a key
+ for (var i in (this.rstack[this.rstack.length - 1])) {
+ if (Thrift.Protocol.RType[i] === null) {
+ continue;
+ }
+
+ ftype = Thrift.Protocol.RType[i];
+ this.rstack[this.rstack.length - 1] =
+ this.rstack[this.rstack.length - 1][i];
+ }
+ }
+
+ r.fname = '';
+ r.ftype = ftype;
+ r.fid = fid;
+
+ return r;
+ },
+
+ /** Deserializes the end of a field. */
+ readFieldEnd: function() {
+ var pos = this.rpos.pop();
+
+ //get back to the right place in the stack
+ while (this.rstack.length > pos) {
+ this.rstack.pop();
+ }
+
+ },
+
+ /**
+ @class
+ @name AnonReadMapBeginReturn
+ @property {Thrift.Type} ktype - The data type of the key.
+ @property {Thrift.Type} vtype - The data type of the value.
+ @property {number} size - The number of elements in the map.
+ */
+ /**
+ * Deserializes the beginning of a map.
+ * @returns {AnonReadMapBeginReturn}
+ */
+ readMapBegin: function() {
+ var map = this.rstack.pop();
+ var first = map.shift();
+ if (first instanceof Array) {
+ this.rstack.push(map);
+ map = first;
+ first = map.shift();
+ }
+
+ var r = {};
+ r.ktype = Thrift.Protocol.RType[first];
+ r.vtype = Thrift.Protocol.RType[map.shift()];
+ r.size = map.shift();
+
+
+ this.rpos.push(this.rstack.length);
+ this.rstack.push(map.shift());
+
+ return r;
+ },
+
+ /** Deserializes the end of a map. */
+ readMapEnd: function() {
+ this.readFieldEnd();
+ },
+
+ /**
+ @class
+ @name AnonReadColBeginReturn
+ @property {Thrift.Type} etype - The data type of the element.
+ @property {number} size - The number of elements in the collection.
+ */
+ /**
+ * Deserializes the beginning of a list.
+ * @returns {AnonReadColBeginReturn}
+ */
+ readListBegin: function() {
+ var list = this.rstack[this.rstack.length - 1];
+
+ var r = {};
+ r.etype = Thrift.Protocol.RType[list.shift()];
+ r.size = list.shift();
+
+ this.rpos.push(this.rstack.length);
+ this.rstack.push(list.shift());
+
+ return r;
+ },
+
+ /** Deserializes the end of a list. */
+ readListEnd: function() {
+ this.readFieldEnd();
+ },
+
+ /**
+ * Deserializes the beginning of a set.
+ * @returns {AnonReadColBeginReturn}
+ */
+ readSetBegin: function(elemType, size) {
+ return this.readListBegin(elemType, size);
+ },
+
+ /** Deserializes the end of a set. */
+ readSetEnd: function() {
+ return this.readListEnd();
+ },
+
+ /** Returns an object with a value property set to
+ * False unless the next number in the protocol buffer
+ * is 1, in which case the value property is True */
+ readBool: function() {
+ var r = this.readI32();
+
+ if (r !== null && r.value == '1') {
+ r.value = true;
+ } else {
+ r.value = false;
+ }
+
+ return r;
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readByte: function() {
+ return this.readI32();
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readI16: function() {
+ return this.readI32();
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readI32: function(f) {
+ if (f === undefined) {
+ f = this.rstack[this.rstack.length - 1];
+ }
+
+ var r = {};
+
+ if (f instanceof Array) {
+ if (f.length === 0) {
+ r.value = undefined;
+ } else {
+ r.value = f.shift();
+ }
+ } else if (f instanceof Object) {
+ for (var i in f) {
+ if (i === null) {
+ continue;
+ }
+ this.rstack.push(f[i]);
+ delete f[i];
+
+ r.value = i;
+ break;
+ }
+ } else {
+ r.value = f;
+ this.rstack.pop();
+ }
+
+ return r;
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readI64: function() {
+ return this.readI32();
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readDouble: function() {
+ return this.readI32();
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readString: function() {
+ var r = this.readI32();
+ return r;
+ },
+
+ /** Returns the an object with a value property set to the
+ next value found in the protocol buffer */
+ readBinary: function() {
+ var r = this.readI32();
+ r.value = atob(r.value);
+ return r;
+ },
+
+ /**
+ * Method to arbitrarily skip over data */
+ skip: function(type) {
+ var ret, i;
+ switch (type) {
+ case Thrift.Type.STOP:
+ return null;
+
+ case Thrift.Type.BOOL:
+ return this.readBool();
+
+ case Thrift.Type.BYTE:
+ return this.readByte();
+
+ case Thrift.Type.I16:
+ return this.readI16();
+
+ case Thrift.Type.I32:
+ return this.readI32();
+
+ case Thrift.Type.I64:
+ return this.readI64();
+
+ case Thrift.Type.DOUBLE:
+ return this.readDouble();
+
+ case Thrift.Type.STRING:
+ return this.readString();
+
+ case Thrift.Type.STRUCT:
+ this.readStructBegin();
+ while (true) {
+ ret = this.readFieldBegin();
+ if (ret.ftype == Thrift.Type.STOP) {
+ break;
+ }
+ this.skip(ret.ftype);
+ this.readFieldEnd();
+ }
+ this.readStructEnd();
+ return null;
+
+ case Thrift.Type.MAP:
+ ret = this.readMapBegin();
+ for (i = 0; i < ret.size; i++) {
+ if (i > 0) {
+ if (this.rstack.length > this.rpos[this.rpos.length - 1] + 1) {
+ this.rstack.pop();
+ }
+ }
+ this.skip(ret.ktype);
+ this.skip(ret.vtype);
+ }
+ this.readMapEnd();
+ return null;
+
+ case Thrift.Type.SET:
+ ret = this.readSetBegin();
+ for (i = 0; i < ret.size; i++) {
+ this.skip(ret.etype);
+ }
+ this.readSetEnd();
+ return null;
+
+ case Thrift.Type.LIST:
+ ret = this.readListBegin();
+ for (i = 0; i < ret.size; i++) {
+ this.skip(ret.etype);
+ }
+ this.readListEnd();
+ return null;
+ }
+ }
+};
+
+
+/**
+ * Initializes a MutilplexProtocol Implementation as a Wrapper for Thrift.Protocol
+ * @constructor
+ */
+Thrift.MultiplexProtocol = function (srvName, trans, strictRead, strictWrite) {
+ Thrift.Protocol.call(this, trans, strictRead, strictWrite);
+ this.serviceName = srvName;
+};
+Thrift.inherits(Thrift.MultiplexProtocol, Thrift.Protocol, 'multiplexProtocol');
+
+/** Override writeMessageBegin method of prototype*/
+Thrift.MultiplexProtocol.prototype.writeMessageBegin = function (name, type, seqid) {
+
+ if (type === Thrift.MessageType.CALL || type === Thrift.MessageType.ONEWAY) {
+ Thrift.Protocol.prototype.writeMessageBegin.call(this, this.serviceName + ":" + name, type, seqid);
+ } else {
+ Thrift.Protocol.prototype.writeMessageBegin.call(this, name, type, seqid);
+ }
+};
+
+Thrift.Multiplexer = function () {
+ this.seqid = 0;
+};
+
+/** Instantiates a multiplexed client for a specific service
+ * @constructor
+ * @param {String} serviceName - The transport to serialize to/from.
+ * @param {Thrift.ServiceClient} SCl - The Service Client Class
+ * @param {Thrift.Transport} transport - Thrift.Transport instance which provides remote host:port
+ * @example
+ * var mp = new Thrift.Multiplexer();
+ * var transport = new Thrift.Transport("http://localhost:9090/foo.thrift");
+ * var protocol = new Thrift.Protocol(transport);
+ * var client = mp.createClient('AuthService', AuthServiceClient, transport);
+*/
+Thrift.Multiplexer.prototype.createClient = function (serviceName, SCl, transport) {
+ if (SCl.Client) {
+ SCl = SCl.Client;
+ }
+ var self = this;
+ SCl.prototype.new_seqid = function () {
+ self.seqid += 1;
+ return self.seqid;
+ };
+ var client = new SCl(new Thrift.MultiplexProtocol(serviceName, transport));
+
+ return client;
+};
+
+
+
+var copyList, copyMap;
+
+copyList = function(lst, types) {
+
+ if (!lst) {return lst; }
+
+ var type;
+
+ if (types.shift === undefined) {
+ type = types;
+ }
+ else {
+ type = types[0];
+ }
+ var Type = type;
+
+ var len = lst.length, result = [], i, val;
+ for (i = 0; i < len; i++) {
+ val = lst[i];
+ if (type === null) {
+ result.push(val);
+ }
+ else if (type === copyMap || type === copyList) {
+ result.push(type(val, types.slice(1)));
+ }
+ else {
+ result.push(new Type(val));
+ }
+ }
+ return result;
+};
+
+copyMap = function(obj, types){
+
+ if (!obj) {return obj; }
+
+ var type;
+
+ if (types.shift === undefined) {
+ type = types;
+ }
+ else {
+ type = types[0];
+ }
+ var Type = type;
+
+ var result = {}, val;
+ for(var prop in obj) {
+ if(obj.hasOwnProperty(prop)) {
+ val = obj[prop];
+ if (type === null) {
+ result[prop] = val;
+ }
+ else if (type === copyMap || type === copyList) {
+ result[prop] = type(val, types.slice(1));
+ }
+ else {
+ result[prop] = new Type(val);
+ }
+ }
+ }
+ return result;
+};
+
+Thrift.copyMap = copyMap;
+Thrift.copyList = copyList;

Event Timeline