blob: 07c3388f9845dca70c758711ed40534b2377ffd3 [file] [log] [blame]
/*
* Copyright 2018-present Open Networking Foundation
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
* http://www.apache.org/licenses/LICENSE-2.0
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package model
import (
"bytes"
"crypto/md5"
"fmt"
"reflect"
"sort"
)
var (
RevisionCache = make(map[string]interface{})
)
type Revision struct {
Config *DataRevision
Children map[string][]*Revision
Hash string
branch *Branch
WeakRef string
}
func NewRevision(branch *Branch, data interface{}, children map[string][]*Revision) *Revision {
cr := &Revision{}
cr.branch = branch
cr.Config = NewDataRevision(data)
cr.Children = children
cr.finalize()
return cr
}
func (cr *Revision) finalize() {
cr.Hash = cr.hashContent()
if _, exists := RevisionCache[cr.Hash]; !exists {
RevisionCache[cr.Hash] = cr
}
if _, exists := RevisionCache[cr.Config.Hash]; !exists {
RevisionCache[cr.Config.Hash] = cr.Config
} else {
cr.Config = RevisionCache[cr.Config.Hash].(*DataRevision)
}
}
func (cr *Revision) hashContent() string {
var buffer bytes.Buffer
var childrenKeys []string
if cr.Config != nil {
buffer.WriteString(cr.Config.Hash)
}
for key, _ := range cr.Children {
childrenKeys = append(childrenKeys, key)
}
sort.Strings(childrenKeys)
if cr.Children != nil && len(cr.Children) > 0 {
// Loop through sorted Children keys
for _, key := range childrenKeys {
for _, child := range cr.Children[key] {
buffer.WriteString(child.Hash)
}
}
}
return fmt.Sprintf("%x", md5.Sum(buffer.Bytes()))[:12]
}
func (cr *Revision) getData() interface{} {
if cr.Config == nil {
return nil
}
return cr.Config.Data
}
func (cr *Revision) getNode() *Node {
return cr.branch.node
}
func (cr *Revision) getType() reflect.Type {
// TODO: what is this returning really?
return reflect.TypeOf(cr.getData())
}
func (cr *Revision) clearHash() {
cr.Hash = ""
}
func (cr *Revision) Get(depth int) interface{} {
originalData := cr.getData()
data := Clone(originalData)
if depth > 0 {
for fieldName, field := range ChildrenFields(cr.getType()) {
if field.IsContainer {
for _, rev := range cr.Children[fieldName] {
childData := rev.Get(depth - 1)
childDataHolder := GetAttributeValue(data, fieldName, 0)
// TODO: merge with childData
fmt.Printf("data:%+v, dataHolder:%+v", childData, childDataHolder)
}
} else {
rev := cr.Children[fieldName][0]
childData := rev.Get(depth - 1)
childDataHolder := GetAttributeValue(data, fieldName, 0)
// TODO: merge with childData
fmt.Printf("data:%+v, dataHolder:%+v", childData, childDataHolder)
}
}
}
return data
}
func (cr *Revision) UpdateData(data interface{}, branch *Branch) *Revision {
newRev := Clone(cr).(*Revision)
newRev.branch = branch
newRev.Config = data.(*DataRevision)
newRev.finalize()
return newRev
}
func (cr *Revision) UpdateChildren(name string, children []*Revision, branch *Branch) *Revision {
newChildren := Clone(cr.Children).(map[string][]*Revision)
newChildren[name] = children
newRev := Clone(cr).(*Revision)
newRev.branch = branch
newRev.Children = newChildren
newRev.finalize()
return newRev
}
func (cr *Revision) UpdateAllChildren(children map[string][]*Revision, branch *Branch) *Revision {
newRev := Clone(cr).(*Revision)
newRev.branch = branch
newRev.Children = children
newRev.finalize()
return newRev
}