aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
-rw-r--r--modules/host-environment.el35
-rw-r--r--tests/test-host-environment.el68
2 files changed, 98 insertions, 5 deletions
diff --git a/modules/host-environment.el b/modules/host-environment.el
index af9248c2..832cc195 100644
--- a/modules/host-environment.el
+++ b/modules/host-environment.el
@@ -12,12 +12,37 @@
+(defun env--battery-status-char-indicates-battery-p (status)
+ "Return non-nil if STATUS indicates a real battery is present.
+STATUS is the value returned by `battery-format' with the \"%B\" spec.
+A real battery reports \"!\" (critical), \"+\" (charging), or \"-\"
+(discharging). Desktops report \"N/A\", \"unknown\", or empty, which
+all mean no battery."
+ (and (stringp status)
+ (member status '("!" "+" "-"))
+ t))
+
+(defun env--power-supply-has-battery-p (power-supply-dir)
+ "Return non-nil if POWER-SUPPLY-DIR contains a BAT* entry.
+Canonical Linux check. A laptop exposes directories like
+/sys/class/power_supply/BAT0; a desktop exposes only AC adapters
+and USB-C power entries."
+ (and (file-directory-p power-supply-dir)
+ (file-expand-wildcards (expand-file-name "BAT*" power-supply-dir))
+ t))
+
(defun env-laptop-p ()
- "Non-nil if a battery is present."
- (when (and (require 'battery nil 'noerror)
- battery-status-function)
- (not (string= "N/A"
- (battery-format "%B" (funcall battery-status-function))))))
+ "Non-nil if the host has a battery.
+On Linux, checks /sys/class/power_supply for BAT* entries, which is the
+canonical signal. On other platforms, falls back to `battery-format'
+\"%B\" and checks for a live battery status char."
+ (cond
+ ((eq system-type 'gnu/linux)
+ (env--power-supply-has-battery-p "/sys/class/power_supply"))
+ ((and (require 'battery nil 'noerror)
+ battery-status-function)
+ (env--battery-status-char-indicates-battery-p
+ (battery-format "%B" (funcall battery-status-function))))))
(defun env-desktop-p ()
"Return t if host is a laptop (has a battery), nil if not."
diff --git a/tests/test-host-environment.el b/tests/test-host-environment.el
new file mode 100644
index 00000000..9290a32f
--- /dev/null
+++ b/tests/test-host-environment.el
@@ -0,0 +1,68 @@
+;;; test-host-environment.el --- Tests for host-environment.el -*- lexical-binding: t; -*-
+
+;;; Commentary:
+;; Unit tests for `env-laptop-p' and its pure helpers.
+;;
+;; Regression driver: upower on a desktop with only an AC adapter reports
+;; `(battery-format "%B" ...)' as "unknown", not "N/A". The previous
+;; implementation treated any non-"N/A" value as "has a battery", so
+;; desktops were misclassified as laptops.
+
+;;; Code:
+
+(require 'ert)
+
+(add-to-list 'load-path (expand-file-name "modules" user-emacs-directory))
+(require 'host-environment)
+
+;;; env--battery-status-char-indicates-battery-p
+
+(ert-deftest test-host-environment-battery-char-live-codes ()
+ "Normal: \"+\", \"-\", and \"!\" indicate a live battery."
+ (should (env--battery-status-char-indicates-battery-p "+"))
+ (should (env--battery-status-char-indicates-battery-p "-"))
+ (should (env--battery-status-char-indicates-battery-p "!")))
+
+(ert-deftest test-host-environment-battery-char-na-returns-nil ()
+ "Boundary: legacy \"N/A\" sentinel means no battery."
+ (should-not (env--battery-status-char-indicates-battery-p "N/A")))
+
+(ert-deftest test-host-environment-battery-char-unknown-returns-nil ()
+ "Boundary: upower \"unknown\" (desktop with AC adapter) means no battery."
+ (should-not (env--battery-status-char-indicates-battery-p "unknown")))
+
+(ert-deftest test-host-environment-battery-char-empty-returns-nil ()
+ "Boundary: empty string means no battery."
+ (should-not (env--battery-status-char-indicates-battery-p "")))
+
+(ert-deftest test-host-environment-battery-char-nil-returns-nil ()
+ "Error: nil input means no battery."
+ (should-not (env--battery-status-char-indicates-battery-p nil)))
+
+;;; env--power-supply-has-battery-p
+
+(ert-deftest test-host-environment-power-supply-with-bat-dir ()
+ "Normal: a directory with BAT0 subdirectory reports a battery."
+ (let ((dir (make-temp-file "power-supply-test-" t)))
+ (unwind-protect
+ (progn
+ (make-directory (expand-file-name "BAT0" dir))
+ (should (env--power-supply-has-battery-p dir)))
+ (delete-directory dir t))))
+
+(ert-deftest test-host-environment-power-supply-no-bat-dir ()
+ "Boundary: a directory with only AC adapter entries reports no battery."
+ (let ((dir (make-temp-file "power-supply-test-" t)))
+ (unwind-protect
+ (progn
+ (make-directory (expand-file-name "ACAD" dir))
+ (make-directory (expand-file-name "ucsi-source-psy-USBC000:001" dir))
+ (should-not (env--power-supply-has-battery-p dir)))
+ (delete-directory dir t))))
+
+(ert-deftest test-host-environment-power-supply-missing-dir-returns-nil ()
+ "Error: nonexistent power-supply directory is treated as no battery."
+ (should-not (env--power-supply-has-battery-p "/nonexistent/path/for-test-12345")))
+
+(provide 'test-host-environment)
+;;; test-host-environment.el ends here