ref: 7f84e2e246aa9a9b5adb28ccd86fc61635d5c0ea
dir: /objectstore/packed/delta_apply.go/
package packed
import (
"fmt"
deltaapply "codeberg.org/lindenii/furgit/format/delta/apply"
packfmt "codeberg.org/lindenii/furgit/format/pack"
"codeberg.org/lindenii/furgit/objecttype"
)
// deltaResolveContent resolves one object's content bytes from its pack location.
func (store *Store) deltaResolveContent(start location) (objecttype.Type, []byte, error) {
chain, err := store.deltaBuildChain(start)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
pack, meta, err := store.entryMetaAt(start)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
declaredSize := meta.size
if !packfmt.IsBaseObjectType(meta.ty) {
declaredSize, err = deltaDeclaredSizeAt(pack, meta.dataOffset)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
}
return store.deltaResolveChain(chain, declaredSize)
}
// deltaResolveChain resolves one object chain into content bytes.
func (store *Store) deltaResolveChain(chain deltaChain, declaredSize int64) (objecttype.Type, []byte, error) {
ty, out, nextDelta, err := store.deltaResolveChainStart(chain)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
for i := nextDelta; i >= 0; i-- {
node := chain.deltas[i]
pack, err := store.openPack(node.loc.packName)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
delta, err := inflateAt(pack, node.dataOffset, -1)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
out, err = deltaapply.Apply(out, delta)
if err != nil {
return objecttype.TypeInvalid, nil, err
}
store.cacheMu.Lock()
store.deltaCache.add(
deltaBaseKey{packName: node.loc.packName, offset: node.loc.offset},
ty,
out,
)
store.cacheMu.Unlock()
}
if int64(len(out)) != declaredSize {
return objecttype.TypeInvalid, nil, fmt.Errorf(
"objectstore/packed: resolved content size mismatch: got %d want %d",
len(out),
declaredSize,
)
}
if ty != chain.baseType {
return objecttype.TypeInvalid, nil, fmt.Errorf(
"objectstore/packed: resolved content type mismatch: got %d want %d",
ty,
chain.baseType,
)
}
return ty, out, nil
}
// deltaResolveChainStart finds the nearest cached chain node or inflates the
// innermost base object. It returns the starting bytes and the next delta index
// to apply in reverse order.
func (store *Store) deltaResolveChainStart(chain deltaChain) (objecttype.Type, []byte, int, error) {
for i, node := range chain.deltas {
store.cacheMu.RLock()
ty, out, ok := store.deltaCache.get(
deltaBaseKey{packName: node.loc.packName, offset: node.loc.offset},
)
store.cacheMu.RUnlock()
if ok {
return ty, out, i - 1, nil
}
}
store.cacheMu.RLock()
ty, out, ok := store.deltaCache.get(
deltaBaseKey{packName: chain.baseLoc.packName, offset: chain.baseLoc.offset},
)
store.cacheMu.RUnlock()
if ok {
return ty, out, len(chain.deltas) - 1, nil
}
pack, meta, err := store.entryMetaAt(chain.baseLoc)
if err != nil {
return objecttype.TypeInvalid, nil, 0, err
}
if !packfmt.IsBaseObjectType(meta.ty) {
return objecttype.TypeInvalid, nil, 0, fmt.Errorf("objectstore/packed: delta chain base is not a base object")
}
base, err := inflateAt(pack, meta.dataOffset, meta.size)
if err != nil {
return objecttype.TypeInvalid, nil, 0, err
}
store.cacheMu.Lock()
store.deltaCache.add(
deltaBaseKey{packName: chain.baseLoc.packName, offset: chain.baseLoc.offset},
meta.ty,
base,
)
store.cacheMu.Unlock()
return meta.ty, base, len(chain.deltas) - 1, nil
}