Metadata-Version: 2.4
Name: cs-obj
Version: 20260526
Summary: Convenience facilities for objects.
Keywords: python2,python3
Author-email: Cameron Simpson <cs@cskk.id.au>
Description-Content-Type: text/markdown
Classifier: Programming Language :: Python
Classifier: Programming Language :: Python :: 3
Classifier: Development Status :: 4 - Beta
Classifier: Intended Audience :: Developers
Classifier: Operating System :: OS Independent
Classifier: Topic :: Software Development :: Libraries :: Python Modules
Classifier: License :: OSI Approved :: GNU General Public License v3 or later (GPLv3+)
Requires-Dist: cs.deco>=20260525
Project-URL: MonoRepo Commits, https://bitbucket.org/cameron_simpson/css/commits/branch/main
Project-URL: Monorepo Git Mirror, https://github.com/cameron-simpson/css
Project-URL: Monorepo Hg/Mercurial Mirror, https://hg.sr.ht/~cameron-simpson/css
Project-URL: Source, https://github.com/cameron-simpson/css/blob/main/lib/python/cs/obj.py

Convenience facilities for objects.

*Latest release 20260526*:
Refreshable mixin class embodying refresh logic.

Short summary:
* `as_dict`: Return a dictionary with keys mapping to the values of the attributes of `o`.
* `copy`: Convenient function to shallow copy an object with simple modifications.
* `flavour`: Return constants indicating the ``flavour'' of an object: * `T_MAP`: DictType, DictionaryType, objects with an __keys__ or keys attribute. * `T_SEQ`: TupleType, ListType, objects with an __iter__ attribute. * `T_SCALAR`: Anything else.
* `O`: The `O` class is now obsolete, please subclass `types.SimpleNamespace` or use a dataclass.
* `O_attritems`: Generator yielding `(attr,value)` for relevant attributes of `o`.
* `O_attrs`: Yield attribute names from `o` which are pertinent to `O_str`.
* `O_merge`: Merge key:value pairs from a mapping into an object.
* `O_str`: Return a `str` representation of the object `o`.
* `obj_as_dict`: OBSOLETE version of obj_as_dict, suggestion: use cs.obj.as_dict.
* `Proxy`: An extremely simple proxy object that passes all unmatched attribute accesses to the proxied object.
* `public_subclasses`: Return a set of the subclasses of `cls` which have public names.
* `Refreshable`: A mixin for refreshable objects. The object must provide a `_refresh(resource)` method which tries to refresh from `resource`.
* `Sentinel`: A simple class for named sentinels whose `str()` is just the name and whose `==` uses `is`.
* `singleton`: Obtain an object for `key` via `registry` (a mapping of `key`=>object). Return `(is_new,object)`.
* `SingletonMixin`: A mixin turning a subclass into a singleton factory.
* `TrackedClassMixin`: A mixin to track all instances of a particular class.

Module contents:
- <a name="as_dict"></a>`as_dict(o, selector=None)`: Return a dictionary with keys mapping to the values of the attributes of `o`.

  Parameters:
  * `o`: the object to map
  * `selector`: the optional selection criterion

  If `selector` is omitted or `None`, select "public" attributes,
  those not commencing with an underscore.

  If `selector` is a `str`, select attributes starting with `selector`.

  Otherwise presume `selector` is callable
  and select attributes `attr` where `selector(attr)` is true.
- <a name="copy"></a>`copy(obj, **kw)`: Convenient function to shallow copy an object with simple modifications.

  Performs a shallow copy of `self` using `copy.copy`.

  Treat all keyword arguments as `(attribute,value)` 2-tuples and
  replace those attributes with the supplied values.
- <a name="flavour"></a>`flavour(obj)`: Return constants indicating the ``flavour'' of an object:
  * `T_MAP`: DictType, DictionaryType, objects with an __keys__ or keys attribute.
  * `T_SEQ`: TupleType, ListType, objects with an __iter__ attribute.
  * `T_SCALAR`: Anything else.
- <a name="O"></a>`class O(types.SimpleNamespace)`: The `O` class is now obsolete, please subclass `types.SimpleNamespace`
  or use a dataclass.
- <a name="O_attritems"></a>`O_attritems(o)`: Generator yielding `(attr,value)` for relevant attributes of `o`.
- <a name="O_attrs"></a>`O_attrs(o)`: Yield attribute names from `o` which are pertinent to `O_str`.

  Note: this calls `getattr(o,attr)` to inspect it in order to
  prune callables.
- <a name="O_merge"></a>`O_merge(o, _conflict=None, _overwrite=False, **kw)`: Merge key:value pairs from a mapping into an object.

  Ignore keys that do not start with a letter.
  New attributes or attributes whose values compare equal are
  merged in. Unequal values are passed to:

      _conflict(o, attr, old_value, new_value)

  to resolve the conflict. If _conflict is omitted or None
  then the new value overwrites the old if _overwrite is true.
