/lsp-intelephense-diagnostic-client-cli

PHP LSP Intelephense Diagnostic Client in Command Line

Primary LanguagePython

LSP Diagnostic Command Line Client

Language server (LSP) spec is used to provide various feature for text editors/IDEs (e.g.: VSCode). Example of LSP features including showing diagnostic (possible error/warning/notice, like unused variable, undefined variable, etc), code highlighting, autocomple, etc.

This project is aimed to display diagnostic from LSP language server from command line instead of a full text editor. Therefore this project can be used to integrate LSP diagnostic inside a CI/CD pipeline (e.g.: fail a build if a diagnostic above certain severity is encountered in the build process)

This project is made for php and intelephense, but should also work for other languages. Getting diagnostic is done by waiting for textDocument/publishDiagnostics notification from LSP language server.

Requirements

  • Python3 with standard library (socket, json, subprocess, os, glob, sys, hashlib)
  • Tested on Linux (should also work on Windows and Mac)

Usage

  • Download client.py
  • Run command: python3 client.py config.json, which config.json is path to a text file containing this project's configuration
  • Example configuration file:
{
    "workingDirectory": "/var/www/html/uks_kristian/web",
    "outputFile": "/var/www/html/uks_kristian/output.json",
    "languageId": "php",
    "includes": [
        "/**/*.php",
    ],
    "excludes": [
        "vendor/**/*.php",
        "class/adodb5/**/*.php",
        "function/phpMailer/*.php"
    ],
    "lspServerCommand": ["docker","exec","intelephense","intelephense","--socket={PORT_NUMBER}"],
    "lspServerStdout": "/tmp/lsp-server-stdout",
    "lspServerStderr": "/tmp/lsp-server-stderr",
    "lspClientPort": 0,
    "lspClientAddress": "0.0.0.0",
    "incrementLineNumber": true,
    "debugPrintTargetFiles": true,
    "debugPrintLspComm": false,
    "debugPrintDiagnostics": false,
    "debugPrintProgress": true,
    "dryRun": false,
    "batchSize": 20,
    "ignoreMessageRegex": ["Undefined type 'Memcached'"],
    "ignoreSeverity": ["Hint"],
    "ignoreCode": ["P1009"],
    "ignoreSource": ["javascript"],
    "additionalInfoCmd": ["bash","-c","cd {WORKING_DIRECTORY} ; git log -s --pretty=format:'%ae//%an//%h' -L {LINE_START},{LINE_END}:{FILE_NAME} | head -n3 | sort | uniq"],
    "cmdBefore": ["pkill","-f","intelephense -i --socket"],
    "cmdAfter": ["bash","-c","cat output.json | jq -r '(map(keys) | add | unique) as $cols | map(. as $row | $cols | map($row[.])) as $rows | $cols, $rows[] | @csv' > output.csv"]
}
  • Configuration file contents:

    • workingDirectory:

      • Directory of the project that will be diagnosed,
      • LSP language server will index this directory and look for symbols, functions, etc.
      • Default value: current directory
    • outputFile:

      • File path to save the diagnostic result
      • file will be created if not exist,
      • file will be truncated if exist
      • Default value: output.json file in workingDirectory, example: /var/www/html/output.json if parameter workingDirectory is /var/www/html
    • languageId:

    • excludes and includes:

      • Array of glob patterns to include and exclude file for searching.
      • Syntax /**/ can be used for recursive glob / also called globstar
      • Default value for includes: all files which have extension of languageId, example: /**/*.php if parameter languageId is php
      • Default value for excludes: empty array
    • lspServerCommand:

      • Command to start LSP server,
      • {PORT_NUMBER} will be substituted with the port number of this client's variable port number
      • Default value: ["intelephense","--socket={PORT_NUMBER}"]
    • lspServerStdout and lspServerStderr:

      • For debugging purpose (e.g.: when LSP server crashes, output of STDOUT/STDERR should be examined).
      • When this option is set to a file path (e.g.: /tmp/stderr and /tmp/stdout), it will log LSP server's STDOUT/STDERR to that file.
      • Example use case is for reporting bug/developing for this client, when this client send incorrect JSON, LSP server may crash and exit instead. The output of the crash may contain clue to fix bug in this client's implementation.
      • Default value: null for both (will discard STDOUT/STDERR output)
    • lspClientPort:

      • Listen to custom port.
      • If not specified (or 0 is passed), this client will listen to random port and pass the port number to {PORT_NUMBER} in lspServerCommand.
      • Default value: 0
    • lspClientAddress:

      • Bind to this address.
      • Default value: 0.0.0.0 (listen to all interfaces)
    • incrementLineNumber:

      • LSP server diagnostic parameter lineStart and lineEnd starts at 0 (first line in a file is line 0, second line is line 1, etc)
      • If set to true, will increment lineStart and lineEnd by 1 (so first in a file is line 1, second line is line 2, etc)
      • Default value: true
    • debugPrintTargetFiles:

      • Toggle print/show target files (from processing option includes and excludes) before running the actual diagnostics
      • Default value: true
    • debugPrintLspComm:

      • Toggle print/show traffic/payload of LSP communications, useful for debugging or learning LSP protocol
      • Default value: false
    • debugPrintDiagnostics:

      • Toggle print/show diagnostics gained from LSP server
      • Default value: false
    • debugPrintProgress:

      • Toggle print/show progress percentage
      • Default value: true
    • dryRun:

      • When set to true, will only enumerate files to process and then exit
      • Default value: false
    • batchSize:

      • Opening too many files at once may crash/overload the LSP server. Batch processing will open X number of files, then read the diagnostics before moving to next batch of files.
      • Lowering this value should decrease resource usage while increasing processing time
      • Increasing this value should increase resource usage while reducing processing time
      • Default value: 20
    • ignoreMessageRegex:

      • Array of strings to ignore diagnostic based on this regex pattern against diagnostic message
      • Default value: [] (empty array)
    • ignoreSeverity:

      • Array of strings to ignore severity, can be filled with: Error,Warning,Information,Hint
      • Default value: [] (empty array)
    • ignoreCode:

      • Array of strings to ignore diagnostic based on diagnostic code
      • Default value: [] (empty array)
    • ignoreSource:

      • Array of strings to ignore diagnostic based on diagnostic source
      • Default value: [] (empty array)
    • additionalInfoCmd:

      • Array of command to be executed to get additional info about this error
      • This command will be run each time a diagnostic error received from LSP server
      • This can be useful to identify who committed this line in version control (e.g.: git), or to run a command when a diagnostic is found (e.g.: failing pipeline or rolling back push/git)
      • Output of this command will be saved to variable addtionalInfo
      • Value can have this parameter to be replaced: {FILE_NAME}, {LINE_START}, {LINE_END}, {CHARACTER_START}, {CHARACTER_END}, {MESSAGE}, {SEVERITY}, {CODE}, {SOURCE}
      • Default value: null (no additional info)
    • cmdBefore:

      • Array of command to be executed (once) before the actual communication with LSP server take place
      • This can be useful to prevent diagnostic to be run too frequently (e.g.: only run diagnostic nightly, if last run output.json was not older than 23 hour then cancel current run), or to prevent diagnostic running when machine load is too high (e.g.: exit if machine load average higher than 50% * number of cpu), or to kill old LSP client process (e.g.: prevent duplicate command to be run)
      • If the return code if this command is not 0, the client will be terminated
      • No parameter replacement will be done
      • Default value: null (no command will be executed)
    • cmdAfter:

      • Array of command to be executed (once) after the actual communication with LSP server take place
      • This can be useful to convert diagnostic data from json format to other format (e.g.: csv)
      • No parameter replacement will be done
      • Default value: null (no command will be executed)
  • Example usage with all default parameters: python3 client.py <(echo '{}')

