sophia ea87b6824d
Upgrade bolt to bbolt
boltdb/bolt is no longer a maintained project. bbolt is the CoreOS
fork that the author of boltdb suggests using as a replacement.
2022-04-25 12:24:34 -05:00

134 lines
3.1 KiB
Go

package singleprocess
import (
bolt "go.etcd.io/bbolt"
"github.com/hashicorp/go-hclog"
"github.com/hashicorp/vagrant-plugin-sdk/proto/vagrant_plugin_sdk"
"github.com/hashicorp/vagrant/internal/server"
"github.com/hashicorp/vagrant/internal/server/proto/vagrant_server"
"github.com/hashicorp/vagrant/internal/server/singleprocess/state"
"github.com/hashicorp/vagrant/internal/serverconfig"
)
// service implements the gRPC service for the server.
type service struct {
// state is the state management interface that provides functions for
// safely mutating server state.
state *state.State
// id is our unique server ID.
id string
vagrant_server.UnimplementedVagrantServer
vagrant_plugin_sdk.UnimplementedTargetServiceServer
vagrant_plugin_sdk.UnimplementedProjectServiceServer
vagrant_plugin_sdk.UnimplementedVagrantfileServiceServer
}
// New returns a Vagrant server implementation that uses BoltDB plus
// in-memory locks to operate safely.
func New(opts ...Option) (vagrant_server.VagrantServer, error) {
var s service
var cfg config
for _, opt := range opts {
if err := opt(&s, &cfg); err != nil {
return nil, err
}
}
log := cfg.log
if log == nil {
log = hclog.L()
}
// Initialize our state
st, err := state.New(log, cfg.db)
if err != nil {
log.Trace("state initialization failed", "error", err)
return nil, err
}
s.state = st
// If we don't have a server ID, set that.
id, err := st.ServerIdGet()
if err != nil {
return nil, err
}
if id == "" {
id, err = server.Id()
if err != nil {
return nil, err
}
if err := st.ServerIdSet(id); err != nil {
return nil, err
}
}
s.id = id
// Set specific server config for the deployment entrypoint binaries
if scfg := cfg.serverConfig; scfg != nil && scfg.CEBConfig != nil && scfg.CEBConfig.Addr != "" {
// only one advertise address can be configured
addr := &vagrant_server.ServerConfig_AdvertiseAddr{
Addr: scfg.CEBConfig.Addr,
Tls: scfg.CEBConfig.TLSEnabled,
TlsSkipVerify: scfg.CEBConfig.TLSSkipVerify,
}
conf := &vagrant_server.ServerConfig{
AdvertiseAddrs: []*vagrant_server.ServerConfig_AdvertiseAddr{addr},
}
if err := s.state.ServerConfigSet(conf); err != nil {
return nil, err
}
}
return &s, nil
}
type config struct {
db *bolt.DB
serverConfig *serverconfig.Config
log hclog.Logger
acceptUrlTerms bool
}
type Option func(*service, *config) error
// WithDB sets the Bolt DB for use with the server.
func WithDB(db *bolt.DB) Option {
return func(s *service, cfg *config) error {
cfg.db = db
return nil
}
}
// WithConfig sets the server config in use with this server.
func WithConfig(scfg *serverconfig.Config) Option {
return func(s *service, cfg *config) error {
cfg.serverConfig = scfg
return nil
}
}
// WithLogger sets the logger for use with the server.
func WithLogger(log hclog.Logger) Option {
return func(s *service, cfg *config) error {
cfg.log = log
return nil
}
}
func WithAcceptURLTerms(accept bool) Option {
return func(s *service, cfg *config) error {
cfg.acceptUrlTerms = true
return nil
}
}
var _ vagrant_server.VagrantServer = (*service)(nil)