style: use prettier

pull/391/head
Felix Becker 2018-11-11 18:32:17 +01:00
parent edd8806ab7
commit 0c2985e776
16 changed files with 260 additions and 196 deletions

View File

@ -1,4 +1,3 @@
<!-- <!--
THERE IS A 99% CHANCE YOU ARE FILING THIS ISSUE IN THE WRONG REPOSITORY. THERE IS A 99% CHANCE YOU ARE FILING THIS ISSUE IN THE WRONG REPOSITORY.

4
.prettierignore Normal file
View File

@ -0,0 +1,4 @@
package.json
package-lock.json
vendor/
out/

22
.prettierrc Normal file
View File

@ -0,0 +1,22 @@
{
"tabWidth": 4,
"printWidth": 120,
"proseWrap": "preserve",
"semi": false,
"trailingComma": "es5",
"singleQuote": true,
"overrides": [
{
"files": "{*.js?(on),*.y?(a)ml,.*.js?(on),.*.y?(a)ml,*.md,.prettierrc,.stylelintrc,.babelrc}",
"options": {
"tabWidth": 2
}
},
{
"files": "composer.{json,lock}",
"options": {
"tabWidth": 4
}
}
]
}

9
.vscode/launch.json vendored
View File

@ -7,9 +7,7 @@
"type": "extensionHost", "type": "extensionHost",
"request": "launch", "request": "launch",
"runtimeExecutable": "${execPath}", "runtimeExecutable": "${execPath}",
"args": [ "args": ["--extensionDevelopmentPath=${workspaceRoot}"],
"--extensionDevelopmentPath=${workspaceRoot}"
],
"stopOnEntry": false, "stopOnEntry": false,
"sourceMaps": true, "sourceMaps": true,
"outDir": "${workspaceRoot}/out" "outDir": "${workspaceRoot}/out"
@ -19,10 +17,7 @@
"type": "extensionHost", "type": "extensionHost",
"request": "launch", "request": "launch",
"runtimeExecutable": "${execPath}", "runtimeExecutable": "${execPath}",
"args": [ "args": ["--extensionDevelopmentPath=${workspaceRoot}", "--extensionTestsPath=${workspaceRoot}/out/test"],
"--extensionDevelopmentPath=${workspaceRoot}",
"--extensionTestsPath=${workspaceRoot}/out/test"
],
"stopOnEntry": false, "stopOnEntry": false,
"sourceMaps": true, "sourceMaps": true,
"outDir": "${workspaceRoot}/out/test" "outDir": "${workspaceRoot}/out/test"

8
.vscode/tasks.json vendored
View File

@ -6,16 +6,12 @@
{ {
"type": "npm", "type": "npm",
"script": "build", "script": "build",
"problemMatcher": [ "problemMatcher": ["$tsc"]
"$tsc"
]
}, },
{ {
"type": "npm", "type": "npm",
"script": "watch", "script": "watch",
"problemMatcher": [ "problemMatcher": ["$tsc-watch"]
"$tsc-watch"
]
} }
] ]
} }

View File

@ -1,2 +1 @@
# See [GitHub releases](https://github.com/felixfbecker/vscode-php-intellisense/releases) # See [GitHub releases](https://github.com/felixfbecker/vscode-php-intellisense/releases)

View File

