Python Forum
How to share a configured logger between running processes?
Thread Rating:
  • 0 Vote(s) - 0 Average
  • 1
  • 2
  • 3
  • 4
  • 5
How to share a configured logger between running processes?
#1
I'm trying to build a small GUI app with a logger that will, at some point, be doing some time consuming manipulation of multiple sets of data (up to several hundred). Naturally, I wanted to use multiprocessing to help speed things up a bit. I was following the example given in the Logging Cookbook (second example) in the Python docs and trying to figure out how to work it into my code. In this stripped down minimal example, clicking the build button should simply log a few messages. The problem, which is hopefully obvious to someone more learned in the topics at hand, is that it doesn't work as intended. The application only prints 3 of the 5 messages to the console and exactly zero are added to the log file.

What I was expecting, obviously, was that all 5 messages would be logged with the logger instance created in gui.py.

I've tried merging methods, moving methods out of the class into module level functions, creating the Queue/loggers in different places, and passing the first logger instance around as an argument. Everything I've tried up to this point either leads to the same results or throws a pickling error and eventually ends in an EOFError. The code given is just the most recent revision that doesn't throw an exception.

I'm just trying to get some direction on "where" I'm messing up. In case it matters, this is on Windows 10 using Python 3.12.

# gui.py

from multiprocessing import Queue
import tkinter as tk
from tkinter import ttk

import builder
import logger

class Gui(tk.Tk):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        btn = ttk.Button(self, text='Build', command=lambda: builder.Build(q))
        btn.pack()
        log.configure(q)
        self.mainloop()

if __name__ == '__main__':
    q = Queue()
    log = logger.Logger()
    Gui()
# logger.py

import logging
import logging.handlers

class Logger:
    def __init__(self):
        self.logger = logging.getLogger('dcre')
        self.logger.setLevel('DEBUG')
        self.log = self.logger.log

    def configure(self, q):
        self.q = q
        self.qh = logging.handlers.QueueHandler(self.q)
        self.file = logging.FileHandler('log.log')
        self.file.setLevel('DEBUG')
        self.logger.addHandler(self.qh)
        self.logger.addHandler(self.file)
        logging.basicConfig(level='INFO')
# builder.py

import logging
from multiprocessing import Process
import threading

class Build:
    def __init__(self, q):
        self.queue = q
        self.logger = logging.getLogger('dcre')

        workers = []
        for i in range(5):
            wp = Process(target=self.foo, args=(i,))
            workers.append(wp)
            wp.start()

            lp = threading.Thread(target=self.logger_thread)
            lp.start()
            for wp in workers:
                wp.join()

            self.queue.put(None)
            lp.join()

    def logger_thread(self):
        while True:
            record = self.queue.get()
            if record is None:
                break
            self.logger.handle(record)

    def foo(self, i):
        msgs = (
            (10, "This is a DEBUG message. You shouldn't see this."),
            (20, 'This is an INFO message. Just so you know.'),
            (30, 'This is a WARNING message. Be careful, yo.'),
            (40, 'This is an ERROR message. Man, you done messed up.'),
            (50, 'This is a CRITICAL message. Game over!')
        )
        self.logger.log(*msgs[i])
Note: The configure method for the logger only existed to delay the configuration until after the GUI was created so that it had access to a Text widget for a custom handler to write to.
Reply
#2
I really hate to be petty (seriously, I do) but, just.. wow. This site is a beautiful example of why one should always read the terms before creating an account anywhere.
Quote:
  • Users are not allowed to delete posts. Requests to delete a post will be denied unless mods/admins find it justifiable. This will not be immediate as your request will be deliberated over.
  • Users are not allowed to delete threads. Requests to delete a thread will be denied unless mods/admins find it justifiable. This will not be immediate as your request will be deliberated over.
  • Requests to delete accounts will be denied unless mods/admins find it justifiable (listed below). This will not be immediate as your request will be deliberated over.
I mean, if you really want to take ownership of my posts, then fine, but at least make it transparent (i.e., mark it as such). Especially considering this little tidbit:
Quote:..you acknowledge that all messages posted on this discussion system express the views and opinions of the original message author and not necessarily the views of this bulletin board. Therefore we take no responsibility ... for any messages posted.
So you take control of the post in perpetuity but refuse any responsibility. Pick one. The post is either mine or yours. It can't be both.

And making me "request" account deletion? Really? At that point, it's obvious that not only are my posts not my own, but neither is my account.

And it's really sad that I feel like I need to clarify this but, since someone involved in the rules making process was apparently traumatized by a "cheater" at some point in their lives, I haven't been a student for 30+ years. So that's not what this is. I'm just a grumpy old man.
Reply
#3
If there is content you want to remove you can edit your post at any time. Other than that, why do you care about your post immortalized and soon forgotten? Maybe someone will solve the problem and write a great reply. That might be too late to help you, but someone else may have a similar problem, run across you post and see the solution. I was going to investigate when I could find some time.

And though we don't take responsibility for the content of posts, we do monitor the forum and edit or remove non-conformant content when it is discovered or reported.
Reply
#4
Apologies for the attitude. I originally simply wanted to delete the post as I'd already solved the issue and when I couldn't find the option to do so and went hunting I was taken aback by the lack of any control I had regarding posts or even the account itself. I should have just went about my day, but the "grumpy old man" was already out of the bottle.

And apologies again if any of that was taken as anything other than a rant against the "system". It was certainly not intended as a slight against the community, especially as regarding the lack of any response to the question (it's not even a day old).
Reply
#5
What was the solution?
Reply
#6
Not doing what the question asked. In other words, not sharing the logger and instead sharing the Queue and making use of the QueueHandler and QueueListener provided in the logging package. And then calling getLogger for each new process.

I don't have enough confidence in the solution yet to say it's the "correct" way to do it, but it's working now as intended and, in retrospect, seems to be what the docs were telling me to do to begin with before I got sidetracked trying to follow the example provided in those docs.

I would post the working code example (and still will if wanted) but I'm away from my PC atm.
Reply


Possibly Related Threads…
Thread Author Replies Views Last Post
  Timestamp of file changes if a share mapped to a server…. tester_V 34 9,680 Jul-04-2023, 05:19 AM
Last Post: tester_V
  logger behaviour setdetnet 1 1,500 Apr-15-2023, 05:20 AM
Last Post: Gribouillis
  how to write exception error into logger mg24 3 1,801 Nov-15-2022, 04:20 PM
Last Post: insharazzak
  python insert blank line in logger mg24 1 4,860 Nov-02-2022, 08:36 AM
Last Post: snippsat
  python logger help ... save logger into different folder mg24 1 3,113 Oct-25-2022, 03:04 PM
Last Post: snippsat
  processes shall be parallel flash77 4 1,938 Sep-20-2022, 11:46 AM
Last Post: DeaD_EyE
  Closing logger from other function problem Paqqno 1 1,768 Apr-25-2022, 11:49 AM
Last Post: Gribouillis
  Closing logger to rename directory malcoverc 1 1,788 Apr-19-2022, 07:06 AM
Last Post: Gribouillis
  access share attributed among several class methods drSlump 0 1,519 Nov-18-2021, 03:02 PM
Last Post: drSlump
  Killing processes via python Lavina 2 4,085 Aug-04-2021, 06:20 AM
Last Post: warnerarc

Forum Jump:

User Panel Messages

Announcements
Announcement #1 8/1/2020
Announcement #2 8/2/2020
Announcement #3 8/6/2020