init.go 4.25 KB
Newer Older
1
package main
Matt Bell's avatar
Matt Bell committed
2 3 4 5

import (
	"encoding/base64"
	"errors"
Brian Tiger Chow's avatar
Brian Tiger Chow committed
6
	"fmt"
Matt Bell's avatar
Matt Bell committed
7 8 9 10 11 12 13 14 15 16 17
	"os"
	"path/filepath"

	cmds "github.com/jbenet/go-ipfs/commands"
	config "github.com/jbenet/go-ipfs/config"
	ci "github.com/jbenet/go-ipfs/crypto"
	peer "github.com/jbenet/go-ipfs/peer"
	u "github.com/jbenet/go-ipfs/util"
)

var initCmd = &cmds.Command{
18 19 20 21
	Description: "Initializes IPFS config file",
	Help: `Initializes IPFS configuration files and generates a new keypair.
`,

Matt Bell's avatar
Matt Bell committed
22
	Options: []cmds.Option{
23
		cmds.IntOption("bits", "b", "Number of bits to use in the generated RSA private key (defaults to 4096)"),
24 25 26
		cmds.StringOption("passphrase", "p", "Passphrase for encrypting the private key"),
		cmds.BoolOption("force", "f", "Overwrite existing config (if it exists)"),
		cmds.StringOption("datastore", "d", "Location for the IPFS data store"),
Matt Bell's avatar
Matt Bell committed
27
	},
28
	Run: func(req cmds.Request) (interface{}, error) {
29

30
		dspathOverride, _, err := req.Option("d").String() // if !found it's okay. Let == ""
31 32
		if err != nil {
			return nil, err
33 34
		}

35
		force, _, err := req.Option("f").Bool() // if !found, it's okay force == false
36 37
		if err != nil {
			return nil, err
38 39
		}

40
		nBitsForKeypair, bitsOptFound, err := req.Option("b").Int()
41 42 43
		if err != nil {
			return nil, err
		}
44
		if !bitsOptFound {
45 46 47
			nBitsForKeypair = 4096
		}

48
		return nil, doInit(req.Context().ConfigRoot, dspathOverride, force, nBitsForKeypair)
49 50
	},
}
Matt Bell's avatar
Matt Bell committed
51

Brian Tiger Chow's avatar
Brian Tiger Chow committed
52
// TODO add default welcome hash: eaa68bedae247ed1e5bd0eb4385a3c0959b976e4
Brian Tiger Chow's avatar
Brian Tiger Chow committed
53 54
// NB: if dspath is not provided, it will be retrieved from the config
func doInit(configRoot string, dspathOverride string, force bool, nBitsForKeypair int) error {
55

56
	u.POut("initializing ipfs node at %s\n", configRoot)
57 58

	configFilename, err := config.Filename(configRoot)
59
	if err != nil {
60
		return errors.New("Couldn't get home directory path")
61 62
	}

63
	fi, err := os.Lstat(configFilename)
64 65
	if fi != nil || (err != nil && !os.IsNotExist(err)) {
		if !force {
66 67
			// TODO multi-line string
			return errors.New("ipfs configuration file already exists!\nReinitializing would overwrite your keys.\n(use -f to force overwrite)")
Matt Bell's avatar
Matt Bell committed
68
		}
69
	}
Matt Bell's avatar
Matt Bell committed
70

Brian Tiger Chow's avatar
Brian Tiger Chow committed
71
	ds, err := datastoreConfig(dspathOverride)
Brian Tiger Chow's avatar
Brian Tiger Chow committed
72
	if err != nil {
73
		return err
74 75
	}

Brian Tiger Chow's avatar
Brian Tiger Chow committed
76 77 78 79
	identity, err := identityConfig(nBitsForKeypair)
	if err != nil {
		return err
	}
80

Brian Tiger Chow's avatar
Brian Tiger Chow committed
81
	conf := config.Config{
82

Brian Tiger Chow's avatar
Brian Tiger Chow committed
83 84 85 86 87 88
		// setup the node addresses.
		Addresses: config.Addresses{
			Swarm: "/ip4/0.0.0.0/tcp/4001",
			API:   "/ip4/127.0.0.1/tcp/5001",
		},

89 90 91 92 93 94 95
		Bootstrap: []*config.BootstrapPeer{
			&config.BootstrapPeer{ // Use these hardcoded bootstrap peers for now.
				// mars.i.ipfs.io
				PeerID:  "QmaCpDMGvV2BGHeYERUEnRQAwe3N8SzbUtfsmvsqQLuvuJ",
				Address: "/ip4/104.131.131.82/tcp/4001",
			},
		},
Brian Tiger Chow's avatar
Brian Tiger Chow committed
96 97 98 99

		Datastore: ds,

		Identity: identity,
100

Brian Tiger Chow's avatar
Brian Tiger Chow committed
101 102 103 104 105
		// setup the node mount points.
		Mounts: config.Mounts{
			IPFS: "/ipfs",
			IPNS: "/ipns",
		},
106

Brian Tiger Chow's avatar
Brian Tiger Chow committed
107 108
		// tracking ipfs version used to generate the init folder and adding
		// update checker default setting.
Brian Tiger Chow's avatar
Brian Tiger Chow committed
109
		Version: config.VersionDefaultValue(),
110 111
	}

Brian Tiger Chow's avatar
Brian Tiger Chow committed
112
	err = config.WriteConfigFile(configFilename, conf)
113
	if err != nil {
114
		return err
115
	}
116
	return nil
Matt Bell's avatar
Matt Bell committed
117
}
Brian Tiger Chow's avatar
Brian Tiger Chow committed
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144

func datastoreConfig(dspath string) (config.Datastore, error) {
	ds := config.Datastore{}
	if len(dspath) == 0 {
		var err error
		dspath, err = config.DataStorePath("")
		if err != nil {
			return ds, err
		}
	}
	ds.Path = dspath
	ds.Type = "leveldb"

	// Construct the data store if missing
	if err := os.MkdirAll(dspath, os.ModePerm); err != nil {
		return ds, err
	}

	// Check the directory is writeable
	if f, err := os.Create(filepath.Join(dspath, "._check_writeable")); err == nil {
		os.Remove(f.Name())
	} else {
		return ds, errors.New("Datastore '" + dspath + "' is not writeable")
	}

	return ds, nil
}
Brian Tiger Chow's avatar
Brian Tiger Chow committed
145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174

func identityConfig(nbits int) (config.Identity, error) {
	// TODO guard higher up
	ident := config.Identity{}
	if nbits < 1024 {
		return ident, errors.New("Bitsize less than 1024 is considered unsafe.")
	}

	fmt.Println("generating key pair...")
	sk, pk, err := ci.GenerateKeyPair(ci.RSA, nbits)
	if err != nil {
		return ident, err
	}

	// currently storing key unencrypted. in the future we need to encrypt it.
	// TODO(security)
	skbytes, err := sk.Bytes()
	if err != nil {
		return ident, err
	}
	ident.PrivKey = base64.StdEncoding.EncodeToString(skbytes)

	id, err := peer.IDFromPubKey(pk)
	if err != nil {
		return ident, err
	}
	ident.PeerID = id.Pretty()

	return ident, nil
}