| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | // Copyright 2018 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 localstore | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | import ( | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	"context" | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | 	"fmt" | 
					
						
							|  |  |  | 	"time" | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	"github.com/ethereum/go-ethereum/log" | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | 	"github.com/ethereum/go-ethereum/metrics" | 
					
						
							| 
									
										
										
										
											2019-06-03 12:28:18 +02:00
										 |  |  | 	"github.com/ethersphere/swarm/chunk" | 
					
						
							|  |  |  | 	"github.com/ethersphere/swarm/shed" | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	"github.com/syndtr/goleveldb/leveldb" | 
					
						
							|  |  |  | ) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // Get returns a chunk from the database. If the chunk is | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | // not found chunk.ErrChunkNotFound will be returned. | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | // All required indexes will be updated required by the | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | // Getter Mode. Get is required to implement chunk.Store | 
					
						
							|  |  |  | // interface. | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | func (db *DB) Get(ctx context.Context, mode chunk.ModeGet, addr chunk.Address) (ch chunk.Chunk, err error) { | 
					
						
							|  |  |  | 	metricName := fmt.Sprintf("localstore.Get.%s", mode) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	metrics.GetOrRegisterCounter(metricName, nil).Inc(1) | 
					
						
							|  |  |  | 	defer totalTimeMetric(metricName, time.Now()) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	defer func() { | 
					
						
							|  |  |  | 		if err != nil { | 
					
						
							| 
									
										
										
										
											2019-05-07 13:46:26 +02:00
										 |  |  | 			metrics.GetOrRegisterCounter(metricName+".error", nil).Inc(1) | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | 		} | 
					
						
							|  |  |  | 	}() | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	out, err := db.get(mode, addr) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		if err == leveldb.ErrNotFound { | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 			return nil, chunk.ErrChunkNotFound | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 		return nil, err | 
					
						
							|  |  |  | 	} | 
					
						
							| 
									
										
										
										
											2019-02-26 16:09:32 +01:00
										 |  |  | 	return chunk.NewChunk(out.Address, out.Data), nil | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // get returns Item from the retrieval index | 
					
						
							|  |  |  | // and updates other indexes. | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | func (db *DB) get(mode chunk.ModeGet, addr chunk.Address) (out shed.Item, err error) { | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	item := addressToItem(addr) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	out, err = db.retrievalDataIndex.Get(item) | 
					
						
							|  |  |  | 	if err != nil { | 
					
						
							|  |  |  | 		return out, err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	switch mode { | 
					
						
							|  |  |  | 	// update the access timestamp and gc index | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	case chunk.ModeGetRequest: | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 		if db.updateGCSem != nil { | 
					
						
							|  |  |  | 			// wait before creating new goroutines | 
					
						
							|  |  |  | 			// if updateGCSem buffer id full | 
					
						
							|  |  |  | 			db.updateGCSem <- struct{}{} | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 		db.updateGCWG.Add(1) | 
					
						
							|  |  |  | 		go func() { | 
					
						
							|  |  |  | 			defer db.updateGCWG.Done() | 
					
						
							|  |  |  | 			if db.updateGCSem != nil { | 
					
						
							|  |  |  | 				// free a spot in updateGCSem buffer | 
					
						
							|  |  |  | 				// for a new goroutine | 
					
						
							|  |  |  | 				defer func() { <-db.updateGCSem }() | 
					
						
							|  |  |  | 			} | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | 			metricName := "localstore.updateGC" | 
					
						
							|  |  |  | 			metrics.GetOrRegisterCounter(metricName, nil).Inc(1) | 
					
						
							|  |  |  | 			defer totalTimeMetric(metricName, time.Now()) | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 			err := db.updateGC(out) | 
					
						
							|  |  |  | 			if err != nil { | 
					
						
							| 
									
										
										
										
											2019-04-25 10:22:57 +02:00
										 |  |  | 				metrics.GetOrRegisterCounter(metricName+".error", nil).Inc(1) | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 				log.Error("localstore update gc", "err", err) | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 			// if gc update hook is defined, call it | 
					
						
							|  |  |  | 			if testHookUpdateGC != nil { | 
					
						
							|  |  |  | 				testHookUpdateGC() | 
					
						
							|  |  |  | 			} | 
					
						
							|  |  |  | 		}() | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// no updates to indexes | 
					
						
							| 
									
										
										
										
											2019-04-10 16:50:58 +02:00
										 |  |  | 	case chunk.ModeGetSync: | 
					
						
							|  |  |  | 	case chunk.ModeGetLookup: | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 	default: | 
					
						
							|  |  |  | 		return out, ErrInvalidMode | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	return out, nil | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // updateGC updates garbage collection index for | 
					
						
							|  |  |  | // a single item. Provided item is expected to have | 
					
						
							|  |  |  | // only Address and Data fields with non zero values, | 
					
						
							|  |  |  | // which is ensured by the get function. | 
					
						
							|  |  |  | func (db *DB) updateGC(item shed.Item) (err error) { | 
					
						
							| 
									
										
										
										
											2019-03-09 00:06:39 +01:00
										 |  |  | 	db.batchMu.Lock() | 
					
						
							|  |  |  | 	defer db.batchMu.Unlock() | 
					
						
							| 
									
										
										
										
											2019-02-07 18:40:26 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 	batch := new(leveldb.Batch) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	// update accessTimeStamp in retrieve, gc | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	i, err := db.retrievalAccessIndex.Get(item) | 
					
						
							|  |  |  | 	switch err { | 
					
						
							|  |  |  | 	case nil: | 
					
						
							|  |  |  | 		item.AccessTimestamp = i.AccessTimestamp | 
					
						
							|  |  |  | 	case leveldb.ErrNotFound: | 
					
						
							|  |  |  | 		// no chunk accesses | 
					
						
							|  |  |  | 	default: | 
					
						
							|  |  |  | 		return err | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	if item.AccessTimestamp == 0 { | 
					
						
							|  |  |  | 		// chunk is not yet synced | 
					
						
							|  |  |  | 		// do not add it to the gc index | 
					
						
							|  |  |  | 		return nil | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 	// delete current entry from the gc index | 
					
						
							|  |  |  | 	db.gcIndex.DeleteInBatch(batch, item) | 
					
						
							|  |  |  | 	// update access timestamp | 
					
						
							|  |  |  | 	item.AccessTimestamp = now() | 
					
						
							|  |  |  | 	// update retrieve access index | 
					
						
							|  |  |  | 	db.retrievalAccessIndex.PutInBatch(batch, item) | 
					
						
							|  |  |  | 	// add new entry to gc index | 
					
						
							|  |  |  | 	db.gcIndex.PutInBatch(batch, item) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	return db.shed.WriteBatch(batch) | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // testHookUpdateGC is a hook that can provide | 
					
						
							|  |  |  | // information when a garbage collection index is updated. | 
					
						
							|  |  |  | var testHookUpdateGC func() |