* core/types, core/vm, eth, tests: regenerate gencodec files * Makefile: update devtools target Install protoc-gen-go and print reminders about npm, solc and protoc. Also switch to github.com/kevinburke/go-bindata because it's more maintained. * contracts/ens: update contracts and regenerate with solidity v0.4.19 The newer upstream version of the FIFSRegistrar contract doesn't set the resolver anymore. The resolver is now deployed separately. * contracts/release: regenerate with solidity v0.4.19 * contracts/chequebook: fix fallback and regenerate with solidity v0.4.19 The contract didn't have a fallback function, payments would be rejected when compiled with newer solidity. References to 'mortal' and 'owned' use the local file system so we can compile without network access. * p2p/discv5: regenerate with recent stringer * cmd/faucet: regenerate * dashboard: regenerate * eth/tracers: regenerate * internal/jsre/deps: regenerate * dashboard: avoid sed -i because it's not portable * accounts/usbwallet/internal/trezor: fix go generate warnings
		
			
				
	
	
		
			310 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
			
		
		
	
	
			310 lines
		
	
	
		
			7.9 KiB
		
	
	
	
		
			Go
		
	
	
	
	
	
| // Copyright 2017 The go-ethereum Authors
 | |
| // This file is part of the go-ethereum library.
 | |
| //
 | |
| // The go-ethereum library is free software: you can redistribute it and/or modify
 | |
| // it under the terms of the GNU Lesser General Public License as published by
 | |
| // the Free Software Foundation, either version 3 of the License, or
 | |
| // (at your option) any later version.
 | |
| //
 | |
| // The go-ethereum library is distributed in the hope that it will be useful,
 | |
| // but WITHOUT ANY WARRANTY; without even the implied warranty of
 | |
| // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
 | |
| // GNU Lesser General Public License for more details.
 | |
| //
 | |
| // You should have received a copy of the GNU Lesser General Public License
 | |
| // along with the go-ethereum library. If not, see <http://www.gnu.org/licenses/>.
 | |
| 
 | |
| package dashboard
 | |
| 
 | |
| //go:generate npm --prefix ./assets install
 | |
| //go:generate ./assets/node_modules/.bin/webpack --config ./assets/webpack.config.js --context ./assets
 | |
| //go:generate go-bindata -nometadata -o assets.go -prefix assets -nocompress -pkg dashboard assets/public/...
 | |
| //go:generate sh -c "sed 's#var _public#//nolint:misspell\\\n&#' assets.go > assets.go.tmp && mv assets.go.tmp assets.go"
 | |
| //go:generate gofmt -w -s assets.go
 | |
| 
 | |
| import (
 | |
| 	"fmt"
 | |
| 	"io/ioutil"
 | |
| 	"net"
 | |
| 	"net/http"
 | |
| 	"path/filepath"
 | |
| 	"sync"
 | |
| 	"sync/atomic"
 | |
| 	"time"
 | |
| 
 | |
| 	"github.com/ethereum/go-ethereum/log"
 | |
| 	"github.com/ethereum/go-ethereum/p2p"
 | |
| 	"github.com/ethereum/go-ethereum/rpc"
 | |
| 	"github.com/rcrowley/go-metrics"
 | |
| 	"golang.org/x/net/websocket"
 | |
| )
 | |
| 
 | |
| const (
 | |
| 	memorySampleLimit  = 200 // Maximum number of memory data samples
 | |
| 	trafficSampleLimit = 200 // Maximum number of traffic data samples
 | |
| )
 | |
| 
 | |
| var nextID uint32 // Next connection id
 | |
| 
 | |
| // Dashboard contains the dashboard internals.
 | |
| type Dashboard struct {
 | |
| 	config *Config
 | |
| 
 | |
| 	listener net.Listener
 | |
| 	conns    map[uint32]*client // Currently live websocket connections
 | |
| 	charts   *HomeMessage
 | |
| 	lock     sync.RWMutex // Lock protecting the dashboard's internals
 | |
| 
 | |
| 	quit chan chan error // Channel used for graceful exit
 | |
| 	wg   sync.WaitGroup
 | |
| }
 | |
| 
 | |
| // client represents active websocket connection with a remote browser.
 | |
| type client struct {
 | |
| 	conn   *websocket.Conn // Particular live websocket connection
 | |
| 	msg    chan Message    // Message queue for the update messages
 | |
| 	logger log.Logger      // Logger for the particular live websocket connection
 | |
| }
 | |
| 
 | |
| // New creates a new dashboard instance with the given configuration.
 | |
| func New(config *Config) (*Dashboard, error) {
 | |
| 	return &Dashboard{
 | |
| 		conns:  make(map[uint32]*client),
 | |
| 		config: config,
 | |
| 		quit:   make(chan chan error),
 | |
| 		charts: &HomeMessage{
 | |
| 			Memory:  &Chart{},
 | |
| 			Traffic: &Chart{},
 | |
| 		},
 | |
| 	}, nil
 | |
| }
 | |
