This really was a lark to see if I could change ALL the files using woccur and a regular expression. Quite please with how simple that was.
		
			
				
	
	
		
			331 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Org Mode
		
	
	
	
	
	
			
		
		
	
	
			331 lines
		
	
	
		
			13 KiB
		
	
	
	
		
			Org Mode
		
	
	
	
	
	
| #+TITLE:  Configuring Python in Emacs
 | ||
| #+AUTHOR: Howard X. Abrams
 | ||
| #+DATE:   2021-11-16
 | ||
| 
 | ||
| A literate programming file for configuring Python.
 | ||
| 
 | ||
| #+begin_src emacs-lisp :exports none
 | ||
|   ;;; ha-programming-python --- Python configuration. -*- lexical-binding: t; -*-
 | ||
|   ;;
 | ||
|   ;; © 2021-2023 Howard X. Abrams
 | ||
|   ;;   Licensed under a Creative Commons Attribution 4.0 International License.
 | ||
|   ;;   See http://creativecommons.org/licenses/by/4.0/
 | ||
|   ;;
 | ||
|   ;; Author: Howard X. Abrams <http://gitlab.com/howardabrams>
 | ||
|   ;; Maintainer: Howard X. Abrams
 | ||
|   ;; Created: November 16, 2021
 | ||
|   ;;
 | ||
|   ;; This file is not part of GNU Emacs.
 | ||
|   ;;
 | ||
|   ;; *NB:* Do not edit this file. Instead, edit the original literate file at:
 | ||
|   ;;            ~/other/hamacs/ha-programming-python.org
 | ||
|   ;;       And tangle the file to recreate this one.
 | ||
|   ;;
 | ||
|   ;;; Code:
 | ||
|   #+end_src
 | ||
| * Introduction
 | ||
| The critical part of Python integration with Emacs is running LSP in Python using [[file:ha-programming.org::*direnv][direnv]]. And the question to ask is if the Python we run it in Docker or in a virtual environment.
 | ||
| 
 | ||
| #+begin_src emacs-lisp
 | ||
|   (general-create-definer ha-python-leader
 | ||
|       :states '(normal visual motion)
 | ||
|       :keymaps 'python-mode-map
 | ||
|       :prefix "SPC m"
 | ||
|       :global-prefix "<f17>"
 | ||
|       :non-normal-prefix "S-SPC")
 | ||
| #+end_src
 | ||
| While Emacs supplies a Python editing environment, we’ll still use =use-package= to grab the latest:
 | ||
| 
 | ||
| #+begin_src emacs-lisp
 | ||
