]> git.eshelyaron.com Git - emacs.git/commitdiff
Document cl-n... set operations consistently
authorBasil L. Contovounesios <basil@contovou.net>
Sun, 2 Feb 2025 17:05:57 +0000 (18:05 +0100)
committerEshel Yaron <me@eshelyaron.com>
Sat, 15 Feb 2025 19:30:33 +0000 (20:30 +0100)
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.

(cherry picked from commit ac143186c04ffd729cfe11abd99f02abdf742f64)

lisp/emacs-lisp/bytecomp.el
lisp/emacs-lisp/cl-seq.el
test/lisp/emacs-lisp/cl-seq-tests.el

index 7a4e646ec68d7a5f45eb20a5e89c3d70c032cca3..3d90e8e7d2e1e7a97ca5cdb4507c9f0a98bd6c82 100644 (file)
@@ -3603,7 +3603,7 @@ This assumes the function has the `important-return-value' property."
          (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-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-nreconc 1)
          (cl-sort 1) (cl-stable-sort 1) (cl-merge 2 3)
index 1878153f8112e54c3c59c00bb2d29cdb0a395606..5b4337ad9cbb371f7d087701cf5f67276f085646 100644 (file)
@@ -864,8 +864,8 @@ to avoid corrupting the original LIST1 and LIST2.
 (defun cl-nintersection (cl-list1 cl-list2 &rest cl-keys)
   "Combine LIST1 and LIST2 using a set-intersection operation.
 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
-whenever possible.
+This is a destructive function; it reuses the storage of LIST1 (but not
+LIST2) whenever possible.
 \nKeywords supported:  :test :test-not :key
 \n(fn LIST1 LIST2 [KEYWORD VALUE]...)"
   (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)
   "Combine LIST1 and LIST2 using a set-difference operation.
 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
-whenever possible.
+This is a destructive function; it reuses the storage of LIST1 (but not
+LIST2) whenever possible.
 \nKeywords supported:  :test :test-not :key
 \n(fn LIST1 LIST2 [KEYWORD VALUE]...)"
   (if (or (null cl-list1) (null cl-list2)) cl-list1
index 2348a7fc81274d5b058890575eaae1d3588b2393..f72596e4a4bbdaafb8077a5fde0dd6b55dff83be 100644 (file)
@@ -914,18 +914,18 @@ Additionally register an `ert-info' to help identify test failures."
 
 (ert-deftest cl-nintersection-test ()
   (should-not (cl-nintersection () ()))
-  (should-not (cl-nintersection () (list 1 2 3)))
-  (should-not (cl-nintersection (list 1 2) (list 3 4)))
-  (should (equal (cl-nintersection (list 1 2 3 4) (list 3 4 5 6))
+  (should-not (cl-nintersection () '(1 2 3)))
+  (should-not (cl-nintersection (list 1 2) '(3 4)))
+  (should (equal (cl-nintersection (list 1 2 3 4) '(3 4 5 6))
                  '(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)))
-  (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)))
   (should (equal (cl-nintersection (list 1 (copy-sequence "two") 3)
-                                   (list 3 "two" 4))
+                                   '(3 "two" 4))
                  '(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))))
 
 (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 ()
   ;; 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)))
     (should (memq 1 diff))
     (should (memq 2 diff))
-    (should (= (length diff) 2))
+    (should (length= diff 2))
     (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)))
     (should (member "1" diff))
     (should (member "2" diff))
-    (should (= (length diff) 2))
+    (should (length= diff 2))
     (should (equal l2 '("3" "4" "5" "6"))))
   (let* ((l1 (list '(a . 1) '(b . 2) '(c . 3) '(d . 4)))
          (l2 (list '(c . 3) '(d . 4) '(e . 5) '(f . 6)))
          (diff (cl-nset-difference l1 l2 :key #'car)))
     (should (member '(a . 1) 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)))
          (l2 (list '("c" . 3) '("d" . 4) '("e" . 5) '("f" . 6)))
          (diff (cl-nset-difference l1 l2 :key #'car :test #'string=)))
     (should (member '("a" . 1) 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 ()
   (should-not (cl-nset-difference () ()))
   (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) (list 3 2 1) :test #'equal))
+  (should-not (cl-nset-difference (list 1 2 3) '(1 2 3)))
+  (should-not (cl-nset-difference (list 1 2 3) '(3 2 1) :test #'equal))
   (should (equal (cl-nset-difference (list 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)))
-  (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)))
-  (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)))
-  (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)))
-  (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"))))
 
 (ert-deftest cl-set-exclusive-or-test ()