| 
 | |
| // Protocols is a meaningless implementation of node.Service.
 | |
| func (db *Dashboard) Protocols() []p2p.Protocol { return nil }
 | |
| 
 | |
| // APIs is a meaningless implementation of node.Service.
 | |
| func (db *Dashboard) APIs() []rpc.API { return nil }
 | |
| 
 | |
| // Start implements node.Service, starting the data collection thread and the listening server of the dashboard.
 | |
| func (db *Dashboard) Start(server *p2p.Server) error {
 | |
| 	db.wg.Add(2)
 | |
| 	go db.collectData()
 | |
| 	go db.collectLogs() // In case of removing this line change 2 back to 1 in wg.Add.
 | |
| 
 | |
| 	http.HandleFunc("/", db.webHandler)
 | |
| 	http.Handle("/api", websocket.Handler(db.apiHandler))
 | |
| 
 | |
| 	listener, err := net.Listen("tcp", fmt.Sprintf("%s:%d", db.config.Host, db.config.Port))
 | |
| 	if err != nil {
 | |
| 		return err
 | |
| 	}
 | |
| 	db.listener = listener
 | |
| 
 | |
| 	go http.Serve(listener, nil)
 | |
| 
 | |
| 	return nil
 | |
| }
 | |
| 
 | |
| // Stop implements node.Service, stopping the data collection thread and the connection listener of the dashboard.
 | |
| func (db *Dashboard) Stop() error {
 | |
| 	// Close the connection listener.
 | |
| 	var errs []error
 | |
| 	if err := db.listener.Close(); err != nil {
 | |
| 		errs = append(errs, err)
 | |
| 	}
 | |
| 	// Close the collectors.
 | |
| 	errc := make(chan error, 1)
 | |
| 	for i := 0; i < 2; i++ {
 | |
| 		db.quit <- errc
 | |
| 		if err := <-errc; err != nil {
 | |
| 			errs = append(errs, err)
 | |
| 		}
 | |
| 	}
 | |
| 	// Close the connections.
 | |
| 	db.lock.Lock()
 | |
| 	for _, c := range db.conns {
 | |
| 		if err := c.conn.Close(); err != nil {
 | |
| 			c.logger.Warn("Failed to close connection", "err", err)
 | |
| 		}
 | |
| 	}
 | |
| 	db.lock.Unlock()
 | |
| 
 | |
| 	// Wait until every goroutine terminates.
 | |
| 	db.wg.Wait()
 | |
| 	log.Info("Dashboard stopped")
 | |
| 
 | |
| 	var err error
 | |
| 	if len(errs) > 0 {
 | |
| 		err = fmt.Errorf("%v", errs)
 | |
| 	}
 | |
| 
 | |
| 	return err
 | |
| }
 | |
| 
 | |
| // webHandler handles all non-api requests, simply flattening and returning the dashboard website.
 | |
| func (db *Dashboard) webHandler(w http.ResponseWriter, r *http.Request) {
 | |
| 	log.Debug("Request", "URL", r.URL)
 | |
| 
 | |
| 	path := r.URL.String()
 | |
| 	if path == "/" {
 | |
| 		path = "/dashboard.html"
 | |
| 	}
 | |
| 	// If the path of the assets is manually set
 | |
| 	if db.config.Assets != "" {
 | |
| 		blob, err := ioutil.ReadFile(filepath.Join(db.config.Assets, path))
 | |
| 		if err != nil {
 | |
| 			log.Warn("Failed to read file", "path", path, "err", err)
 | |
| 			http.Error(w, "not found", http.StatusNotFound)
 | |
| 			return
 | |
| 		}
 | |
| 		w.Write(blob)
 | |
| 		return
 | |
| 	}
 | |
| 	blob, err := Asset(filepath.Join("public", path))
 | |
| 	if err != nil {
 | |
| 		log.Warn("Failed to load the asset", "path", path, "err", err)
 | |
| 		http.Error(w, "not found", http.StatusNotFound)
 | |
| 		return
 | |
| 	}
 | |
| 	w.Write(blob)
 | |
| }
 | |
| 
 | |
| // apiHandler handles requests for the dashboard.
 | |
