#104, fix ifelse, fix foreach, add catch, add tests

This commit is contained in:
Basile Burg 2016-11-16 12:37:41 +01:00
parent f1a6c31fb6
commit 204d0fb22c
No known key found for this signature in database
GPG Key ID: 1868039F415CB8CF
1 changed files with 146 additions and 6 deletions

View File

@ -1,7 +1,7 @@
module halstead;
import
std.meta, std.traits, std.algorithm.iteration, std.json, std.conv;
std.meta, std.algorithm.iteration, std.json, std.conv;
import
dparse.lexer, dparse.parser, dparse.ast, dparse.rollback_allocator;
import
@ -300,15 +300,15 @@ private final class HalsteadMetric: ASTVisitor
override void visit(const(IfStatement) st)
{
++operators["if"];
ifStatement = true;
st.accept(this);
ifStatement = false;
if (st.thenStatement)
++operators["then"];
if (st.elseStatement)
++operators["else"];
}
override void visit(const(DeclarationOrStatement) st)
{
if (ifStatement && st.statement)
++operators["thenOrElse"];
st.accept(this);
}
@ -327,6 +327,11 @@ private final class HalsteadMetric: ASTVisitor
override void visit(const(ForeachStatement) st)
{
++operators["foreach"];
if (st.foreachTypeList)
foreach(ft; st.foreachTypeList.items)
++operands[ft.identifier.text];
if (st.foreachType)
++operands[st.foreachType.identifier.text];
st.accept(this);
}
@ -368,7 +373,7 @@ private final class HalsteadMetric: ASTVisitor
override void visit(const(CaseRangeStatement) st)
{
++operators["case"];
++++operators["case"];
st.accept(this);
}
@ -390,6 +395,14 @@ private final class HalsteadMetric: ASTVisitor
st.accept(this);
}
override void visit(const(Catch) c)
{
++operators["catch"];
c.accept(this);
if (c.identifier.text)
++operands[c.identifier.text];
}
override void visit(const(VariableDeclaration) decl)
{
if (decl.declarators)
@ -738,6 +751,133 @@ version(unittest)
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
foreach(i,a;z){}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 3);
assert(r.operators.length == 1);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
foreach(i; l..h){}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 3);
assert(r.operators.length == 1);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
for(i = 0; i < len; i++){}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 3);
assert(r.operators.length == 4);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
for(;;){continue;}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 0);
assert(r.operators.length == 2);
r.destruct;
}
unittest
{
string src =
q{
int foo()
{
while(true) {return 0;}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 2);
assert(r.operators.length == 2);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
switch(a)
{
default: break;
case 1: return;
case 2: .. case 8: ;
}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 4);
assert(r.operators.length == 4);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
try a();
catch(Exception e)
throw v;
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 2);
assert(r.operators.length == 4);
r.destruct;
}
unittest
{
string src =
q{
void foo()
{
if (true) {} else {i = 0;}
}
};
HalsteadMetric r = src.parseAndVisit!HalsteadMetric;
assert(r.operands.length == 3);
assert(r.operators.length == 4);
r.destruct;
}
unittest
{
// TODO: detect function call w/o parens