Skip to content

Commit 0e3f6c4

Browse files
committed
fix #2238 if cwd() cannot be determined always return "" instead of None
1 parent b070015 commit 0e3f6c4

File tree

9 files changed

+34
-27
lines changed

9 files changed

+34
-27
lines changed

HISTORY.rst

+6-2
Original file line numberDiff line numberDiff line change
@@ -15,6 +15,10 @@
1515
empty string)
1616
- The function is faster since it no longer iterates over all processes.
1717
- No longer produces duplicate connection entries.
18+
- 2238_: there are cases where `Process.cwd()`_ cannot be determined
19+
(e.g. directory no longer exists), in which case we returned either ``None``
20+
or an empty string. This was consolidated and we now return ``""`` on all
21+
platforms.
1822

1923
**Bug fixes**
2024

@@ -39,8 +43,8 @@
3943
*used* are too high. We now match values shown by *htop* CLI utility.
4044
- 2236_, [NetBSD]: `Process.num_threads()`_ and `Process.threads()`_ return
4145
threads that are already terminated.
42-
- 2237_, [OpenBSD]: `Process.cwd()`_ may raise ``FileNotFoundError`` if cwd no
43-
longer exists. Return ``None`` instead.
46+
- 2237_, [OpenBSD], [NetBSD]: `Process.cwd()`_ may raise ``FileNotFoundError``
47+
if cwd no longer exists. Return an empty string instead.
4448

4549
5.9.4
4650
=====

docs/index.rst

+7-4
Original file line numberDiff line numberDiff line change
@@ -1180,9 +1180,10 @@ Process class
11801180

11811181
.. method:: exe()
11821182

1183-
The process executable as an absolute path.
1184-
On some systems this may also be an empty string.
1185-
The return value is cached after first call.
1183+
The process executable as an absolute path. On some systems, if exe cannot
1184+
be determined for some internal reason (e.g. system process or path no
1185+
longer exists), this may be an empty string. The return value is cached
1186+
after first call.
11861187

11871188
>>> import psutil
11881189
>>> psutil.Process().exe()
@@ -1281,7 +1282,9 @@ Process class
12811282

12821283
.. method:: cwd()
12831284

1284-
The process current working directory as an absolute path.
1285+
The process current working directory as an absolute path. If cwd cannot be
1286+
determined for some internal reason (e.g. system process or directiory no
1287+
longer exists) it may return an empty string.
12851288

12861289
.. versionchanged:: 5.6.4 added support for NetBSD
12871290

psutil/_psaix.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -487,7 +487,7 @@ def cwd(self):
487487
return result.rstrip('/')
488488
except FileNotFoundError:
489489
os.stat("%s/%s" % (procfs_path, self.pid)) # raise NSP or AD
490-
return None
490+
return ""
491491

492492
@wrap_exceptions
493493
def memory_info(self):

