aboutsummaryrefslogtreecommitdiffstats
path: root/cmd/swarm/global-store/global_store.go
blob: f93b464dbc4f35db607bbedf205843de5d1f6de2 (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
// Copyright 2019 The go-ethereum Authors
// This file is part of go-ethereum.
//
// go-ethereum is free software: you can redistribute it and/or modify
// it under the terms of the GNU General Public License as published by
// the Free Software Foundation, either version 3 of the License, or
// (at your option) any later version.
//
// go-ethereum 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 General Public License for more details.
//
// You should have received a copy of the GNU General Public License
// along with go-ethereum. If not, see <http://www.gnu.org/licenses/>.

package main

import (
    "io"
    "net"
    "net/http"
    "os"

    "github.com/ethereum/go-ethereum/log"
    "github.com/ethereum/go-ethereum/rpc"
    "github.com/ethereum/go-ethereum/swarm/storage/mock"
    "github.com/ethereum/go-ethereum/swarm/storage/mock/db"
    "github.com/ethereum/go-ethereum/swarm/storage/mock/mem"
    cli "gopkg.in/urfave/cli.v1"
)

// startHTTP starts a global store with HTTP RPC server.
// It is used for "http" cli command.
func startHTTP(ctx *cli.Context) (err error) {
    server, cleanup, err := newServer(ctx)
    if err != nil {
        return err
    }
    defer cleanup()

    listener, err := net.Listen("tcp", ctx.String("addr"))
    if err != nil {
        return err
    }
    log.Info("http", "address", listener.Addr().String())

    return http.Serve(listener, server)
}

// startWS starts a global store with WebSocket RPC server.
// It is used for "websocket" cli command.
func startWS(ctx *cli.Context) (err error) {
    server, cleanup, err := newServer(ctx)
    if err != nil {
        return err
    }
    defer cleanup()

    listener, err := net.Listen("tcp", ctx.String("addr"))
    if err != nil {
        return err
    }
    origins := ctx.StringSlice("origins")
    log.Info("websocket", "address", listener.Addr().String(), "origins", origins)

    return http.Serve(listener, server.WebsocketHandler(origins))
}

// newServer creates a global store and starts a chunk explorer server if configured.
// Returned cleanup function should be called only if err is nil.
func newServer(ctx *cli.Context) (server *rpc.Server, cleanup func(), err error) {
    log.PrintOrigins(true)
    log.Root().SetHandler(log.LvlFilterHandler(log.Lvl(ctx.Int("verbosity")), log.StreamHandler(os.Stdout, log.TerminalFormat(false))))

    cleanup = func() {}
    var globalStore mock.GlobalStorer
    dir := ctx.String("dir")
    if dir != "" {
        dbStore, err := db.NewGlobalStore(dir)
        if err != nil {
            return nil, nil, err
        }
        cleanup = func() {
            if err := dbStore.Close(); err != nil {
                log.Error("global store: close", "err", err)
            }
        }
        globalStore = dbStore
        log.Info("database global store", "dir", dir)
    } else {
        globalStore = mem.NewGlobalStore()
        log.Info("in-memory global store")
    }

    server = rpc.NewServer()
    if err := server.RegisterName("mockStore", globalStore); err != nil {
        cleanup()
        return nil, nil, err
    }

    shutdown, err := serveChunkExplorer(ctx, globalStore)
    if err != nil {
        cleanup()
        return nil, nil, err
    }
    if shutdown != nil {
        cleanup = func() {
            shutdown()

            if c, ok := globalStore.(io.Closer); ok {
                if err := c.Close(); err != nil {
                    log.Error("global store: close", "err", err)
                }
            }
        }
    }

    return server, cleanup, nil
}