- <a name="O_str"></a>`O_str(o, no_recurse=False, seen=None)`: Return a `str` representation of the object `o`.

  Parameters:
  * `o`: the object to describe.
  * `no_recurse`: if true, do not recurse into the object's structure.
    Default: `False`.
  * `seen`: a set of previously sighted objects
    to prevent recursion loops.
- <a name="obj_as_dict"></a>`obj_as_dict(o, **kw)`: OBSOLETE version of obj_as_dict, suggestion: use cs.obj.as_dict

  OBSOLETE convesion of an object to a `dict`. Please us `cs.obj.as_dict`.
- <a name="Proxy"></a>`class Proxy`: An extremely simple proxy object
  that passes all unmatched attribute accesses to the proxied object.

  Note that setattr and delattr work directly on the proxy, not the proxied object.
- <a name="public_subclasses"></a>`public_subclasses(cls, extras=())`: Return a set of the subclasses of `cls` which have public names.
- <a name="Refreshable"></a>`class Refreshable`: A mixin for refreshable objects.
  The object must provide a `_refresh(resource)` method
  which tries to refresh from `resource`.

  This mixin provides a `.refresh()` method which implements the refresh policy.

*`Refreshable.refresh(self, resource: Optional = None, *, force=False, lifespan: float = None, ratelimit: float = None, recurse=False, seen=None, **_refresh_kw) -> bool`*:
Refresh this object; if it is stale attempt to refresh via `self._refresh()`.
Return `True` if the object was updated, `False` otherwise.

The refresh policy is: if `force` or (the object's information
is stale and the rate limit does not preclude a refresh),
call `self._refresh(resource)`.
This is measured by `self.refresh_needed()`.

State about the refresh poll times is kept in
`self.refresh_last_poll` and `self.refresh_last_update`.

Parameters:
* `resource`: the reference resource, default from `self.refresh_resource`
* `force`: optional flag, default `False`; if true attempt
  a refresh regardless of staleness or the rate limit
* `lifespan`: how many seconds before updated information
  is considered no stale, default from `self.refresh_lifespan`,
  or `type(self).REFRESH_RATELIMIT`
* `ratelimit`: how many seconds should elapsed before
  attempting a refresh, default from `self.refresh_ratelimit`
  or `type(self).REFRESH_RATELIMIT`
* `recurse`: optional flag, default `False`; if true the
  recursively refresh the objects from `self.refresh_related()`
* `seen`: optional set of keys to prevent unbound recursion
Other keyword parameters are passed to `self._refresh()` if it is called.

*`Refreshable.refresh_key(self)`*:
Return the key value for the `seen` set used in a recursive refresh.
This default returns `id(self)`; classes like `HasTags`
would use a characteristic value like `self.name`.

*`Refreshable.refresh_needed(self, *, lifespan: float = None, now: float = None) -> bool`*:
Test whether `self` is considered stale.

Parameters:
* `lifespan`: how many seconds before updated information
  is considered no stale, default from `self.refresh_lifespan`,
  or `type(self).REFRESH_LIFESPAN`
 * `now`: optional reference time, default from `time.time()`

*`Refreshable.refresh_related(self)`*:
Return the related objects which should also be refreshed in recursive refreshes.

*`Refreshable.refreshed(self, now: float = None)`*:
Mark `self` as refreshed as of `now`, default `time.time()`.
- <a name="Sentinel"></a>`class Sentinel`: A simple class for named sentinels whose `str()` is just the name
  and whose `==` uses `is`.

  Example:

      >>> from cs.obj import Sentinel
      >>> MISSING = Sentinel("MISSING")
      >>> print(MISSING)
      MISSING
      >>> other = Sentinel("other")
      >>> MISSING == other
      False
      >>> MISSING == MISSING
      True
- <a name="singleton"></a>`singleton(registry, key, factory, fargs, fkwargs)`: Obtain an object for `key` via `registry` (a mapping of `key`=>object).
  Return `(is_new,object)`.

  If the `key` exists in the registry, return the associated object.
  Otherwise create a new object by calling `factory(*fargs,**fkwargs)`
  and store it as `key` in the `registry`.

  The `registry` may be any mapping of `key`s to objects
  but will usually be a `weakref.WeakValueDictionary`
  in order that object references expire as normal,
  allowing garbage collection.

  *Note*: this function *is not* thread safe.
  Multithreaded users should hold a mutex.

  See the `SingletonMixin` class for a simple mixin to create
  singleton classes,
  which does provide thread safe operations.