|   (use-package python
 | ||
|     :after projectile flycheck
 | ||
|     :mode ("[./]flake8\\'" . conf-mode)
 | ||
|     :mode ("/Pipfile\\'" . conf-mode)
 | ||
|     :init
 | ||
|     (setq python-indent-guess-indent-offset-verbose nil
 | ||
|           flycheck-flake8-maximum-line-length 120)
 | ||
|     :config
 | ||
|     (when (and (executable-find "python3")
 | ||
|                (string= python-shell-interpreter "python"))
 | ||
|         (setq python-shell-interpreter "python3"))
 | ||
| 
 | ||
|     ;; While `setup.py' and `requirements.txt' are already added, I often
 | ||
|     ;; create these files for my Python projects:
 | ||
|     (add-to-list 'projectile-project-root-files "requirements-dev.txt")
 | ||
|     (add-to-list 'projectile-project-root-files "requirements-test.txt")
 | ||
| 
 | ||
|     (flycheck-add-next-checker 'python-pylint 'python-pycompile 'append))
 | ||
| #+end_src
 | ||
| 
 | ||
| Note: Install the following checks:
 | ||
| #+begin_src sh
 | ||
|   pip install flake8 pylint pyright mypy pycompile
 | ||
| #+end_src
 | ||
| Or better yet, add those to the =requirements-dev.txt= file.
 | ||
| ** Virtual Environment
 | ||
| For a local virtual machine, put the following in your =.envrc= file:
 | ||
| #+begin_src conf
 | ||
| layout_python3
 | ||
| #+end_src
 | ||
| That is pretty slick and simple.
 | ||
| 
 | ||
| The old way, that we still use if you need a particular version of Python, is to install [[https://github.com/pyenv/pyenv][pyenv]] globally:
 | ||
| #+begin_src sh
 | ||
| pip install pyenv
 | ||
| #+end_src
 | ||
| 
 | ||
| And have this in your =.envrc= file:
 | ||
| #+begin_src conf
 | ||
| use python 3.7.1
 | ||
| #+end_src
 | ||
| 
 | ||
| Also, you need the following in your =~/.config/direnv/direnvrc= file (which I have):
 | ||
| #+begin_src shell
 | ||
| use_python() {
 | ||
|   local python_root=$(pyenv root)/versions/$1
 | ||
|   load_prefix "$python_root"
 | ||
|   if [[ -x "$python_root/bin/python" ]]; then
 | ||
|     layout python "$python_root/bin/python"
 | ||
|   else
 | ||
|     echo "Error: $python_root/bin/python can't be executed."
 | ||
|     exit
 | ||
|   fi
 | ||
| }
 | ||
| #+end_src
 | ||
| ** Editing Python Code
 | ||
| Let’s integrate this [[https://github.com/wbolster/evil-text-object-python][Python support for evil-text-object]] project:
 | ||
| #+begin_src emacs-lisp
 | ||
|   (use-package evil-text-object-python
 | ||
|     :hook (python-mode . evil-text-object-python-add-bindings))
 | ||
| 
 | ||
| 
 | ||
| #+end_src
 | ||
| This allows me to delete a Python “block” using ~dal~.
 | ||
| ** Docker Environment
 | ||
| Docker really allows you to isolate your project's environment. The downside is that you are using Docker and probably a bloated container. On my work laptop, a Mac, this creates a behemoth virtual machine that immediately spins the fans like a wind tunnel.
 | ||
| 
 | ||
| But, but... think of the dependencies!
 | ||
| 
 | ||
| Enough of the rant (I go back and forth), after getting Docker installed and running (ooo Podman ... shiny), and you've created a =Dockerfile= for your project, let's install [[https://github.com/snbuback/container-env][container-env]].
 | ||
| 
 | ||
| Your project's =.envrc= file would contain something like:
 | ||
| #+begin_src shell
 | ||
| CONTAINER_NAME=my-docker-container
 | ||
| CONTAINER_WRAPPERS=(python3 pip3 yamllint)
 | ||
| CONTAINER_EXTRA_ARGS="--env SOME_ENV_VAR=${SOME_ENV_VAR}"
 | ||
| 
 | ||
| container_layout
 | ||
| #+end_src
 | ||
| ** Unit Tests
 | ||
| #+begin_src emacs-lisp
 | ||
|   (use-package python-pytest
 | ||
|     :after python
 | ||
|     :commands python-pytest-dispatch
 | ||
|     :init
 | ||
|     (ha-python-leader
 | ||
|       "t" '(:ignore t :which-key "tests")
 | ||
|       "t a" '("all" . python-pytest)
 | ||
|       "t f" '("file dwim" . python-pytest-file-dwim)
 | ||
|       "t F" '("file" . python-pytest-file)
 | ||
|       "t t" '("function-dwim" . python-pytest-function-dwim)
 | ||
|       "t T" '("function" . python-pytest-function)
 | ||
|       "t r" '("repeat" . python-pytest-repeat)
 | ||
|       "t p" '("dispatch" . python-pytest-dispatch)))
 | ||
| #+end_src
 | ||
| ** Python Dependencies
 | ||
| Each Python project's =requirements-dev.txt= file would reference the [[https://pypi.org/project/python-lsp-server/][python-lsp-server]] (not the /unmaintained/ project, =python-language-server=):
 | ||
| 
 | ||
| #+begin_src conf :tangle no
 | ||
| python-lsp-server[all]
 | ||
| #+end_src
 | ||
| 
 | ||
| *Note:* This does mean, you would have a =tox.ini= with this line:
 | ||
| #+begin_src conf
 | ||
|   [tox]
 | ||
|   minversion = 1.6
 | ||
|   skipsdist = True
 | ||
|   envlist = linters
 | ||
|   ignore_basepython_conflict = True
 | ||
| 
 | ||
|   [testenv]
 | ||
|   basepython = python3
 | ||
|   install_command = pip install {opts} {packages}
 | ||
|   deps = -r{toxinidir}/test-requirements.txt
 | ||
|   commands = stestr run {posargs}
 | ||
|              stestr slowest
 | ||
|   # ...
 | ||
| #+end_src
 | ||
| *** Pyright
 | ||
| I’m using the Microsoft-supported [[https://github.com/Microsoft/pyright][pyright]] package instead. Adding this to my =requirements.txt= files:
 | ||
| #+begin_src conf :tangle no
 | ||
| pyright
 | ||
| #+end_src
 | ||
| 
 | ||
| The [[https://github.com/emacs-lsp/lsp-pyright][pyright package]] works with LSP.
 | ||
| 
 | ||
| #+begin_src emacs-lisp :tangle no
 | ||
| (use-package lsp-pyright
 | ||
|     :hook (python-mode . (lambda () (require 'lsp-pyright)))
 | ||
|     :init (when (executable-find "python3")
 | ||
|               (setq lsp-pyright-python-executable-cmd "python3")))
 | ||
| #+end_src
 | ||
| * LSP Integration of Python
 | ||
| Now that the [[file:ha-programming.org::*Language Server Protocol (LSP) Integration][LSP Integration]] is complete, we can stitch the two projects together, by calling =lsp=. I oscillate between automatically turning on LSP mode with every Python file, but I sometimes run into issues when starting, so I turn it on with ~SPC m w s~.
 | ||
| 
 | ||
| #+begin_src emacs-lisp :tangle no
 | ||
|   (use-package lsp-mode
 | ||
|     ;; :hook ((python-mode . lsp)))
 | ||
|     :config
 | ||
|     (ha-python-leader
 | ||
|       "0" '("treemacs" . lsp-treemacs-symbols)
 | ||
| 
 | ||
|       "/" '("complete" . completion-at-point)
 | ||
|       "k" '("check code" . python-check)
 | ||
|       "]" '("shift left" . python-indent-shift-left)
 | ||
|       "[" '("shift right" . python-indent-shift-right)
 | ||
| 
 | ||
|       ;; actions
 | ||
|       "a" '(:ignore t :which-key "code actions")
 | ||
|       "aa" '("code actions" . lsp-execute-code-action)
 | ||
|       "ah" '("highlight symbol" . lsp-document-highlight)
 | ||
|       "al" '("lens" . lsp-avy-lens)
 | ||
| 
 | ||
|       ;; formatting
 | ||
|       "=" '(:ignore t :which-key "formatting")
 | ||
|       "==" '("format buffer" . lsp-format-buffer)
 | ||
|       "=r" '("format region" . lsp-format-region)
 | ||
| 
 | ||
|       "e" '(:ignore t :which-key "eval")
 | ||
|       "e P" '("run python" . run-python)
 | ||
|       "e e" '("send statement" . python-shell-send-statement)
 | ||
|       "e b" '("send buffer" . python-shell-send-buffer)
 | ||
|       "e f" '("send defun" . python-shell-send-defun)
 | ||
|       "e F" '("send file" . python-shell-send-file)
 | ||
|       "e r" '("send region" . python-shell-send-region)
 | ||
|       "e ;" '("expression" . python-shell-send-string)
 | ||
|       "e p" '("switch-to-shell" . python-shell-switch-to-shell)
 | ||
| 
 | ||
|       ;; folders
 | ||
|       "F" '(:ignore t :which-key "folders")
 | ||
|       "Fa" '("add folder" . lsp-workspace-folders-add)
 | ||
|       "Fb" '("un-blacklist folder" . lsp-workspace-blacklist-remove)
 | ||
|       "Fr" '("remove folder" . lsp-workspace-folders-remove)
 | ||
| 
 | ||
|       ;; goto
 | ||
|       "g" '(:ignore t :which-key "goto")
 | ||
|       "ga" '("find symbol in workspace" . xref-find-apropos)
 | ||
|       "gd" '("find declarations" . lsp-find-declaration)
 | ||
|       "ge" '("show errors" . lsp-treemacs-errors-list)
 | ||
|       "gg" '("find definitions" . lsp-find-definition)
 | ||
|       "gh" '("call hierarchy" . lsp-treemacs-call-hierarchy)
 | ||
|       "gi" '("find implementations" . lsp-find-implementation)
 | ||
|       "gm" '("imenu" . lsp-ui-imenu)
 | ||
|       "gr" '("find references" . lsp-find-references)
 | ||
|       "gt" '("find type definition" . lsp-find-type-definition)
 | ||
| 
 | ||
|       ;; peeks
 | ||
|       "G" '(:ignore t :which-key "peek")
 | ||
|       "Gg" '("peek definitions" . lsp-ui-peek-find-definitions)
 | ||
|       "Gi" '("peek implementations" . lsp-ui-peek-find-implementation)
 | ||
|       "Gr" '("peek references" . lsp-ui-peek-find-references)
 | ||
|       "Gs" '("peek workspace symbol" . lsp-ui-peek-find-workspace-symbol)
 | ||
| 
 | ||
|       ;; help
 | ||
|       "h" '(:ignore t :which-key "help")
 | ||
|       "he" '("eldoc" . python-eldoc-at-point)
 | ||
|       "hg" '("glance symbol" . lsp-ui-doc-glance)
 | ||
|       "hh" '("describe symbol at point" . lsp-describe-thing-at-point)
 | ||
|       "gH" '("describe python symbol" . python-describe-at-point)
 | ||
|       "hs" '("signature help" . lsp-signature-activate)
 | ||
| 
 | ||
|       "i" 'imenu
 | ||
| 
 | ||
|       ;; refactoring
 | ||
|       "r" '(:ignore t :which-key "refactor")
 | ||
|       "ro" '("organize imports" . lsp-organize-imports)
 | ||
|       "rr" '("rename" . lsp-rename)
 | ||
| 
 | ||
|       ;; toggles
 | ||
|       "t" '(:ignore t :which-key "toggle")
 | ||
|       "tD" '("toggle modeline diagnostics" . lsp-modeline-diagnostics-mode)
 | ||
|       "tL" '("toggle log io" . lsp-toggle-trace-io)
 | ||
|       "tS" '("toggle sideline" . lsp-ui-sideline-mode)
 | ||
|       "tT" '("toggle treemacs integration" . lsp-treemacs-sync-mode)
 | ||
|       "ta" '("toggle modeline code actions" . lsp-modeline-code-actions-mode)
 | ||
|       "tb" '("toggle breadcrumb" . lsp-headerline-breadcrumb-mode)
 | ||
|       "td" '("toggle documentation popup" . lsp-ui-doc-mode)
 | ||
|       "tf" '("toggle on type formatting" . lsp-toggle-on-type-formatting)
 | ||
|       "th" '("toggle highlighting" . lsp-toggle-symbol-highlight)
 | ||
|       "tl" '("toggle lenses" . lsp-lens-mode)
 | ||
|       "ts" '("toggle signature" . lsp-toggle-signature-auto-activate)
 | ||
| 
 | ||
|       ;; workspaces
 | ||
|       "w" '(:ignore t :which-key "workspaces")
 | ||
|       "wD" '("disconnect" . lsp-disconnect)
 | ||
|       "wd" '("describe session" . lsp-describe-session)
 | ||
|       "wq" '("shutdown server" . lsp-workspace-shutdown)
 | ||
|       "wr" '("restart server" . lsp-workspace-restart)
 | ||
|       "ws" '("start server" . lsp)))
 | ||
| #+end_src
 | ||
| * Project Configuration
 | ||
| I work with a lot of projects with my team where I need to /configure/ the project such that LSP and my Emacs setup works. Let's suppose I could point a function at a project directory, and have it /set it up/:
 | ||
| 
 | ||
| #+begin_src emacs-lisp
 | ||
|   (defun ha-python-configure-project (proj-directory)
 | ||
|     "Configure PROJ-DIRECTORY for LSP and Python."
 | ||
|     (interactive "DPython Project: ")
 | ||
| 
 | ||
|     (let ((default-directory proj-directory))
 | ||
|       (unless (f-exists? ".envrc")
 | ||
|         (message "Configuring direnv")
 | ||
|         (with-temp-file ".envrc"
 | ||
|           ;; (insert "use_python 3.7.4\n")
 | ||
|           (insert "layout_python3\n"))
 | ||
|         (direnv-allow))
 | ||
| 
 | ||
|       (unless (f-exists? ".pip.conf")
 | ||
|         (message "Configuring pip")
 | ||
|         (with-temp-file ".pip.conf"
 | ||
|           (insert "[global]\n")
 | ||
|           (insert "index-url = https://pypi.python.org/simple\n"))
 | ||
|         (shell-command "pipconf --local")
 | ||
|         (shell-command "pip install --upgrade pip"))
 | ||
| 
 | ||
|       (message "Configuring pip for LSP")
 | ||
|       (with-temp-file "requirements-dev.txt"
 | ||
|         (insert "python-lsp-server[all]\n")
 | ||
| 
 | ||
|         ;; Let's install these extra packages individually ...
 | ||
|         (insert "pyls-flake8\n")
 | ||
|         ;; (insert "pylsp-mypy")
 | ||
|         ;; (insert "pyls-isort")
 | ||
|         ;; (insert "python-lsp-black")
 | ||
|         ;; (insert "pyls-memestra")
 | ||
|         (insert "pylsp-rope\n"))
 | ||
|       (shell-command "pip install -r requirements-dev.txt")
 | ||
| 
 | ||
|       (unless (f-exists? ".projectile")
 | ||
|         (with-temp-file ".projectile"))
 | ||
| 
 | ||
|       (unless (f-exists? ".dir-locals.el")
 | ||
|         (with-temp-file ".dir-locals.el"
 | ||
|           (insert "((nil . ((projectile-enable-caching . t))))")))))
 | ||
| #+end_src
 | ||
| * Technical Artifacts                                :noexport:
 | ||
| Let's =provide= a name so we can =require= this file:
 | ||
| 
 | ||
| #+begin_src emacs-lisp :exports none
 | ||
|   (provide 'ha-programming-python)
 | ||
|   ;;; ha-programming-python.el ends here
 | ||
|   #+end_src
 | ||
| 
 | ||
| #+DESCRIPTION: A literate programming file for configuring Python.
 | ||
| 
 | ||
| #+PROPERTY:    header-args:sh :tangle no
 | ||
| #+PROPERTY:    header-args:emacs-lisp  :tangle yes
 | ||
| #+PROPERTY:    header-args    :results none :eval no-export :comments no mkdirp yes
 | ||
| 
 | ||
| #+OPTIONS:     num:nil toc:nil todo:nil tasks:nil tags:nil date:nil
 | ||
| #+OPTIONS:     skip:nil author:nil email:nil creator:nil timestamp:nil
 | ||
| #+INFOJS_OPT:  view:nil toc:nil ltoc:t mouse:underline buttons:0 path:http://orgmode.org/org-info.js
 |