Fix issue #47: Input is now read from stdin where it should be. Also made --sloc and --tokenCount work with multiple files and the recursive option

This commit is contained in:
Hackerpilot 2013-08-25 11:14:29 +00:00
parent 270cd6d9a1
commit 001118e730
4 changed files with 112 additions and 44 deletions

View File

@ -1,5 +1,5 @@
#dmd *.d stdx/d/*.d -release -inline -noboundscheck -O -w -wi -m64 -property -ofdscanner-dmd
dmd *.d stdx/d/*.d -g -m64 -wi -ofdscanner
dmd main.d stats.d imports.d highlighter.d ctags.d astprinter.d stdx/d/*.d -g -m64 -wi -ofdscanner
#ldc2 -O3 *.d stdx/d/*.d -of=dscanner-ldc -release -m64
#ldc2 *.d stdx/d/*.d -of=dscanner -unittest -m64 -g
#/opt/gdc/bin/gdc -O3 -odscanner-gdc -fno-bounds-check -frelease -m64 *.d stdx/d/*.d

133
main.d
View File

@ -89,54 +89,62 @@ int main(string[] args)
LexerConfig config;
config.iterStyle = IterationStyle.everything;
config.tokenStyle = TokenStyle.source;
File f = args.length == 1 ? stdin : File(args[1]);
ubyte[] buffer = uninitializedArray!(ubyte[])(to!size_t(f.size));
highlighter.highlight(byToken(f.rawRead(buffer), config),
bool usingStdin = args.length == 1;
ubyte[] bytes = usingStdin ? readStdin() : readFile(args[1]);
highlighter.highlight(byToken(bytes, config),
args.length == 1 ? "stdin" : args[1]);
return 0;
}
else if (ctags)
{
if (recursive)
{
stdout.printCtags(dirEntries(args[1], SpanMode.depth)
.filter!(a => a.name.endsWith(".d") || a.name.endsWith(".di"))()
.map!(a => a.name)().array());
}
else
stdout.printCtags(args[1 .. $]);
stdout.printCtags(expandArgs(args, recursive));
}
else
{
LexerConfig config;
bool usingStdin = args.length == 3;
config.fileName = usingStdin ? "stdin" : args[1];
File f = usingStdin ? stdin : File(args[1]);
auto bytes = usingStdin ? cast(ubyte[]) [] : uninitializedArray!(ubyte[])(to!size_t(f.size));
f.rawRead(bytes);
auto tokens = byToken(bytes, config);
if (sloc)
bool usingStdin = args.length == 1;
if (sloc || tokenCount)
{
printLineCount(stdout, tokens);
if (usingStdin)
{
auto tokens = byToken(readStdin(), config);
if (tokenCount)
printTokenCount(stdout, "stdin", tokens);
else
printLineCount(stdout, "stdin", tokens);
}
else if (tokenCount)
else
{
printTokenCount(stdout, tokens);
ulong count;
foreach (f; expandArgs(args, recursive))
{
auto tokens = byToken(readFile(f), config);
if (tokenCount)
count += printTokenCount(stdout, f, tokens);
else
count += printLineCount(stdout, f, tokens);
}
writefln("total:\t%d", count);
}
}
else if (syntaxCheck)
{
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
parseModule(tokens.array(), config.fileName);
}
else if (imports)
{
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
auto mod = parseModule(tokens.array(), config.fileName);
auto visitor = new ImportPrinter;
visitor.visit(mod);
}
else if (ast)
{
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
auto mod = parseModule(tokens.array(), config.fileName);
auto printer = new XMLPrinter;
printer.output = stdout;
@ -146,9 +154,59 @@ int main(string[] args)
return 0;
}
string[] expandArgs(string[] args, bool recursive)
{
if (recursive)
{
string[] rVal;
foreach (arg; args[1 ..$])
{
if (isFile(arg) && arg.endsWith(`.d`) || arg.endsWith(`.di`))
rVal ~= arg;
else foreach (item; dirEntries(arg, SpanMode.breadth).map!(a => a.name))
{
if (isFile(item) && (item.endsWith(`.d`) || item.endsWith(`.di`)))
rVal ~= item;
else
continue;
}
}
return rVal;
}
else
return args[1 .. $];
}
ubyte[] readStdin()
{
auto sourceCode = appender!(ubyte[])();
ubyte[4096] buf;
while (true)
{
auto b = stdin.rawRead(buf);
if (b.length == 0)
break;
sourceCode.put(b);
}
return sourceCode.data;
}
ubyte[] readFile(string fileName)
{
if (!exists(fileName))
{
stderr.writefln("%s does not exist", fileName);
return [];
}
File f = File(fileName);
ubyte[] sourceCode = uninitializedArray!(ubyte[])(to!size_t(f.size));
f.rawRead(sourceCode);
return sourceCode;
}
void printHelp(string programName)
{
writefln(
stderr.writefln(
`
Usage: %s options
@ -156,22 +214,26 @@ options:
--help | -h
Prints this help message
--sloc | -l [sourceFile]
--sloc | -l [sourceFiles]
Prints the number of logical lines of code in the given
source file. If no files are specified, a file is read from stdin.
source files. If no files are specified, input is read from stdin.
--tokenCount | t [sourceFile]
Prints the number of tokens in the given source file.
--tokenCount | t [sourceFiles]
Prints the number of tokens in the given source files. If no files are
specified, input is read from stdin.
--highlight [sourceFile] - Syntax-highlight the given source file. The
resulting HTML will be written to standard output.
resulting HTML will be written to standard output. If no files are
specified, input is read from stdin.
--imports | -i [sourceFile]
Prints modules imported by the given source file.
Prints modules imported by the given source file. If no files are
specified, input is read from stdin.
--syntaxCheck | -s [sourceFile]
Lexes and parses sourceFile, printing the line and column number of any
syntax errors to stdout. One error or warning is printed per line.
If no files are specified, input is read from stdin.
--ctags | -c sourceFile
Generates ctags information from the given source code file. Note that
@ -179,11 +241,12 @@ options:
of a filename.
--ast | --xml sourceFile
Generates an XML representation of the source files abstract syntax tree
Generates an XML representation of the source files abstract syntax
tree. If no files are specified, input is read from stdin.
--recursive | -R | -r directory
When used with --ctags, dscanner will produce ctags output for all .d
and .di files contained within the given directory and its
sub-directories.`,
--recursive | -R | -r
When used with --ctags, --tokenCount, or --sloc, dscanner will produce
ctags output for all .d and .di files contained within the given
directories and its sub-directories.`,
programName);
}

11
stats.d
View File

@ -30,12 +30,14 @@ pure nothrow bool isLineOfCode(TokenType t)
}
}
void printTokenCount(Tokens)(File output, ref Tokens tokens)
ulong printTokenCount(Tokens)(File output, string fileName, ref Tokens tokens)
{
output.writefln("%d", tokens.count!(a => true));
ulong c = tokens.count!(a => true);
output.writefln("%s:\t%d", fileName, c);
return c;
}
void printLineCount(Tokens)(File output, ref Tokens tokens)
ulong printLineCount(Tokens)(File output, string fileName, ref Tokens tokens)
{
ulong count;
foreach (t; tokens)
@ -43,5 +45,6 @@ void printLineCount(Tokens)(File output, ref Tokens tokens)
if (isLineOfCode(t.type))
++count;
}
output.writefln("%d", count);
output.writefln("%s:\t%d", fileName, count);
return count;
}

View File

@ -2776,6 +2776,8 @@ body {} // six
while (moreTokens())
{
auto single = parseSingleImport();
if (single is null)
return null;
if (currentIs(TokenType.colon))
{
node.importBindings = parseImportBindings(single);