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 -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 -O3 *.d stdx/d/*.d -of=dscanner-ldc -release -m64
#ldc2 *.d stdx/d/*.d -of=dscanner -unittest -m64 -g #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 #/opt/gdc/bin/gdc -O3 -odscanner-gdc -fno-bounds-check -frelease -m64 *.d stdx/d/*.d

141
main.d
View File

@ -77,9 +77,9 @@ int main(string[] args)
{ {
stderr.writeln("Too many options specified"); stderr.writeln("Too many options specified");
return 1; return 1;
} }
else if (optionCount < 1) else if (optionCount < 1)
{ {
printHelp(args[0]); printHelp(args[0]);
return 1; return 1;
} }
@ -89,54 +89,62 @@ int main(string[] args)
LexerConfig config; LexerConfig config;
config.iterStyle = IterationStyle.everything; config.iterStyle = IterationStyle.everything;
config.tokenStyle = TokenStyle.source; config.tokenStyle = TokenStyle.source;
File f = args.length == 1 ? stdin : File(args[1]); bool usingStdin = args.length == 1;
ubyte[] buffer = uninitializedArray!(ubyte[])(to!size_t(f.size)); ubyte[] bytes = usingStdin ? readStdin() : readFile(args[1]);
highlighter.highlight(byToken(f.rawRead(buffer), config), highlighter.highlight(byToken(bytes, config),
args.length == 1 ? "stdin" : args[1]); args.length == 1 ? "stdin" : args[1]);
return 0; return 0;
} }
else if (ctags) else if (ctags)
{ {
if (recursive) stdout.printCtags(expandArgs(args, 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 .. $]);
} }
else else
{ {
LexerConfig config; LexerConfig config;
bool usingStdin = args.length == 1;
bool usingStdin = args.length == 3; if (sloc || tokenCount)
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)
{ {
printLineCount(stdout, tokens); if (usingStdin)
} {
else if (tokenCount) auto tokens = byToken(readStdin(), config);
{ if (tokenCount)
printTokenCount(stdout, tokens); printTokenCount(stdout, "stdin", tokens);
else
printLineCount(stdout, "stdin", tokens);
}
else
{
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) else if (syntaxCheck)
{ {
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
parseModule(tokens.array(), config.fileName); parseModule(tokens.array(), config.fileName);
} }
else if (imports) else if (imports)
{ {
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
auto mod = parseModule(tokens.array(), config.fileName); auto mod = parseModule(tokens.array(), config.fileName);
auto visitor = new ImportPrinter; auto visitor = new ImportPrinter;
visitor.visit(mod); visitor.visit(mod);
} }
else if (ast) else if (ast)
{ {
auto tokens = byToken(usingStdin ? readStdin() : readFile(args[1]),
config);
auto mod = parseModule(tokens.array(), config.fileName); auto mod = parseModule(tokens.array(), config.fileName);
auto printer = new XMLPrinter; auto printer = new XMLPrinter;
printer.output = stdout; printer.output = stdout;
@ -146,9 +154,59 @@ int main(string[] args)
return 0; 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) void printHelp(string programName)
{ {
writefln( stderr.writefln(
` `
Usage: %s options Usage: %s options
@ -156,22 +214,26 @@ options:
--help | -h --help | -h
Prints this help message Prints this help message
--sloc | -l [sourceFile] --sloc | -l [sourceFiles]
Prints the number of logical lines of code in the given 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] --tokenCount | t [sourceFiles]
Prints the number of tokens in the given source file. 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 --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] --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] --syntaxCheck | -s [sourceFile]
Lexes and parses sourceFile, printing the line and column number of any Lexes and parses sourceFile, printing the line and column number of any
syntax errors to stdout. One error or warning is printed per line. 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 --ctags | -c sourceFile
Generates ctags information from the given source code file. Note that Generates ctags information from the given source code file. Note that
@ -179,11 +241,12 @@ options:
of a filename. of a filename.
--ast | --xml sourceFile --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 --recursive | -R | -r
When used with --ctags, dscanner will produce ctags output for all .d When used with --ctags, --tokenCount, or --sloc, dscanner will produce
and .di files contained within the given directory and its ctags output for all .d and .di files contained within the given
sub-directories.`, directories and its sub-directories.`,
programName); 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; ulong count;
foreach (t; tokens) foreach (t; tokens)
@ -43,5 +45,6 @@ void printLineCount(Tokens)(File output, ref Tokens tokens)
if (isLineOfCode(t.type)) if (isLineOfCode(t.type))
++count; ++count;
} }
output.writefln("%d", count); output.writefln("%s:\t%d", fileName, count);
return count;
} }

View File

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