You cannot select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
phpinspect.el/phpinspect-imports.el

395 lines
18 KiB
EmacsLisp

; phpinspect-imports.el --- PHP parsing and completion package -*- lexical-binding: t; -*-
;; Copyright (C) 2021-2023 Free Software Foundation, Inc
;; Author: Hugo Thunnissen <devel@hugot.nl>
;; Keywords: php, languages, tools, convenience
;; Version: 1.2.0
;; This program is free software; you can redistribute it and/or modify
;; it under the terms of the GNU General Public License as published by
;; the Free Software Foundation, either version 3 of the License, or
;; (at your option) any later version.
;; This program is distributed in the hope that it will be useful,
;; but WITHOUT ANY WARRANTY; without even the implied warranty of
;; MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
;; GNU General Public License for more details.
;; You should have received a copy of the GNU General Public License
;; along with this program. If not, see <https://www.gnu.org/licenses/>.
;;; Commentary:
;; See docstrings for documentation, starting with `phpinspect-mode'.
;;; Code:
(require 'phpinspect-token-predicates)
(require 'phpinspect-index)
(require 'phpinspect-autoload)
(require 'phpinspect-buffer)
(require 'phpinspect-cache)
(require 'phpinspect-util)
(require 'phpinspect-type)
(defun phpinspect-insert-at-point (point data)
(save-excursion
(goto-char point)
(insert data)))
(defun phpinspect-find-first-use (token-meta)
(if (and (phpinspect-namespace-p (phpinspect-meta-token token-meta))
(phpinspect-namespace-is-blocked-p (phpinspect-meta-token token-meta)))
(phpinspect-find-first-use (phpinspect-meta-last-child token-meta))
(phpinspect-meta-find-first-child-matching
token-meta (phpinspect-meta-wrap-token-pred #'phpinspect-use-p))))
(defun phpinspect-add-use (fqn buffer &optional namespace-meta)
"Add use statement for FQN to BUFFER.
If NAMESPACE-TOKEN is non-nil, it is assumed to be a token that
was parsed from BUFFER and its location will be used to find a
buffer position to insert the use statement at."
(when (string-match "^\\\\" fqn)
(setq fqn (string-trim-left fqn "\\\\")))
(if namespace-meta
(let* ((namespace-block (and (phpinspect-namespace-is-blocked-p
(phpinspect-meta-token namespace-meta))
(phpinspect-meta-last-child namespace-meta)))
(existing-use (phpinspect-find-first-use namespace-meta)))
(if existing-use
(phpinspect-insert-at-point
(phpinspect-meta-start existing-use) (format "use %s;%c" fqn ?\n))
(if namespace-block
(phpinspect-insert-at-point
(+ 1 (phpinspect-meta-start namespace-block))
(format "%c%cuse %s;%c" ?\n ?\n fqn ?\n))
(phpinspect-insert-at-point
(phpinspect-meta-end
(phpinspect-meta-find-first-child-matching
namespace-meta (phpinspect-meta-wrap-token-pred #'phpinspect-terminator-p)))
(format "%c%cuse %s;%c" ?\n ?\n fqn ?\n)))))
;; else
(let ((existing-use (phpinspect-meta-find-first-child-matching
(phpinspect-buffer-root-meta buffer)
(phpinspect-meta-wrap-token-pred #'phpinspect-use-p))))
(if existing-use
(phpinspect-insert-at-point
(phpinspect-meta-start existing-use)
(format "use %s;%c" fqn ?\n))
(let* ((first-token (phpinspect-meta-first-child (phpinspect-buffer-root-meta buffer)))
token-after)
(when (and (phpinspect-word-p (phpinspect-meta-token first-token))
(string= "declare" (cadr (phpinspect-meta-token first-token))))
(progn
(setq token-after first-token)
(while (and token-after (not (phpinspect-terminator-p
(phpinspect-meta-token token-after))))
(setq token-after (phpinspect-meta-find-right-sibling token-after)))))
(if token-after
(phpinspect-insert-at-point
(phpinspect-meta-end token-after) (format "%c%cuse %s;%c" ?\n ?\n fqn ?\n))
(phpinspect-insert-at-point
(phpinspect-meta-start first-token)
(format "%c%cuse %s;%c%c" ?\n ?\n fqn ?\n ?\n))))))))
(defun phpinspect-add-use-interactive (typename buffer project &optional namespace-token)
(let* ((autoloader (phpinspect-project-autoload project))
(fqns (phpinspect-autoloader-get-type-bag autoloader typename)))
(cond ((= 1 (length fqns))
(phpinspect-add-use (phpinspect-name-string (car fqns)) buffer namespace-token))
((> (length fqns) 1)
(phpinspect-add-use (completing-read "Class: " (phpinspect-names-to-alist fqns))
buffer namespace-token))
(t (phpinspect-message "No import found for type %s" typename)))))
(defun phpinspect-namespace-part-of-typename (typename)
(string-trim-right typename "\\\\?[^\\]+"))
(defalias 'phpinspect-fix-uses-interactive #'phpinspect-fix-imports
"Alias for backwards compatibility")
(defsubst phpinspect-namespace-meta-body (namespace-meta)
"Return the token metadata of NAMESPACE-META's body.
More specifically, returns the token itself if it is a namespace
without block. If the namespace is defined with a block ('{}'),
NAMESPACE-META itself is returned without alterations."
(if (phpinspect-block-p (caddr (phpinspect-meta-token namespace-meta)))
(phpinspect-meta-find-first-child-matching-token namespace-meta #'phpinspect-block-p)
namespace-meta))
(defun phpinspect-find-use-statement-for-import (parent-token import-type)
(phpinspect-meta-find-first-child-matching-token
(if (phpinspect-namespace-p (phpinspect-meta-token parent-token))
(phpinspect-namespace-meta-body parent-token)
parent-token)
(lambda (token)
(and (phpinspect-use-p token)
(eq (car (phpinspect--use-to-type-cons token))
(phpinspect--type-bare-name-sym import-type))))))
(define-inline phpinspect-codify-token-delimiters (token)
(inline-letevals (token)
(inline-quote
(cond
((phpinspect-namespace-p ,token) '("namespace" . ""))
((phpinspect-use-p ,token) '("use" . ""))
((phpinspect-annotation-p ,token) '("@" . ""))
((phpinspect-root-p ,token) '("<?php" . ""))
((phpinspect-token-type-p ,token :string) '("'" . "'"))
((phpinspect-block-p ,token) '("{" . "}"))
((phpinspect-list-p ,token) '("(" . ")"))
((phpinspect-array-p ,token) '("[" . "]"))
((phpinspect-object-attrib-p ,token) '("->" . ""))
((phpinspect-variable-p ,token) '("$" . ""))
((phpinspect-doc-block-p ,token) '("/**" . "**/"))))))
(defun phpinspect-codify-token (token-meta)
"Pretty print TOKEN-META.
Note: this function is very imcomplete and not guaranteed to work
correctly. It might result in deleted code when used in
unforeseen scenario's."
(let ((delimiters (phpinspect-codify-token-delimiters (phpinspect-meta-token token-meta))))
(when delimiters
(insert (car delimiters)))
;; FIXME: if this code is ever to be used for broader cases than formatting
;; use statements, it will have to support a much wider array of token types
;; and should be tested extensively.
(phpinspect-splayt-traverse-lr (child (phpinspect-meta-children token-meta))
(let ((token (phpinspect-meta-token child)))
(cond ((phpinspect-word-p token)
(insert (concat " " (cadr token))))
((phpinspect-terminator-p token)
(insert (cadr token)))
((and (listp token) (keywordp (car token)))
(phpinspect-codify-token child))
(t (error "Unable to determine code format of object %s" token)))))
(when delimiters
(insert (cdr delimiters)))))
(defun phpinspect-codify-token-to-string (token-meta)
(with-temp-buffer
(phpinspect-codify-token token-meta)
(buffer-string)))
(defun phpinspect-remove-unneeded-use-statements (types buffer imports parent-token)
(dolist (import imports)
;; Namespace must be inferred within the loop, see comments in
;; `phpinspect-add-use-statements-for-missing-types' for context.
(let ((namespace (if (phpinspect-namespace-p (phpinspect-meta-token parent-token))
parent-token
(phpinspect-meta-find-parent-matching-token parent-token #'phpinspect-namespace-p))))
(unless (member (car import) types)
(when-let ((use-meta (phpinspect-find-use-statement-for-import namespace (cdr import))))
(let ((start-point (phpinspect-meta-start use-meta))
(use-before (phpinspect-meta-find-left-sibling use-meta)))
(if (phpinspect-use-p (phpinspect-meta-token use-before))
;; left-sibling is another use statement, remove all preceding whitespace
(setq start-point (- start-point (length (phpinspect-meta-whitespace-before use-meta))))
;; left-sibling isn't a use statement, just remove a newline if
;; any whitespace is present
(when (length> (phpinspect-meta-whitespace-before use-meta) 0)
(setq start-point (- start-point 1))))
(delete-region start-point (phpinspect-meta-end use-meta))
(phpinspect-buffer-parse buffer 'no-interrupt)))))))
(defun phpinspect-add-use-statements-for-missing-types (types buffer imports project parent-token)
"Add use statements to BUFFER for TYPES if not already included in IMPORTS.
Uses PROJECT's autoloader to determine available types for import.
PARENT-TOKEN must be a `token-meta' object and is used to
determine the scope of the imports (global or local namespace)."
(dolist (type types)
;; Namespace token must be inferred within the loop, as the ancestors of
;; PARENT-TOKEN may change after a buffer reparse (which happens after each
;; insert)
(let* ((namespace (if (phpinspect-namespace-p (phpinspect-meta-token parent-token))
parent-token
(phpinspect-meta-find-parent-matching-token
parent-token #'phpinspect-namespace-p)))
(namespace-name (if namespace
(phpinspect-namespace-name (phpinspect-meta-token namespace))
"")))
;; Add use statements for types that aren't imported or already referenced
;; with a fully qualified name.
(unless (or (or (alist-get type imports))
(gethash (phpinspect-intern-name
(phpinspect--resolve-type-name
nil namespace-name (phpinspect-name-string type)))
(phpinspect-autoloader-types
(phpinspect-project-autoload project))))
(unless (member (phpinspect-name-string type) phpinspect-native-typenames)
(phpinspect-add-use-interactive type buffer project namespace)
(phpinspect-buffer-parse buffer 'no-interrupt))))))
(defcustom phpinspect-import-sort-comparison #'string<
"The comparison function used to sort use statements."
:type 'function
:group 'phpinspect)
(defun phpinspect-sort-compare-import-strings (str1 str2)
(funcall phpinspect-import-sort-comparison str1 str2))
(defun phpinspect-seq-sorted-p (pred seq)
(let (prev)
(catch 'sorted
(seq-doseq (el seq)
(when prev
(unless (funcall pred prev el)
(throw 'sorted nil)))
(setq prev el))
t)))
(defun phpinspect-format-use-statements (buffer first-meta)
"Format a group of use statements to be sorted in alphabetical
order and have the right amount of whitespace.
BUFFER should be the buffer (`phpinspect-buffer-p') the use
statements are located in.
FIRST-META should be the metadata of the first use token of the
group."
(when first-meta
(let ((statements
(list (cons (phpinspect-use-name-string (phpinspect-meta-token first-meta))
first-meta)))
(start (phpinspect-meta-start first-meta))
(end (phpinspect-meta-end first-meta))
(current first-meta))
(while (and (setq current (phpinspect-meta-find-right-sibling current))
(phpinspect-use-p (phpinspect-meta-token current)))
(setq end (phpinspect-meta-end current))
(push (cons (phpinspect-use-name-string (phpinspect-meta-token current))
current)
statements))
(setq statements (nreverse statements))
(with-current-buffer (phpinspect-buffer-buffer buffer)
(save-excursion
;; Check if use statements are already sorted
(if (phpinspect-seq-sorted-p
(lambda (a b)
(phpinspect-sort-compare-import-strings (car a) (car b)))
statements)
;; statements are sorted, skip to the end of the region
(goto-char end)
;; else: sort and re-insert
(sort statements (lambda (a b) (phpinspect-sort-compare-import-strings (car a) (car b))))
(goto-char start)
(combine-after-change-calls
(delete-region start end)
(dolist (statement statements)
(phpinspect-codify-token (cdr statement))
(insert-char ?\n)))
;; go char backward for accurate whitespace detection in the code
;; below that removes/adds whitespace.
(backward-char))
(if (looking-at "[[:blank:]\n]+")
;; Delete excess trailing whitespace (there's more than 2 between the
;; last use statement and the next token)
(when (< 2 (- (match-end 0) (match-beginning 0)))
(delete-region (match-beginning 0) (match-end 0))
(insert-char ?\n 2))
;; Insert an extra newline (there's only one between the last use
;; statement and the next token)
(insert-char ?\n)))))))
(defun phpinspect-fix-imports ()
"Find types that are used in the current buffer and make sure
that there are import (\"use\") statements for them."
(interactive)
(if phpinspect-current-buffer
(let* ((buffer phpinspect-current-buffer)
;; use buffer-reparse to ensure fully structurally correct
;; tree. (at the time of writing, incremental parsing has some
;; limitations causing reused tokens to lose their special meaning
;; when they are reused. For example the "class" keyword being
;; reused as just a word instead of a special keyword marking the
;; start of a class)
;;
;; FIXME: Change to buffer-parse when this particular problem in
;; incremental parsing has been solved
(tree (phpinspect-buffer-reparse-if-not-fresh buffer))
(index (phpinspect--index-tokens
tree nil (phpinspect-buffer-location-resolver buffer)))
(namespaces (alist-get 'namespaces index))
(imports (alist-get 'imports index))
(project (phpinspect-buffer-project buffer))
(used-types (alist-get 'used-types index))
namespace-tokens)
;; First collect tokens in the buffer via which the namespace tokens can
;; be found. The edits we do to add imports will invalidate the
;; namespace region bounds, making this hard to do during the loop that
;; adds them.
(dolist (namespace namespaces)
(let ((region (alist-get 'location namespace)))
;; Use the first child of the namespace token. The namespace token
;; itself will be tainted and reparsed after every edit (inserted
;; use statement), making its reference useless as it will be in an
;; overlayed tree. The first token in the namespace, however will be
;; adopted into the new tree as long as it isn't altered. This
;; allows a lookup of the new namespace token by getting this
;; token's (newly assigned) parent after every edit.
(push (cons (phpinspect-meta-find-first-child-matching-token
(phpinspect-meta-find-parent-matching-token
(phpinspect-bmap-last-token-before-point
(phpinspect-buffer-map buffer)
(+ (phpinspect-region-start region) 1))
#'phpinspect-namespace-p)
#'phpinspect-word-p)
namespace)
namespace-tokens)))
(phpinspect-add-use-statements-for-missing-types
used-types buffer imports project (phpinspect-buffer-root-meta buffer))
(phpinspect-remove-unneeded-use-statements
used-types buffer imports (phpinspect-buffer-root-meta buffer))
(phpinspect-format-use-statements
buffer (phpinspect-find-first-use (phpinspect-buffer-root-meta buffer)))
(phpinspect-buffer-parse buffer 'no-interrupt)
(dolist (cell namespace-tokens)
(let* ((namespace (cdr cell))
(namespace-name (car namespace))
(token-meta (car cell))
(used-types (alist-get 'used-types namespace))
(namespace-imports (alist-get 'imports namespace)))
(unless token-meta
(error "Unable to find token for namespace %s" namespace-name))
(phpinspect-add-use-statements-for-missing-types
used-types buffer (append imports namespace-imports) project token-meta)
(with-current-buffer (get-buffer-create "attempt-before-namespace-removal-code")
(erase-buffer)
(insert (with-current-buffer (phpinspect-buffer-buffer buffer) (buffer-string))))
(phpinspect-remove-unneeded-use-statements
used-types buffer (append imports namespace-imports) token-meta)
(let ((parent (phpinspect-meta-find-parent-matching-token
token-meta #'phpinspect-namespace-or-root-p)))
(phpinspect-format-use-statements buffer (phpinspect-find-first-use parent))
(phpinspect-buffer-parse buffer 'no-interrupt)))))))
(provide 'phpinspect-imports)