LSST Applications g00d0e8bbd7+edbf708997,g03191d30f7+9ce8016dbd,g1955dfad08+0bd186d245,g199a45376c+5137f08352,g1fd858c14a+a888a50aa2,g262e1987ae+45f9aba685,g29ae962dfc+1c7d47a24f,g2cef7863aa+73c82f25e4,g35bb328faa+edbf708997,g3fd5ace14f+eed17d2c67,g47891489e3+6dc8069a4c,g53246c7159+edbf708997,g64539dfbff+c4107e45b5,g67b6fd64d1+6dc8069a4c,g74acd417e5+f452e9c21a,g786e29fd12+af89c03590,g7ae74a0b1c+a25e60b391,g7aefaa3e3d+2025e9ce17,g7cc15d900a+2d158402f9,g87389fa792+a4172ec7da,g89139ef638+6dc8069a4c,g8d4809ba88+c4107e45b5,g8d7436a09f+e96c132b44,g8ea07a8fe4+db21c37724,g98df359435+aae6d409c1,ga2180abaac+edbf708997,gac66b60396+966efe6077,gb632fb1845+88945a90f8,gbaa8f7a6c5+38b34f4976,gbf99507273+edbf708997,gca7fc764a6+6dc8069a4c,gd7ef33dd92+6dc8069a4c,gda68eeecaf+7d1e613a8d,gdab6d2f7ff+f452e9c21a,gdbb4c4dda9+c4107e45b5,ge410e46f29+6dc8069a4c,ge41e95a9f2+c4107e45b5,geaed405ab2+e194be0d2b,w.2025.47
LSST Data Management Base Package
Loading...
Searching...
No Matches
utils.py
Go to the documentation of this file.
1#!/usr/bin/env python
2#
3# LSST Data Management System
4#
5# Copyright 2016 AURA/LSST.
6#
7# This product includes software developed by the
8# LSST Project (http://www.lsst.org/).
9#
10# This program is free software: you can redistribute it and/or modify
11# it under the terms of the GNU General Public License as published by
12# the Free Software Foundation, either version 3 of the License, or
13# (at your option) any later version.
14#
15# This program is distributed in the hope that it will be useful,
16# but WITHOUT ANY WARRANTY; without even the implied warranty of
17# MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
18# GNU General Public License for more details.
19#
20# You should have received a copy of the LSST License Statement and
21# the GNU General Public License along with this program. If not,
22# see <https://www.lsstcorp.org/LegalNotices/>.
23#
24
25__all__ = [
26 "traceSetAt",
27 "temporaryLogLevel",
28 "LogRedirect",
29 "enable_notebook_logging",
30 "disable_notebook_logging",
31]
32
33from contextlib import contextmanager
34import os
35import sys
36import threading
37
38from lsst.log import Log
39
40
41def traceSetAt(name, number):
42 """Adjusts logging level to display messages with the trace number being
43 less than or equal to the provided value.
44
45 Parameters
46 ----------
47 name : `str`
48 Name of the logger.
49 number : `int`
50 The trace number threshold for display.
51 """
52 for i in range(6):
53 level = Log.INFO if i > number else Log.DEBUG
54 Log.getLogger('TRACE%d.%s' % (i, name)).setLevel(level)
55
56
57@contextmanager
58def temporaryLogLevel(name, level):
59 """A context manager that temporarily sets the level of a `Log`.
60
61 Parameters
62 ----------
63 name : `str`
64 Name of the log to modify.
65 level : `int`
66 Integer enumeration constant indicating the temporary log level.
67 """
68 log = Log.getLogger(name)
69 old = log.getLevel()
70 log.setLevel(level)
71 try:
72 yield
73 finally:
74 log.setLevel(old)
75
76
78 """Redirect a logging file descriptor to a Python stream.
79
80 Parameters
81 ----------
82 fd : `int`
83 File descriptor number, usually 1 for standard out, the default log
84 output location.
85 dest : `io.TextIOBase`
86 Destination text stream, often `sys.stderr` for ipython or Jupyter
87 notebooks.
88 encoding : `str`
89 Text encoding of the data written to fd.
90 errors : `str`
91 Encoding error handling.
92
93 Notes
94 -----
95 Inspired by `this Stack Overflow answer
96 <https://stackoverflow.com/questions/41216215>`_
97 """
98
99 def __init__(self, fd=1, dest=sys.stderr, encoding="utf-8", errors="strict"):
100 self._fd = fd
101 self._dest = dest
102 # Save original filehandle so we can restore it later.
103 self._filehandle = os.dup(fd)
104
105 # Redirect `fd` to the write end of the pipe.
106 pipe_read, pipe_write = os.pipe()
107 os.dup2(pipe_write, fd)
108 os.close(pipe_write)
109
110 # This thread reads from the read end of the pipe.
111 def consumer_thread(f, data):
112 while True:
113 buf = os.read(f, 1024)
114 if not buf:
115 break
116 data.write(buf.decode(encoding, errors))
117 os.close(f)
118 return
119
120 # Spawn consumer thread with the desired destination stream.
121 self._thread = threading.Thread(target=consumer_thread, args=(pipe_read, dest))
122 self._thread.start()
123
124 def finish(self):
125 """Stop redirecting output.
126 """
127
128 # Cleanup: flush streams, restore `fd`
129 self._dest.flush()
130 # This dup2 closes the saved file descriptor, which is now the write
131 # end of the pipe, causing the thread's read to terminate
132 os.dup2(self._filehandle, self._fd)
133 os.close(self._filehandle)
134 self._thread.join()
135
136
137_redirect = None
138
139
140def enable_notebook_logging(dest=sys.stderr):
141 """Enable notebook output for log4cxx messages."""
142 global _redirect
143 if _redirect is None:
144 _redirect = LogRedirect(dest=dest)
145
146
148 """Stop notebook output for log4cxx messages."""
149 global _redirect
150 if _redirect is not None:
151 _redirect.finish()
152 _redirect = None
__init__(self, fd=1, dest=sys.stderr, encoding="utf-8", errors="strict")
Definition utils.py:99
setLevel(loggername, level)
disable_notebook_logging()
Definition utils.py:147
enable_notebook_logging(dest=sys.stderr)
Definition utils.py:140
traceSetAt(name, number)
Definition utils.py:41
temporaryLogLevel(name, level)
Definition utils.py:58