// Copyright 2014 Google Inc. All rights reserved. // // 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 parser import ( "fmt" "io" "math" "sort" ) func AddStringToList(list *List, s string) (modified bool) { for _, v := range list.Values { if v.Type() != StringType { panic(fmt.Errorf("expected string in list, got %s", v.Type())) } if sv, ok := v.(*String); ok && sv.Value == s { // string already exists return false } } list.Values = append(list.Values, &String{ LiteralPos: list.RBracePos, Value: s, }) return true } func RemoveStringFromList(list *List, s string) (modified bool) { for i, v := range list.Values { if v.Type() != StringType { panic(fmt.Errorf("expected string in list, got %s", v.Type())) } if sv, ok := v.(*String); ok && sv.Value == s { list.Values = append(list.Values[:i], list.Values[i+1:]...) return true } } return false } // A Patch represents a region of a text buffer to be replaced [Start, End) and its Replacement type Patch struct { Start, End int Replacement string } // A PatchList is a list of sorted, non-overlapping Patch objects type PatchList []Patch type PatchOverlapError error // Add adds a Patch to a PatchList. It returns a PatchOverlapError if the patch cannot be added. func (list *PatchList) Add(start, end int, replacement string) error { patch := Patch{start, end, replacement} if patch.Start > patch.End { return fmt.Errorf("invalid patch, start %d is after end %d", patch.Start, patch.End) } for _, p := range *list { if (patch.Start >= p.Start && patch.Start < p.End) || (patch.End >= p.Start && patch.End < p.End) || (p.Start >= patch.Start && p.Start < patch.End) || (p.Start == patch.Start && p.End == patch.End) { return PatchOverlapError(fmt.Errorf("new patch %d-%d overlaps with existing patch %d-%d", patch.Start, patch.End, p.Start, p.End)) } } *list = append(*list, patch) list.sort() return nil } func (list *PatchList) sort() { sort.SliceStable(*list, func(i, j int) bool { return (*list)[i].Start < (*list)[j].Start }) } // Apply applies all the Patch objects in PatchList to the data from an input ReaderAt to an output Writer. func (list *PatchList) Apply(in io.ReaderAt, out io.Writer) error { var offset int64 for _, patch := range *list { toWrite := int64(patch.Start) - offset written, err := io.Copy(out, io.NewSectionReader(in, offset, toWrite)) if err != nil { return err } offset += toWrite if written != toWrite { return fmt.Errorf("unexpected EOF at %d", offset) } _, err = io.WriteString(out, patch.Replacement) if err != nil { return err } offset += int64(patch.End - patch.Start) } _, err := io.Copy(out, io.NewSectionReader(in, offset, math.MaxInt64-offset)) return err }