| func (db *Dashboard) apiHandler(conn *websocket.Conn) {
 | |
| 	id := atomic.AddUint32(&nextID, 1)
 | |
| 	client := &client{
 | |
| 		conn:   conn,
 | |
| 		msg:    make(chan Message, 128),
 | |
| 		logger: log.New("id", id),
 | |
| 	}
 | |
| 	done := make(chan struct{})
 | |
| 
 | |
| 	// Start listening for messages to send.
 | |
| 	db.wg.Add(1)
 | |
| 	go func() {
 | |
| 		defer db.wg.Done()
 | |
| 
 | |
| 		for {
 | |
| 			select {
 | |
| 			case <-done:
 | |
| 				return
 | |
| 			case msg := <-client.msg:
 | |
| 				if err := websocket.JSON.Send(client.conn, msg); err != nil {
 | |
| 					client.logger.Warn("Failed to send the message", "msg", msg, "err", err)
 | |
| 					client.conn.Close()
 | |
| 					return
 | |
| 				}
 | |
| 			}
 | |
| 		}
 | |
| 	}()
 | |
| 	// Send the past data.
 | |
| 	client.msg <- Message{
 | |
| 		Home: &HomeMessage{
 | |
| 			Memory: &Chart{
 | |
| 				History: db.charts.Memory.History,
 | |
| 			},
 | |
| 			Traffic: &Chart{
 | |
| 				History: db.charts.Traffic.History,
 | |
| 			},
 | |
| 		},
 | |
| 	}
 | |
| 	// Start tracking the connection and drop at connection loss.
 | |
| 	db.lock.Lock()
 | |
| 	db.conns[id] = client
 | |
| 	db.lock.Unlock()
 | |
| 	defer func() {
 | |
| 		db.lock.Lock()
 | |
| 		delete(db.conns, id)
 | |
| 		db.lock.Unlock()
 | |
| 	}()
 | |
| 	for {
 | |
| 		fail := []byte{}
 | |
| 		if _, err := conn.Read(fail); err != nil {
 | |
| 			close(done)
 | |
| 			return
 | |
| 		}
 | |
| 		// Ignore all messages
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // collectData collects the required data to plot on the dashboard.
 | |
| func (db *Dashboard) collectData() {
 | |
| 	defer db.wg.Done()
 | |
| 
 | |
| 	for {
 | |
| 		select {
 | |
| 		case errc := <-db.quit:
 | |
| 			errc <- nil
 | |
| 			return
 | |
| 		case <-time.After(db.config.Refresh):
 | |
| 			inboundTraffic := metrics.DefaultRegistry.Get("p2p/InboundTraffic").(metrics.Meter).Rate1()
 | |
| 			memoryInUse := metrics.DefaultRegistry.Get("system/memory/inuse").(metrics.Meter).Rate1()
 | |
| 			now := time.Now()
 | |
| 			memory := &ChartEntry{
 | |
| 				Time:  now,
 | |
| 				Value: memoryInUse,
 | |
| 			}
 | |
| 			traffic := &ChartEntry{
 | |
| 				Time:  now,
 | |
| 				Value: inboundTraffic,
 | |
| 			}
 | |
| 			first := 0
 | |
| 			if len(db.charts.Memory.History) == memorySampleLimit {
 | |
| 				first = 1
 | |
| 			}
 | |
| 			db.charts.Memory.History = append(db.charts.Memory.History[first:], memory)
 | |
| 			first = 0
 | |
| 			if len(db.charts.Traffic.History) == trafficSampleLimit {
 | |
| 				first = 1
 | |
| 			}
 | |
| 			db.charts.Traffic.History = append(db.charts.Traffic.History[first:], traffic)
 | |
| 
 | |
| 			db.sendToAll(&Message{
 | |
| 				Home: &HomeMessage{
 | |
| 					Memory: &Chart{
 | |
| 						New: memory,
 | |
| 					},
 | |
| 					Traffic: &Chart{
 | |
| 						New: traffic,
 | |
| 					},
 | |
| 				},
 | |
| 			})
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // collectLogs collects and sends the logs to the active dashboards.
 | |
| func (db *Dashboard) collectLogs() {
 | |
| 	defer db.wg.Done()
 | |
| 
 | |
| 	id := 1
 | |
| 	// TODO (kurkomisi): log collection comes here.
 | |
| 	for {
 | |
| 		select {
 | |
| 		case errc := <-db.quit:
 | |
| 			errc <- nil
 | |
| 			return
 | |
| 		case <-time.After(db.config.Refresh / 2):
 | |
| 			db.sendToAll(&Message{
 | |
| 				Logs: &LogsMessage{
 | |
| 					Log: fmt.Sprintf("%-4d: This is a fake log.", id),
 | |
| 				},
 | |
| 			})
 | |
| 			id++
 | |
| 		}
 | |
| 	}
 | |
| }
 | |
| 
 | |
| // sendToAll sends the given message to the active dashboards.
 | |
| func (db *Dashboard) sendToAll(msg *Message) {
 | |
| 	db.lock.Lock()
 | |
| 	for _, c := range db.conns {
 | |
| 		select {
 | |
| 		case c.msg <- *msg:
 | |
| 		default:
 | |
| 			c.conn.Close()
 | |
| 		}
 | |
| 	}
 | |
| 	db.lock.Unlock()
 | |
| }
 |