Intelephense From Docker

To ease installation of Intelephense, which depends on NPM and node.js, author has prepared a docker image (~90MB) to ease the installation on system which have docker installed. On system which already have Intelephense installed, this step is optional.

docker run \
    --net=host \
    -v '/var/www/html:/var/www/html' \
    --name intelephense \
    krstian/intelephense-node22-alpine:latest
  • option --net=host is used to enable LSP language client and LSP language server to communicate in same port number with host machine
  • option -v should mount the project directory (inside workingDirectory configuration), so that Intelephense can index the project directory for symbols/functions/classes declarations
  • dockerhub: https://hub.docker.com/r/krstian/intelephense-node22-alpine

Intelephense and CLI Client From Docker

Another docker image which already have NPM, node.js, intelephense, python3 and this client (~110MB) can be used on pipeline.

  • Example Usage:
    • mount /etc/intelephense-cli-config.json to json configuration file
    • mount php project
  • Example project:
    • directory /home/kristian/www/my-php-project contains a php project
    • directory /home/kristian/www/my-php-project/web is pointed to DocumentRoot of apache web server
    • directory /home/kristian/www/my-php-project/intelephense-cli-config.json is a json file containing LSP client configuration (see above for documentation)
  • Example (minimal) json configuration:
{
    "workingDirectory": "/var/www/html",
    "outputFile": "/var/www/html/diagnostic-output.json",
    "languageId": "php"
}
  • For complete json configuration, see documentation above
  • Example docker usage:
docker run --rm \
  --name lsp-cli \
  -v /home/kristian/www/my-php-project/intelephense-cli-config.json:/etc/intelephense-cli-config.json \
  -v /home/kristian/www/my-php-project/web:/var/www/html \
  krstian/intelephense-lspclientcli-python3-node22-alpine