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

feat(gres)!: add mappings for managing GRESName and GRESNode objects #40

Merged

Conversation

NucciTheBoss
Copy link
Member

This PR adds the GRESNameMapping and GRESNodeMapping objects to make it easier to manage gres.conf data model objects. Rather than throw lists and dictionaries out there all loosey goosey for GRESConfig, instead these new maps help facilitate interacting with the configured generic resources and easily share them between the slurmctld and slurmd operators. Couple key changes:

  • There's now stricter validation for the gres.conf related models by using jsonschema. Helps make sure the input we're putting into the maps are valid.
  • Methods for encoding and decoding dictionaries with nested data models. Makes it easier to validate with jsonschema and send "across the wire," but still enable us to take advantage of the data models and the descriptors + methods that they provide.
    • The expand function is an anamorphism that recursively expands a dictionary with nested data model objects. This makes it easier to punch objects into a JSON schema validator. Custom object hooks are used with the JSON decoder to parse the data models back into Python objects.
  • I introduced a BaseMapping object to provide methods necessary for GRESNameMapping and GRESNodeMapping, but in the future I could see BaseMapping becoming the new BaseModel. The classes do similar things, but BaseMapping is more flexible and uses JSON schemas to validate the data models.

Breaking changes

The names and nodes properties from GRESConfig now return GRESNameMapping and GRESNodeMapping objects respectively rather than list and dict types.

Related issues

Misc.

  • I updated the annotations for some functions to be compliant with recommendations set forth by Python 3.10. Big thing is that the many of the types provided by the typing module are now deprecated, so their imports are now updated. Another thing is the inclusion of the typing-extensions module. This is an official module maintained by the Python team. I bring it in as dependency since it provides the Self type which wasn't introduced into Python until 3.11. Since py310 is still a supported Python version, figured it doesn't hurt to bring typing-extensions in.

    MutableMapping and Sequence are the recommend annotations for arguments that are either type dict or list respectively.

`jsonschema` is now included as a dependency to help validate Slurm
data models as they're exchanged between classes and read in.
`typing-extensions` is added because the `Self` type annotation
is not available within Python 3.10 add is still a support version
of Python for security maintenance and Jammy Jellyfish.

Signed-off-by: Jason C. Nucciarone <[email protected]>
Changes:

* Change `names` and `nodes` property on `GRESConfig` to return `GRESNameMapping` and `GRESNodeMapping` respectively.
* Add `_slice` method to enable quick slicing of the internal data store dictionaries.
* Add `BaseMapping` ABC that can be used to create other mappings that can be used within slurm.conf files.
* Add `expand` method for recursively expanding dictionaries with nested Slurm data model objects.
* Introduce JSON schemas for validating `GRESNameMapping` and `GRESNodeMapping` objects.
* Define JSON decoders that can be used to parse dictionary objects into Slurm data models.

Signed-off-by: Jason C. Nucciarone <[email protected]>
Changes:

* Add addition test case where the same `NodeName` can have multiple generic resource configurations.

Signed-off-by: Jason C. Nucciarone <[email protected]>
@NucciTheBoss NucciTheBoss added the Type: Enhancement Proposes a new feature to be added to the project. label Dec 20, 2024
Copy link

@dsloanm dsloanm left a comment

Choose a reason for hiding this comment

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

Looks good and solves the issue I was hitting!

I've not spotted the stricter validation, however. I accidentally generated a few invalid gres.conf files when testing since I can get away with things like:

$ python
Python 3.12.3 (main, Nov  6 2024, 18:32:19) [GCC 13.2.0] on linux
Type "help", "copyright", "credits" or "license" for more information.
>>> from slurmutils.editors import gresconfig
>>> with gresconfig.edit("gres-testing.conf") as config:
...   config.nodes["123"] = [456]
... 
>>> 
$ cat gres-testing.conf
456
$

README.md Show resolved Hide resolved
@NucciTheBoss
Copy link
Member Author

NucciTheBoss commented Dec 20, 2024

I've not spotted the stricter validation, however. I accidentally generated a few invalid gres.conf files when testing since I can get away with things like:

Ah, that's because the validation is only done when the mapping object is first created. If you tried to initialize a new GRESNodeMapping with that dictionary, it would "explode in your face":

>>> from slurmutils.models import GRESNodeMapping
>>> GRESNodeMapping({"123": [456]})
Traceback (most recent call last):
  File "/home/nucci/git/slurmutils/slurmutils/models/model.py", line 252, in __init__
    validate(d, schema=self._schema)
  File "/home/nucci/.cache/pypoetry/virtualenvs/slurmutils-d46I0P7a-py3.12/lib/python3.12/site-packages/jsonschema/validators.py", line 1332, in validate
    raise error
jsonschema.exceptions.ValidationError: 456 is not of type 'object'
Failed validating 'type' in schema['patternProperties']['^.+$']['items']:
    {'$schema': 'https://json-schema.org/draft/2020-12/schema',
     'type': 'object',
     'properties': {'NodeName': {'type': 'string'},
                    'AutoDetect': {'type': 'string'},
                    'Count': {'type': 'string'},
                    'Cores': {'type': 'array',
                              'items': {'type': 'string'},
                              'uniqueItems': True},
                    'File': {'type': 'string'},
                    'Flags': {'type': 'array',
                              'items': {'type': 'string'},
                              'uniqueItems': True},
                    'Links': {'type': 'array', 'items': {'type': 'string'}},
                    'MultipleFiles': {'type': 'string'},
                    'Name': {'type': 'string'},
                    'Type': {'type': 'string'}},
     'additionalProperties': False}
On instance['123'][0]:
    456
During handling of the above exception, another exception occurred:
Traceback (most recent call last):
  File "<input>", line 1, in <module>
  File "/home/nucci/git/slurmutils/slurmutils/models/model.py", line 255, in __init__
    raise ModelError(e.message)
slurmutils.exceptions.ModelError: 456 is not of type 'object'

Validation is performed when the mappings are created — I still need to extend it to the other data models — but I'd like to eventually get it where validation is also performed before the object is serialized such as when we're writing back out to a configuration file or converting to a JSON string for "sending across the wire" . That's for a future PR however, otherwise the diff for this PR would become massive 😅

@NucciTheBoss NucciTheBoss merged commit 3f1aaa3 into charmed-hpc:main Dec 20, 2024
3 checks passed
@NucciTheBoss
Copy link
Member Author

v0.11.0 is now published to PyPI: https://pypi.org/project/slurmutils/

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
Type: Enhancement Proposes a new feature to be added to the project.
Projects
None yet
Development

Successfully merging this pull request may close these issues.

Support for multiple lines with same NodeName in gres.conf
2 participants