// Mgmt // Copyright (C) 2013-2018+ James Shubin and the project contributors // Written by James Shubin and the project contributors // // This program is free software: you can redistribute it and/or modify // it under the terms of the GNU General Public License as published by // the Free Software Foundation, either version 3 of the License, or // (at your option) any later version. // // This program is distributed in the hope that it will be useful, // but WITHOUT ANY WARRANTY; without even the implied warranty of // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the // GNU General Public License for more details. // // You should have received a copy of the GNU General Public License // along with this program. If not, see . package engine import ( "fmt" "github.com/purpleidea/mgmt/pgraph" ) // ResCmp compares two resources by checking multiple aspects. This is the main // entry point for running all the compare steps on two resource. func ResCmp(r1, r2 Res) error { if r1.Kind() != r2.Kind() { return fmt.Errorf("kind differs") } if r1.Name() != r2.Name() { return fmt.Errorf("name differs") } if err := r1.Cmp(r2); err != nil { return err } // compare meta params for resources with auto edges r1e, ok1 := r1.(EdgeableRes) r2e, ok2 := r2.(EdgeableRes) if ok1 != ok2 { return fmt.Errorf("edgeable differs") // they must be different (optional) } if ok1 && ok2 { if r1e.AutoEdgeMeta().Cmp(r2e.AutoEdgeMeta()) != nil { return fmt.Errorf("autoedge differs") } } // compare meta params for resources with auto grouping r1g, ok1 := r1.(GroupableRes) r2g, ok2 := r2.(GroupableRes) if ok1 != ok2 { return fmt.Errorf("groupable differs") // they must be different (optional) } if ok1 && ok2 { if r1g.AutoGroupMeta().Cmp(r2g.AutoGroupMeta()) != nil { return fmt.Errorf("autogroup differs") } // if resources are grouped, are the groups the same? if i, j := r1g.GetGroup(), r2g.GetGroup(); len(i) != len(j) { return fmt.Errorf("autogroup groups differ") } else if len(i) > 0 { // trick the golinter // Sort works with Res, so convert the lists to that iRes := []Res{} for _, r := range i { res := r.(Res) iRes = append(iRes, res) } jRes := []Res{} for _, r := range j { res := r.(Res) jRes = append(jRes, res) } ix, jx := Sort(iRes), Sort(jRes) // now sort :) for k := range ix { // compare sub resources if err := ResCmp(ix[k], jx[k]); err != nil { return err } } } } return nil } // VertexCmpFn returns if two vertices are equivalent. It errors if they can't // be compared because one is not a vertex. This returns true if equal. // TODO: shouldn't the first argument be an `error` instead? func VertexCmpFn(v1, v2 pgraph.Vertex) (bool, error) { r1, ok := v1.(Res) if !ok { return false, fmt.Errorf("v1 is not a Res") } r2, ok := v2.(Res) if !ok { return false, fmt.Errorf("v2 is not a Res") } if ResCmp(r1, r2) != nil { return false, nil } return true, nil } // EdgeCmpFn returns if two edges are equivalent. It errors if they can't be // compared because one is not an edge. This returns true if equal. // TODO: shouldn't the first argument be an `error` instead? func EdgeCmpFn(e1, e2 pgraph.Edge) (bool, error) { edge1, ok := e1.(*Edge) if !ok { return false, fmt.Errorf("e1 is not an Edge") } edge2, ok := e2.(*Edge) if !ok { return false, fmt.Errorf("e2 is not an Edge") } return edge1.Cmp(edge2) == nil, nil }