config.go 4.17 KB
Newer Older
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
1 2 3
package config

import (
4 5 6 7
	"crypto"
	"crypto/x509"
	"encoding/base64"
	"errors"
8
	"os"
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
9
	"path/filepath"
10 11

	u "github.com/jbenet/go-ipfs/util"
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
12 13
)

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
14
// Identity tracks the configuration of the local node's identity.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
15
type Identity struct {
16
	PeerID  string
17
	PrivKey string
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
18 19
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
20
// Datastore tracks the configuration of the datastore.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
21
type Datastore struct {
Juan Batiz-Benet's avatar
gofmt  
Juan Batiz-Benet committed
22 23
	Type string
	Path string
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
24 25
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
26 27 28 29 30 31
// Addresses stores the (string) multiaddr addresses for the node.
type Addresses struct {
	Swarm string // address for the swarm network
	API   string // address for the local API (RPC)
}

32 33 34 35 36 37
// Mounts stores the (string) mount points
type Mounts struct {
	IPFS string
	IPNS string
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
38 39
// BootstrapPeer is a peer used to bootstrap the network.
type BootstrapPeer struct {
40
	Address string
41
	PeerID  string // until multiaddr supports ipfs, use another field.
42 43
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
44 45 46 47
func (bp *BootstrapPeer) String() string {
	return bp.Address + "/" + bp.PeerID
}

48 49
// Updates regulates checking and downloading for application's most recent version
type Updates struct {
50 51
	Check   string `json:"check"`    // "ignore" for do not check, "warn" and "error" for reacting when obsolete
	Version string `json: "version"` // ipfs version for which config was generated
52 53
}

Juan Batiz-Benet's avatar
go lint  
Juan Batiz-Benet committed
54
// Config is used to load IPFS config files.
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
55
type Config struct {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
56 57 58
	Identity  Identity         // local node's peer identity
	Datastore Datastore        // local node's storage
	Addresses Addresses        // local node's addresses
59
	Mounts    Mounts           // local node's mount points
60
	Updates   Updates          // local node's version management
61
	Bootstrap []*BootstrapPeer // local nodes's bootstrap peers
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
62 63
}

Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
64
// DefaultPathRoot is the path to the default config dir location.
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
65
const DefaultPathRoot = "~/.go-ipfs"
Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
66 67

// DefaultConfigFile is the filename of the configuration file
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
68
const DefaultConfigFile = "config"
Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
69 70

// DefaultDataStoreDirectory is the directory to store all the local IPFS data.
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
71
const DefaultDataStoreDirectory = "datastore"
Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
72 73

// EnvDir is the environment variable used to change the path root.
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
74 75 76 77 78
const EnvDir = "IPFS_DIR"

// PathRoot returns the default configuration root directory
func PathRoot() (string, error) {
	dir := os.Getenv(EnvDir)
79 80
	var err error
	if len(dir) == 0 {
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
81
		dir, err = u.TildeExpansion(DefaultPathRoot)
82 83 84
	}
	return dir, err
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
85

Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
86 87 88
// Path returns the path `extension` relative to the configuration root. If an
// empty string is provided for `configroot`, the default root is used.
func Path(configroot, extension string) (string, error) {
89
	if len(configroot) == 0 {
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
90
		dir, err := PathRoot()
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
91 92 93
		if err != nil {
			return "", err
		}
Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
94
		return filepath.Join(dir, extension), nil
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
95

96
	}
Juan Batiz-Benet's avatar
lint  
Juan Batiz-Benet committed
97
	return filepath.Join(configroot, extension), nil
98
}
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
99

Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
100 101 102
// DataStorePath returns the default data store path given a configuration root
// (set an empty string to have the default configuration root)
func DataStorePath(configroot string) (string, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
103
	return Path(configroot, DefaultDataStoreDirectory)
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
104 105 106 107 108
}

// Filename returns the configuration file path given a configuration root
// directory. If the configuration root directory is empty, use the default one
func Filename(configroot string) (string, error) {
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
109
	return Path(configroot, DefaultConfigFile)
Shanti Bouchez-Mongardé's avatar
Shanti Bouchez-Mongardé committed
110 111
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
112
// DecodePrivateKey is a helper to decode the users PrivateKey
113 114 115 116 117 118
func (i *Identity) DecodePrivateKey(passphrase string) (crypto.PrivateKey, error) {
	pkb, err := base64.StdEncoding.DecodeString(i.PrivKey)
	if err != nil {
		return nil, err
	}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
119 120
	// currently storing key unencrypted. in the future we need to encrypt it.
	// TODO(security)
121 122 123
	return x509.ParsePKCS1PrivateKey(pkb)
}

Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
124 125
// Load reads given file and returns the read config, or error.
func Load(filename string) (*Config, error) {
126
	// if nothing is there, fail. User must run 'ipfs init'
Juan Batiz-Benet's avatar
gofmt  
Juan Batiz-Benet committed
127
	if _, err := os.Stat(filename); os.IsNotExist(err) {
128
		return nil, errors.New("ipfs not initialized, please run 'ipfs init'")
Juan Batiz-Benet's avatar
gofmt  
Juan Batiz-Benet committed
129 130 131
	}

	var cfg Config
132
	err := ReadConfigFile(filename, &cfg)
133 134 135 136 137 138
	if err != nil {
		return nil, err
	}

	// tilde expansion on datastore path
	cfg.Datastore.Path, err = u.TildeExpansion(cfg.Datastore.Path)
Juan Batiz-Benet's avatar
gofmt  
Juan Batiz-Benet committed
139 140 141 142 143
	if err != nil {
		return nil, err
	}

	return &cfg, err
Juan Batiz-Benet's avatar
Juan Batiz-Benet committed
144
}
145 146 147

// Set sets the value of a particular config key
func Set(filename, key, value string) error {
148
	return WriteConfigKey(filename, key, value)
149
}