2016-01-25 06:41:39 +01:00
|
|
|
// Copyright 2016 The Chihaya Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by the BSD 2-Clause license,
|
|
|
|
// which can be found in the LICENSE file.
|
|
|
|
|
|
|
|
package store
|
|
|
|
|
|
|
|
import (
|
|
|
|
"errors"
|
|
|
|
"log"
|
|
|
|
"sync"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"gopkg.in/yaml.v2"
|
|
|
|
|
2016-03-03 02:18:55 +01:00
|
|
|
"github.com/chihaya/chihaya"
|
2016-01-25 06:41:39 +01:00
|
|
|
"github.com/chihaya/chihaya/server"
|
|
|
|
"github.com/chihaya/chihaya/tracker"
|
|
|
|
)
|
|
|
|
|
|
|
|
var theStore *Store
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
server.Register("store", constructor)
|
|
|
|
}
|
|
|
|
|
2016-03-03 02:18:55 +01:00
|
|
|
func constructor(srvcfg *chihaya.ServerConfig, tkr *tracker.Tracker) (server.Server, error) {
|
2016-01-25 06:41:39 +01:00
|
|
|
if theStore == nil {
|
|
|
|
cfg, err := newConfig(srvcfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, errors.New("store: invalid store config: " + err.Error())
|
|
|
|
}
|
|
|
|
|
2016-03-11 21:09:49 +01:00
|
|
|
ps, err := OpenPeerStore(&cfg.PeerStore)
|
2016-02-16 21:47:40 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-03-11 21:09:49 +01:00
|
|
|
ips, err := OpenIPStore(&cfg.IPStore)
|
2016-02-16 21:47:40 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-03-11 21:09:49 +01:00
|
|
|
ss, err := OpenStringStore(&cfg.StringStore)
|
2016-03-03 09:39:19 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2016-01-25 06:41:39 +01:00
|
|
|
theStore = &Store{
|
2016-02-16 21:47:40 +01:00
|
|
|
cfg: cfg,
|
|
|
|
tkr: tkr,
|
2016-03-02 22:18:34 +01:00
|
|
|
shutdown: make(chan struct{}),
|
2016-02-16 21:47:40 +01:00
|
|
|
PeerStore: ps,
|
|
|
|
IPStore: ips,
|
2016-03-03 09:39:19 +01:00
|
|
|
StringStore: ss,
|
2016-01-25 06:41:39 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return theStore, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
type Config struct {
|
2016-03-11 21:09:49 +01:00
|
|
|
Addr string `yaml:"addr"`
|
|
|
|
RequestTimeout time.Duration `yaml:"request_timeout"`
|
|
|
|
ReadTimeout time.Duration `yaml:"read_timeout"`
|
|
|
|
WriteTimeout time.Duration `yaml:"write_timeout"`
|
|
|
|
GCAfter time.Duration `yaml:"gc_after"`
|
|
|
|
PeerStore DriverConfig `yaml:"peer_store"`
|
|
|
|
IPStore DriverConfig `yaml:"ip_store"`
|
|
|
|
StringStore DriverConfig `yaml:"string_store"`
|
|
|
|
}
|
|
|
|
|
|
|
|
type DriverConfig struct {
|
|
|
|
Name string `yaml:"name"`
|
|
|
|
Config interface{} `yaml:"config"`
|
2016-01-25 06:41:39 +01:00
|
|
|
}
|
|
|
|
|
2016-03-03 02:18:55 +01:00
|
|
|
func newConfig(srvcfg *chihaya.ServerConfig) (*Config, error) {
|
2016-03-02 22:15:48 +01:00
|
|
|
bytes, err := yaml.Marshal(srvcfg.Config)
|
2016-01-25 06:41:39 +01:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
var cfg Config
|
|
|
|
err = yaml.Unmarshal(bytes, &cfg)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return &cfg, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// MustGetStore is used by middleware to access the store.
|
|
|
|
//
|
|
|
|
// This function calls log.Fatal if a server hasn't been already created by
|
|
|
|
// the server package.
|
|
|
|
func MustGetStore() *Store {
|
|
|
|
if theStore == nil {
|
|
|
|
log.Fatal("store middleware used without store server")
|
|
|
|
}
|
|
|
|
return theStore
|
|
|
|
}
|
|
|
|
|
|
|
|
type Store struct {
|
|
|
|
cfg *Config
|
|
|
|
tkr *tracker.Tracker
|
|
|
|
shutdown chan struct{}
|
|
|
|
wg sync.WaitGroup
|
|
|
|
|
|
|
|
PeerStore
|
2016-02-16 21:46:40 +01:00
|
|
|
IPStore
|
2016-03-03 09:39:19 +01:00
|
|
|
StringStore
|
2016-01-25 06:41:39 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Store) Start() {
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s *Store) Stop() {
|
|
|
|
close(s.shutdown)
|
|
|
|
s.wg.Wait()
|
|
|
|
}
|