psutil/_psbsd.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -839,7 +839,7 @@ def cwd(self):
839839
elif NETBSD or HAS_PROC_OPEN_FILES:
840840
# FreeBSD < 8 does not support functions based on
841841
# kinfo_getfile() and kinfo_getvmmap()
842-
return cext.proc_cwd(self.pid) or None
842+
return cext.proc_cwd(self.pid) or ""
843843
else:
844844
raise NotImplementedError(
845845
"supported only starting from FreeBSD 8" if

psutil/_pssunos.py

+1-1
Original file line numberDiff line numberDiff line change
@@ -542,7 +542,7 @@ def cwd(self):
542542
return os.readlink("%s/%s/path/cwd" % (procfs_path, self.pid))
543543
except FileNotFoundError:
544544
os.stat("%s/%s" % (procfs_path, self.pid)) # raise NSP or AD
545-
return None
545+
return ""
546546

547547
@wrap_exceptions
548548
def memory_info(self):

psutil/arch/netbsd/proc.c

+6-3
Original file line numberDiff line numberDiff line change
@@ -115,10 +115,13 @@ psutil_proc_cwd(PyObject *self, PyObject *args) {
115115
ssize_t len = readlink(buf, path, sizeof(path) - 1);
116116
free(buf);
117117
if (len == -1) {
118-
if (errno == ENOENT)
119-
NoSuchProcess("readlink -> ENOENT");
120-
else
118+
if (errno == ENOENT) {
119+
psutil_debug("sysctl(KERN_PROC_CWD) -> ENOENT converted to ''");
120+
return Py_BuildValue("", "");
121+
}
122+
else {
121123
PyErr_SetFromErrno(PyExc_OSError);
124+
}
122125
return NULL;
123126
}
124127
path[len] = '\0';

psutil/arch/openbsd/proc.c

+2-2
Original file line numberDiff line numberDiff line change
@@ -306,8 +306,8 @@ psutil_proc_cwd(PyObject *self, PyObject *args) {
306306
int name[] = { CTL_KERN, KERN_PROC_CWD, pid };
307307
if (sysctl(name, 3, path, &pathlen, NULL, 0) != 0) {
308308
if (errno == ENOENT) {
309-
psutil_debug("sysctl(KERN_PROC_CWD) -> ENOENT converted to None");
310-
Py_RETURN_NONE; // mimic os.cpu_count()
309+
psutil_debug("sysctl(KERN_PROC_CWD) -> ENOENT converted to ''");
310+
return Py_BuildValue("", "");
311311
}
312312
else {
313313
PyErr_SetFromErrno(PyExc_OSError);

psutil/tests/test_contracts.py

+9-7
Original file line numberDiff line numberDiff line change
@@ -452,10 +452,9 @@ def cmdline(self, ret, info):
452452
self.assertIsInstance(part, str)
453453

454454
def exe(self, ret, info):
455-
self.assertIsInstance(ret, (str, unicode, type(None)))
456-
if not ret:
457-
self.assertEqual(ret, '')
458-
else:
455+
self.assertIsInstance(ret, (str, unicode))
456+
self.assertEqual(ret.strip(), ret)
457+
if ret:
459458
if WINDOWS and not ret.endswith('.exe'):
460459
return # May be "Registry", "MemCompression", ...
461460
assert os.path.isabs(ret), ret
@@ -520,7 +519,8 @@ def gids(self, ret, info):
520519

521520
def username(self, ret, info):
522521
self.assertIsInstance(ret, str)
523-
assert ret
522+
self.assertEqual(ret.strip(), ret)
523+
assert ret.strip()
524524

525525
def status(self, ret, info):
526526
self.assertIsInstance(ret, str)
@@ -619,6 +619,7 @@ def open_files(self, ret, info):
619619
for f in ret:
620620
self.assertIsInstance(f.fd, int)
621621
self.assertIsInstance(f.path, str)
622+
self.assertEqual(f.path.strip(), f.path)
622623
if WINDOWS:
623624
self.assertEqual(f.fd, -1)
624625
elif LINUX:
@@ -651,8 +652,9 @@ def connections(self, ret, info):
651652
check_connection_ntuple(conn)
652653

653654
def cwd(self, ret, info):
654-
if ret: # 'ret' can be None or empty
655-
self.assertIsInstance(ret, str)
655+
self.assertIsInstance(ret, (str, unicode))
656+
self.assertEqual(ret.strip(), ret)
657+
if ret:
656658
assert os.path.isabs(ret), ret
657659
try:
658660
st = os.stat(ret)

psutil/tests/test_posix.py

+1-6
Original file line numberDiff line numberDiff line change
@@ -351,12 +351,7 @@ def test_users(self):
351351
self.assertEqual(u.name, users[idx])
352352
self.assertEqual(u.terminal, terminals[idx])
353353
if u.pid is not None: # None on OpenBSD
354-
p = psutil.Process(u.pid)
355-
# on macOS time is off by ~47 secs for some reason, but
356-
# the next test against 'who' CLI succeeds
357-
delta = 60 if MACOS else 1
358-
self.assertAlmostEqual(
359-
u.started, p.create_time(), delta=delta)
354+
psutil.Process(u.pid)
360355

361356
@retry_on_failure()
362357
def test_users_started(self):

0 commit comments

Comments
 (0)