emacs/test/lisp/erc/erc-track-tests.el

414 lines
16 KiB
EmacsLisp
Raw Normal View History

;;; erc-track-tests.el --- Tests for erc-track. -*- lexical-binding:t -*-
2024-01-02 09:47:10 +08:00
;; Copyright (C) 2016-2024 Free Software Foundation, Inc.
;; Author: Mario Lang <mlang@delysid.org>
2016-04-21 14:44:31 -07:00
;; Author: Vivek Dasmohapatra <vivek@etla.org>
;; This file is part of GNU Emacs.
;; GNU Emacs 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.
;; GNU Emacs 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 GNU Emacs. If not, see <https://www.gnu.org/licenses/>.
;;; Code:
(require 'erc-track)
(require 'ert-x)
(eval-and-compile
(let ((load-path (cons (ert-resource-directory) load-path)))
(require 'erc-tests-common)))
(ert-deftest erc-track--shorten-aggressive-nil ()
"Test non-aggressive erc track buffer name shortening."
(let (erc-track-shorten-aggressively)
(should
(equal (erc-unique-channel-names '("#emacs" "#vi" "#electronica" "#folk")
'("#emacs" "#vi"))
'("#em" "#vi")))
(should
(equal (erc-unique-channel-names '("#linux-de" "#linux-fr")
'("#linux-de" "#linux-fr"))
'("#linux-de" "#linux-fr")))
(should
(equal (erc-unique-channel-names
'("#dunnet" "#lisp" "#sawfish" "#fsf" "#guile" "#testgnome"
"#gnu" "#fsbot" "#hurd" "#hurd-bunny" "#emacs")
'("#hurd-bunny" "#hurd" "#sawfish" "#lisp"))
'("#hurd-" "#hurd" "#s" "#l")))
(should
(equal (erc-unique-substrings '("#emacs" "#vi" "#electronica" "#folk"))
'("#em" "#vi" "#el" "#f")))
(should
(equal (erc-unique-channel-names
'("#emacs" "#burse" "+linux.de" "#starwars"
"#bitlbee" "+burse" "#ratpoison")
'("+linux.de" "#starwars" "#burse"))
'("+l" "#s" "#bu")))
(should
(equal (erc-unique-channel-names '("fsbot" "#emacs" "deego") '("fsbot"))
'("fs")))
(should
(equal (erc-unique-channel-names '("fsbot" "#emacs" "deego")
'("fsbot")
(lambda (s) (> (length s) 4)) 1)
'("f")))
(should
(equal (erc-unique-channel-names '("fsbot" "#emacs" "deego")
'("fsbot")
(lambda (s) (> (length s) 4)) 2)
'("fs")))
(should
(equal (erc-unique-channel-names '("deego" "#hurd" "#hurd-bunny" "#emacs")
'("#hurd" "#hurd-bunny"))
'("#hurd" "#hurd-")))
(should
(and
(equal (erc-unique-substring-1 "abc" '("ab" "abcd")) "abcd")
(not (erc-unique-substring-1 "a" '("xyz" "xab")))
(equal (erc-unique-substrings '("abc" "xyz" "xab")) '("abc" "xyz" "xab"))
(equal (erc-unique-substrings '("abc" "abcdefg")) '("abc" "abcd")))) ))
(ert-deftest erc-track--shorten-aggressive-t ()
"Test aggressive erc track buffer name shortening."
(let ((erc-track-shorten-aggressively t))
(should
(equal (erc-unique-channel-names '("#emacs" "#vi" "#electronica" "#folk")
'("#emacs" "#vi"))
'("#em" "#v")))
(should
(equal (erc-unique-channel-names '("#linux-de" "#linux-fr")
'("#linux-de" "#linux-fr"))
'("#linux-d" "#linux-f")))
(should
(equal (erc-unique-substrings '("#emacs" "#vi" "#electronica" "#folk"))
'("#em" "#v" "#el" "#f")))
(should
(and
(equal (erc-unique-substring-1 "abc" '("ab" "abcd")) "abcd")
(not (erc-unique-substring-1 "a" '("xyz" "xab")))
(equal (erc-unique-substrings '("abc" "xyz" "xab")) '("ab" "xy" "xa"))
(equal (erc-unique-substrings '("abc" "abcdefg")) '("abc" "abcd")))) ))
(ert-deftest erc-track--shorten-aggressive-max ()
"Test maximally aggressive erc track buffer name shortening."
(let ((erc-track-shorten-aggressively 'max))
(should
(equal (erc-unique-channel-names '("#emacs" "#vi" "#electronica" "#folk")
'("#emacs" "#vi"))
'("#e" "#v"))) ))
(ert-deftest erc-track--shortened-names ()
(let (erc-track--shortened-names
erc-track--shortened-names-current-hash
results)
(with-memoization (erc-track--shortened-names-get
'("apple" "banana" "cherries"))
'("a" "b" "c"))
(should (integerp (car erc-track--shortened-names)))
(should (equal (cdr erc-track--shortened-names) '("a" "b" "c")))
(push erc-track--shortened-names results)
;; Redundant call doesn't run.
(with-memoization (erc-track--shortened-names-get
'("apple" "banana" "cherries"))
(should-not 'run)
'("a" "b" "c"))
(should (equal erc-track--shortened-names (car results)))
;; Change in environment or context forces run.
(with-temp-buffer
(with-memoization (erc-track--shortened-names-get
'("apple" "banana" "cherries"))
'("x" "y" "z")))
(should (and (integerp (car erc-track--shortened-names))
(/= (car erc-track--shortened-names) (caar results))))
(should (equal (cdr erc-track--shortened-names) '("x" "y" "z")))
(push erc-track--shortened-names results)
(with-memoization (erc-track--shortened-names-get
'("apple" "banana" "cherries"))
'("1" "2" "3"))
(should (and (integerp (car erc-track--shortened-names))
(/= (car erc-track--shortened-names) (caar results))))
(should (equal (cdr erc-track--shortened-names) '("1" "2" "3")))))
(ert-deftest erc-track--erc-faces-in ()
"`erc-faces-in' should pick up both 'face and 'font-lock-face properties."
(let ((str0 (copy-sequence "is bold"))
(str1 (copy-sequence "is bold")))
;; Turn on Font Lock mode: this initialize `char-property-alias-alist'
;; to '((face font-lock-face)). Note that `font-lock-mode' don't
;; turn on the mode if the test is run on batch mode or if the
;; buffer name starts with ?\s (Bug#23954).
(unless font-lock-mode (font-lock-default-function 1))
(put-text-property 3 (length str0) 'font-lock-face
'(bold erc-current-nick-face) str0)
(put-text-property 3 (length str1) 'face
'(bold erc-current-nick-face) str1)
(should (erc-faces-in str0))
(should (erc-faces-in str1)) ))
Promote "normal" faces in erc-track * etc/ERC-NEWS: Add entry for new behavior involving the option `erc-track-faces-normal-list'. * lisp/erc/erc-button.el (erc-button-nick-default-face): New face to serve as default for `erc-button-nickname-face'. (erc-button-nickname-face): Change default value to `erc-button-nick-default-face'. * lisp/erc/erc-track.el (erc-track--massage-nick-button-faces): New function to serve as Custom :set function for priority and "normal" face-list options. (erc-track-faces-normal-list): Fix Custom :type by loading `erc-button' during validation so Customize chooses the correct UI instead of a generic form field with "(mismatch)" printed alongside the "STATE" button. (erc-track-faces-priority-list, erc-track-faces-normal-list): Remove values for "buttonized" `match' module faces that, if retained, would need updating to feature `erc-button-nick-default-face' instead of `erc-nick-default-face'. However, as noted in the NEWS entry, this ordering of button face atop match face is not possible. Use :set function to massage saved user values. (erc-track-ignore-normal-contenders-p): New compatibility switch to access pre-5.6 behavior, in which faces in `erc-track-faces-normal-list' were only considered for promotion to the mode line if the current face occupying that pole position wasn't present. (erc-track-mode, erc-track-enable, erc-track-disable): Add comments regarding perceived futility of hooking on `erc-server-001-functions' and likely unneeded hook removal. Run common buffer-local setup and teardown. (erc-track--normal-faces): New local variable, a snapshot of `erc-track-faces-normal-list'. (erc-track--setup): New function to stash `erc-track-faces-normal-list' on init. (erc-track-select-mode-line-face): Offer alternate explanation of certain particulars in doc string. (erc-track--alt-normals-function): New function-valued variable to allow other modules to intervene in deciding whether to pursue and promote a "normal" contending face. (erc-track--select-mode-line-face): New function similar to its public namesake except that it considers other viable candidates among the "normal" alternatives. (erc-track-modified-channels): Only run face selection portion when faces are actually found. Use `erc-track--select-mode-line-face' instead of `erc-track-select-mode-line-face'. * test/lisp/erc/erc-track-tests.el (erc-track-select-mode-line-face): New test. (erc-track-tests--select-mode-line-face): New fixture function. (erc-track--select-mode-line-face): New test. (Bug#67767)
2023-12-10 05:33:48 -08:00
;; This simulates an alternating bold/non-bold [#c] in the mode-line,
;; i.e., an `erc-modified-channels-alist' that vacillates between
;;
;; ((#<buffer #chan> 42 . erc-default-face))
;;
;; and
;;
;; ((#<buffer #chan> 42 erc-nick-default-face erc-default-face))
;;
;; This is a fairly typical scenario where consecutive messages
;; feature speaker and addressee button highlighting and otherwise
;; plain message bodies. This mapping of phony to real faces
;; describes the picture in 5.6:
;;
;; `1': (erc-button erc-default-face) ; URL
;; `2': (erc-nick-default-face erc-default-face) ; mention
;; `3': erc-default-face ; body
;; `_': (erc-nick-default-face erc-nick-default-face) ; speaker
;;
;; The `_' represents a commonly occurring face (a <speaker>) that's
;; not present in either option's default (standard) value. It's a
;; no-op from the POV of `erc-track-select-mode-line-face'.
(ert-deftest erc-track-select-mode-line-face ()
;; Observed (see key above).
(let ((erc-track-faces-priority-list '(1 2 3))
(erc-track-faces-normal-list '(1 2 3)))
(should (equal 2 (erc-track-select-mode-line-face 3 '(2 _ 3))))
(should (equal 2 (erc-track-select-mode-line-face 2 '(2 _ 3))))
(should (equal 3 (erc-track-select-mode-line-face 2 '(_ 3))))
(should (equal 2 (erc-track-select-mode-line-face 3 '(2 3))))
(should (equal 3 (erc-track-select-mode-line-face 2 '(3))))
(should (equal 1 (erc-track-select-mode-line-face 1 '(2 1 3))))
(should (equal 1 (erc-track-select-mode-line-face 1 '(1 3))))
(should (equal 1 (erc-track-select-mode-line-face 1 '(1 3 2))))
(should (equal 1 (erc-track-select-mode-line-face 1 '(3 1)))))
;; When the current face outranks all new faces and doesn't appear
;; among them, it's eligible to be replaced with a fellow "normal"
;; from those new faces. But if it does appear among them, it's
;; never replaced.
(let ((erc-track-faces-priority-list '(a b))
(erc-track-faces-normal-list '(a b)))
(should (equal 'a (erc-track-select-mode-line-face 'a '(b a))))
(should (equal 'a (erc-track-select-mode-line-face 'a '(a b))))
(should (equal 'a (erc-track-select-mode-line-face 'b '(b a))))
(should (equal 'a (erc-track-select-mode-line-face 'b '(a b))))
(should (equal 'a (erc-track-select-mode-line-face 'b '(a))))
(should (equal 'b (erc-track-select-mode-line-face 'a '(b)))))
;; The ordering of the "normal" list doesn't matter.
(let ((erc-track-faces-priority-list '(a b))
(erc-track-faces-normal-list '(b a)))
(should (equal 'a (erc-track-select-mode-line-face 'a '(b a))))
(should (equal 'a (erc-track-select-mode-line-face 'a '(a b))))
(should (equal 'a (erc-track-select-mode-line-face 'b '(b a))))
(should (equal 'a (erc-track-select-mode-line-face 'b '(a b))))))
(defun erc-track-tests--select-mode-line-face (ranked normals cases)
(setq normals (map-into (mapcar (lambda (f) (cons f t)) normals)
'(hash-table :test equal)))
(pcase-dolist (`(,want ,cur-face ,new-faces) cases)
(ert-info ((format "Observed: {cur: %S, new: %S, want: %S}"
cur-face new-faces want))
(setq new-faces (cons (map-into
(mapcar (lambda (f) (cons f t)) new-faces)
'(hash-table :test equal))
(reverse new-faces)))
(should (equal want (funcall #'erc-track--select-mode-line-face
cur-face new-faces ranked normals))))))
;; The main difference between these variants is that with the above,
;; when given alternating lines like
;;
;; CUR NEW CHOICE
;; text (mention $speaker text) => mention
;; mention ($speaker text) => text
;;
;; we see the effect of alternating faces in the indicator. But when
;; given consecutive lines with a similar composition, like
;;
;; text (mention $speaker text) => mention
;; text (mention $speaker text) => mention
;;
;; we lose the effect. With the variant below, we get
;;
;; text (mention $speaker text) => mention
;; text (mention $speaker text) => text
;;
(ert-deftest erc-track--select-mode-line-face ()
(should-not erc-track-ignore-normal-contenders-p)
;; These are the same test cases from the previous test. The syntax
;; is (expected cur-face new-faces).
(erc-track-tests--select-mode-line-face
'(1 2 3) '(1 2 3)
'((2 3 (2 _ 3))
(3 2 (2 _ 3))
(3 2 (_ 3))
(2 3 (2 3))
(3 2 (3))
(2 1 (2 1 3))
(3 1 (1 3))
(2 1 (1 3 2))
(3 1 (3 1))))
(erc-track-tests--select-mode-line-face
'(a b) '(a b)
'((b a (b a))
(b a (a b))
(a b (b a))
(a b (a b))
(a b (a))
(b a (b))))
(erc-track-tests--select-mode-line-face
'(a b) '(b a)
'((b a (b a))
(b a (a b))
(a b (b a))
(a b (a b)))))
(ert-deftest erc-track--collect-faces-in ()
(with-current-buffer (get-buffer-create "*erc-track--get-faces-in*")
(erc-tests-common-prep-for-insertion)
(goto-char (point-min))
(skip-chars-forward "\n")
(let ((ts #("[04:37]"
0 1 ( erc--msg 0 field erc-timestamp
font-lock-face erc-timestamp-face)
1 7 ( field erc-timestamp
font-lock-face erc-timestamp-face)))
bounds)
(with-silent-modifications
(push (list (point)) bounds)
(insert ; JOIN
ts " " ; iniital `fill' indentation lacks properties
#("*** You have joined channel #chan" 0 33
(font-lock-face erc-notice-face))
"\n")
(setcdr (car bounds) (point))
(push (list (point)) bounds)
(insert ; 353
ts " "
#("*** Users on #chan: bob alice dummy tester"
0 30 (font-lock-face erc-notice-face)
30 35 (font-lock-face erc-current-nick-face)
35 42 (font-lock-face erc-notice-face))
"\n" #(" @fsbot" ; but intervening HAS properties
0 23 (font-lock-face erc-notice-face)))
(setcdr (car bounds) (point))
(push (list (point)) bounds)
(insert ; PRIVMSG
"\n" ts " "
#("<alice> bob: Thou canst not come to me: I come to"
0 1 (font-lock-face erc-default-face)
;; erc-dangerous-host-face -> erc-nicks-alice-face (undefined)
1 6 (font-lock-face (erc-dangerous-host-face erc-nick-default-face))
6 8 (font-lock-face erc-default-face)
;; erc-pal-face -> erc-nicks-bob-face (undefined)
8 11 (font-lock-face (erc-pal-face erc-default-face))
11 49 (font-lock-face erc-default-face))
"\n" #(" thee."
0 22 (font-lock-face erc-default-face))
"\n")
(setcdr (car bounds) (point)))
(goto-char (point-max))
(should (equal (setq bounds (nreverse bounds))
'((3 . 50) (50 . 129) (129 . 212))))
;; For these result assertions, the insertion order of the table
;; elements should mirror that of the consed lists.
;; Baseline
(narrow-to-region 1 3)
(let ((result (erc-track--collect-faces-in)))
(should-not (map-pairs (car result)))
(should-not (cdr result)))
;; JOIN
(narrow-to-region (car (nth 0 bounds)) (cdr (nth 0 bounds)))
(let ((result (erc-track--collect-faces-in)))
(should (seq-set-equal-p
(map-pairs (car result)) '((erc-timestamp-face . t)
(erc-notice-face . t))))
(should (equal (cdr result) '(erc-notice-face erc-timestamp-face))))
;; 353
(narrow-to-region (car (nth 1 bounds)) (cdr (nth 1 bounds)))
(let ((result (erc-track--collect-faces-in)))
(should (seq-set-equal-p (map-pairs (car result))
'((erc-timestamp-face . t)
(erc-notice-face . t)
(erc-current-nick-face . t))))
(should (equal (cdr result) '(erc-current-nick-face
erc-notice-face
erc-timestamp-face))))
;; PRIVMSG
(narrow-to-region (car (nth 2 bounds)) (cdr (nth 2 bounds)))
(let ((result (erc-track--collect-faces-in)))
(should (seq-set-equal-p
(map-pairs (car result))
'((erc-timestamp-face . t)
(erc-default-face . t)
((erc-dangerous-host-face erc-nick-default-face) . t)
((erc-pal-face erc-default-face) . t))))
(should (equal (cdr result)
'((erc-pal-face erc-default-face)
(erc-dangerous-host-face erc-nick-default-face)
erc-default-face
erc-timestamp-face))))
;; Entire buffer.
(narrow-to-region (car (nth 0 bounds)) erc-insert-marker)
(let ((result (erc-track--collect-faces-in)))
(should (seq-set-equal-p
(map-pairs (car result))
'((erc-timestamp-face . t)
(erc-notice-face . t)
(erc-current-nick-face . t)
(erc-default-face . t)
((erc-dangerous-host-face erc-nick-default-face) . t)
((erc-pal-face erc-default-face) . t))))
(should (equal (cdr result)
'((erc-pal-face erc-default-face)
(erc-dangerous-host-face erc-nick-default-face)
erc-default-face
erc-current-nick-face
erc-notice-face
erc-timestamp-face)))))
(widen)
(when noninteractive
(kill-buffer))))
;;; erc-track-tests.el ends here