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:
parent
270cd6d9a1
commit
001118e730
2
build.sh
2
build.sh
|
@ -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
141
main.d
|
@ -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
11
stats.d
|
@ -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;
|
||||||
}
|
}
|
||||||
|
|
|
@ -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);
|
||||||
|
|
Loading…
Reference in New Issue