@ -15,50 +15,63 @@ I recommend to disable VS Code's built-in PHP IntelliSense by setting `php.sugge
## Features ## Features
### Completion ### Completion
![Completion search demo](images/completion.gif) ![Completion search demo](images/completion.gif)
### Signature Help ### Signature Help
![Signature help demo](images/signatureHelp.gif) ![Signature help demo](images/signatureHelp.gif)
### Workspace symbol search ### Workspace symbol search
![Workspace symbol search demo](images/workspaceSymbol.gif) ![Workspace symbol search demo](images/workspaceSymbol.gif)
### Find all References ### Find all References
![Find References demo](images/references.png) ![Find References demo](images/references.png)
### Go to Definition ### Go to Definition
![Go To Definition demo](images/definition.gif) ![Go To Definition demo](images/definition.gif)
### Hover ### Hover
![Hover class demo](images/hoverClass.png) ![Hover class demo](images/hoverClass.png)
![Hover parameter demo](images/hoverParam.png) ![Hover parameter demo](images/hoverParam.png)
### Find all symbols ### Find all symbols
![Find all symbols demo](images/documentSymbol.gif) ![Find all symbols demo](images/documentSymbol.gif)
### Column-accurate error reporting ### Column-accurate error reporting
![Error reporting demo](images/publishDiagnostics.png) ![Error reporting demo](images/publishDiagnostics.png)
### Code style linting ### Code style linting
Please use a dedicated extension like [PHP CodeSniffer](https://marketplace.visualstudio.com/items?itemName=ikappas.phpcs). Please use a dedicated extension like [PHP CodeSniffer](https://marketplace.visualstudio.com/items?itemName=ikappas.phpcs).
### Format code ### Format code
Please use a dedicated extension like [PHP CS Fixer](https://marketplace.visualstudio.com/items?itemName=junstyle.php-cs-fixer). Please use a dedicated extension like [PHP CS Fixer](https://marketplace.visualstudio.com/items?itemName=junstyle.php-cs-fixer).
## Todo ## Todo
- Rename
- Signature help - Rename
- Signature help
## Contributing ## Contributing
Clone whole repository and in root directory execute: Clone whole repository and in root directory execute:
```bash ```bash
composer install composer install
npm install npm install
npm run build npm run build
code . code .
``` ```
The last command will open the folder in VS Code. Hit `F5` to launch an Extension Development Host with the extension. The last command will open the folder in VS Code. Hit `F5` to launch an Extension Development Host with the extension.
For working on the language server, the easiest way is to replace the language server installation from composer in `vendor/felixfbecker/language-server` with a symlink to your local clone. For working on the language server, the easiest way is to replace the language server installation from composer in `vendor/felixfbecker/language-server` with a symlink to your local clone.

View File

@ -1,35 +1,33 @@
collectors: collectors:
- type: js-npm
- type: js-npm
path: / path: /
actors: actors:
# pull requests for new major versions # pull requests for new major versions
- type: js-npm - type: js-npm
versions: "Y.0.0" versions: 'Y.0.0'
settings: settings:
# these updates should generally not have an effect on users # these updates should generally not have an effect on users
commit_message_prefix: "chore: " commit_message_prefix: 'chore: '
# receive pull requests to update composer.lock for any update to the language server or dependencies of it # receive pull requests to update composer.lock for any update to the language server or dependencies of it
# after merging, depending on whether it was a patch or feature update, # after merging, depending on whether it was a patch or feature update,
# semantic-release will bundle a new patch or feature version of the extension # semantic-release will bundle a new patch or feature version of the extension
- type: php-composer - type: php-composer
path: / path: /
settings: settings:
collect_transitive: true collect_transitive: true
actors: actors:
- type: php-composer
versions: 'L.L.Y'
settings:
commit_message_prefix: 'fix: '
- type: php-composer - type: php-composer
versions: "L.L.Y" versions: 'L.Y.0'
settings: settings:
commit_message_prefix: "fix: " commit_message_prefix: 'feat: '
- type: php-composer - type: php-composer
versions: "L.Y.0" versions: 'Y.0.0'
settings: settings:
commit_message_prefix: "feat: " commit_message_prefix: 'chore: '
- type: php-composer
versions: "Y.0.0"
settings:
commit_message_prefix: "chore: "

View File

@ -47,7 +47,9 @@
"build": "tsc -p .", "build": "tsc -p .",
"watch": "tsc -w -p .", "watch": "tsc -w -p .",
"postinstall": "node ./node_modules/vscode/bin/install", "postinstall": "node ./node_modules/vscode/bin/install",
"lint": "tslint -c tslint.json \"src/**/*.ts\"", "lint": "npm run prettier && npm run tslint",
"tslint": "tslint -p tsconfig.json -c tslint.json \"src/**/*.ts\"",
"prettier": "prettier '**/{*.{js?(on),ts,yml},.*.js?(on),.*.yml,.prettierrc,*.md}' --write --list-different",
"semantic-release": "semantic-release" "semantic-release": "semantic-release"
}, },
"devDependencies": { "devDependencies": {
@ -59,9 +61,11 @@
"@types/node": "8.10.29", "@types/node": "8.10.29",
"@types/semver": "5.5.0", "@types/semver": "5.5.0",
"husky": "1.1.3", "husky": "1.1.3",
"prettier": "1.15.2",
"semantic-release": "15.10.8", "semantic-release": "15.10.8",
"semantic-release-vsce": "2.1.2", "semantic-release-vsce": "2.1.2",
"tslint": "5.11.0", "tslint": "5.11.0",
"tslint-config-prettier": "^1.15.0",
"typescript": "3.1.6", "typescript": "3.1.6",
"vscode": "1.1.21" "vscode": "1.1.21"
}, },

