mirror of
				https://github.com/1Password/onepassword-operator.git
				synced 2025-10-24 16:30:47 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			337 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			337 lines
		
	
	
		
			8.1 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| package jsonpatch
 | |
| 
 | |
| import (
 | |
| 	"bytes"
 | |
| 	"encoding/json"
 | |
| 	"fmt"
 | |
| 	"reflect"
 | |
| 	"strings"
 | |
| )
 | |
| 
 | |
| var errBadJSONDoc = fmt.Errorf("invalid JSON Document")
 | |
| 
 | |
| type JsonPatchOperation = Operation
 | |
| 
 | |
| type Operation struct {
 | |
| 	Operation string      `json:"op"`
 | |
| 	Path      string      `json:"path"`
 | |
| 	Value     interface{} `json:"value,omitempty"`
 | |
| }
 | |
| 
 | |
| func (j *Operation) Json() string {
 | |
| 	b, _ := json.Marshal(j)
 | |
| 	return string(b)
 | |
| }
 | |
| 
 | |
| func (j *Operation) MarshalJSON() ([]byte, error) {
 | |
| 	var b bytes.Buffer
 | |
| 	b.WriteString("{")
 | |
| 	b.WriteString(fmt.Sprintf(`"op":"%s"`, j.Operation))
 | |
| 	b.WriteString(fmt.Sprintf(`,"path":"%s"`, j.Path))
 | |
| 	// Consider omitting Value for non-nullable operations.
 | |
| 	if j.Value != nil || j.Operation == "replace" || j.Operation == "add" {
 | |
| 		v, err := json.Marshal(j.Value)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 		b.WriteString(`,"value":`)
 | |
| 		b.Write(v)
 | |
| 	}
 | |
| 	b.WriteString("}")
 | |
| 	return b.Bytes(), nil
 | |
| }
 | |
| 
 | |
| type ByPath []Operation
 | |
| 
 | |
| func (a ByPath) Len() int           { return len(a) }
 | |
| func (a ByPath) Swap(i, j int)      { a[i], a[j] = a[j], a[i] }
 | |
| func (a ByPath) Less(i, j int) bool { return a[i].Path < a[j].Path }
 | |
| 
 | |
| func NewPatch(operation, path string, value interface{}) Operation {
 | |
| 	return Operation{Operation: operation, Path: path, Value: value}
 | |
| }
 | |
| 
 | |
| // CreatePatch creates a patch as specified in http://jsonpatch.com/
 | |
| //
 | |
| // 'a' is original, 'b' is the modified document. Both are to be given as json encoded content.
 | |
| // The function will return an array of JsonPatchOperations
 | |
| //
 | |
| // An error will be returned if any of the two documents are invalid.
 | |
| func CreatePatch(a, b []byte) ([]Operation, error) {
 | |
| 	var aI interface{}
 | |
| 	var bI interface{}
 | |
| 	err := json.Unmarshal(a, &aI)
 | |
| 	if err != nil {
 | |
| 		return nil, errBadJSONDoc
 | |
| 	}
 | |
| 	err = json.Unmarshal(b, &bI)
 | |
| 	if err != nil {
 | |
| 		return nil, errBadJSONDoc
 | |
| 	}
 | |
| 	return handleValues(aI, bI, "", []Operation{})
 | |
| }
 | |
| 
 | |
| // Returns true if the values matches (must be json types)
 | |
| // The types of the values must match, otherwise it will always return false
 | |
| // If two map[string]interface{} are given, all elements must match.
 | |
| func matchesValue(av, bv interface{}) bool {
 | |
| 	if reflect.TypeOf(av) != reflect.TypeOf(bv) {
 | |
| 		return false
 | |
| 	}
 | |
| 	switch at := av.(type) {
 | |
| 	case string:
 | |
| 		bt, ok := bv.(string)
 | |
| 		if ok && bt == at {
 | |
| 			return true
 | |
| 		}
 | |
| 	case float64:
 | |
| 		bt, ok := bv.(float64)
 | |
| 		if ok && bt == at {
 | |
| 			return true
 | |
| 		}
 | |
| 	case bool:
 | |
| 		bt, ok := bv.(bool)
 | |
| 		if ok && bt == at {
 | |
| 			return true
 | |
| 		}
 | |
| 	case map[string]interface{}:
 | |
| 		bt, ok := bv.(map[string]interface{})
 | |
| 		if !ok {
 | |
| 			return false
 | |
| 		}
 | |
| 		for key := range at {
 | |
| 			if !matchesValue(at[key], bt[key]) {
 | |
| 				return false
 | |
| 			}
 | |
| 		}
 | |
| 		for key := range bt {
 | |
| 			if !matchesValue(at[key], bt[key]) {
 | |
| 				return false
 | |
| 			}
 | |
| 		}
 | |
| 		return true
 | |
| 	case []interface{}:
 | |
| 		bt, ok := bv.([]interface{})
 | |
| 		if !ok {
 | |
| 			return false
 | |
| 		}
 | |
| 		if len(bt) != len(at) {
 | |
| 			return false
 | |
| 		}
 | |
| 		for key := range at {
 | |
| 			if !matchesValue(at[key], bt[key]) {
 | |
| 				return false
 | |
| 			}
 | |
| 		}
 | |
| 		for key := range bt {
 | |
| 			if !matchesValue(at[key], bt[key]) {
 | |
| 				return false
 | |
| 			}
 | |
| 		}
 | |
| 		return true
 | |
| 	}
 | |
| 	return false
 | |
| }
 | |
