1 // Copyright (C) The Arvados Authors. All rights reserved.
3 // SPDX-License-Identifier: AGPL-3.0
10 "git.arvados.org/arvados.git/sdk/go/arvados"
13 // Replica is a file on disk (or object in an S3 bucket, or blob in an
14 // Azure storage container, etc.) as reported in a keepstore index
21 // BlockState indicates the desired storage class and number of
22 // replicas (according to the collections we know about) and the
23 // replicas actually stored (according to the keepstore indexes we
25 type BlockState struct {
26 Refs map[string]bool // pdh => true (only tracked when len(Replicas)==0)
29 Desired map[string]int
30 // TODO: Support combinations of classes ("private + durable")
31 // by replacing the map[string]int with a map[*[]string]int
32 // here, where the map keys come from a pool of semantically
33 // distinct class combinations.
35 // TODO: Use a pool of semantically distinct Desired maps to
36 // conserve memory (typically there are far more BlockState
37 // objects in memory than distinct Desired profiles).
40 var defaultClasses = []string{"default"}
42 func (bs *BlockState) addReplica(r Replica) {
43 bs.Replicas = append(bs.Replicas, r)
44 // Free up memory wasted by tracking PDHs that will never be
45 // reported (see comment in increaseDesired)
49 func (bs *BlockState) increaseDesired(pdh string, classes []string, n int) {
50 if pdh != "" && len(bs.Replicas) == 0 {
51 // Note we only track PDHs if there's a possibility
52 // that we will report the list of referring PDHs,
53 // i.e., if we haven't yet seen a replica.
55 bs.Refs = map[string]bool{}
60 if len(classes) == 0 {
61 classes = defaultClasses
63 for _, class := range classes {
64 if bs.Desired == nil {
65 bs.Desired = map[string]int{class: n}
66 } else if d, ok := bs.Desired[class]; !ok || d < n {
72 // BlockStateMap is a goroutine-safe wrapper around a
73 // map[arvados.SizedDigest]*BlockState.
74 type BlockStateMap struct {
75 entries map[arvados.SizedDigest]*BlockState
79 // NewBlockStateMap returns a newly allocated BlockStateMap.
80 func NewBlockStateMap() *BlockStateMap {
81 return &BlockStateMap{
82 entries: make(map[arvados.SizedDigest]*BlockState),
86 // return a BlockState entry, allocating a new one if needed. (Private
87 // method: not goroutine-safe.)
88 func (bsm *BlockStateMap) get(blkid arvados.SizedDigest) *BlockState {
89 // TODO? Allocate BlockState structs a slice at a time,
90 // instead of one at a time.
91 blk := bsm.entries[blkid]
94 bsm.entries[blkid] = blk
99 // Apply runs f on each entry in the map.
100 func (bsm *BlockStateMap) Apply(f func(arvados.SizedDigest, *BlockState)) {
102 defer bsm.mutex.Unlock()
104 for blkid, blk := range bsm.entries {
109 // AddReplicas updates the map to indicate that mnt has a replica of
110 // each block in idx.
111 func (bsm *BlockStateMap) AddReplicas(mnt *KeepMount, idx []arvados.KeepServiceIndexEntry) {
113 defer bsm.mutex.Unlock()
115 for _, ent := range idx {
116 bsm.get(ent.SizedDigest).addReplica(Replica{
123 // IncreaseDesired updates the map to indicate the desired replication
124 // for the given blocks in the given storage class is at least n.
126 // If pdh is non-empty, it will be tracked and reported in the "lost
128 func (bsm *BlockStateMap) IncreaseDesired(pdh string, classes []string, n int, blocks []arvados.SizedDigest) {
130 defer bsm.mutex.Unlock()
132 for _, blkid := range blocks {
133 bsm.get(blkid).increaseDesired(pdh, classes, n)
137 // GetConfirmedReplication returns the replication level of the given
138 // blocks, considering only the specified storage classes.
140 // If len(classes)==0, returns the replication level without regard to
143 // Safe to call concurrently with other calls to GetCurrent, but not
144 // with different BlockStateMap methods.
145 func (bsm *BlockStateMap) GetConfirmedReplication(blkids []arvados.SizedDigest, classes []string) int {
146 defaultClasses := map[string]bool{"default": true}
148 for _, blkid := range blkids {
150 perclass := make(map[string]int, len(classes))
151 for _, c := range classes {
154 for _, r := range bsm.get(blkid).Replicas {
155 total += r.KeepMount.Replication
156 mntclasses := r.KeepMount.StorageClasses
157 if len(mntclasses) == 0 {
158 mntclasses = defaultClasses
160 for c := range mntclasses {
163 // Don't care about this storage class
166 perclass[c] = n + r.KeepMount.Replication
172 for _, n := range perclass {
176 if n < min || min == 0 {
180 if len(perclass) == 0 && (min == 0 || min > total) {