1
Fork 0
mirror of git://git.sv.gnu.org/emacs.git synced 2025-12-24 06:20:43 -08:00

Document cl-n... set operations consistently

The docstrings of cl-nintersection and cl-nset-difference have been
inconsistent with their manual entries since the beginning of
emacs.git history (bug#76017).  This patch settles on the weaker and
thus backward-compatible requirement that only their first argument
be safe to mutate.

* lisp/emacs-lisp/bytecomp.el: Include only first argument in
mutates-arguments property.
* lisp/emacs-lisp/cl-seq.el (cl-nintersection, cl-nset-difference):
Make docstring consistent with manual in that the second argument is
not modified.
* test/lisp/emacs-lisp/cl-seq-tests.el (cl-nintersection-test)
(cl-nset-difference-test): Simplify.
(cl-nset-difference): Pass fresh list as second argument, otherwise
destructive modifications to it could go undetected.
This commit is contained in:
Basil L. Contovounesios 2025-02-02 18:05:57 +01:00
parent 0edf094e54
commit ac143186c0
3 changed files with 27 additions and 25 deletions

View file

@ -3583,7 +3583,7 @@ This assumes the function has the `important-return-value' property."
(cl-nsubst 3) (cl-nsubst-if 3) (cl-nsubst-if-not 3) (cl-nsubst 3) (cl-nsubst-if 3) (cl-nsubst-if-not 3)
(cl-nsubstitute 3) (cl-nsubstitute-if 3) (cl-nsubstitute-if-not 3) (cl-nsubstitute 3) (cl-nsubstitute-if 3) (cl-nsubstitute-if-not 3)
(cl-nsublis 2) (cl-nsublis 2)
(cl-nunion 1 2) (cl-nintersection 1 2) (cl-nset-difference 1 2) (cl-nunion 1 2) (cl-nintersection 1) (cl-nset-difference 1)
(cl-nset-exclusive-or 1 2) (cl-nset-exclusive-or 1 2)
(cl-nreconc 1) (cl-nreconc 1)
(cl-sort 1) (cl-stable-sort 1) (cl-merge 2 3) (cl-sort 1) (cl-stable-sort 1) (cl-merge 2 3)

View file

@ -864,8 +864,8 @@ to avoid corrupting the original LIST1 and LIST2.
(defun cl-nintersection (cl-list1 cl-list2 &rest cl-keys) (defun cl-nintersection (cl-list1 cl-list2 &rest cl-keys)
"Combine LIST1 and LIST2 using a set-intersection operation. "Combine LIST1 and LIST2 using a set-intersection operation.
The resulting list contains all items that appear in both LIST1 and LIST2. The resulting list contains all items that appear in both LIST1 and LIST2.
This is a destructive function; it reuses the storage of LIST1 and LIST2 This is a destructive function; it reuses the storage of LIST1 (but not
whenever possible. LIST2) whenever possible.
\nKeywords supported: :test :test-not :key \nKeywords supported: :test :test-not :key
\n(fn LIST1 LIST2 [KEYWORD VALUE]...)" \n(fn LIST1 LIST2 [KEYWORD VALUE]...)"
(and cl-list1 cl-list2 (apply 'cl-intersection cl-list1 cl-list2 cl-keys))) (and cl-list1 cl-list2 (apply 'cl-intersection cl-list1 cl-list2 cl-keys)))
@ -894,8 +894,8 @@ to avoid corrupting the original LIST1 and LIST2.
(defun cl-nset-difference (cl-list1 cl-list2 &rest cl-keys) (defun cl-nset-difference (cl-list1 cl-list2 &rest cl-keys)
"Combine LIST1 and LIST2 using a set-difference operation. "Combine LIST1 and LIST2 using a set-difference operation.
The resulting list contains all items that appear in LIST1 but not LIST2. The resulting list contains all items that appear in LIST1 but not LIST2.
This is a destructive function; it reuses the storage of LIST1 and LIST2 This is a destructive function; it reuses the storage of LIST1 (but not
whenever possible. LIST2) whenever possible.
\nKeywords supported: :test :test-not :key \nKeywords supported: :test :test-not :key
\n(fn LIST1 LIST2 [KEYWORD VALUE]...)" \n(fn LIST1 LIST2 [KEYWORD VALUE]...)"
(if (or (null cl-list1) (null cl-list2)) cl-list1 (if (or (null cl-list1) (null cl-list2)) cl-list1

View file

@ -914,18 +914,18 @@ Additionally register an `ert-info' to help identify test failures."
(ert-deftest cl-nintersection-test () (ert-deftest cl-nintersection-test ()
(should-not (cl-nintersection () ())) (should-not (cl-nintersection () ()))
(should-not (cl-nintersection () (list 1 2 3))) (should-not (cl-nintersection () '(1 2 3)))
(should-not (cl-nintersection (list 1 2) (list 3 4))) (should-not (cl-nintersection (list 1 2) '(3 4)))
(should (equal (cl-nintersection (list 1 2 3 4) (list 3 4 5 6)) (should (equal (cl-nintersection (list 1 2 3 4) '(3 4 5 6))
'(4 3))) '(4 3)))
(should (equal (cl-nintersection (list 1 2 3) (list 1 2 3)) (should (equal (cl-nintersection (list 1 2 3) '(1 2 3))
'(1 2 3))) '(1 2 3)))
(should (equal (cl-nintersection (list 1 1 2 2 3) (list 2 2 3 4)) (should (equal (cl-nintersection (list 1 1 2 2 3) '(2 2 3 4))
'(3 2 2))) '(3 2 2)))
(should (equal (cl-nintersection (list 1 (copy-sequence "two") 3) (should (equal (cl-nintersection (list 1 (copy-sequence "two") 3)
(list 3 "two" 4)) '(3 "two" 4))
'(3))) '(3)))
(should (equal (cl-nintersection (list 1 2 3) (list 3 2 1) :test #'equal) (should (equal (cl-nintersection (list 1 2 3) '(3 2 1) :test #'equal)
'(1 2 3)))) '(1 2 3))))
(ert-deftest cl-set-difference-test () (ert-deftest cl-set-difference-test ()
@ -961,47 +961,49 @@ Additionally register an `ert-info' to help identify test failures."
(ert-deftest cl-nset-difference () (ert-deftest cl-nset-difference ()
;; Our nset-difference doesn't preserve order. ;; Our nset-difference doesn't preserve order.
(let* ((l1 (list 1 2 3 4)) (l2 '(3 4 5 6)) (let* ((l1 (list 1 2 3 4)) (l2 (list 3 4 5 6))
(diff (cl-nset-difference l1 l2))) (diff (cl-nset-difference l1 l2)))
(should (memq 1 diff)) (should (memq 1 diff))
(should (memq 2 diff)) (should (memq 2 diff))
(should (= (length diff) 2)) (should (length= diff 2))
(should (equal l2 '(3 4 5 6)))) (should (equal l2 '(3 4 5 6))))
(let* ((l1 (list "1" "2" "3" "4")) (l2 '("3" "4" "5" "6")) (let* ((l1 (list "1" "2" "3" "4")) (l2 (list "3" "4" "5" "6"))
(diff (cl-nset-difference l1 l2 :test #'equal))) (diff (cl-nset-difference l1 l2 :test #'equal)))
(should (member "1" diff)) (should (member "1" diff))
(should (member "2" diff)) (should (member "2" diff))
(should (= (length diff) 2)) (should (length= diff 2))
(should (equal l2 '("3" "4" "5" "6")))) (should (equal l2 '("3" "4" "5" "6"))))
(let* ((l1 (list '(a . 1) '(b . 2) '(c . 3) '(d . 4))) (let* ((l1 (list '(a . 1) '(b . 2) '(c . 3) '(d . 4)))
(l2 (list '(c . 3) '(d . 4) '(e . 5) '(f . 6))) (l2 (list '(c . 3) '(d . 4) '(e . 5) '(f . 6)))
(diff (cl-nset-difference l1 l2 :key #'car))) (diff (cl-nset-difference l1 l2 :key #'car)))
(should (member '(a . 1) diff)) (should (member '(a . 1) diff))
(should (member '(b . 2) diff)) (should (member '(b . 2) diff))
(should (= (length diff) 2))) (should (length= diff 2))
(should (equal l2 '((c . 3) (d . 4) (e . 5) (f . 6)))))
(let* ((l1 (list '("a" . 1) '("b" . 2) '("c" . 3) '("d" . 4))) (let* ((l1 (list '("a" . 1) '("b" . 2) '("c" . 3) '("d" . 4)))
(l2 (list '("c" . 3) '("d" . 4) '("e" . 5) '("f" . 6))) (l2 (list '("c" . 3) '("d" . 4) '("e" . 5) '("f" . 6)))
(diff (cl-nset-difference l1 l2 :key #'car :test #'string=))) (diff (cl-nset-difference l1 l2 :key #'car :test #'string=)))
(should (member '("a" . 1) diff)) (should (member '("a" . 1) diff))
(should (member '("b" . 2) diff)) (should (member '("b" . 2) diff))
(should (= (length diff) 2)))) (should (length= diff 2))
(should (equal l2 '(("c" . 3) ("d" . 4) ("e" . 5) ("f" . 6))))))
(ert-deftest cl-nset-difference-test () (ert-deftest cl-nset-difference-test ()
(should-not (cl-nset-difference () ())) (should-not (cl-nset-difference () ()))
(should-not (cl-nset-difference () (list 1 2 3))) (should-not (cl-nset-difference () (list 1 2 3)))
(should-not (cl-nset-difference (list 1 2 3) (list 1 2 3))) (should-not (cl-nset-difference (list 1 2 3) '(1 2 3)))
(should-not (cl-nset-difference (list 1 2 3) (list 3 2 1) :test #'equal)) (should-not (cl-nset-difference (list 1 2 3) '(3 2 1) :test #'equal))
(should (equal (cl-nset-difference (list 1 2 3) ()) (should (equal (cl-nset-difference (list 1 2 3) ())
'(1 2 3))) '(1 2 3)))
(should (equal (cl-nset-difference (list 1 2 3 4) (list 3 4 5 6)) (should (equal (cl-nset-difference (list 1 2 3 4) '(3 4 5 6))
'(1 2))) '(1 2)))
(should (equal (cl-nset-difference (list 1 1 2 2 3) (list 3 4 5)) (should (equal (cl-nset-difference (list 1 1 2 2 3) '(3 4 5))
'(1 1 2 2))) '(1 1 2 2)))
(should (equal (cl-nset-difference (list 1 2 3) (list 3 2 4)) (should (equal (cl-nset-difference (list 1 2 3) '(3 2 4))
'(1))) '(1)))
(should (equal (cl-nset-difference (list 1 2 3 4 5) (list 3 4 5 6 7)) (should (equal (cl-nset-difference (list 1 2 3 4 5) '(3 4 5 6 7))
'(1 2))) '(1 2)))
(should (equal (cl-nset-difference (list 1 (copy-sequence "a")) (list 1 "a")) (should (equal (cl-nset-difference (list 1 (copy-sequence "a")) '(1 "a"))
'("a")))) '("a"))))
(ert-deftest cl-set-exclusive-or-test () (ert-deftest cl-set-exclusive-or-test ()