| 
 | |
| // From http://tools.ietf.org/html/rfc6901#section-4 :
 | |
| //
 | |
| // Evaluation of each reference token begins by decoding any escaped
 | |
| // character sequence.  This is performed by first transforming any
 | |
| // occurrence of the sequence '~1' to '/', and then transforming any
 | |
| // occurrence of the sequence '~0' to '~'.
 | |
| //   TODO decode support:
 | |
| //   var rfc6901Decoder = strings.NewReplacer("~1", "/", "~0", "~")
 | |
| 
 | |
| var rfc6901Encoder = strings.NewReplacer("~", "~0", "/", "~1")
 | |
| 
 | |
| func makePath(path string, newPart interface{}) string {
 | |
| 	key := rfc6901Encoder.Replace(fmt.Sprintf("%v", newPart))
 | |
| 	if path == "" {
 | |
| 		return "/" + key
 | |
| 	}
 | |
| 	if strings.HasSuffix(path, "/") {
 | |
| 		return path + key
 | |
| 	}
 | |
| 	return path + "/" + key
 | |
| }
 | |
| 
 | |
| // diff returns the (recursive) difference between a and b as an array of JsonPatchOperations.
 | |
| func diff(a, b map[string]interface{}, path string, patch []Operation) ([]Operation, error) {
 | |
| 	for key, bv := range b {
 | |
| 		p := makePath(path, key)
 | |
| 		av, ok := a[key]
 | |
| 		// value was added
 | |
| 		if !ok {
 | |
| 			patch = append(patch, NewPatch("add", p, bv))
 | |
| 			continue
 | |
| 		}
 | |
| 		// Types are the same, compare values
 | |
| 		var err error
 | |
| 		patch, err = handleValues(av, bv, p, patch)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	}
 | |
| 	// Now add all deleted values as nil
 | |
| 	for key := range a {
 | |
| 		_, found := b[key]
 | |
| 		if !found {
 | |
| 			p := makePath(path, key)
 | |
| 
 | |
| 			patch = append(patch, NewPatch("remove", p, nil))
 | |
| 		}
 | |
| 	}
 | |
| 	return patch, nil
 | |
| }
 | |
| 
 | |