- <a name="SingletonMixin"></a>`class SingletonMixin`: A mixin turning a subclass into a singleton factory.

  *Note*: this mixin overrides `object.__new__`
  and may not play well with other classes which override `__new__`.

  *Warning*: because of the mechanics of `__new__`,
  the instance's `__init__` method will always be called
  after `__new__`,
  even when a preexisting object is returned.
  Therefore that method should be sensible
  even for an already initialised
  and probably subsequently modified object.

  My suggested approach is to access some attribute,
  and preemptively return if it already exists.
  Example:

      def __init__(self, x, y):
          if 'x' in self.__dict__:
              return
          self.x = x
          self.y = y

  *Note*: we probe `self.__dict__` above to accomodate classes
  with a `__getattr__` method.

  *Note*: each class registry has a lock,
  which ensures that reuse of an object
  in multiple threads will call the `__init__` method
  in a thread safe serialised fashion.

  Implementation requirements:
  a subclass should:
  * provide a method `_singleton_key(*args,**kwargs)`
    returning a key for use in the single registry,
    computed from the positional and keyword arguments
    supplied on instance creation
    i.e. those which `__init__` would normally receive.
    This should have the same signature as `__init__`
    but using `cls` instead of `self`.
  * provide a normal `__init__` method
    which can be safely called again
    after some earlier initialisation.

  This class is thread safe for the registry operations.

  Example:

      class Pool(SingletonMixin):

          @classmethod
          def _singleton_key(cls, foo, bah=3):
              return foo, bah

          def __init__(self, foo, bah=3):
              if hasattr(self, 'foo'):
                  return
             ... normal __init__ stuff here ...
             self.foo = foo
             ...

*`SingletonMixin.__hash__(self)`*:
default hash and equality methods

*`SingletonMixin.singleton_also_by(also_key, key)`*:
Obtain a singleton by a secondary key.
Return the instance or `None`.

Parameters:
* `also_key`: the name of the secondary key index
* `key`: the key for the index
- <a name="TrackedClassMixin"></a>`class TrackedClassMixin`: A mixin to track all instances of a particular class.

  This is aimed at checking the global state of objects of a
  particular type, particularly states like counters. The
  tracking is attached to the class itself.

  The class to be tracked includes this mixin as a superclass and calls:

      TrackedClassMixin.__init__(class_to_track)

  from its __init__ method. Note that `class_to_track` is
  typically the class name itself, not `type(self)` which would
  track the specific subclass. At some relevant point one can call:

      self.tcm_dump(class_to_track[, file])

  `class_to_track` needs a `tcm_get_state` method to return the
  salient information, such as this from cs.resources.MultiOpenMixin:

      def tcm_get_state(self):
          return {'opened': self.opened, 'opens': self._opens}

  See cs.resources.MultiOpenMixin for example use.

*`TrackedClassMixin.tcm_all_state(klass)`*:
Generator yielding tracking information
for objects of type `klass`
in the form `(o,state)`
where `o` if a tracked object
and `state` is the object's `get_tcm_state` method result.

*`TrackedClassMixin.tcm_dump(klass, f=None)`*:
Dump the tracking information for `klass` to the file `f`
(default `sys.stderr`).

# Release Log



*Release 20260526*:
Refreshable mixin class embodying refresh logic.

*Release 20250306*:
* Remove cs.py3 dependency, this module has not worked in python2 for quite a while.
* copy: drop use of positional parameters.

*Release 20250103*:
public_subclasses: new optional extras= parameter for additional classes to scan, now returns a set.

*Release 20241009*:
public_subclasses: catch TypeError from "type.__subclasses__", which is just a function.

*Release 20241005*:
New public_subclasses(cls) returning all subclasses with public names.

*Release 20220918*:
* SingletonMixin: change example to probe self__dict__ instead of hasattr, faster and less fragile.
* New Sentinel class for named sentinel objects, equal only to their own instance.

*Release 20220530*:
SingletonMixin: add default __hash__ and __eq__ methods to support dict and set membership.

*Release 20210717*:
SingletonMixin: if cls._singleton_key returns None we always make a new instance and do not register it.

*Release 20210306*:
SingletonMixin: make singleton_also_by() a public method.

*Release 20210131*:
SingletonMixin: new _singleton_also_indexmap method to return a mapping of secondary keys to values to secondary lookup, _singleton_also_index() to update these indices, _singleton_also_by to look up a secondary index.

*Release 20210122*:
SingletonMixin: new _singleton_instances() method returning a list of the current instances.

*Release 20201227*:
SingletonMixin: correctly invoke __new__, a surprisingly fiddly task to get right.

*Release 20201021*:
* @OBSOLETE(obj_as_dict), recommend "as_dict()".
* [BREAKING] change as_dict() to accept a single optional selector instead of various mutually exclusive keywords.

*Release 20200716*:
SingletonMixin: no longer require special _singleton_init method, reuse default __init__ implicitly through __new__ mechanics.

*Release 20200517*:
Documentation improvements.

*Release 20200318*:
* Replace obsolete O class with a new subclass of SimpleNamespace which issues a warning.
* New singleton() generic factory function and SingletonMixin mixin class for making singleton classes.

*Release 20190103*:
* New mixin class TrackedClassMixin to track all instances of a particular class.
* Documentation updates.

*Release 20170904*:
Minor cleanups.

*Release 20160828*:
* Use "install_requires" instead of "requires" in DISTINFO.
* Minor tweaks.

*Release 20150118*:
move long_description into cs/README-obj.rst

*Release 20150110*:
cleaned out some old junk, readied metadata for PyPI