View File

@ -4,12 +4,10 @@
"rangeStrategy": "pin", "rangeStrategy": "pin",
"semanticCommits": true, "semanticCommits": true,
"npm": { "npm": {
"semanticCommitType": "chore",
"packageRules": [ "packageRules": [
{ {
"packagePatterns": [ "packagePatterns": ["*"],
"*" "semanticCommitType": "chore",
],
"minor": { "minor": {
"groupName": "all non-major dependencies", "groupName": "all non-major dependencies",
"groupSlug": "all-minor-patch" "groupSlug": "all-minor-patch"

View File

@ -1,134 +1,135 @@
import execa from 'execa'
import * as path from 'path'; import { ChildProcess, spawn } from 'mz/child_process'
import { spawn, ChildProcess } from 'mz/child_process'; import * as net from 'net'
import * as vscode from 'vscode'; import * as path from 'path'
import { LanguageClient, LanguageClientOptions, StreamInfo } from 'vscode-languageclient'; import * as semver from 'semver'
import * as semver from 'semver'; import * as url from 'url'
import * as net from 'net'; import * as vscode from 'vscode'
import * as url from 'url'; import { LanguageClient, LanguageClientOptions, StreamInfo } from 'vscode-languageclient'
import execa from 'execa';
export async function activate(context: vscode.ExtensionContext): Promise<void> { export async function activate(context: vscode.ExtensionContext): Promise<void> {
const conf = vscode.workspace.getConfiguration('php')
const conf = vscode.workspace.getConfiguration('php'); const executablePath =
const executablePath = conf.get<string>('executablePath') || conf.get<string>('executablePath') ||
conf.get<string>('validate.executablePath') || conf.get<string>('validate.executablePath') ||
(process.platform === 'win32' ? 'php.exe' : 'php'); (process.platform === 'win32' ? 'php.exe' : 'php')
const memoryLimit = conf.get<string>('memoryLimit') || '4095M'; const memoryLimit = conf.get<string>('memoryLimit') || '4095M'
if (memoryLimit !== '-1' && !/^\d+[KMG]?$/.exec(memoryLimit)) { if (memoryLimit !== '-1' && !/^\d+[KMG]?$/.exec(memoryLimit)) {
const selected = await vscode.window.showErrorMessage( const selected = await vscode.window.showErrorMessage(
'The memory limit you\'d provided is not numeric, nor "-1" nor valid php shorthand notation!', 'The memory limit you\'d provided is not numeric, nor "-1" nor valid php shorthand notation!',
'Open settings' 'Open settings'
); )
if (selected === 'Open settings') { if (selected === 'Open settings') {
await vscode.commands.executeCommand('workbench.action.openGlobalSettings'); await vscode.commands.executeCommand('workbench.action.openGlobalSettings')
} }
return; return
} }
// Check path (if PHP is available and version is ^7.0.0) // Check path (if PHP is available and version is ^7.0.0)
let stdout: string; let stdout: string
try { try {
stdout = await execa.stdout(executablePath, ['--version']); stdout = await execa.stdout(executablePath, ['--version'])
} catch (err) { } catch (err) {
if (err.code === 'ENOENT') { if (err.code === 'ENOENT') {
const selected = await vscode.window.showErrorMessage( const selected = await vscode.window.showErrorMessage(
'PHP executable not found. Install PHP 7 and add it to your PATH or set the php.executablePath setting', 'PHP executable not found. Install PHP 7 and add it to your PATH or set the php.executablePath setting',
'Open settings' 'Open settings'
); )
if (selected === 'Open settings') { if (selected === 'Open settings') {
await vscode.commands.executeCommand('workbench.action.openGlobalSettings'); await vscode.commands.executeCommand('workbench.action.openGlobalSettings')
} }
} else { } else {
vscode.window.showErrorMessage('Error spawning PHP: ' + err.message); vscode.window.showErrorMessage('Error spawning PHP: ' + err.message)
console.error(err); console.error(err)
} }
return; return
} }
// Parse version and discard OS info like 7.0.8--0ubuntu0.16.04.2 // Parse version and discard OS info like 7.0.8--0ubuntu0.16.04.2
const match = stdout.match(/^PHP ([^\s]+)/m); const match = stdout.match(/^PHP ([^\s]+)/m)
if (!match) { if (!match) {
vscode.window.showErrorMessage('Error parsing PHP version. Please check the output of php --version'); vscode.window.showErrorMessage('Error parsing PHP version. Please check the output of php --version')
return; return
} }
let version = match[1].split('-')[0]; let version = match[1].split('-')[0]
// Convert PHP prerelease format like 7.0.0rc1 to 7.0.0-rc1 // Convert PHP prerelease format like 7.0.0rc1 to 7.0.0-rc1
if (!/^\d+.\d+.\d+$/.test(version)) { if (!/^\d+.\d+.\d+$/.test(version)) {
version = version.replace(/(\d+.\d+.\d+)/, '$1-'); version = version.replace(/(\d+.\d+.\d+)/, '$1-')
} }
if (semver.lt(version, '7.0.0')) { if (semver.lt(version, '7.0.0')) {
vscode.window.showErrorMessage('The language server needs at least PHP 7 installed. Version found: ' + version); vscode.window.showErrorMessage('The language server needs at least PHP 7 installed. Version found: ' + version)
return; return
} }
let client: LanguageClient; let client: LanguageClient
const serverOptions = () => new Promise<ChildProcess | StreamInfo>((resolve, reject) => { const serverOptions = () =>
new Promise<ChildProcess | StreamInfo>((resolve, reject) => {
// Use a TCP socket because of problems with blocking STDIO // Use a TCP socket because of problems with blocking STDIO
const server = net.createServer(socket => { const server = net.createServer(socket => {
// 'connection' listener // 'connection' listener
console.log('PHP process connected'); console.log('PHP process connected')
socket.on('end', () => { socket.on('end', () => {
console.log('PHP process disconnected'); console.log('PHP process disconnected')
}); })
server.close(); server.close()
resolve({ reader: socket, writer: socket }); resolve({ reader: socket, writer: socket })
}); })
// Listen on random port // Listen on random port
server.listen(0, '127.0.0.1', () => { server.listen(0, '127.0.0.1', () => {
// The server is implemented in PHP // The server is implemented in PHP
const childProcess = spawn(executablePath, [ const childProcess = spawn(executablePath, [
context.asAbsolutePath(path.join('vendor', 'felixfbecker', 'language-server', 'bin', 'php-language-server.php')), context.asAbsolutePath(
path.join('vendor', 'felixfbecker', 'language-server', 'bin', 'php-language-server.php')
),
'--tcp=127.0.0.1:' + server.address().port, '--tcp=127.0.0.1:' + server.address().port,
'--memory-limit=' + memoryLimit '--memory-limit=' + memoryLimit,
]); ])
childProcess.stderr.on('data', (chunk: Buffer) => { childProcess.stderr.on('data', (chunk: Buffer) => {
const str = chunk.toString(); const str = chunk.toString()
console.log('PHP Language Server:', str); console.log('PHP Language Server:', str)
client.outputChannel.appendLine(str); client.outputChannel.appendLine(str)
}); })
// childProcess.stdout.on('data', (chunk: Buffer) => { // childProcess.stdout.on('data', (chunk: Buffer) => {
// console.log('PHP Language Server:', chunk + ''); // console.log('PHP Language Server:', chunk + '');
// }); // });
childProcess.on('exit', (code, signal) => { childProcess.on('exit', (code, signal) => {
client.outputChannel.appendLine(`Language server exited ` + (signal ? `from signal ${signal}` : `with exit code ${code}`)); client.outputChannel.appendLine(
`Language server exited ` + (signal ? `from signal ${signal}` : `with exit code ${code}`)
)
if (code !== 0) { if (code !== 0) {
client.outputChannel.show(); client.outputChannel.show()
} }
}); })
return childProcess; return childProcess
}); })
}); })
// Options to control the language client // Options to control the language client
const clientOptions: LanguageClientOptions = { const clientOptions: LanguageClientOptions = {
// Register the server for php documents // Register the server for php documents
documentSelector: [ documentSelector: [{ scheme: 'file', language: 'php' }, { scheme: 'untitled', language: 'php' }],
{ scheme: 'file', language: 'php' },
{ scheme: 'untitled', language: 'php' }
],
uriConverters: { uriConverters: {
// VS Code by default %-encodes even the colon after the drive letter // VS Code by default %-encodes even the colon after the drive letter
// NodeJS handles it much better // NodeJS handles it much better
code2Protocol: uri => url.format(url.parse(uri.toString(true))), code2Protocol: uri => url.format(url.parse(uri.toString(true))),
protocol2Code: str => vscode.Uri.parse(str) protocol2Code: str => vscode.Uri.parse(str),
}, },
synchronize: { synchronize: {
// Synchronize the setting section 'php' to the server // Synchronize the setting section 'php' to the server
configurationSection: 'php', configurationSection: 'php',
// Notify the server about changes to PHP files in the workspace // Notify the server about changes to PHP files in the workspace
fileEvents: vscode.workspace.createFileSystemWatcher('**/*.php') fileEvents: vscode.workspace.createFileSystemWatcher('**/*.php'),
},
} }
};
// Create the language client and start the client. // Create the language client and start the client.
client = new LanguageClient('PHP Language Server', serverOptions, clientOptions); client = new LanguageClient('PHP Language Server', serverOptions, clientOptions)
const disposable = client.start(); const disposable = client.start()
// Push the disposable to the context's subscriptions so that the // Push the disposable to the context's subscriptions so that the
// client can be deactivated on extension deactivation // client can be deactivated on extension deactivation
context.subscriptions.push(disposable); context.subscriptions.push(disposable)
} }

View File

@ -4,7 +4,7 @@
// //
// The module 'assert' provides assertion methods from node // The module 'assert' provides assertion methods from node
import * as assert from 'assert'; import * as assert from 'assert'
// You can import and use all API from the 'vscode' module // You can import and use all API from the 'vscode' module
// as well as import your extension to test it // as well as import your extension to test it
@ -13,10 +13,9 @@ import * as assert from 'assert';
// Defines a Mocha test suite to group tests of similar kind together // Defines a Mocha test suite to group tests of similar kind together
suite('Extension Tests', () => { suite('Extension Tests', () => {
// Defines a Mocha unit test // Defines a Mocha unit test
test('Something 1', () => { test('Something 1', () => {
assert.equal(-1, [1, 2, 3].indexOf(5)); assert.equal(-1, [1, 2, 3].indexOf(5))
assert.equal(-1, [1, 2, 3].indexOf(0)); assert.equal(-1, [1, 2, 3].indexOf(0))
}); })
}); })

View File

@ -10,13 +10,13 @@
// to report the results back to the caller. When the tests are finished, return // to report the results back to the caller. When the tests are finished, return
// a possible error to the callback or null if none. // a possible error to the callback or null if none.
const testRunner = require('vscode/lib/testrunner'); const testRunner = require('vscode/lib/testrunner')
// You can directly control Mocha options by uncommenting the following lines // You can directly control Mocha options by uncommenting the following lines
// See https://github.com/mochajs/mocha/wiki/Using-mocha-programmatically#set-options for more info // See https://github.com/mochajs/mocha/wiki/Using-mocha-programmatically#set-options for more info
testRunner.configure({ testRunner.configure({
ui: 'tdd', // the TDD UI is being used in extension.test.ts (suite, test, etc.) ui: 'tdd', // the TDD UI is being used in extension.test.ts (suite, test, etc.)
useColors: true // colored output from test results useColors: true, // colored output from test results
}); })
module.exports = testRunner; module.exports = testRunner

View File

@ -5,9 +5,7 @@
"moduleResolution": "node", "moduleResolution": "node",
"outDir": "out", "outDir": "out",
"esModuleInterop": true, "esModuleInterop": true,
"lib": [ "lib": ["es6"],
"es6"
],
"noImplicitAny": true, "noImplicitAny": true,
"sourceMap": true, "sourceMap": true,
"rootDir": "src", "rootDir": "src",
@ -18,9 +16,5 @@
"allowUnreachableCode": false, "allowUnreachableCode": false,
"allowUnusedLabels": false "allowUnusedLabels": false
}, },
"exclude": [ "exclude": ["node_modules", "out", "vendor"]
"node_modules",
"out",
"vendor"
]
} }

View File

@ -1,43 +1,85 @@
{ {
"extends": ["tslint:recommended", "tslint-config-prettier"],
"rules": { "rules": {
"class-name": true, "adjacent-overload-signatures": true,
"comment-format": [true, "check-space"], "array-type": [true, "array"],
"indent": [true, "spaces"], "arrow-return-shorthand": [true, "multiline"],
"no-eval": true, "await-promise": [true, "Thenable"],
"no-internal-module": true, "ban": [
"no-trailing-whitespace": true,
"no-var-keyword": true,
"one-line": [true, "check-catch", "check-finally", "check-else", "check-open-brace", "check-whitespace"],
"quotemark": [true, "single"],
"semicolon": [true, "always"],
"triple-equals": [true, "allow-null-check"],
"typedef-whitespace": [
true, true,
{ {
"call-signature": "nospace", "name": ["*", "forEach"]
"index-signature": "nospace",
"parameter": "nospace",
"property-declaration": "nospace",
"variable-declaration": "nospace"
}, },
["describe", "only"],
["it", "only"]
],
"callable-types": true,
"class-name": true,
"comment-format": [true, "check-space"],
"curly": true,
"deprecation": {
"severity": "warning"
},
"interface-name": [false],
"interface-over-type-literal": true,
"jsdoc-format": true,
"max-classes-per-file": false,
"member-access": [true, "check-accessor"],
"member-ordering": [false],
"no-angle-bracket-type-assertion": true,
"no-arg": true,
"no-bitwise": false,
"no-boolean-literal-compare": true,
"no-conditional-assignment": true,
"no-console": [false],
"no-construct": true,
"no-debugger": true,
"no-duplicate-super": true,
"no-duplicate-variable": true,
"no-empty": true,
"no-empty-interface": false,
"no-eval": true,
"no-floating-promises": [true],
"no-for-in-array": true,
"no-inferrable-types": [true],
"no-inferred-empty-object-type": true,
"no-internal-module": true,
"no-invalid-template-strings": true,
"no-magic-numbers": false,
"no-misused-new": true,
"no-namespace": [false, "allow-declarations"],
"no-reference-import": true,
"no-shadowed-variable": false,
"no-sparse-arrays": true,
"no-string-literal": true,
"no-string-throw": true,
"no-unbound-method": true,
"no-unnecessary-callback-wrapper": false,
"no-unnecessary-qualifier": true,
"no-unnecessary-type-assertion": false,
"no-unsafe-any": false,
"no-unsafe-finally": true,
"no-unused-expression": true,
"no-use-before-declare": true,
"no-var-keyword": true,
"no-var-requires": false,
"no-void-expression": false,
"object-literal-shorthand": true,
"object-literal-sort-keys": false,
"one-variable-per-declaration": [true, "ignore-for-loop"],
"only-arrow-functions": [true, "allow-declarations", "allow-named-functions"],
"prefer-const": [
true,
{ {
"call-signature": "onespace", "destructuring": "all"
"index-signature": "onespace",
"parameter": "onespace",
"property-declaration": "onespace",
"variable-declaration": "onespace"
} }
], ],
"variable-name": [true, "ban-keywords"], "prefer-for-of": true,
"whitespace": [true, "check-branch", "check-decl", "check-operator", "check-separator", "check-type"], "prefer-template": [false, "allow-single-concat"],
"jsdoc-format": true, "return-undefined": true,
"no-consecutive-blank-lines": true, "triple-equals": [true],
"one-variable-per-declaration": [true, "ignore-for-loop"], "typedef": [true, "call-signature"],
"curly": true, "unified-signatures": true,
"no-empty": true, "variable-name": [true, "ban-keywords"]
"no-unused-expression": true,
"eofline": true,
"trailing-comma": [true, {"singleline": "never", "multiline": "never"}],
"align": [true, "parameters", "statements"]
} }
} }