merge: pull request #5292 from elken/feature/php-improvements
Improve `:lang php`
This commit is contained in:
commit
06dc65f906
3 changed files with 158 additions and 28 deletions
|
@ -15,8 +15,13 @@
|
||||||
- [[#opensuse][openSUSE]]
|
- [[#opensuse][openSUSE]]
|
||||||
- [[#dependencies][Dependencies]]
|
- [[#dependencies][Dependencies]]
|
||||||
- [[#features][Features]]
|
- [[#features][Features]]
|
||||||
|
- [[#lsp-support][LSP Support]]
|
||||||
|
- [[#phpunit][PHPUnit]]
|
||||||
|
- [[#composer][Composer]]
|
||||||
- [[#configuration][Configuration]]
|
- [[#configuration][Configuration]]
|
||||||
|
- [[#docker-compose][Docker Compose]]
|
||||||
- [[#troubleshooting][Troubleshooting]]
|
- [[#troubleshooting][Troubleshooting]]
|
||||||
|
- [[#im-missing-functionality-on-lsp-mode]["I'm missing functionality on lsp-mode"]]
|
||||||
|
|
||||||
* Description
|
* Description
|
||||||
This module adds support for PHP 5.3+ (including PHP7).
|
This module adds support for PHP 5.3+ (including PHP7).
|
||||||
|
@ -27,6 +32,7 @@ This module adds support for PHP 5.3+ (including PHP7).
|
||||||
+ Code refactoring commands (~php-refactor-mode~)
|
+ Code refactoring commands (~php-refactor-mode~)
|
||||||
+ Unit-test commands (~phpunit~)
|
+ Unit-test commands (~phpunit~)
|
||||||
+ Support for ~laravel~ and ~composer~ projects (with project-specific snippets)
|
+ Support for ~laravel~ and ~composer~ projects (with project-specific snippets)
|
||||||
|
+ Shortcuts for composer commands
|
||||||
+ [[../../editor/file-templates/templates/php-mode][File templates]]
|
+ [[../../editor/file-templates/templates/php-mode][File templates]]
|
||||||
+ [[https://github.com/hlissner/doom-snippets/tree/master/php-mode][Snippets]]
|
+ [[https://github.com/hlissner/doom-snippets/tree/master/php-mode][Snippets]]
|
||||||
|
|
||||||
|
@ -45,8 +51,8 @@ This module has no dedicated maintainers.
|
||||||
|
|
||||||
** Module Flags
|
** Module Flags
|
||||||
+ =+hack= Add support for the [[https://hacklang.org/][Hack dialect of PHP]] by Facebook.
|
+ =+hack= Add support for the [[https://hacklang.org/][Hack dialect of PHP]] by Facebook.
|
||||||
+ =+lsp= Enable LSP support through phpactor or intelephense. Requires the
|
+ =+lsp= Enable LSP support through phpactor or intelephense. Requires the ~:tools
|
||||||
~:tools lsp~ module and the [[https://phpactor.readthedocs.io/en/develop/usage/standalone.html][phpactor server]] to be installed on your system.
|
lsp~ module and the [[https://phpactor.readthedocs.io/en/develop/usage/standalone.html][phpactor server]] to be installed on your system.
|
||||||
** Plugins
|
** Plugins
|
||||||
+ [[https://github.com/tomterl/php-boris][async]]
|
+ [[https://github.com/tomterl/php-boris][async]]
|
||||||
+ [[https://github.com/tomterl/php-boris][php-boris]]
|
+ [[https://github.com/tomterl/php-boris][php-boris]]
|
||||||
|
@ -54,6 +60,7 @@ This module has no dedicated maintainers.
|
||||||
+ [[https://github.com/emacs-php/php-mode][php-mode]]
|
+ [[https://github.com/emacs-php/php-mode][php-mode]]
|
||||||
+ [[https://github.com/keelerm84/php-refactor-mode.el][php-refactor-mode]]
|
+ [[https://github.com/keelerm84/php-refactor-mode.el][php-refactor-mode]]
|
||||||
+ [[https://github.com/nlamirault/phpunit.el][phpunit]]
|
+ [[https://github.com/nlamirault/phpunit.el][phpunit]]
|
||||||
|
+ [[https://github.com/emacs-php/composer.el][composer.el]]
|
||||||
+ =+hack=
|
+ =+hack=
|
||||||
+ [[https://github.com/hhvm/hack-mode][hack-mode]]
|
+ [[https://github.com/hhvm/hack-mode][hack-mode]]
|
||||||
+ =+lsp=
|
+ =+lsp=
|
||||||
|
@ -71,9 +78,10 @@ Note for =+lsp=:
|
||||||
2. If you use intelephense, ~node~ and ~npm~ are needed.
|
2. If you use intelephense, ~node~ and ~npm~ are needed.
|
||||||
|
|
||||||
*** MacOS
|
*** MacOS
|
||||||
PHP 5.5 comes prepackaged with newer versions of MacOS. These instructions are provided for reference:
|
PHP 5.5 comes prepackaged with newer versions of MacOS. These instructions are
|
||||||
|
provided for reference:
|
||||||
|
|
||||||
#+BEGIN_SRC sh :tangle (if (doom-system-os 'macos) "yes")
|
#+begin_src sh :tangle (if (doom-system-os 'macos) "yes")
|
||||||
brew tap homebrew/homebrew-php
|
brew tap homebrew/homebrew-php
|
||||||
brew install php71 # or php53, php54, php55
|
brew install php71 # or php53, php54, php55
|
||||||
brew install composer
|
brew install composer
|
||||||
|
@ -81,23 +89,23 @@ brew install composer
|
||||||
# If you use intelephense:
|
# If you use intelephense:
|
||||||
brew install node
|
brew install node
|
||||||
brew install npm
|
brew install npm
|
||||||
#+END_SRC
|
#+end_src
|
||||||
|
|
||||||
*** Arch Linux
|
*** Arch Linux
|
||||||
#+BEGIN_SRC sh :dir /sudo:: :tangle (if (doom-system-os 'arch) "yes")
|
#+begin_src sh :dir /sudo:: :tangle (if (doom-system-os 'arch) "yes")
|
||||||
sudo pacman --needed --noconfirm -S php composer # or php53, php54, php55
|
sudo pacman --needed --noconfirm -S php composer # or php53, php54, php55
|
||||||
|
|
||||||
# If you use intelephense:
|
# If you use intelephense:
|
||||||
sudo pacman -S nodejs npm
|
sudo pacman -S nodejs npm
|
||||||
#+END_SRC
|
#+end_src
|
||||||
|
|
||||||
*** openSUSE
|
*** openSUSE
|
||||||
#+BEGIN_SRC sh :dir /sudo::
|
#+begin_src sh :dir /sudo::
|
||||||
sudo zypper install php-composer
|
sudo zypper install php-composer
|
||||||
|
|
||||||
# If you use intelephense:
|
# If you use intelephense:
|
||||||
sudo zypper install nodejs npm
|
sudo zypper install nodejs npm
|
||||||
#+END_SRC
|
#+end_src
|
||||||
|
|
||||||
** Dependencies
|
** Dependencies
|
||||||
This module has no required dependencies, but it has a couple optional ones.
|
This module has no required dependencies, but it has a couple optional ones.
|
||||||
|
@ -106,40 +114,129 @@ This module has no required dependencies, but it has a couple optional ones.
|
||||||
+ ~phpctags~ (better code completion)
|
+ ~phpctags~ (better code completion)
|
||||||
+ ~phpunit~ (unit test commands)
|
+ ~phpunit~ (unit test commands)
|
||||||
+ ~php-cs-fixer~ and ~@prettier/plugin-php~ (for code formatting)
|
+ ~php-cs-fixer~ and ~@prettier/plugin-php~ (for code formatting)
|
||||||
|
+ ~phpactor~ (for LSP if intelephense isn't desired)
|
||||||
|
|
||||||
#+BEGIN_SRC sh
|
#+begin_src sh
|
||||||
composer global require \
|
composer global require \
|
||||||
d11wtq/boris \
|
d11wtq/boris \
|
||||||
phpunit/phpunit \
|
phpunit/phpunit \
|
||||||
techlivezheng/phpctags \
|
techlivezheng/phpctags \
|
||||||
friendsofphp/php-cs-fixer
|
friendsofphp/php-cs-fixer \
|
||||||
|
phpactor/phpactor
|
||||||
|
|
||||||
# Needed by php-cs-fixer, otherwise you'll get "Couldn't resolve parser
|
# Needed by php-cs-fixer, otherwise you'll get "Couldn't resolve parser
|
||||||
# 'php'" errors
|
# 'php'" errors
|
||||||
npm install -g @prettier/plugin-php
|
npm install -g @prettier/plugin-php
|
||||||
#+END_SRC
|
#+end_src
|
||||||
|
|
||||||
You must ensure that ~\~/.composer/vendor/bin~ is in ~PATH~, so these
|
You must ensure that ~~/.composer/vendor/bin~ is in ~PATH~, so these executables are
|
||||||
executables are visible to Emacs:
|
visible to Emacs:
|
||||||
|
|
||||||
#+BEGIN_SRC sh
|
#+begin_src sh
|
||||||
# place this in your profile file, like ~/.bash_profile or ~/.zshenv
|
# place this in your profile file, like ~/.bash_profile or ~/.zshenv
|
||||||
export PATH="~/.composer/vendor/bin:$PATH"
|
export PATH="~/.composer/vendor/bin:$PATH"
|
||||||
#+END_SRC
|
#+end_src
|
||||||
|
|
||||||
You may also need to regenerate your envvar file by running ~doom env~ on the
|
You may also need to regenerate your envvar file by running ~doom env~ on the
|
||||||
command line.
|
command line.
|
||||||
|
|
||||||
#+begin_quote
|
*NOTE* phpactor doesn't have to be installed via =composer=, just has to exist in
|
||||||
To use intelephense instead of , run =M-x lsp-install-server= and choose ~iph~ to
|
your =$PATH=.
|
||||||
install lsp-intelephense.
|
|
||||||
#+end_quote
|
|
||||||
|
|
||||||
* TODO Features
|
* Features
|
||||||
# An in-depth list of features, how to use them, and their dependencies.
|
** LSP Support
|
||||||
|
There are a number of currently supported LSP servers:
|
||||||
|
|
||||||
* TODO Configuration
|
+ [[https://emacs-lsp.github.io/lsp-mode/page/lsp-intelephense/][Intelephense]] (_Recommended_)
|
||||||
# How to configure this module, including common problems and how to address them.
|
+ [[https://emacs-lsp.github.io/lsp-mode/page/lsp-phpactor/][phpactor]]
|
||||||
|
+ [[https://emacs-lsp.github.io/lsp-mode/page/lsp-serenata/][Serenata]]
|
||||||
|
+ [[https://emacs-lsp.github.io/lsp-mode/page/lsp-php/][felixbecker]] (Considered unsupported)
|
||||||
|
|
||||||
* TODO Troubleshooting
|
Intelephense is currently the only server that supports automatic installation,
|
||||||
# Common issues and their solution, or places to look for help.
|
which will trigger either when you open a PHP project or manually invoke
|
||||||
|
=lsp-install-server= through =M-x=.
|
||||||
|
|
||||||
|
The others have to be installed manually and added to your =$PATH=.
|
||||||
|
|
||||||
|
** PHPUnit
|
||||||
|
This module provides an interface to PHPUnit through a number of commands as
|
||||||
|
detailed below. By default, it loads configuration from the root ~phpunit.xml~.
|
||||||
|
|
||||||
|
+ ~phpunit-current-project~ Launch all tests for the project
|
||||||
|
+ ~phpunit-current-class~ Launch all tests for the current class/fixture
|
||||||
|
+ ~phpunit-current-test~ Launch the current test at point
|
||||||
|
|
||||||
|
If for some reason, the default ~phpunit.xml~ is in a different location (or you
|
||||||
|
use the ~phpunit.xml.dist~ convention) , the path can be changed via
|
||||||
|
=phpunit-configuration-file=
|
||||||
|
|
||||||
|
#+begin_src emacs-lisp
|
||||||
|
(setq phpunit-configuration-file "phpunit.xml")
|
||||||
|
#+end_src
|
||||||
|
|
||||||
|
** Composer
|
||||||
|
This module provides several convenience methods for triggering composer
|
||||||
|
commands:
|
||||||
|
|
||||||
|
| Binding | Function |
|
||||||
|
|---------------------+---------------------------------|
|
||||||
|
| ~<localleader> m c c~ | ~composer~ |
|
||||||
|
| ~<localleader> m c i~ | ~composer-install~ |
|
||||||
|
| ~<localleader> m c r~ | ~composer-require~ |
|
||||||
|
| ~<localleader> m c u~ | ~composer-update~ |
|
||||||
|
| ~<localleader> m c d~ | ~composer-dump-autoload~ |
|
||||||
|
| ~<localleader> m c s~ | ~composer-run-scripts~ |
|
||||||
|
| ~<localleader> m c v~ | ~composer-run-vendor-bin-command~ |
|
||||||
|
| ~<localleader> m c o~ | ~composer-find-json-file~ |
|
||||||
|
| ~<localleader> m c l~ | ~composer-view-lock-file~ |
|
||||||
|
|
||||||
|
These are all invokable via =M-x= too.
|
||||||
|
|
||||||
|
* Configuration
|
||||||
|
** Docker Compose
|
||||||
|
A lot of projects rely on running inside docker compose (ie Laravel), and as
|
||||||
|
such a minor mode has been configured to attempt to run tests inside the =php-fpm=
|
||||||
|
(by default) container.
|
||||||
|
|
||||||
|
If for some reason you wish to specify a different container, modify the
|
||||||
|
~+php/default-docker-container~ variable (ideally inside a ~.dir-locals.el~ file)
|
||||||
|
|
||||||
|
#+begin_src emacs-lisp
|
||||||
|
((php-mode . ((+php/default-docker-container . "php-octane"))))
|
||||||
|
#+end_src
|
||||||
|
|
||||||
|
* Troubleshooting
|
||||||
|
** "I'm missing functionality on lsp-mode"
|
||||||
|
Unfortunately, [[https://intelephense.com/][intelephense]] currently operates under a "freemium" model, and as
|
||||||
|
such requires a license for extended features. Once purchased, this can be
|
||||||
|
(insecurely) added directly to your config:
|
||||||
|
|
||||||
|
#+begin_src emacs-lisp
|
||||||
|
(setq lsp-intelephense-licence-key "<key>")
|
||||||
|
#+end_src
|
||||||
|
|
||||||
|
A more recommended approach would be to utilise Emacs' own ~auth-sources~ for
|
||||||
|
storing authentication info, which can also be encrypted.
|
||||||
|
|
||||||
|
Create a file in your home directory (which can optionally be encrypted, verify
|
||||||
|
your ~auth-sources~ has the correct values) called ~~/.authinfo~
|
||||||
|
|
||||||
|
#+begin_src
|
||||||
|
machine * login intelephense password <key>
|
||||||
|
#+end_src
|
||||||
|
|
||||||
|
And add the following to your config
|
||||||
|
|
||||||
|
#+begin_src emacs-lisp
|
||||||
|
(defun my-fetch-password (&rest params)
|
||||||
|
(require 'auth-source)
|
||||||
|
(let ((match (car (apply #'auth-source-search params))))
|
||||||
|
(if match
|
||||||
|
(let ((secret (plist-get match :secret)))
|
||||||
|
(if (functionp secret)
|
||||||
|
(funcall secret)
|
||||||
|
secret))
|
||||||
|
(error "Password not found for %S" params))))
|
||||||
|
|
||||||
|
(setq lsp-intelephense-license-key (my-fetch-password :user intelephense))
|
||||||
|
#+end_src
|
||||||
|
|
|
@ -3,6 +3,12 @@
|
||||||
(defvar +php--company-backends nil
|
(defvar +php--company-backends nil
|
||||||
"List of company backends to use in `php-mode'.")
|
"List of company backends to use in `php-mode'.")
|
||||||
|
|
||||||
|
(defvar +php-default-docker-container "php-fpm"
|
||||||
|
"The default docker container to run commands in.")
|
||||||
|
|
||||||
|
(defvar +php-default-docker-compose "docker-compose.yml"
|
||||||
|
"Path to docker-compose file.")
|
||||||
|
|
||||||
(after! projectile
|
(after! projectile
|
||||||
(add-to-list 'projectile-project-root-files "composer.json"))
|
(add-to-list 'projectile-project-root-files "composer.json"))
|
||||||
|
|
||||||
|
@ -12,10 +18,11 @@
|
||||||
|
|
||||||
(use-package! php-mode
|
(use-package! php-mode
|
||||||
:mode "\\.inc\\'"
|
:mode "\\.inc\\'"
|
||||||
|
:hook (php-mode . rainbow-delimiters-mode)
|
||||||
:config
|
:config
|
||||||
;; Disable HTML compatibility in php-mode. `web-mode' has superior support for
|
;; Disable HTML compatibility in php-mode. `web-mode' has superior support for
|
||||||
;; php+html. Use the .phtml extension instead.
|
;; php+html. Use the .phtml extension instead.
|
||||||
(setq php-template-compatibility nil)
|
(setq php-mode-template-compatibility nil)
|
||||||
|
|
||||||
(set-docsets! 'php-mode "PHP" "PHPUnit" "Laravel" "CakePHP" "CodeIgniter" "Doctrine_ORM")
|
(set-docsets! 'php-mode "PHP" "PHPUnit" "Laravel" "CakePHP" "CodeIgniter" "Doctrine_ORM")
|
||||||
(set-repl-handler! 'php-mode #'php-boris)
|
(set-repl-handler! 'php-mode #'php-boris)
|
||||||
|
@ -23,7 +30,7 @@
|
||||||
(set-formatter! 'php-mode #'php-cs-fixer-fix)
|
(set-formatter! 'php-mode #'php-cs-fixer-fix)
|
||||||
(set-ligatures! 'php-mode
|
(set-ligatures! 'php-mode
|
||||||
;; Functional
|
;; Functional
|
||||||
:lambda "function()"
|
:lambda "function()" :lambda "fn"
|
||||||
:def "function"
|
:def "function"
|
||||||
;; Types
|
;; Types
|
||||||
:null "null"
|
:null "null"
|
||||||
|
@ -127,6 +134,24 @@
|
||||||
:mode "\\.hh$")
|
:mode "\\.hh$")
|
||||||
|
|
||||||
|
|
||||||
|
(use-package! composer
|
||||||
|
:defer t
|
||||||
|
:init
|
||||||
|
(map! :after php-mode
|
||||||
|
:localleader
|
||||||
|
:map php-mode-map
|
||||||
|
:prefix ("c" . "composer")
|
||||||
|
"c" #'composer
|
||||||
|
"i" #'composer-install
|
||||||
|
"r" #'composer-require
|
||||||
|
"u" #'composer-update
|
||||||
|
"d" #'composer-dump-autoload
|
||||||
|
"s" #'composer-run-script
|
||||||
|
"v" #'composer-run-vendor-bin-command
|
||||||
|
"o" #'composer-find-json-file
|
||||||
|
"l" #'composer-view-lock-file))
|
||||||
|
|
||||||
|
|
||||||
;;
|
;;
|
||||||
;; Projects
|
;; Projects
|
||||||
|
|
||||||
|
@ -137,3 +162,10 @@
|
||||||
(def-project-mode! +php-composer-mode
|
(def-project-mode! +php-composer-mode
|
||||||
:modes '(web-mode php-mode)
|
:modes '(web-mode php-mode)
|
||||||
:files ("composer.json"))
|
:files ("composer.json"))
|
||||||
|
|
||||||
|
(def-project-mode! +phpunit-docker-compose-mode
|
||||||
|
:modes '(php-mode docker-compose-mode)
|
||||||
|
:files (and "phpunit.xml" (or +php-default-docker-compose "docker-compose.yml"))
|
||||||
|
:on-enter
|
||||||
|
(setq phpunit-args `("exec" ,+php-default-docker-container "php" "vendor/bin/phpunit")
|
||||||
|
phpunit-executable (executable-find "docker-compose")))
|
||||||
|
|
|
@ -8,6 +8,7 @@
|
||||||
(package! php-mode :pin "535aec81739e8e766e0420fda616efc8846f2911")
|
(package! php-mode :pin "535aec81739e8e766e0420fda616efc8846f2911")
|
||||||
(package! php-refactor-mode :pin "7a794b0618df2882b1bd586fdd698dba0bc5130d")
|
(package! php-refactor-mode :pin "7a794b0618df2882b1bd586fdd698dba0bc5130d")
|
||||||
(package! phpunit :pin "fe6bc91c3bd8b329c6d26ad883a025f06b5121ee")
|
(package! phpunit :pin "fe6bc91c3bd8b329c6d26ad883a025f06b5121ee")
|
||||||
|
(package! composer :pin "7c7f89df226cac69664d7eca5e913b544dc475c5")
|
||||||
|
|
||||||
(when (featurep! +hack)
|
(when (featurep! +hack)
|
||||||
(package! hack-mode
|
(package! hack-mode
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue