-
Notifications
You must be signed in to change notification settings - Fork 175
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
Use Native Python Types #329
Comments
While I completely agree that work like this needs to be done, it would likely need to be part of a much larger modernization refactoring effort. Currently the pyTenable Tenable.sc and Tenable.io packages still rely on a lot of Python 2.x compatible code that would need to be modernized along with this. |
That's perfectly reasonable, and I appreciate the consideration. Please let me know if there is any support needed to accelerate that type of effort. EDIT: This comment assumes that the goal would be to do so soon, due to Python 2 being end-of-life. If realistically that is not a goal of the project due to enterprise or other customers, I fully understand. |
this will be part of a bigger refactoring to Python 3.x code standards that is on our roadmap. |
So in diving into this further, and when looking into what would be needed to support all of Tenable.sc 5.x, this may be a much bigger addition than expected. This would also mean the library would have to be revved for every version of Tenable.sc, or any other products when they get revved to add new fields. I'm not sure is native datatypes are the right approach. I do understand that SC's responses are almost entirely string values, and it's a pain having to recast them. my concern is that by forcing folks to upgrade the library potentially every rev of any product may create other issues. There may be some alternative validation and transformation libraries that can handle data in a less lossy way. |
so Marshmallow might be a more appropriate path forward here, as it can handle unknown fields in a more appropriate manner. https://marshmallow.readthedocs.io/en/stable/ In any case, implementing response attribute recasting across the whole library will dramatically increase the time to refactor the code on the pathway to the larger v2 refactor. |
So please excuse the stream of posts here, but I'll be treating this issue as a bit of a scratch pad while I research this. In either case, marshmallow shows the most promise so far. I was already looking into using it to replace a lot of madness in the IO package around filtering. Custom Fields: https://marshmallow.readthedocs.io/en/stable/custom_fields.html Potential issues/thoughts:
|
@kkirsche so the approach here will be to refactor parts of the library using marshmallow to handle the data typing and transformation. This will effectively replace a lot of the old constructor code. It'll be happening organically, and the first part of that was to replace the old, inflexible connection class with something that's better tested. #457 Details this initial work. We didn't want to have to touch everything multiple times, but this enables a pattern like so (below). It's a work in progress, just keeping you posted. Example Tenable.io Access Groups v2 API Schemaimport re
from marshmallow import (
Schema,
fields,
pre_load,
validate as v,
validates_schema,
)
from tenable.base.schema.fields import LowerCase, UpperCase
from tenable.io.base.schemas.filters.base import BaseFilterSchema
from tenable.io.base.session import TenableIO
class RuleSchema(BaseFilterSchema):
'''
Marshmallow schema for processing the Access List rule definitions.
'''
type = fields.Str(required=True)
operator = fields.Str(required=True)
terms = fields.List(fields.Str(), required=True)
_filters = None
@classmethod
def populate_filters(cls, tio: TenableIO):
'''
Pre-populates the asset filter definitions into the RuleSchema class.
'''
super().populate_filters(tio, 'filters/workbenches/assets')
@validates_schema
def filter_validation(self, data, **kwargs): # noqa: PLW0613
'''
Handles validation of the filter provided against the asset filter
definitions.
'''
self.validate_filter(data.get('type'),
data.get('operator'),
data.get('terms'))
@pre_load(pass_many=False)
def tuple_expansion(self, data, **kwargs): # noqa: PLW0613
'''
Handles expanding a tuple definition into the dictionary equivalent.
'''
if isinstance(data, tuple):
return {
'type': data[0],
'operator': data[1],
'terms': data[2]
}
return data
class PrincipalSchema(Schema):
principal_name = fields.Str()
principal_id = fields.UUID()
type = LowerCase(
fields.Str(validate=v.OneOf(['all_users', 'user', 'group'])))
permissions = fields.List(UpperCase(
fields.Str(validate=v.OneOf(['CAN_VIEW', 'CAN_SCAN']))))
@pre_load(pass_many=False)
def pre_process(self, data, **kwargs):
if isinstance(data, tuple):
# unpack the tuple into the appropriate fields. Supporting format
#
# ('TYPE', 'NAME', ['PERMS'])
#
# or
#
# ('TYPE', 'UUID', ['PERMS'])
newdata = {
'type': data[0],
'permissions': data[2]
}
if re.match(r'[0-9a-f]{8}(?:-[0-9a-f]{4}){3}-[0-9a-f]{12}',
data[1],
re.I):
newdata['principal_id'] = data[1]
else:
newdata['principal_name'] = data[1]
return newdata
return data
class AccessGroupsSchema(Schema): # noqa: PLR0903
'''
'''
name = fields.Str(required=True)
access_group_type = UpperCase(fields.Str(validate=v.OneOf(['MANAGE_ASSETS',
'SCAN_TARGETS',
'ALL'
])))
all_users = fields.Bool()
principals = fields.List(fields.Nested(PrincipalSchema))
rules = fields.List(fields.Nested(RuleSchema))
container_uuid = fields.UUID(load_only=True)
created_by_uuid = fields.UUID(load_only=True)
updated_by_uuid = fields.UUID(load_only=True)
created_by_name = fields.Str(load_only=True)
updated_by_name = fields.Str(load_only=True)
created_at = fields.DateTime(load_only=True)
updated_at = fields.DateTime(load_only=True)
id = fields.UUID(load_only=True)
status = fields.Str(load_only=True)
all_assets = fields.Bool(load_only=True)
processing_percent_complete = fields.Int(load_only=True) |
Is your feature request related to a problem? Please describe.
In the Tenable.SC python module, the library returns strings for various data types complicating interactions. As a result, end-users need to be aware of all the various data types in the API to properly interact with and manipulate the data. For example, calculating differences in times between various scans, processing things like ID's as integers, etc.
Describe the solution you'd like
Use native python types
Describe alternatives you've considered
I have written my own custom integration layer to handle type conversions to native python types and key styles.
Additional context
Example from my integration layer:
The text was updated successfully, but these errors were encountered: