DCD/src/dcd/client/client.d

424 lines
11 KiB
D

/**
* This file is part of DCD, a development tool for the D programming language.
* Copyright (C) 2014 Brian Schott
*
* This program is free software: you can redistribute it and/or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation, either version 3 of the License, or
* (at your option) any later version.
*
* This program is distributed in the hope that it will be useful,
* but WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License
* along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
module dcd.client.client;
import std.socket;
import std.stdio;
import std.getopt;
import std.array;
import std.process;
import std.algorithm;
import std.path;
import std.file;
import std.conv;
import std.string;
import std.experimental.logger;
import dcd.common.messages;
import dcd.common.dcd_version;
import dcd.common.socket;
int main(string[] args)
{
sharedLog.fatalHandler = () {};
size_t cursorPos = size_t.max;
string[] importPaths;
ushort port;
bool help;
bool shutdown;
bool clearCache;
bool symbolLocation;
bool doc;
bool query;
bool printVersion;
bool listImports;
bool getIdentifier;
bool localUse;
bool fullOutput;
string search;
version(Windows)
{
bool useTCP = true;
string socketFile;
}
else
{
bool useTCP = false;
string socketFile = generateSocketName();
}
try
{
getopt(args, "cursorPos|c", &cursorPos, "I", &importPaths,
"port|p", &port, "help|h", &help, "shutdown", &shutdown,
"clearCache", &clearCache, "symbolLocation|l", &symbolLocation,
"doc|d", &doc, "query|status|q", &query, "search|s", &search,
"version", &printVersion, "listImports", &listImports,
"tcp", &useTCP, "socketFile", &socketFile,
"getIdentifier", &getIdentifier,
"localUse|u", &localUse, "extended|x", &fullOutput);
}
catch (ConvException e)
{
fatal(e.msg);
printHelp(args[0]);
return 1;
}
AutocompleteRequest request;
if (help)
{
printHelp(args[0]);
return 0;
}
if (printVersion)
{
version (Windows)
writeln(DCD_VERSION);
else version(built_with_dub)
writeln(DCD_VERSION);
else
write(DCD_VERSION, " ", GIT_HASH);
return 0;
}
version (Windows) if (socketFile !is null)
{
fatal("UNIX domain sockets not supported on Windows");
return 1;
}
// If the user specified a port number, assume that they wanted a TCP
// connection. Otherwise set the port number to the default and let the
// useTCP flag deterimen what to do later.
if (port != 0)
useTCP = true;
else
port = DEFAULT_PORT_NUMBER;
if (useTCP)
socketFile = null;
if (query)
{
if (serverIsRunning(useTCP, socketFile, port))
{
writeln("Server is running");
return 0;
}
else
{
writeln("Server is not running");
return 1;
}
}
else if (shutdown || clearCache)
{
if (shutdown)
request.kind = RequestKind.shutdown;
else if (clearCache)
request.kind = RequestKind.clearCache;
Socket socket = createSocket(socketFile, port);
scope (exit) { socket.shutdown(SocketShutdown.BOTH); socket.close(); }
return sendRequest(socket, request) ? 0 : 1;
}
else if (importPaths.length > 0)
{
request.kind |= RequestKind.addImport;
request.importPaths = importPaths.map!(a => absolutePath(a)).array;
if (cursorPos == size_t.max)
{
Socket socket = createSocket(socketFile, port);
scope (exit) { socket.shutdown(SocketShutdown.BOTH); socket.close(); }
if (!sendRequest(socket, request))
return 1;
return 0;
}
}
else if (listImports)
{
request.kind |= RequestKind.listImports;
Socket socket = createSocket(socketFile, port);
scope (exit) { socket.shutdown(SocketShutdown.BOTH); socket.close(); }
sendRequest(socket, request);
AutocompleteResponse response = getResponse(socket);
printImportList(response);
return 0;
}
else if (search == null && cursorPos == size_t.max)
{
// cursor position is a required argument
printHelp(args[0]);
return 1;
}
// Read in the source
immutable bool usingStdin = args.length <= 1;
string fileName = usingStdin ? "stdin" : args[1];
if (!usingStdin && !exists(args[1]))
{
stderr.writefln("%s does not exist", args[1]);
return 1;
}
ubyte[] sourceCode;
if (usingStdin)
{
ubyte[4096] buf;
while (true)
{
auto b = stdin.rawRead(buf);
if (b.length == 0)
break;
sourceCode ~= b;
}
}
else
{
if (!exists(args[1]))
{
stderr.writeln("Could not find ", args[1]);
return 1;
}
File f = File(args[1]);
sourceCode = uninitializedArray!(ubyte[])(to!size_t(f.size));
f.rawRead(sourceCode);
}
request.fileName = fileName;
request.importPaths = importPaths;
request.sourceCode = sourceCode;
request.cursorPosition = cursorPos;
request.searchName = search;
if (symbolLocation | getIdentifier)
request.kind |= RequestKind.symbolLocation;
else if (doc)
request.kind |= RequestKind.doc;
else if (search)
request.kind |= RequestKind.search;
else if(localUse)
request.kind |= RequestKind.localUse;
else
request.kind |= RequestKind.autocomplete;
// Send message to server
Socket socket = createSocket(socketFile, port);
scope (exit) { socket.shutdown(SocketShutdown.BOTH); socket.close(); }
if (!sendRequest(socket, request))
return 1;
AutocompleteResponse response = getResponse(socket);
if (symbolLocation)
printLocationResponse(response);
else if (getIdentifier)
printIdentifierResponse(response);
else if (doc)
printDocResponse(response);
else if (search !is null)
printSearchResponse(response);
else if (localUse)
printLocalUse(response);
else
printCompletionResponse(response, fullOutput);
return 0;
}
private:
void printHelp(string programName)
{
writefln(
`
Usage: %1$s [Options] [FILENAME]
A file name is optional. If it is given, autocomplete information will be
given for the file specified. If it is missing, input will be read from
stdin instead.
Source code is assumed to be UTF-8 encoded and must not exceed 4 megabytes.
Options:
--help | -h
Displays this help message
--cursorPos | -c position
Provides auto-completion at the given cursor position. The cursor
position is measured in bytes from the beginning of the source code.
--clearCache
Instructs the server to clear out its autocompletion cache.
--shutdown
Instructs the server to shut down.
--symbolLocation | -l
Get the file name and position that the symbol at the cursor location
was defined.
--doc | -d
Gets documentation comments associated with the symbol at the cursor
location.
--search | -s symbolName
Searches for symbolName in both stdin / the given file name as well as
others files cached by the server.
--localUse | -u
Searches for all the uses of the symbol at the cursor location
in the given filename (or stdin).
--extended | -x
Includes more information with a slightly different format for
calltips when autocompleting.
--query | -q | --status
Query the server statis. Returns 0 if the server is running. Returns
1 if the server could not be contacted.
-I PATH
Instructs the server to add PATH to its list of paths searched for
imported modules.
--version
Prints the version number and then exits.
--port PORTNUMBER | -p PORTNUMBER
Uses PORTNUMBER to communicate with the server instead of the default
port 9166. Only used on Windows or when the --tcp option is set.
--tcp
Send requests on a TCP socket instead of a UNIX domain socket. This
switch has no effect on Windows.
--socketFile FILENAME
Use the given FILENAME as the path to the UNIX domain socket. Using
this switch is an error on Windows.`, programName);
}
Socket createSocket(string socketFile, ushort port)
{
import core.time : dur;
Socket socket;
if (socketFile is null)
{
socket = new TcpSocket(AddressFamily.INET);
socket.connect(new InternetAddress("localhost", port));
}
else
{
version(Windows)
{
// should never be called with non-null socketFile on Windows
assert(false);
}
else
{
socket = new Socket(AddressFamily.UNIX, SocketType.STREAM);
socket.connect(new UnixAddress(socketFile));
}
}
socket.setOption(SocketOptionLevel.SOCKET, SocketOption.RCVTIMEO, dur!"seconds"(5));
socket.blocking = true;
return socket;
}
void printDocResponse(ref const AutocompleteResponse response)
{
import std.algorithm : each;
response.completions.each!(a => a.documentation.escapeConsoleOutputString(true).writeln);
}
void printIdentifierResponse(ref const AutocompleteResponse response)
{
if (response.completions.length == 0)
return;
writeln(makeTabSeparated(response.completions[0].identifier, response.symbolIdentifier.to!string));
}
void printLocationResponse(ref const AutocompleteResponse response)
{
if (response.symbolFilePath is null)
writeln("Not found");
else
writeln(makeTabSeparated(response.symbolFilePath, response.symbolLocation.to!string));
}
void printCompletionResponse(ref const AutocompleteResponse response, bool extended)
{
if (response.completions.length > 0)
{
writeln(response.completionType);
auto app = appender!(string[])();
if (response.completionType == CompletionType.identifiers || extended)
{
foreach (ref completion; response.completions)
{
if (extended)
app.put(makeTabSeparated(
completion.identifier,
completion.kind == char.init ? "" : "" ~ completion.kind,
completion.definition,
completion.symbolFilePath.length ? completion.symbolFilePath ~ " " ~ completion.symbolLocation.to!string : "",
completion.documentation
));
else
app.put(makeTabSeparated(completion.identifier, "" ~ completion.kind));
}
}
else
{
foreach (completion; response.completions)
app.put(completion.definition);
}
// Deduplicate overloaded methods
foreach (line; app.data.sort().uniq)
writeln(line);
}
}
void printSearchResponse(const AutocompleteResponse response)
{
foreach(ref completion; response.completions)
writeln(makeTabSeparated(completion.identifier, "" ~ completion.kind, completion.symbolLocation.to!string));
}
void printLocalUse(const AutocompleteResponse response)
{
if (response.symbolFilePath.length)
{
writeln(makeTabSeparated(response.symbolFilePath, response.symbolLocation.to!string));
foreach(loc; response.completions)
writeln(loc.symbolLocation);
}
else write("00000");
}
void printImportList(const AutocompleteResponse response)
{
import std.algorithm.iteration : each;
response.importPaths.each!(a => writeln(a));
}