315 lines
10 KiB
TypeScript
315 lines
10 KiB
TypeScript
|
/*---------------------------------------------------------------------------------------------
|
||
|
* Copyright (c) Microsoft Corporation. All rights reserved.
|
||
|
* Licensed under the MIT License. See License.txt in the project root for license information.
|
||
|
*--------------------------------------------------------------------------------------------*/
|
||
|
|
||
|
import * as cp from 'child_process';
|
||
|
import { StringDecoder } from 'string_decoder';
|
||
|
|
||
|
import * as vscode from 'vscode';
|
||
|
|
||
|
import { ThrottledDelayer } from './utils/async';
|
||
|
|
||
|
import * as nls from 'vscode-nls';
|
||
|
let localize = nls.loadMessageBundle();
|
||
|
|
||
|
const enum Setting {
|
||
|
Run = 'php.validate.run',
|
||
|
CheckedExecutablePath = 'php.validate.checkedExecutablePath',
|
||
|
Enable = 'php.validate.enable',
|
||
|
ExecutablePath = 'php.validate.executablePath',
|
||
|
}
|
||
|
|
||
|
export class LineDecoder {
|
||
|
private stringDecoder: StringDecoder;
|
||
|
private remaining: string | null;
|
||
|
|
||
|
constructor(encoding: string = 'utf8') {
|
||
|
this.stringDecoder = new StringDecoder(encoding);
|
||
|
this.remaining = null;
|
||
|
}
|
||
|
|
||
|
public write(buffer: Buffer): string[] {
|
||
|
let result: string[] = [];
|
||
|
let value = this.remaining
|
||
|
? this.remaining + this.stringDecoder.write(buffer)
|
||
|
: this.stringDecoder.write(buffer);
|
||
|
|
||
|
if (value.length < 1) {
|
||
|
return result;
|
||
|
}
|
||
|
let start = 0;
|
||
|
let ch: number;
|
||
|
while (start < value.length && ((ch = value.charCodeAt(start)) === 13 || ch === 10)) {
|
||
|
start++;
|
||
|
}
|
||
|
let idx = start;
|
||
|
while (idx < value.length) {
|
||
|
ch = value.charCodeAt(idx);
|
||
|
if (ch === 13 || ch === 10) {
|
||
|
result.push(value.substring(start, idx));
|
||
|
idx++;
|
||
|
while (idx < value.length && ((ch = value.charCodeAt(idx)) === 13 || ch === 10)) {
|
||
|
idx++;
|
||
|
}
|
||
|
start = idx;
|
||
|
} else {
|
||
|
idx++;
|
||
|
}
|
||
|
}
|
||
|
this.remaining = start < value.length ? value.substr(start) : null;
|
||
|
return result;
|
||
|
}
|
||
|
|
||
|
public end(): string | null {
|
||
|
return this.remaining;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
enum RunTrigger {
|
||
|
onSave,
|
||
|
onType
|
||
|
}
|
||
|
|
||
|
namespace RunTrigger {
|
||
|
export let strings = {
|
||
|
onSave: 'onSave',
|
||
|
onType: 'onType'
|
||
|
};
|
||
|
export let from = function (value: string): RunTrigger {
|
||
|
if (value === 'onType') {
|
||
|
return RunTrigger.onType;
|
||
|
} else {
|
||
|
return RunTrigger.onSave;
|
||
|
}
|
||
|
};
|
||
|
}
|
||
|
|
||
|
export default class PHPValidationProvider {
|
||
|
|
||
|
private static MatchExpression: RegExp = /(?:(?:Parse|Fatal) error): (.*)(?: in )(.*?)(?: on line )(\d+)/;
|
||
|
private static BufferArgs: string[] = ['-l', '-n', '-d', 'display_errors=On', '-d', 'log_errors=Off'];
|
||
|
private static FileArgs: string[] = ['-l', '-n', '-d', 'display_errors=On', '-d', 'log_errors=Off', '-f'];
|
||
|
|
||
|
private validationEnabled: boolean;
|
||
|
private executableIsUserDefined: boolean | undefined;
|
||
|
private executable: string | undefined;
|
||
|
private trigger: RunTrigger;
|
||
|
private pauseValidation: boolean;
|
||
|
|
||
|
private documentListener: vscode.Disposable | null = null;
|
||
|
private diagnosticCollection?: vscode.DiagnosticCollection;
|
||
|
private delayers?: { [key: string]: ThrottledDelayer<void> };
|
||
|
|
||
|
constructor(private workspaceStore: vscode.Memento) {
|
||
|
this.executable = undefined;
|
||
|
this.validationEnabled = true;
|
||
|
this.trigger = RunTrigger.onSave;
|
||
|
this.pauseValidation = false;
|
||
|
}
|
||
|
|
||
|
public activate(subscriptions: vscode.Disposable[]) {
|
||
|
this.diagnosticCollection = vscode.languages.createDiagnosticCollection();
|
||
|
subscriptions.push(this);
|
||
|
vscode.workspace.onDidChangeConfiguration(this.loadConfiguration, this, subscriptions);
|
||
|
this.loadConfiguration();
|
||
|
|
||
|
vscode.workspace.onDidOpenTextDocument(this.triggerValidate, this, subscriptions);
|
||
|
vscode.workspace.onDidCloseTextDocument((textDocument) => {
|
||
|
this.diagnosticCollection!.delete(textDocument.uri);
|
||
|
delete this.delayers![textDocument.uri.toString()];
|
||
|
}, null, subscriptions);
|
||
|
subscriptions.push(vscode.commands.registerCommand('php.untrustValidationExecutable', this.untrustValidationExecutable, this));
|
||
|
}
|
||
|
|
||
|
public dispose(): void {
|
||
|
if (this.diagnosticCollection) {
|
||
|
this.diagnosticCollection.clear();
|
||
|
this.diagnosticCollection.dispose();
|
||
|
}
|
||
|
if (this.documentListener) {
|
||
|
this.documentListener.dispose();
|
||
|
this.documentListener = null;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private loadConfiguration(): void {
|
||
|
let section = vscode.workspace.getConfiguration();
|
||
|
let oldExecutable = this.executable;
|
||
|
if (section) {
|
||
|
this.validationEnabled = section.get<boolean>(Setting.Enable, true);
|
||
|
let inspect = section.inspect<string>(Setting.ExecutablePath);
|
||
|
if (inspect && inspect.workspaceValue) {
|
||
|
this.executable = inspect.workspaceValue;
|
||
|
this.executableIsUserDefined = false;
|
||
|
} else if (inspect && inspect.globalValue) {
|
||
|
this.executable = inspect.globalValue;
|
||
|
this.executableIsUserDefined = true;
|
||
|
} else {
|
||
|
this.executable = undefined;
|
||
|
this.executableIsUserDefined = undefined;
|
||
|
}
|
||
|
this.trigger = RunTrigger.from(section.get<string>(Setting.Run, RunTrigger.strings.onSave));
|
||
|
}
|
||
|
if (this.executableIsUserDefined !== true && this.workspaceStore.get<string | undefined>(Setting.CheckedExecutablePath, undefined) !== undefined) {
|
||
|
vscode.commands.executeCommand('setContext', 'php.untrustValidationExecutableContext', true);
|
||
|
}
|
||
|
this.delayers = Object.create(null);
|
||
|
if (this.pauseValidation) {
|
||
|
this.pauseValidation = oldExecutable === this.executable;
|
||
|
}
|
||
|
if (this.documentListener) {
|
||
|
this.documentListener.dispose();
|
||
|
this.documentListener = null;
|
||
|
}
|
||
|
this.diagnosticCollection!.clear();
|
||
|
if (this.validationEnabled) {
|
||
|
if (this.trigger === RunTrigger.onType) {
|
||
|
this.documentListener = vscode.workspace.onDidChangeTextDocument((e) => {
|
||
|
this.triggerValidate(e.document);
|
||
|
});
|
||
|
} else {
|
||
|
this.documentListener = vscode.workspace.onDidSaveTextDocument(this.triggerValidate, this);
|
||
|
}
|
||
|
// Configuration has changed. Reevaluate all documents.
|
||
|
vscode.workspace.textDocuments.forEach(this.triggerValidate, this);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private untrustValidationExecutable() {
|
||
|
this.workspaceStore.update(Setting.CheckedExecutablePath, undefined);
|
||
|
vscode.commands.executeCommand('setContext', 'php.untrustValidationExecutableContext', false);
|
||
|
}
|
||
|
|
||
|
private triggerValidate(textDocument: vscode.TextDocument): void {
|
||
|
if (textDocument.languageId !== 'php' || this.pauseValidation || !this.validationEnabled) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
interface MessageItem extends vscode.MessageItem {
|
||
|
id: string;
|
||
|
}
|
||
|
|
||
|
let trigger = () => {
|
||
|
let key = textDocument.uri.toString();
|
||
|
let delayer = this.delayers![key];
|
||
|
if (!delayer) {
|
||
|
delayer = new ThrottledDelayer<void>(this.trigger === RunTrigger.onType ? 250 : 0);
|
||
|
this.delayers![key] = delayer;
|
||
|
}
|
||
|
delayer.trigger(() => this.doValidate(textDocument));
|
||
|
};
|
||
|
|
||
|
if (this.executableIsUserDefined !== undefined && !this.executableIsUserDefined) {
|
||
|
let checkedExecutablePath = this.workspaceStore.get<string | undefined>(Setting.CheckedExecutablePath, undefined);
|
||
|
if (!checkedExecutablePath || checkedExecutablePath !== this.executable) {
|
||
|
vscode.window.showInformationMessage<MessageItem>(
|
||
|
localize('php.useExecutablePath', 'Do you allow {0} (defined as a workspace setting) to be executed to lint PHP files?', this.executable),
|
||
|
{
|
||
|
title: localize('php.yes', 'Allow'),
|
||
|
id: 'yes'
|
||
|
},
|
||
|
{
|
||
|
title: localize('php.no', 'Disallow'),
|
||
|
isCloseAffordance: true,
|
||
|
id: 'no'
|
||
|
}
|
||
|
).then(selected => {
|
||
|
if (!selected || selected.id === 'no') {
|
||
|
this.pauseValidation = true;
|
||
|
} else if (selected.id === 'yes') {
|
||
|
this.workspaceStore.update(Setting.CheckedExecutablePath, this.executable);
|
||
|
vscode.commands.executeCommand('setContext', 'php.untrustValidationExecutableContext', true);
|
||
|
trigger();
|
||
|
}
|
||
|
});
|
||
|
return;
|
||
|
}
|
||
|
}
|
||
|
trigger();
|
||
|
}
|
||
|
|
||
|
private doValidate(textDocument: vscode.TextDocument): Promise<void> {
|
||
|
return new Promise<void>((resolve) => {
|
||
|
let executable = this.executable || 'php';
|
||
|
let decoder = new LineDecoder();
|
||
|
let diagnostics: vscode.Diagnostic[] = [];
|
||
|
let processLine = (line: string) => {
|
||
|
let matches = line.match(PHPValidationProvider.MatchExpression);
|
||
|
if (matches) {
|
||
|
let message = matches[1];
|
||
|
let line = parseInt(matches[3]) - 1;
|
||
|
let diagnostic: vscode.Diagnostic = new vscode.Diagnostic(
|
||
|
new vscode.Range(line, 0, line, Number.MAX_VALUE),
|
||
|
message
|
||
|
);
|
||
|
diagnostics.push(diagnostic);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
let options = (vscode.workspace.workspaceFolders && vscode.workspace.workspaceFolders[0]) ? { cwd: vscode.workspace.workspaceFolders[0].uri.fsPath } : undefined;
|
||
|
let args: string[];
|
||
|
if (this.trigger === RunTrigger.onSave) {
|
||
|
args = PHPValidationProvider.FileArgs.slice(0);
|
||
|
args.push(textDocument.fileName);
|
||
|
} else {
|
||
|
args = PHPValidationProvider.BufferArgs;
|
||
|
}
|
||
|
try {
|
||
|
let childProcess = cp.spawn(executable, args, options);
|
||
|
childProcess.on('error', (error: Error) => {
|
||
|
if (this.pauseValidation) {
|
||
|
resolve();
|
||
|
return;
|
||
|
}
|
||
|
this.showError(error, executable);
|
||
|
this.pauseValidation = true;
|
||
|
resolve();
|
||
|
});
|
||
|
if (childProcess.pid) {
|
||
|
if (this.trigger === RunTrigger.onType) {
|
||
|
childProcess.stdin.write(textDocument.getText());
|
||
|
childProcess.stdin.end();
|
||
|
}
|
||
|
childProcess.stdout.on('data', (data: Buffer) => {
|
||
|
decoder.write(data).forEach(processLine);
|
||
|
});
|
||
|
childProcess.stdout.on('end', () => {
|
||
|
let line = decoder.end();
|
||
|
if (line) {
|
||
|
processLine(line);
|
||
|
}
|
||
|
this.diagnosticCollection!.set(textDocument.uri, diagnostics);
|
||
|
resolve();
|
||
|
});
|
||
|
} else {
|
||
|
resolve();
|
||
|
}
|
||
|
} catch (error) {
|
||
|
this.showError(error, executable);
|
||
|
}
|
||
|
});
|
||
|
}
|
||
|
|
||
|
private async showError(error: any, executable: string): Promise<void> {
|
||
|
let message: string | null = null;
|
||
|
if (error.code === 'ENOENT') {
|
||
|
if (this.executable) {
|
||
|
message = localize('wrongExecutable', 'Cannot validate since {0} is not a valid php executable. Use the setting \'php.validate.executablePath\' to configure the PHP executable.', executable);
|
||
|
} else {
|
||
|
message = localize('noExecutable', 'Cannot validate since no PHP executable is set. Use the setting \'php.validate.executablePath\' to configure the PHP executable.');
|
||
|
}
|
||
|
} else {
|
||
|
message = error.message ? error.message : localize('unknownReason', 'Failed to run php using path: {0}. Reason is unknown.', executable);
|
||
|
}
|
||
|
if (!message) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const openSettings = localize('goToSetting', 'Open Settings');
|
||
|
if (await vscode.window.showInformationMessage(message, openSettings) === openSettings) {
|
||
|
vscode.commands.executeCommand('workbench.action.openSettings', Setting.ExecutablePath);
|
||
|
}
|
||
|
}
|
||
|
}
|