Done with: ack '2022+' -l | xargs sed -i -e 's/2022+/2023+/g' Checked manually with: git add -p Hello to future James from 2024, and Happy Hacking!
141 lines
3.9 KiB
Go
141 lines
3.9 KiB
Go
// Mgmt
|
|
// Copyright (C) 2013-2023+ James Shubin and the project contributors
|
|
// Written by James Shubin <james@shubin.ca> 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 <http://www.gnu.org/licenses/>.
|
|
|
|
package autogroup
|
|
|
|
import (
|
|
"fmt"
|
|
|
|
"github.com/purpleidea/mgmt/pgraph"
|
|
)
|
|
|
|
// baseGrouper is the base type for implementing the AutoGrouper interface.
|
|
type baseGrouper struct {
|
|
graph *pgraph.Graph // store a pointer to the graph
|
|
vertices []pgraph.Vertex // cached list of vertices
|
|
i int
|
|
j int
|
|
done bool
|
|
}
|
|
|
|
// Name provides a friendly name for the logs to see.
|
|
func (ag *baseGrouper) Name() string {
|
|
return "baseGrouper"
|
|
}
|
|
|
|
// Init is called only once and before using other AutoGrouper interface methods
|
|
// the name method is the only exception: call it any time without side effects!
|
|
func (ag *baseGrouper) Init(g *pgraph.Graph) error {
|
|
if ag.graph != nil {
|
|
return fmt.Errorf("the init method has already been called")
|
|
}
|
|
ag.graph = g // pointer
|
|
ag.vertices = ag.graph.VerticesSorted() // cache in deterministic order!
|
|
ag.i = 0
|
|
ag.j = 0
|
|
if len(ag.vertices) == 0 { // empty graph
|
|
ag.done = true
|
|
return nil
|
|
}
|
|
return nil
|
|
}
|
|
|
|
// VertexNext is a simple iterator that loops through vertex (pair) combinations
|
|
// an intelligent algorithm would selectively offer only valid pairs of vertices
|
|
// these should satisfy logical grouping requirements for the autogroup designs!
|
|
// the desired algorithms can override, but keep this method as a base iterator!
|
|
func (ag *baseGrouper) VertexNext() (v1, v2 pgraph.Vertex, err error) {
|
|
// this does a for v... { for w... { return v, w }} but stepwise!
|
|
l := len(ag.vertices)
|
|
if ag.i < l {
|
|
v1 = ag.vertices[ag.i]
|
|
}
|
|
if ag.j < l {
|
|
v2 = ag.vertices[ag.j]
|
|
}
|
|
|
|
// in case the vertex was deleted
|
|
if !ag.graph.HasVertex(v1) {
|
|
v1 = nil
|
|
}
|
|
if !ag.graph.HasVertex(v2) {
|
|
v2 = nil
|
|
}
|
|
|
|
// two nested loops...
|
|
if ag.j < l {
|
|
ag.j++
|
|
}
|
|
if ag.j == l {
|
|
ag.j = 0
|
|
if ag.i < l {
|
|
ag.i++
|
|
}
|
|
if ag.i == l {
|
|
ag.done = true
|
|
}
|
|
}
|
|
// TODO: is this index swap better or even valid?
|
|
//if ag.i < l {
|
|
// ag.i++
|
|
//}
|
|
//if ag.i == l {
|
|
// ag.i = 0
|
|
// if ag.j < l {
|
|
// ag.j++
|
|
// }
|
|
// if ag.j == l {
|
|
// ag.done = true
|
|
// }
|
|
//}
|
|
|
|
return
|
|
}
|
|
|
|
// VertexCmp can be used in addition to an overridding implementation.
|
|
func (ag *baseGrouper) VertexCmp(v1, v2 pgraph.Vertex) error {
|
|
if v1 == nil || v2 == nil {
|
|
return fmt.Errorf("the vertex is nil")
|
|
}
|
|
if v1 == v2 { // skip yourself
|
|
return fmt.Errorf("the vertices are the same")
|
|
}
|
|
|
|
return nil // success
|
|
}
|
|
|
|
// VertexMerge needs to be overridden to add the actual merging functionality.
|
|
func (ag *baseGrouper) VertexMerge(v1, v2 pgraph.Vertex) (v pgraph.Vertex, err error) {
|
|
return nil, fmt.Errorf("vertexMerge needs to be overridden")
|
|
}
|
|
|
|
// EdgeMerge can be overridden, since it just simply returns the first edge.
|
|
func (ag *baseGrouper) EdgeMerge(e1, e2 pgraph.Edge) pgraph.Edge {
|
|
return e1 // noop
|
|
}
|
|
|
|
// VertexTest processes the results of the grouping for the algorithm to know
|
|
// return an error if something went horribly wrong, and bool false to stop.
|
|
func (ag *baseGrouper) VertexTest(b bool) (bool, error) {
|
|
// NOTE: this particular baseGrouper version doesn't track what happens
|
|
// because since we iterate over every pair, we don't care which merge!
|
|
if ag.done {
|
|
return false, nil
|
|
}
|
|
return true, nil
|
|
}
|