| func handleValues(av, bv interface{}, p string, patch []Operation) ([]Operation, error) {
 | |
| 	{
 | |
| 		at := reflect.TypeOf(av)
 | |
| 		bt := reflect.TypeOf(bv)
 | |
| 		if at == nil && bt == nil {
 | |
| 			// do nothing
 | |
| 			return patch, nil
 | |
| 		} else if at == nil && bt != nil {
 | |
| 			return append(patch, NewPatch("add", p, bv)), nil
 | |
| 		} else if at != bt {
 | |
| 			// If types have changed, replace completely (preserves null in destination)
 | |
| 			return append(patch, NewPatch("replace", p, bv)), nil
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	var err error
 | |
| 	switch at := av.(type) {
 | |
| 	case map[string]interface{}:
 | |
| 		bt := bv.(map[string]interface{})
 | |
| 		patch, err = diff(at, bt, p, patch)
 | |
| 		if err != nil {
 | |
| 			return nil, err
 | |
| 		}
 | |
| 	case string, float64, bool:
 | |
| 		if !matchesValue(av, bv) {
 | |
| 			patch = append(patch, NewPatch("replace", p, bv))
 | |
| 		}
 | |
| 	case []interface{}:
 | |
| 		bt := bv.([]interface{})
 | |
| 		if isSimpleArray(at) && isSimpleArray(bt) {
 | |
| 			patch = append(patch, compareEditDistance(at, bt, p)...)
 | |
| 		} else {
 | |
| 			n := min(len(at), len(bt))
 | |
| 			for i := len(at) - 1; i >= n; i-- {
 | |
| 				patch = append(patch, NewPatch("remove", makePath(p, i), nil))
 | |
| 			}
 | |
| 			for i := n; i < len(bt); i++ {
 | |
| 				patch = append(patch, NewPatch("add", makePath(p, i), bt[i]))
 | |
| 			}
 | |
| 			for i := 0; i < n; i++ {
 | |
| 				var err error
 | |
| 				patch, err = handleValues(at[i], bt[i], makePath(p, i), patch)
 | |
| 				if err != nil {
 | |
| 					return nil, err
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	default:
 | |
| 		panic(fmt.Sprintf("Unknown type:%T ", av))
 | |
| 	}
 | |
| 	return patch, nil
 | |
| }
 | |
| 
 | |
| func isBasicType(a interface{}) bool {
 | |
| 	switch a.(type) {
 | |
| 	case string, float64, bool:
 | |
| 	default:
 | |
| 		return false
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| func isSimpleArray(a []interface{}) bool {
 | |
| 	for i := range a {
 | |
| 		switch a[i].(type) {
 | |
| 		case string, float64, bool:
 | |
| 		default:
 | |
| 			val := reflect.ValueOf(a[i])
 | |
| 			if val.Kind() == reflect.Map {
 | |
| 				for _, k := range val.MapKeys() {
 | |
| 					av := val.MapIndex(k)
 | |
| 					if av.Kind() == reflect.Ptr || av.Kind() == reflect.Interface {
 | |
| 						if av.IsNil() {
 | |
| 							continue
 | |
| 						}
 | |
| 						av = av.Elem()
 | |
| 					}
 | |
| 					if av.Kind() != reflect.String && av.Kind() != reflect.Float64 && av.Kind() != reflect.Bool {
 | |
| 						return false
 | |
| 					}
 | |
| 				}
 | |
| 				return true
 | |
| 			}
 | |
| 			return false
 | |
| 		}
 | |
| 	}
 | |
| 	return true
 | |
| }
 | |
| 
 | |
| // https://en.wikipedia.org/wiki/Wagner%E2%80%93Fischer_algorithm
 | |
| // Adapted from https://github.com/texttheater/golang-levenshtein
 | |
| func compareEditDistance(s, t []interface{}, p string) []Operation {
 | |
| 	m := len(s)
 | |
| 	n := len(t)
 | |
| 
 | |
| 	d := make([][]int, m+1)
 | |
| 	for i := 0; i <= m; i++ {
 | |
| 		d[i] = make([]int, n+1)
 | |
| 		d[i][0] = i
 | |
| 	}
 | |
| 	for j := 0; j <= n; j++ {
 | |
| 		d[0][j] = j
 | |
| 	}
 | |
| 
 | |
| 	for j := 1; j <= n; j++ {
 | |
| 		for i := 1; i <= m; i++ {
 | |
| 			if reflect.DeepEqual(s[i-1], t[j-1]) {
 | |
| 				d[i][j] = d[i-1][j-1] // no op required
 | |
| 			} else {
 | |
| 				del := d[i-1][j] + 1
 | |
| 				add := d[i][j-1] + 1
 | |
| 				rep := d[i-1][j-1] + 1
 | |
| 				d[i][j] = min(rep, min(add, del))
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return backtrace(s, t, p, m, n, d)
 | |
| }
 | |
| 
 | |
| func min(x int, y int) int {
 | |
| 	if y < x {
 | |
| 		return y
 | |
| 	}
 | |
| 	return x
 | |
| }
 | |
| 
 | |
| func backtrace(s, t []interface{}, p string, i int, j int, matrix [][]int) []Operation {
 | |
| 	if i > 0 && matrix[i-1][j]+1 == matrix[i][j] {
 | |
| 		op := NewPatch("remove", makePath(p, i-1), nil)
 | |
| 		return append([]Operation{op}, backtrace(s, t, p, i-1, j, matrix)...)
 | |
| 	}
 | |
| 	if j > 0 && matrix[i][j-1]+1 == matrix[i][j] {
 | |
| 		op := NewPatch("add", makePath(p, i), t[j-1])
 | |
| 		return append([]Operation{op}, backtrace(s, t, p, i, j-1, matrix)...)
 | |
| 	}
 | |
| 	if i > 0 && j > 0 && matrix[i-1][j-1]+1 == matrix[i][j] {
 | |
| 		if isBasicType(s[0]) {
 | |
| 			op := NewPatch("replace", makePath(p, i-1), t[j-1])
 | |
| 			return append([]Operation{op}, backtrace(s, t, p, i-1, j-1, matrix)...)
 | |
| 		}
 | |
| 
 | |
| 		p2, _ := handleValues(s[i-1], t[j-1], makePath(p, i-1), []Operation{})
 | |
| 		return append(p2, backtrace(s, t, p, i-1, j-1, matrix)...)
 | |
| 	}
 | |
| 	if i > 0 && j > 0 && matrix[i-1][j-1] == matrix[i][j] {
 | |
| 		return backtrace(s, t, p, i-1, j-1, matrix)
 | |
| 	}
 | |
| 	return []Operation{}
 | |
| }
 | 
