| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // Copyright 2015 The go-ethereum Authors | 
					
						
							| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // This file is part of the go-ethereum library. | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // 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 | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // the Free Software Foundation, either version 3 of the License, or | 
					
						
							|  |  |  | // (at your option) any later version. | 
					
						
							|  |  |  | // | 
					
						
							| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // The go-ethereum library is distributed in the hope that it will be useful, | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // but WITHOUT ANY WARRANTY; without even the implied warranty of | 
					
						
							|  |  |  | // MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
					
						
							| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // GNU Lesser General Public License for more details. | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // | 
					
						
							| 
									
										
										
										
											2016-04-14 18:18:24 +02:00
										 |  |  | // 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/>. | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | package downloader | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2017-03-22 18:20:33 +01:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 	"sync" | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-06 12:39:14 +03:00
										 |  |  | 	ethereum "github.com/ethereum/go-ethereum" | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/event" | 
					
						
							| 
									
										
										
										
											2015-12-16 10:58:01 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/rpc" | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-03-15 11:27:49 -07:00
										 |  |  | // PublicDownloaderAPI provides an API which gives information about the current synchronisation status. | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | // It offers only methods that operates on data that can be available to anyone without security risks. | 
					
						
							|  |  |  | type PublicDownloaderAPI struct { | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	d                         *Downloader | 
					
						
							|  |  |  | 	mux                       *event.TypeMux | 
					
						
							|  |  |  | 	installSyncSubscription   chan chan interface{} | 
					
						
							|  |  |  | 	uninstallSyncSubscription chan *uninstallSyncSubscriptionRequest | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // NewPublicDownloaderAPI create a new PublicDownloaderAPI. The API has an internal event loop that | 
					
						
							|  |  |  | // listens for events from the downloader through the global event mux. In case it receives one of | 
					
						
							|  |  |  | // these events it broadcasts it to all syncing subscriptions that are installed through the | 
					
						
							|  |  |  | // installSyncSubscription channel. | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | func NewPublicDownloaderAPI(d *Downloader, m *event.TypeMux) *PublicDownloaderAPI { | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	api := &PublicDownloaderAPI{ | 
					
						
							|  |  |  | 		d:   d, | 
					
						
							|  |  |  | 		mux: m, | 
					
						
							|  |  |  | 		installSyncSubscription:   make(chan chan interface{}), | 
					
						
							|  |  |  | 		uninstallSyncSubscription: make(chan *uninstallSyncSubscriptionRequest), | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	go api.eventLoop() | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	return api | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // eventLoop runs an loop until the event mux closes. It will install and uninstall new | 
					
						
							|  |  |  | // sync subscriptions and broadcasts sync status updates to the installed sync subscriptions. | 
					
						
							|  |  |  | func (api *PublicDownloaderAPI) eventLoop() { | 
					
						
							|  |  |  | 	var ( | 
					
						
							|  |  |  | 		sub               = api.mux.Subscribe(StartEvent{}, DoneEvent{}, FailedEvent{}) | 
					
						
							|  |  |  | 		syncSubscriptions = make(map[chan interface{}]struct{}) | 
					
						
							|  |  |  | 	) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	for { | 
					
						
							|  |  |  | 		select { | 
					
						
							|  |  |  | 		case i := <-api.installSyncSubscription: | 
					
						
							|  |  |  | 			syncSubscriptions[i] = struct{}{} | 
					
						
							|  |  |  | 		case u := <-api.uninstallSyncSubscription: | 
					
						
							|  |  |  | 			delete(syncSubscriptions, u.c) | 
					
						
							|  |  |  | 			close(u.uninstalled) | 
					
						
							|  |  |  | 		case event := <-sub.Chan(): | 
					
						
							|  |  |  | 			if event == nil { | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 			var notification interface{} | 
					
						
							|  |  |  | 			switch event.Data.(type) { | 
					
						
							|  |  |  | 			case StartEvent: | 
					
						
							| 
									
										
										
										
											2016-09-06 12:39:14 +03:00
										 |  |  | 				notification = &SyncingResult{ | 
					
						
							|  |  |  | 					Syncing: true, | 
					
						
							|  |  |  | 					Status:  api.d.Progress(), | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 			case DoneEvent, FailedEvent: | 
					
						
							|  |  |  | 				notification = false | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// broadcast | 
					
						
							|  |  |  | 			for c := range syncSubscriptions { | 
					
						
							|  |  |  | 				c <- notification | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	} | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Syncing provides information when this nodes starts synchronising with the Ethereum network and when it's finished. | 
					
						
							|  |  |  | func (api *PublicDownloaderAPI) Syncing(ctx context.Context) (*rpc.Subscription, error) { | 
					
						
							|  |  |  | 	notifier, supported := rpc.NotifierFromContext(ctx) | 
					
						
							|  |  |  | 	if !supported { | 
					
						
							|  |  |  | 		return &rpc.Subscription{}, rpc.ErrNotificationsUnsupported | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	rpcSub := notifier.CreateSubscription() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	go func() { | 
					
						
							|  |  |  | 		statuses := make(chan interface{}) | 
					
						
							|  |  |  | 		sub := api.SubscribeSyncStatus(statuses) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			select { | 
					
						
							|  |  |  | 			case status := <-statuses: | 
					
						
							|  |  |  | 				notifier.Notify(rpcSub.ID, status) | 
					
						
							|  |  |  | 			case <-rpcSub.Err(): | 
					
						
							|  |  |  | 				sub.Unsubscribe() | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			case <-notifier.Closed(): | 
					
						
							|  |  |  | 				sub.Unsubscribe() | 
					
						
							|  |  |  | 				return | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return rpcSub, nil | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // SyncingResult provides information about the current synchronisation status for this node. | 
					
						
							|  |  |  | type SyncingResult struct { | 
					
						
							| 
									
										
										
										
											2016-09-06 12:39:14 +03:00
										 |  |  | 	Syncing bool                  `json:"syncing"` | 
					
						
							|  |  |  | 	Status  ethereum.SyncProgress `json:"status"` | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // uninstallSyncSubscriptionRequest uninstalles a syncing subscription in the API event loop. | 
					
						
							|  |  |  | type uninstallSyncSubscriptionRequest struct { | 
					
						
							|  |  |  | 	c           chan interface{} | 
					
						
							|  |  |  | 	uninstalled chan interface{} | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // SyncStatusSubscription represents a syncing subscription. | 
					
						
							|  |  |  | type SyncStatusSubscription struct { | 
					
						
							|  |  |  | 	api       *PublicDownloaderAPI // register subscription in event loop of this api instance | 
					
						
							|  |  |  | 	c         chan interface{}     // channel where events are broadcasted to | 
					
						
							|  |  |  | 	unsubOnce sync.Once            // make sure unsubscribe logic is executed once | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // Unsubscribe uninstalls the subscription from the DownloadAPI event loop. | 
					
						
							|  |  |  | // The status channel that was passed to subscribeSyncStatus isn't used anymore | 
					
						
							|  |  |  | // after this method returns. | 
					
						
							|  |  |  | func (s *SyncStatusSubscription) Unsubscribe() { | 
					
						
							|  |  |  | 	s.unsubOnce.Do(func() { | 
					
						
							|  |  |  | 		req := uninstallSyncSubscriptionRequest{s.c, make(chan interface{})} | 
					
						
							|  |  |  | 		s.api.uninstallSyncSubscription <- &req | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 		for { | 
					
						
							|  |  |  | 			select { | 
					
						
							|  |  |  | 			case <-s.c: | 
					
						
							|  |  |  | 				// drop new status events until uninstall confirmation | 
					
						
							|  |  |  | 				continue | 
					
						
							|  |  |  | 			case <-req.uninstalled: | 
					
						
							|  |  |  | 				return | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	}) | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2016-03-29 15:07:40 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-27 17:47:46 +02:00
										 |  |  | // SubscribeSyncStatus creates a subscription that will broadcast new synchronisation updates. | 
					
						
							|  |  |  | // The given channel must receive interface values, the result can either | 
					
						
							|  |  |  | func (api *PublicDownloaderAPI) SubscribeSyncStatus(status chan interface{}) *SyncStatusSubscription { | 
					
						
							|  |  |  | 	api.installSyncSubscription <- status | 
					
						
							|  |  |  | 	return &SyncStatusSubscription{api: api, c: status} | 
					
						
							| 
									
										
										
										
											2015-10-15 16:07:19 +02:00
										 |  |  | } |