[Pacemaker] Patch for bugzilla 2541: Shell should warn if parameter uniqueness is violated

Dejan Muhamedagic dejanmm at fastmail.fm
Tue Mar 8 12:07:55 CET 2011


Hi Holger,

On Tue, Mar 08, 2011 at 09:15:01AM +0100, Holger Teutsch wrote:
> On Fri, 2011-03-04 at 13:06 +0100, Holger Teutsch wrote:
> > On Thu, 2011-03-03 at 10:55 +0100, Florian Haas wrote:
> > > On 2011-03-03 10:43, Holger Teutsch wrote:
> > > > Hi,
> > > > I submit a patch for
> > > > "bugzilla 2541: Shell should warn if parameter uniqueness is violated"
> > > > for discussion.
> > > 
> > > I'll leave it do Dejan to review the code, but I love the functionality.
> > > Thanks a lot for tackling this. My only suggestion for an improvement is
> > > to make the warning message a bit more terse, as in:
> > > 
> > > WARNING: Resources ip1a, ip1b violate uniqueness for parameter "ip":
> > > "1.2.3.4"
> > > 
> > 
> > Florian,
> > I see your point. Although my formatting allows for an unlimited number
> > of collisions ( 8-) ) in real life we will only have 2 or 3. Will change
> > this together with Dejan's hints.
> > 
> > > Cheers,
> > > Florian
> > > 
> Florian + Dejan,
> here the version with terse output. The code got terser as well.

It looks good, just a few notes. The check function should move
to the CibObjectSetRaw class and be invoked from
semantic_check(). There's

        rc1 = set_obj_verify.verify()
        if user_prefs.check_frequency != "never":
			rc2 = set_obj_semantic.semantic_check()

The last should be changed to:

			rc2 = set_obj_semantic.semantic_check(set_obj_verify)

set_obj_verify always contains all CIB elements (well, that means
that its name should probably be changed too :). Now, the code
should check _only_ new and changed primitives which are
contained in set_obj_semantic. That's because we don't want to
repeatedly print warnings for all objects on commit, but only for
those which were added/changed in the meantime. On the other
hand, verify is an explicit check and in that case the whole CIB
is always verified.

> - holger
> 
> crm(live)configure# primitive ip1a ocf:heartbeat:IPaddr2 params ip="1.2.3.4" meta target-role="stopped"
> crm(live)configure# primitive ip1b ocf:heartbeat:IPaddr2 params ip="1.2.3.4" meta target-role="stopped"
> crm(live)configure# primitive ip2a ocf:heartbeat:IPaddr2 params ip="1.2.3.5" meta target-role="stopped"
> crm(live)configure# primitive ip2b ocf:heartbeat:IPaddr2 params ip="1.2.3.5" meta target-role="stopped"
> crm(live)configure# primitive ip3 ocf:heartbeat:IPaddr2 params ip="1.2.3.6" meta target-role="stopped"
> crm(live)configure# primitive dummy_1 ocf:heartbeat:Dummy params fake="abc" meta target-role="stopped"
> crm(live)configure# primitive dummy_2 ocf:heartbeat:Dummy params fake="abc" meta target-role="stopped"
> crm(live)configure# primitive dummy_3 ocf:heartbeat:Dummy meta target-role="stopped"
> crm(live)configure# commit
> WARNING: Resources ip1a,ip1b violate uniqueness for parameter "ip": "1.2.3.4"
> WARNING: Resources ip2a,ip2b violate uniqueness for parameter "ip": "1.2.3.5"
> Do you still want to commit? 
> 
> 

> diff -r cf4e9febed8e shell/modules/ui.py.in
> --- a/shell/modules/ui.py.in	Wed Feb 23 14:52:34 2011 +0100
> +++ b/shell/modules/ui.py.in	Tue Mar 08 09:11:38 2011 +0100
> @@ -1509,6 +1509,55 @@
>              return False
>          set_obj = mkset_obj("xml")
>          return ptestlike(set_obj.ptest,'vv',cmd,*args)
> +
> +    def __check_unique_clash(self):
> +        'Check whether resource parameters with attribute "unique" clash'
> +
> +        def process_primitive(prim, clash_dict):
> +            '''
> +            Update dict clash_dict with
> +            (ra_class, ra_provider, ra_type, name, value) -> [ resourcename ]
> +            if parameter "name" should be unique
> +            '''
> +            ra_class = prim.getAttribute("class")
> +            ra_provider = prim.getAttribute("provider")
> +            ra_type = prim.getAttribute("type")
> +            ra_id = prim.getAttribute("id")
> +
> +            ra = RAInfo(ra_class, ra_type, ra_provider)

There's a convenience function get_ra(node) for this.

> +            if ra == None:
> +                return
> +            ra_params = ra.params()
> +
> +            attributes = prim.getElementsByTagName("instance_attributes")
> +            if len(attributes) == 0:
> +                return
> +
> +            for p in attributes[0].getElementsByTagName("nvpair"):
> +                name = p.getAttribute("name")
> +                if ra_params[ name ]['unique'] == '1':
> +                    value = p.getAttribute("value")
> +                    k = (ra_class, ra_provider, ra_type, name, value)
> +                    try:
> +                        clash_dict[k].append(ra_id)
> +                    except:
> +                        clash_dict[k] = [ra_id]
> +            return
> +
> +        clash_dict = {}
> +        for p in cib_factory.mkobj_list("xml","type:primitive"):

This would become:

           for p in all_obj_list: # passed from _verify()
			   if is_primitive(p.node):

> +            process_primitive(p.node, clash_dict)

Or perhaps to loop through self.obj_list and build clash_dict
against all elements? Otherwise, you'll need to skip elements
which don't pass the check but are not new/changed (in
self.obj_list).

Many thanks for the effort. Good work!

Cheers,

Dejan

> +
> +        no_clash = 1
> +        for param, resources in clash_dict.items():
> +            if len(resources) > 1:
> +                no_clash = 0
> +                msg = 'Resources %s violate uniqueness for parameter "%s": "%s"' %\
> +                        (",".join(sorted(resources)), param[3], param[4])
> +                common_warning(msg)
> +
> +        return no_clash
> +
>      def commit(self,cmd,force = None):
>          "usage: commit [force]"
>          if force and force != "force":
> @@ -1523,7 +1572,8 @@
>          rc1 = cib_factory.is_current_cib_equal()
>          rc2 = cib_factory.is_cib_empty() or \
>              self._verify(mkset_obj("xml","changed"),mkset_obj("xml"))
> -        if rc1 and rc2:
> +        rc3 = self.__check_unique_clash()
> +        if rc1 and rc2 and rc3:
>              return cib_factory.commit()
>          if force or user_prefs.get_force():
>              common_info("commit forced")



> _______________________________________________
> Pacemaker mailing list: Pacemaker at oss.clusterlabs.org
> http://oss.clusterlabs.org/mailman/listinfo/pacemaker
> 
> Project Home: http://www.clusterlabs.org
> Getting started: http://www.clusterlabs.org/doc/Cluster_from_Scratch.pdf
> Bugs: http://developerbugs.linux-foundation.org/enter_bug.cgi?product=Pacemaker




More information about the Pacemaker mailing list