Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Gp/fix/act flags #947

Open
wants to merge 25 commits into
base: master
Choose a base branch
from
Open

Gp/fix/act flags #947

wants to merge 25 commits into from

Conversation

iparask
Copy link
Member

@iparask iparask commented Sep 9, 2024

This PR makes the glitch path configurable for glitch flags. It assumes that the flags are under a nested path and traverses it based on a separator. This can later be expanded to accept operations over paths.

@iparask iparask requested review from mhasself and chervias September 9, 2024 19:40
@mhasself
Copy link
Member

I think we need to rethink this -- the additions actually break the abstraction (note the SignalCut inherits from Signal); MLMapmaker caches several Signal objects in self.signals, and loops over them.

So I suspect the right thing to do is have the glitch_flags setting enter through the SignalCut constructor, and be saved as an instance variable.

@iparask
Copy link
Member Author

iparask commented Sep 23, 2024

I moved the path information in the init of the signal classes.

@chervias
Copy link
Contributor

I think going through the Signal objects is fine. We could also go through the init of the MLMapmaker object and pass it there as a "global" parameter that applies to all sub-signal objects included.

@iparask
Copy link
Member Author

iparask commented Sep 24, 2024

I think going through the Signal objects is fine. We could also go through the init of the MLMapmaker object and pass it there as a "global" parameter that applies to all sub-signal objects included.

Included. Also the ACT tod that was causing errors now works.

Copy link
Contributor

@chervias chervias left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I tested the latest version and works fine.

sotodlib/mapmaking/utilities.py Outdated Show resolved Hide resolved
sotodlib/mapmaking/utilities.py Outdated Show resolved Hide resolved
sotodlib/mapmaking/utilities.py Outdated Show resolved Hide resolved
sotodlib/mapmaking/utilities.py Outdated Show resolved Hide resolved
- sep: separator. Defaults to `.`
"""

flags = aman.copy()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Unnecessary copy.

sotodlib/mapmaking/ml_mapmaker.py Outdated Show resolved Hide resolved
@@ -161,7 +170,8 @@ def __init__(self, name, ofmt, output, ext):
self.ext = ext
self.dof = None
self.ready = False
def add_obs(self, id, obs, nmat, Nd): pass
self.glitch_flags = glitch_flags
def add_obs(self, id, obs, nmat, Nd, glitch_flags:Optional[str]): pass
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Same question.

Can it be replaced with a generic **kwargs, to imply "yes sometimes a subclass will accept and process other random args."

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes, I can

sotodlib/mapmaking/ml_mapmaker.py Outdated Show resolved Hide resolved
@@ -361,9 +373,9 @@ def transeval(self, id, obs, other, map, tod):

class SignalCut(Signal):
def __init__(self, comm, name="cut", ofmt="{name}_{rank:02}", dtype=np.float32,
output=False, cut_type=None):
output=False, cut_type=None, glitch_flags:str ="flags.glitch_flags"):
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I would prefer these default to None, as they do in SignalMap. Then convert None -> flags.glitch_flags, to handle the default.

When you have cascading kwargs, e.g.:

  SignalMap(..., glitch_flags=None)
     -> Signal(glitch_flags=glitch_flags)
          ->  self.glitch_flags = glitch_flags

it tends to defeat the use of default value declarations for optional parameters. My practice has been to prefer this format for setting default args:

    class Signal:
        def __init__(self, ..., glitch_flags=None):
            if glitch_flags is None:
                glitch_flags = 'flags.glitch_flags'

Then subclasses, or whatever, can pass in their default value of None and it propagates cleanly all the way down to whatever base function is comfortable with setting the default value.

However... this isn't necessary relevant if we take glitch_flags out of the base class Signal. In that case you won't have cascading. I still think it's a good practice, in code that might become cascade, to use "None" to represent "the default value specified by some lower level". (Though this makes it harder to use "None" to represent "disable this feature" -- I would normally use False to signal that in cases where it is needed.)

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Generally, I prefer having the actual default value over a check and set in the code. It makes it very easy know the default value, and it avoids any question as to what is happening. I agree the base class needs to have a default as None. Using kargs will resolve that and make it much cleaner for sure.

@iparask
Copy link
Member Author

iparask commented Oct 23, 2024

@mhasself let's move forward with this PR. ML mapmaker's multipass is not yet used from sotodlib mapmaking script and I still have not figured out what is happening with I try to run multiple passes with ACT data. I have not forgotten the bug and I will open an issue soon to keep track of that.

@mhasself mhasself self-requested a review November 7, 2024 19:19
Copy link
Member

@mhasself mhasself left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Ok, I think this is going to be ok now. Thanks!

A few more minor comments though.

@@ -5,12 +5,15 @@
import logging
import numpy as np

from typing import Union, Dict, Tuple, List
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

The changes to context.py are just noise; please revert.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I will. I put them when I was trying to understand what was happening.


def setUp(self):
# Mock objects for testing
obj1 = MagicMock()
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Why mocks for these tests? You're not going to test on a real AxisManager even once?

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

It is an real Axismanager. Lines 20 to 25 make sure the first high level object is an AxisManager.

Oh... Lines 12-18 are not needed. Sorry I forgot to remove them

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This is for testing with an arbitrary object that has a dict representation and it is accessed from test_parameter_exists_in_dict.

test_parameter_exists_in_fields tests with an AxisManager object:

        flags = MagicMock(spec=AxisManager)
        flags._fields = {"glitch_flags": "some_value"}
        aman = MagicMock(spec=AxisManager)
        aman._fields = {"flags": flags,
                        "other_key": "another_value"}

I am not sure it makes sense to create a whole AxisManager object. I can give it a try if you think it is necessary

@iparask
Copy link
Member Author

iparask commented Nov 12, 2024

I will also fix the conflict with master and sync again.

@@ -1,6 +1,5 @@
import so3g.proj
import numpy as np
import scipy
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was not used anywhere in this file.

@@ -6,7 +6,7 @@
"""

from spt3g import core
from so3g.spt3g import core
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This was an import bug.

Comment on lines -113 to +115
from threadpoolctl import threadpool_limitse
from threadpoolctl import threadpool_limits
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Typo fixed

Comment on lines -106 to +110
pmat = coords.pmat.P.for_tod(obs, comps=comps, geom=(map.shape, map.wcs), rot=rot, threads="domdir", interpol=self.interpol)
pmat = coords.pmat.P.for_tod(obs, comps=comps, geom=(map.shape, map.wcs), rot=rot, threads="domdir", interpol=interpol)
# And perform the actual injection
pmat.from_map(map.extract(shape, wcs), dest=obs.signal)
pmat.from_map(map.extract(map.shape, map.wcs), dest=obs.signal)
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@chervias git blame said you are the one that added this lines. I corrected based on my understanding, but can you let me know if this is correct?

@iparask iparask requested a review from amaurea December 19, 2024 20:01
Comment on lines 381 to 392
try:
pmap.from_map(dest=tod, signal_map=map_work, comps=self.comps)
except RuntimeError as e:
raise RuntimeError(f"{e}.\nPossibly caused by the assumption that exactly
the same tiles will be hit each pass, which can in rare
cases break when downsampling by different amounts in
different passes when a tile is just barely hit by a
single sample. This can be fixed by adding support for
constructing coords.pmat.P which treats hits to a
missing tile as zero instead of as an error. This also
requires minor changes to so3g Projection.cxx. TODO.")

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

@mhasself this is Sigurd's suggestion based on the discussion I had with him. Should we move forward?

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

3 participants