2018-07-04 10:51:47 +00:00
|
|
|
package libp2p
|
|
|
|
|
|
|
|
// This file contains all libp2p configuration options (except the defaults,
|
2021-06-16 20:19:45 +00:00
|
|
|
// those are in defaults.go).
|
2018-07-04 10:51:47 +00:00
|
|
|
|
|
|
|
import (
|
2021-06-16 20:19:45 +00:00
|
|
|
"errors"
|
2018-07-04 10:51:47 +00:00
|
|
|
"fmt"
|
|
|
|
"net"
|
2021-06-16 20:19:45 +00:00
|
|
|
"time"
|
2018-07-04 10:51:47 +00:00
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
circuit "github.com/libp2p/go-libp2p-circuit"
|
2019-06-09 07:24:20 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/connmgr"
|
|
|
|
"github.com/libp2p/go-libp2p-core/crypto"
|
|
|
|
"github.com/libp2p/go-libp2p-core/metrics"
|
2021-06-16 20:19:45 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/network"
|
|
|
|
"github.com/libp2p/go-libp2p-core/peer"
|
2019-06-09 07:24:20 +00:00
|
|
|
"github.com/libp2p/go-libp2p-core/peerstore"
|
|
|
|
"github.com/libp2p/go-libp2p-core/pnet"
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
"github.com/libp2p/go-libp2p/config"
|
2018-07-04 10:51:47 +00:00
|
|
|
bhost "github.com/libp2p/go-libp2p/p2p/host/basic"
|
2021-06-16 20:19:45 +00:00
|
|
|
autorelay "github.com/libp2p/go-libp2p/p2p/host/relay"
|
2018-07-04 10:51:47 +00:00
|
|
|
|
|
|
|
ma "github.com/multiformats/go-multiaddr"
|
|
|
|
)
|
|
|
|
|
|
|
|
// ListenAddrStrings configures libp2p to listen on the given (unparsed)
|
|
|
|
// addresses.
|
|
|
|
func ListenAddrStrings(s ...string) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
for _, addrstr := range s {
|
|
|
|
a, err := ma.NewMultiaddr(addrstr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.ListenAddrs = append(cfg.ListenAddrs, a)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ListenAddrs configures libp2p to listen on the given addresses.
|
|
|
|
func ListenAddrs(addrs ...ma.Multiaddr) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.ListenAddrs = append(cfg.ListenAddrs, addrs...)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Security configures libp2p to use the given security transport (or transport
|
|
|
|
// constructor).
|
|
|
|
//
|
|
|
|
// Name is the protocol name.
|
|
|
|
//
|
|
|
|
// The transport can be a constructed security.Transport or a function taking
|
|
|
|
// any subset of this libp2p node's:
|
|
|
|
// * Public key
|
|
|
|
// * Private key
|
|
|
|
// * Peer ID
|
|
|
|
// * Host
|
|
|
|
// * Network
|
|
|
|
// * Peerstore
|
|
|
|
func Security(name string, tpt interface{}) Option {
|
|
|
|
stpt, err := config.SecurityConstructor(tpt)
|
|
|
|
err = traceError(err, 1)
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if cfg.Insecure {
|
|
|
|
return fmt.Errorf("cannot use security transports with an insecure libp2p configuration")
|
|
|
|
}
|
|
|
|
cfg.SecurityTransports = append(cfg.SecurityTransports, config.MsSecC{SecC: stpt, ID: name})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NoSecurity is an option that completely disables all transport security.
|
|
|
|
// It's incompatible with all other transport security protocols.
|
|
|
|
var NoSecurity Option = func(cfg *Config) error {
|
|
|
|
if len(cfg.SecurityTransports) > 0 {
|
|
|
|
return fmt.Errorf("cannot use security transports with an insecure libp2p configuration")
|
|
|
|
}
|
|
|
|
cfg.Insecure = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// Muxer configures libp2p to use the given stream multiplexer (or stream
|
|
|
|
// multiplexer constructor).
|
|
|
|
//
|
|
|
|
// Name is the protocol name.
|
|
|
|
//
|
|
|
|
// The transport can be a constructed mux.Transport or a function taking any
|
|
|
|
// subset of this libp2p node's:
|
|
|
|
// * Peer ID
|
|
|
|
// * Host
|
|
|
|
// * Network
|
|
|
|
// * Peerstore
|
|
|
|
func Muxer(name string, tpt interface{}) Option {
|
|
|
|
mtpt, err := config.MuxerConstructor(tpt)
|
|
|
|
err = traceError(err, 1)
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.Muxers = append(cfg.Muxers, config.MsMuxC{MuxC: mtpt, ID: name})
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Transport configures libp2p to use the given transport (or transport
|
|
|
|
// constructor).
|
|
|
|
//
|
|
|
|
// The transport can be a constructed transport.Transport or a function taking
|
|
|
|
// any subset of this libp2p node's:
|
|
|
|
// * Transport Upgrader (*tptu.Upgrader)
|
|
|
|
// * Host
|
|
|
|
// * Stream muxer (muxer.Transport)
|
|
|
|
// * Security transport (security.Transport)
|
|
|
|
// * Private network protector (pnet.Protector)
|
|
|
|
// * Peer ID
|
|
|
|
// * Private Key
|
|
|
|
// * Public Key
|
|
|
|
// * Address filter (filter.Filter)
|
|
|
|
// * Peerstore
|
|
|
|
func Transport(tpt interface{}) Option {
|
|
|
|
tptc, err := config.TransportConstructor(tpt)
|
|
|
|
err = traceError(err, 1)
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.Transports = append(cfg.Transports, tptc)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Peerstore configures libp2p to use the given peerstore.
|
2019-06-09 07:24:20 +00:00
|
|
|
func Peerstore(ps peerstore.Peerstore) Option {
|
2018-07-04 10:51:47 +00:00
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.Peerstore != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple peerstore options")
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Peerstore = ps
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// PrivateNetwork configures libp2p to use the given private network protector.
|
2021-06-16 20:19:45 +00:00
|
|
|
func PrivateNetwork(psk pnet.PSK) Option {
|
2018-07-04 10:51:47 +00:00
|
|
|
return func(cfg *Config) error {
|
2021-06-16 20:19:45 +00:00
|
|
|
if cfg.PSK != nil {
|
2018-07-04 10:51:47 +00:00
|
|
|
return fmt.Errorf("cannot specify multiple private network options")
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
cfg.PSK = psk
|
2018-07-04 10:51:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// BandwidthReporter configures libp2p to use the given bandwidth reporter.
|
|
|
|
func BandwidthReporter(rep metrics.Reporter) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.Reporter != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple bandwidth reporter options")
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.Reporter = rep
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Identity configures libp2p to use the given private key to identify itself.
|
|
|
|
func Identity(sk crypto.PrivKey) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.PeerKey != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple identities")
|
|
|
|
}
|
|
|
|
|
|
|
|
cfg.PeerKey = sk
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConnectionManager configures libp2p to use the given connection manager.
|
2019-06-09 07:24:20 +00:00
|
|
|
func ConnectionManager(connman connmgr.ConnManager) Option {
|
2018-07-04 10:51:47 +00:00
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.ConnManager != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple connection managers")
|
|
|
|
}
|
|
|
|
cfg.ConnManager = connman
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AddrsFactory configures libp2p to use the given address factory.
|
|
|
|
func AddrsFactory(factory config.AddrsFactory) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.AddrsFactory != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple address factories")
|
|
|
|
}
|
|
|
|
cfg.AddrsFactory = factory
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
// EnableRelay configures libp2p to enable the relay transport with
|
|
|
|
// configuration options. By default, this option only configures libp2p to
|
|
|
|
// accept inbound connections from relays and make outbound connections
|
|
|
|
// _through_ relays when requested by the remote peer. (default: enabled)
|
|
|
|
//
|
|
|
|
// To _act_ as a relay, pass the circuit.OptHop option.
|
2018-07-04 10:51:47 +00:00
|
|
|
func EnableRelay(options ...circuit.RelayOpt) Option {
|
|
|
|
return func(cfg *Config) error {
|
2019-06-09 07:24:20 +00:00
|
|
|
cfg.RelayCustom = true
|
2018-07-04 10:51:47 +00:00
|
|
|
cfg.Relay = true
|
|
|
|
cfg.RelayOpts = options
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-06-09 07:24:20 +00:00
|
|
|
// DisableRelay configures libp2p to disable the relay transport.
|
|
|
|
func DisableRelay() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.RelayCustom = true
|
|
|
|
cfg.Relay = false
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
// EnableAutoRelay configures libp2p to enable the AutoRelay subsystem.
|
|
|
|
//
|
|
|
|
// Dependencies:
|
|
|
|
// * Relay (enabled by default)
|
|
|
|
// * Routing (to find relays), or StaticRelays/DefaultStaticRelays.
|
2019-06-09 07:24:20 +00:00
|
|
|
//
|
|
|
|
// This subsystem performs two functions:
|
|
|
|
//
|
|
|
|
// 1. When this libp2p node is configured to act as a relay "hop"
|
|
|
|
// (circuit.OptHop is passed to EnableRelay), this node will advertise itself
|
|
|
|
// as a public relay using the provided routing system.
|
|
|
|
// 2. When this libp2p node is _not_ configured as a relay "hop", it will
|
|
|
|
// automatically detect if it is unreachable (e.g., behind a NAT). If so, it will
|
|
|
|
// find, configure, and announce a set of public relays.
|
|
|
|
func EnableAutoRelay() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.EnableAutoRelay = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-06-16 20:19:45 +00:00
|
|
|
// StaticRelays configures known relays for autorelay; when this option is enabled
|
|
|
|
// then the system will use the configured relays instead of querying the DHT to
|
|
|
|
// discover relays.
|
|
|
|
func StaticRelays(relays []peer.AddrInfo) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.StaticRelays = append(cfg.StaticRelays, relays...)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// DefaultStaticRelays configures the static relays to use the known PL-operated relays.
|
|
|
|
func DefaultStaticRelays() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
for _, addr := range autorelay.DefaultRelays {
|
|
|
|
a, err := ma.NewMultiaddr(addr)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
pi, err := peer.AddrInfoFromP2pAddr(a)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
cfg.StaticRelays = append(cfg.StaticRelays, *pi)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ForceReachabilityPublic overrides automatic reachability detection in the AutoNAT subsystem,
|
|
|
|
// forcing the local node to believe it is reachable externally.
|
|
|
|
func ForceReachabilityPublic() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
public := network.Reachability(network.ReachabilityPublic)
|
|
|
|
cfg.AutoNATConfig.ForceReachability = &public
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ForceReachabilityPrivate overrides automatic reachability detection in the AutoNAT subsystem,
|
|
|
|
// forceing the local node to believe it is behind a NAT and not reachable externally.
|
|
|
|
func ForceReachabilityPrivate() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
private := network.Reachability(network.ReachabilityPrivate)
|
|
|
|
cfg.AutoNATConfig.ForceReachability = &private
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// EnableNATService configures libp2p to provide a service to peers for determining
|
|
|
|
// their reachability status. When enabled, the host will attempt to dial back
|
|
|
|
// to peers, and then tell them if it was successful in making such connections.
|
|
|
|
func EnableNATService() Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.AutoNATConfig.EnableService = true
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// AutoNATServiceRateLimit changes the default rate limiting configured in helping
|
|
|
|
// other peers determine their reachability status. When set, the host will limit
|
|
|
|
// the number of requests it responds to in each 60 second period to the set
|
|
|
|
// numbers. A value of '0' disables throttling.
|
|
|
|
func AutoNATServiceRateLimit(global, perPeer int, interval time.Duration) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.AutoNATConfig.ThrottleGlobalLimit = global
|
|
|
|
cfg.AutoNATConfig.ThrottlePeerLimit = perPeer
|
|
|
|
cfg.AutoNATConfig.ThrottleInterval = interval
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
// FilterAddresses configures libp2p to never dial nor accept connections from
|
2019-10-04 15:21:24 +00:00
|
|
|
// the given addresses. FilterAddresses should be used for cases where the
|
|
|
|
// addresses you want to deny are known ahead of time.
|
2021-06-16 20:19:45 +00:00
|
|
|
//
|
|
|
|
// Note: Using Filter + FilterAddresses at the same time is fine, but you cannot
|
|
|
|
// configure a both ConnectionGater and filtered addresses.
|
|
|
|
//
|
|
|
|
// Deprecated: Please use ConnectionGater() instead.
|
2018-07-04 10:51:47 +00:00
|
|
|
func FilterAddresses(addrs ...*net.IPNet) Option {
|
|
|
|
return func(cfg *Config) error {
|
2021-06-16 20:19:45 +00:00
|
|
|
var f *filtersConnectionGater
|
|
|
|
|
|
|
|
// preserve backwards compatibility.
|
|
|
|
// if we have a connection gater, try to cast it to a *filtersConnectionGater.
|
|
|
|
if cfg.ConnectionGater != nil {
|
|
|
|
var ok bool
|
|
|
|
if f, ok = cfg.ConnectionGater.(*filtersConnectionGater); !ok {
|
|
|
|
return errors.New("cannot configure both Filters and Connection Gater. " +
|
|
|
|
"\n Please consider configuring just a ConnectionGater instead.")
|
|
|
|
}
|
2018-07-04 10:51:47 +00:00
|
|
|
}
|
2021-06-16 20:19:45 +00:00
|
|
|
|
|
|
|
if f == nil {
|
|
|
|
f = (*filtersConnectionGater)(ma.NewFilters())
|
|
|
|
cfg.ConnectionGater = f
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
for _, addr := range addrs {
|
2021-06-16 20:19:45 +00:00
|
|
|
(*ma.Filters)(f).AddFilter(*addr, ma.ActionDeny)
|
2018-07-04 10:51:47 +00:00
|
|
|
}
|
2021-06-16 20:19:45 +00:00
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-04 15:21:24 +00:00
|
|
|
// Filters configures libp2p to use the given filters for accepting/denying
|
2021-06-16 20:19:45 +00:00
|
|
|
// certain addresses. Filters offers more control and should be used when the
|
2019-10-04 15:21:24 +00:00
|
|
|
// addresses you want to accept/deny are not known ahead of time and can
|
|
|
|
// dynamically change.
|
2021-06-16 20:19:45 +00:00
|
|
|
//
|
|
|
|
// Note: You cannot configure both a ConnectionGater and a Filter at the same
|
|
|
|
// time. Under the hood, the Filters object is converted to a ConnectionGater.
|
|
|
|
//
|
|
|
|
// Deprecated: use ConnectionGater() instead.
|
|
|
|
func Filters(filters *ma.Filters) Option {
|
2019-10-04 15:21:24 +00:00
|
|
|
return func(cfg *Config) error {
|
2021-06-16 20:19:45 +00:00
|
|
|
if cfg.ConnectionGater != nil {
|
|
|
|
return errors.New("cannot configure both Filters and Connection Gater. " +
|
|
|
|
"\n Please consider configuring just a ConnectionGater instead.")
|
|
|
|
|
|
|
|
}
|
|
|
|
cfg.ConnectionGater = (*filtersConnectionGater)(filters)
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// ConnectionGater configures libp2p to use the given ConnectionGater
|
|
|
|
// to actively reject inbound/outbound connections based on the lifecycle stage
|
|
|
|
// of the connection.
|
|
|
|
//
|
|
|
|
// For more information, refer to go-libp2p-core.ConnectionGater.
|
|
|
|
func ConnectionGater(cg connmgr.ConnectionGater) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.ConnectionGater != nil {
|
|
|
|
return errors.New("cannot configure multiple connection gaters, or cannot configure both Filters and ConnectionGater")
|
|
|
|
}
|
|
|
|
cfg.ConnectionGater = cg
|
2019-10-04 15:21:24 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-04 10:51:47 +00:00
|
|
|
// NATPortMap configures libp2p to use the default NATManager. The default
|
|
|
|
// NATManager will attempt to open a port in your network's firewall using UPnP.
|
|
|
|
func NATPortMap() Option {
|
|
|
|
return NATManager(bhost.NewNATManager)
|
|
|
|
}
|
|
|
|
|
|
|
|
// NATManager will configure libp2p to use the requested NATManager. This
|
|
|
|
// function should be passed a NATManager *constructor* that takes a libp2p Network.
|
|
|
|
func NATManager(nm config.NATManagerC) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.NATManager != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple NATManagers")
|
|
|
|
}
|
|
|
|
cfg.NATManager = nm
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
2019-06-09 07:24:20 +00:00
|
|
|
|
|
|
|
// Ping will configure libp2p to support the ping service; enable by default.
|
|
|
|
func Ping(enable bool) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.DisablePing = !enable
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Routing will configure libp2p to use routing.
|
|
|
|
func Routing(rt config.RoutingC) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
if cfg.Routing != nil {
|
|
|
|
return fmt.Errorf("cannot specify multiple routing options")
|
|
|
|
}
|
|
|
|
cfg.Routing = rt
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// NoListenAddrs will configure libp2p to not listen by default.
|
|
|
|
//
|
|
|
|
// This will both clear any configured listen addrs and prevent libp2p from
|
|
|
|
// applying the default listen address option. It also disables relay, unless the
|
|
|
|
// user explicitly specifies with an option, as the transport creates an implicit
|
|
|
|
// listen address that would make the node dialable through any relay it was connected to.
|
|
|
|
var NoListenAddrs = func(cfg *Config) error {
|
|
|
|
cfg.ListenAddrs = []ma.Multiaddr{}
|
|
|
|
if !cfg.RelayCustom {
|
|
|
|
cfg.RelayCustom = true
|
|
|
|
cfg.Relay = false
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// NoTransports will configure libp2p to not enable any transports.
|
|
|
|
//
|
|
|
|
// This will both clear any configured transports (specified in prior libp2p
|
|
|
|
// options) and prevent libp2p from applying the default transports.
|
|
|
|
var NoTransports = func(cfg *Config) error {
|
|
|
|
cfg.Transports = []config.TptC{}
|
|
|
|
return nil
|
|
|
|
}
|
2019-10-04 15:21:24 +00:00
|
|
|
|
|
|
|
// UserAgent sets the libp2p user-agent sent along with the identify protocol
|
|
|
|
func UserAgent(userAgent string) Option {
|
|
|
|
return func(cfg *Config) error {
|
|
|
|
cfg.UserAgent = userAgent
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|