blob: a6282d7f81ea6e6b6f56dcbb152b67a96725b077 [file] [log] [blame]
// Browspr is the browser version of WSPR, intended to communicate with javascript through postMessage.
package browspr
import (
"net"
"sync"
"time"
"veyron.io/veyron/veyron2"
"veyron.io/veyron/veyron2/ipc"
"veyron.io/veyron/veyron2/rt"
"veyron.io/veyron/veyron2/vlog"
"veyron.io/wspr/veyron/services/wsprd/account"
"veyron.io/wspr/veyron/services/wsprd/lib"
"veyron.io/wspr/veyron/services/wsprd/namespace"
"veyron.io/wspr/veyron/services/wsprd/principal"
)
func init() {
namespace.EpFormatter = lib.EndpointsToWs
}
// Browspr is an intermediary between our javascript code and the veyron network that allows our javascript library to use veyron.
type Browspr struct {
rt veyron2.Runtime
profileFactory func() veyron2.Profile
listenSpec ipc.ListenSpec
identdEP string
namespaceRoots []string
logger vlog.Logger
accountManager *account.AccountManager
postMessage func(instanceId int32, ty, msg string)
// Locks activeInstances
mu sync.Mutex
activeInstances map[int32]*pipe
}
// Create a new Browspr instance.
func NewBrowspr(postMessage func(instanceId int32, ty, msg string), profileFactory func() veyron2.Profile, listenSpec ipc.ListenSpec, identdEP string, namespaceRoots []string, opts ...veyron2.ROpt) *Browspr {
if listenSpec.Proxy == "" {
vlog.Fatalf("a veyron proxy must be set")
}
if identdEP == "" {
vlog.Fatalf("an identd server must be set")
}
runtime, err := rt.New(opts...)
if err != nil {
vlog.Fatalf("rt.New failed: %s", err)
}
wsNamespaceRoots, err := lib.EndpointsToWs(runtime, namespaceRoots)
if err != nil {
vlog.Fatal(err)
}
runtime.Namespace().SetRoots(wsNamespaceRoots...)
browspr := &Browspr{
profileFactory: profileFactory,
listenSpec: listenSpec,
identdEP: identdEP,
namespaceRoots: wsNamespaceRoots,
postMessage: postMessage,
rt: runtime,
logger: runtime.Logger(),
activeInstances: make(map[int32]*pipe),
}
// TODO(nlacasse, bjornick) use a serializer that can actually persist.
var principalManager *principal.PrincipalManager
if principalManager, err = principal.NewPrincipalManager(runtime.Principal(), &principal.InMemorySerializer{}); err != nil {
vlog.Fatalf("principal.NewPrincipalManager failed: %s", err)
}
browspr.accountManager = account.NewAccountManager(runtime, identdEP, principalManager)
return browspr
}
func (browspr *Browspr) Shutdown() {
browspr.rt.Cleanup()
}
// tcpKeepAliveListener sets TCP keep-alive timeouts on accepted connections.
// It's used by ListenAndServe and ListenAndServeTLS so dead TCP connections
// (e.g. closing laptop mid-download) eventually go away.
// Copied from http/server.go, since it's not exported.
type tcpKeepAliveListener struct {
*net.TCPListener
}
func (ln tcpKeepAliveListener) Accept() (c net.Conn, err error) {
tc, err := ln.AcceptTCP()
if err != nil {
return
}
tc.SetKeepAlive(true)
tc.SetKeepAlivePeriod(3 * time.Minute)
return tc, nil
}
// HandleMessage handles most messages from javascript and forwards them to a
// Controller.
func (b *Browspr) HandleMessage(instanceId int32, msg string) error {
b.mu.Lock()
defer b.mu.Unlock()
instance, ok := b.activeInstances[instanceId]
if !ok {
instance = newPipe(b, instanceId)
b.activeInstances[instanceId] = instance
}
return instance.handleMessage(msg)
}
// HandleCleanupMessage cleans up the specified instance state. (For instance,
// when a browser tab is closed)
func (b *Browspr) HandleCleanupMessage(instanceId int32) {
b.mu.Lock()
defer b.mu.Unlock()
if instance, ok := b.activeInstances[instanceId]; ok {
delete(b.activeInstances, instanceId)
// NOTE(nlacasse): Calling cleanup() on the main thread locks
// browspr, so we must do it in a goroutine.
// TODO(nlacasse): Consider running all the message handlers in
// goroutines.
go instance.cleanup()
}
}
// HandleCreateAccountMessage creates an account for the specified instance.
func (b *Browspr) HandleCreateAccountMessage(instanceId int32, accessToken string) error {
account, err := b.accountManager.CreateAccount(accessToken)
if err != nil {
b.postMessage(instanceId, "createAccountFailedResponse", err.Error())
return err
}
b.postMessage(instanceId, "createAccountResponse", account)
return nil
}
// HandleAssociateAccountMessage associates an account with the specified origin.
func (b *Browspr) HandleAssociateAccountMessage(origin, account string, cavs []account.Caveat) error {
if err := b.accountManager.AssociateAccount(origin, account, cavs); err != nil {
return err
}
return nil
}