/*--------------------------------------------------------------------------------------------- * 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 }; 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(Setting.Enable, true); let inspect = section.inspect(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(Setting.Run, RunTrigger.strings.onSave)); } if (this.executableIsUserDefined !== true && this.workspaceStore.get(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(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(Setting.CheckedExecutablePath, undefined); if (!checkedExecutablePath || checkedExecutablePath !== this.executable) { vscode.window.showInformationMessage( 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 { return new Promise((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 { 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); } } }