diff --git a/docs/compatibility.rst b/docs/compatibility.rst index 8b87a4d4..0c0a4860 100644 --- a/docs/compatibility.rst +++ b/docs/compatibility.rst @@ -87,6 +87,213 @@ or added if it does not exist. The tag *!ZenPackSpec* is not necessary and should be removed from your yaml definitions. +Convert Relations +----------------- + +Class relationships need to be redefined when converting from prior ZP versions to ZP2. +Below is the old style of defining class relationships; Note that the relationship must be defined in both python classes. For instance, class 'Lightsaber' has a definition for the relation to class 'KyberCrystal'. But you will also find the inverted relation defined in the class 'KyberCrystal' for class "Lightsaber'. + +.. code-block:: python + + from StarWars.weapons.Device import Saber as LightSaberDevice + + class Lightsaber (LightSaberDevice) + _relations = ManagedEntity._relations + ( + ('crystal', ToMany( + ToOne, MODULE_NAME['KyberCrystal'], 'saber')), + ('diatium', ToOne( + ToOne, MODULE_NAME['DiatiumPowerCell'], 'saber')), + ('emitter', ToMany( + ToOne, MODULE_NAME['BladeEmitter'], 'saber')), + ('energizers', ToMany( + ToOne, MODULE_NAME['CyclingFieldEnergizers'], 'saber')), + ) + + + from StarWars.items.Component import Crystal as KyberCrystalComponent + + class KyberCrystal (KyberCrystalComponent) + _relations = ManagedEntity._relations + ( + ('saber', ToOne( + ToMany, MODULE_NAME['Lightsaber'], 'crystal')), + ('color', ToManyCont( + ToOne, MODULE_NAME['BladeColor'], 'crystal')), + ) + + +The syntax for defining relations is not pretty, but is redundent, inverted, and error prone. +To resolve this, class relations are now defined just once in the ZP2 yaml file as seen below. + +.. code-block:: yaml + + Defining class_relationships in the yaml: + - Lightsaber(crystal) 1:M KyberCrystal(saber) + - Lightsaber(diatium) 1:1 DiatiumPowerCell(saber) + - Lightsaber(emitter) 1:M BladeEmitter(saber) + - Lightsaber(energizers) 1:M CyclingFieldEnergizers(saber) + - KyberCrystal(color) 1:MC BladeColor(crystal) + +For each class in your ZenPack, you'll need to convert the _relations section to yaml definitions. +Once the relations have been defined you can comment out or delete the code from the python code. + +Convert Classes +--------------- + +Any classes defined in the class_relationships section or has older zp1 style properties and/or meta_types (seen in below sections) +should be updated to ZP2. To do this, we'll add a 'classes' section to the yaml. The classes definition must have a 'DEFAULTS' section with a base that inherites from zenpacklib.Component. From there, python classes can be defined in the yaml, as seen here. +If the python class is derived from a parent object, be sure to define the parent as the base. + +.. code-block:: yaml + + classes: + DEFAULTS: + base: [zenpacklib.Component] + Lightsaber: + base: [StarWars.weapons.Device.Saber] + KyberCrystal: + base: [StarWars.items.Component.Crystal] + +Then, back in the python class def, we'll need to make a couple changes. First, import the yaml defined schema. Second, have the class now inherite from the schema class. + +.. code-block:: python + + import schema from . + class Lightsaber (schema.LightSaber) + ... + + import schema from . + class KyberCrystal (schema.KyberCrystal) + ... + + +In some cases, this inheritance pattern will do something a little different, essentially inheriting from zenpacklib. For example: + +.. code-block:: python + + from Products.ZenModel.DeviceComponent import DeviceComponent + class RestrainingBolt (DeviceComponent) + ... + +Will now resemble something closer to this: + +.. code-block:: python + + from schema import . + class RestrainingBolt (schema.RestrainingBolt) + ... + + classes: + ... + RestrainingBolt: + base: [zenpacklib.Component] + ... + +Here's a short list of a few classes that change the base inheritance: + Products.ZenModel.DeviceComponent.DeviceComponent --> zenpacklib.Component + Products.ZenModel.HWComponent.HWComponent --> zenpacklib.HWComponent + Products.ZenModel.OSComponent.OSComponent --> zenpacklib.OSComponent + + +Convert Properties +------------------ + +Class properties were originally defined in the class, and referenced the variables of the class that were +accessable; ie, they defined what was visible in the console. Below is an old style of defining properties in python. + +.. code-block:: python + + class Lightsaber + _properties = ManagedEntity._properties + LightsaberObject._properties + ( + {'id': 'owner', 'type': 'string', 'label': 'Jedi or Sith Owner'}, + {'id': 'color', 'type': 'string', 'label': 'Blade Color'}, + {'id': 'length', 'type': 'int', 'label': 'Length of Blade'}, + ) + owner = "" + color = "" + length = 0 + +However, with ZP2 style zenpacks these properties have been moved to the yaml file as part of the class definitions. +Hence the properties section can be commented out or removed from the class as well as any of the defined variables. +Note that 'type: string' is not transcribed, this is becuse 'string' is the default value for the field 'type'. + +.. code-block:: yaml + + classes: + ... + Lightsaber: + base: [] + properties: + owner: + label: Jedi or Sith Owner + color: + label: Blade Color + length: + type: int + label: Blade Length + ... + +Update Z-Properties +------------------- + +The __init__.py file may have additional zProperties defined. These properties need to be transferred to the zenpack.yaml and removed from the __init__.py file. + +Update Meta Types +----------------- + +Some classes may have a meta type defined. You can find the meta type definition in the class for older zenpacks. + +.. code-block:: python + + portal_type = meta_type = 'SWLightSaber' + +To updated this to be ZP2 compatible, simple add a 'meta_type' line to the yaml, then comment out or remove the old code. + +.. code-block:: yaml + + classes: + ... + Lightsaber: + meta_type: SWLightSaber + ... + +Debugging help and tools +------------------------ + +One of the most difficult and tedious pieces to converting a zenpack is getting all the relationships to load, build, and link correctly when modeling. Especially when the ZenPack is inheriting from other zenpacks which are still using the older format. If you cannot model the device without getting errors, first go into the console and disable all but one module. It will be best to work through them one at a time. In the associated source files to the active module, locate any of the following snipits of code and comment them out: + +.. code-block:: python + + maps.Append( RelationshipMap( ... ) ) + updateToOne( ... ) + updateTOMany( ... ) + ... etc. + +Re-enable them one at a time, running the modeler, until errors are encountered. There's no easy way to know exactly what is going wrong, however, most of the errors can be fixed by changing the relationship names in the yaml file. Check through the source code and any base classes for clues and hints as to what names should be set to. + +To check whether or not the yaml files contain errors you can use the zendmd command to import/load the zenpack: + +.. code-block:: python + + from ZenPacks.zenoss.Lightsaber import CFG,schema + +After the zenpack has been loaded you can check various aspects, such as the defined yaml relationships, use the following commands: + +.. code-block:: python + + spec = CFG.classes.get("KyberCrystal") + ob = spec.model_class + ob._relations + +Once a device has been modeled, you can inspect the device, see a list of discovered components, view other properties such as the meta_type, and print any relationships that the component may retain: + +.. code-block:: python + + dev=find('16.32.64.128') + dc_list=dev.getDeviceComponents() + comp=dc_list[0] + comp.meta_type + comp._relationships + .. _new-logging: *******************