forked from vitessio/vitess
/
tablet_provider.go
86 lines (69 loc) · 2.72 KB
/
tablet_provider.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
package worker
import (
"fmt"
"golang.org/x/net/context"
"github.com/youtube/vitess/go/vt/discovery"
"github.com/youtube/vitess/go/vt/proto/topodata"
"github.com/youtube/vitess/go/vt/topo"
"github.com/youtube/vitess/go/vt/topo/topoproto"
topodatapb "github.com/youtube/vitess/go/vt/proto/topodata"
)
// tabletProvider defines an interface to pick a tablet for reading data.
type tabletProvider interface {
// getTablet returns a tablet.
getTablet() (*topodatapb.Tablet, error)
// returnTablet must be called after the tablet is no longer used and e.g.
// TabletTracker.Untrack() should get called for it.
returnTablet(*topodata.Tablet)
// description returns a string which can be used in error messages e.g.
// the name of the keyspace and the shard.
description() string
}
// singleTabletProvider implements the tabletProvider interface and always
// returns the one tablet which was set at creation.
type singleTabletProvider struct {
ctx context.Context
ts topo.Server
alias *topodatapb.TabletAlias
}
func newSingleTabletProvider(ctx context.Context, ts topo.Server, alias *topodatapb.TabletAlias) *singleTabletProvider {
return &singleTabletProvider{ctx, ts, alias}
}
func (p *singleTabletProvider) getTablet() (*topodatapb.Tablet, error) {
shortCtx, cancel := context.WithTimeout(p.ctx, *remoteActionsTimeout)
tablet, err := p.ts.GetTablet(shortCtx, p.alias)
cancel()
if err != nil {
return nil, fmt.Errorf("failed to resolve tablet alias: %v err: %v", topoproto.TabletAliasString(p.alias), err)
}
return tablet.Tablet, err
}
func (p *singleTabletProvider) returnTablet(*topodata.Tablet) {}
func (p *singleTabletProvider) description() string {
return topoproto.TabletAliasString(p.alias)
}
// shardTabletProvider returns a random healthy RDONLY tablet for a given
// keyspace and shard. It uses the HealthCheck module to retrieve the tablets.
type shardTabletProvider struct {
tsc *discovery.TabletStatsCache
tracker *TabletTracker
keyspace string
shard string
}
func newShardTabletProvider(tsc *discovery.TabletStatsCache, tracker *TabletTracker, keyspace, shard string) *shardTabletProvider {
return &shardTabletProvider{tsc, tracker, keyspace, shard}
}
func (p *shardTabletProvider) getTablet() (*topodatapb.Tablet, error) {
// Pick any healthy serving tablet.
tablets := p.tsc.GetHealthyTabletStats(p.keyspace, p.shard, topodatapb.TabletType_RDONLY)
if len(tablets) == 0 {
return nil, fmt.Errorf("%v: no healthy RDONLY tablets available", p.description())
}
return p.tracker.Track(tablets), nil
}
func (p *shardTabletProvider) returnTablet(t *topodatapb.Tablet) {
p.tracker.Untrack(t.Alias)
}
func (p *shardTabletProvider) description() string {
return topoproto.KeyspaceShardString(p.keyspace, p.shard)
}