mirror of
https://github.com/aljazceru/lightning.git
synced 2026-01-11 10:04:28 +01:00
daemon: socket code.
At the moment, if you connect it just says Hello! and closes the socket. Signed-off-by: Rusty Russell <rusty@rustcorp.com.au>
This commit is contained in:
@@ -16,6 +16,7 @@ DAEMON_LIB_OBJS := $(DAEMON_LIB_SRC:.c=.o)
|
||||
DAEMON_SRC := \
|
||||
daemon/jsonrpc.c \
|
||||
daemon/lightningd.c \
|
||||
daemon/peer.c \
|
||||
daemon/timeout.c
|
||||
DAEMON_OBJS := $(DAEMON_SRC:.c=.o)
|
||||
|
||||
@@ -31,6 +32,7 @@ DAEMON_HEADERS := \
|
||||
daemon/jsonrpc.h \
|
||||
daemon/lightningd.h \
|
||||
daemon/log.h \
|
||||
daemon/peer.h \
|
||||
daemon/pseudorand.h \
|
||||
daemon/timeout.h
|
||||
|
||||
|
||||
@@ -2,6 +2,7 @@
|
||||
#include "jsonrpc.h"
|
||||
#include "lightningd.h"
|
||||
#include "log.h"
|
||||
#include "peer.h"
|
||||
#include "timeout.h"
|
||||
#include <ccan/container_of/container_of.h>
|
||||
#include <ccan/err/err.h>
|
||||
@@ -12,6 +13,7 @@
|
||||
#include <ccan/timer/timer.h>
|
||||
#include <errno.h>
|
||||
#include <string.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/stat.h>
|
||||
#include <sys/types.h>
|
||||
#include <unistd.h>
|
||||
@@ -25,6 +27,7 @@ static struct lightningd_state *lightningd_state(void)
|
||||
state->base_log = new_log(state, state->log_record,
|
||||
"lightningd(%u):", (int)getpid());
|
||||
|
||||
list_head_init(&state->peers);
|
||||
timers_init(&state->timers, time_now());
|
||||
return state;
|
||||
}
|
||||
@@ -52,6 +55,7 @@ int main(int argc, char *argv[])
|
||||
{
|
||||
struct lightningd_state *state = lightningd_state();
|
||||
struct timer *expired;
|
||||
unsigned int portnum = 0;
|
||||
|
||||
err_set_progname(argv[0]);
|
||||
opt_set_alloc(opt_allocfn, tal_reallocfn, tal_freefn);
|
||||
@@ -60,6 +64,8 @@ int main(int argc, char *argv[])
|
||||
"\n"
|
||||
"A bitcoin lightning daemon.",
|
||||
"Print this message.");
|
||||
opt_register_arg("--port", opt_set_uintval, NULL, &portnum,
|
||||
"Port to bind to (otherwise, dynamic port is used)");
|
||||
opt_register_logging(state->base_log);
|
||||
opt_register_version();
|
||||
|
||||
@@ -94,6 +100,9 @@ int main(int argc, char *argv[])
|
||||
/* Create RPC socket (if any) */
|
||||
setup_jsonrpc(state, state->rpc_filename);
|
||||
|
||||
/* Set up connections from peers. */
|
||||
setup_listeners(state, portnum);
|
||||
|
||||
log_info(state->base_log, "Hello world!");
|
||||
|
||||
/* If io_loop returns NULL, either a timer expired, or all fds closed */
|
||||
|
||||
@@ -1,6 +1,7 @@
|
||||
#ifndef LIGHTNING_DAEMON_LIGHTNING_H
|
||||
#define LIGHTNING_DAEMON_LIGHTNING_H
|
||||
#include "config.h"
|
||||
#include <ccan/list/list.h>
|
||||
#include <ccan/timer/timer.h>
|
||||
#include <stdio.h>
|
||||
|
||||
@@ -16,6 +17,9 @@ struct lightningd_state {
|
||||
char *rpc_filename;
|
||||
|
||||
/* Any pending timers. */
|
||||
struct timers timers;
|
||||
struct timers timers;
|
||||
|
||||
/* Our peers. */
|
||||
struct list_head peers;
|
||||
};
|
||||
#endif /* LIGHTNING_DAEMON_LIGHTNING_H */
|
||||
|
||||
178
daemon/peer.c
Normal file
178
daemon/peer.c
Normal file
@@ -0,0 +1,178 @@
|
||||
#include "lightningd.h"
|
||||
#include "log.h"
|
||||
#include "peer.h"
|
||||
#include <arpa/inet.h>
|
||||
#include <ccan/io/io.h>
|
||||
#include <ccan/noerr/noerr.h>
|
||||
#include <ccan/short_types/short_types.h>
|
||||
#include <ccan/tal/str/str.h>
|
||||
#include <ccan/tal/tal.h>
|
||||
#include <errno.h>
|
||||
#include <netinet/in.h>
|
||||
#include <stdlib.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/types.h>
|
||||
|
||||
static u16 get_port(const struct netaddr *addr)
|
||||
{
|
||||
switch (addr->saddr.s.sa_family) {
|
||||
case AF_INET:
|
||||
return ntohs(addr->saddr.ipv4.sin_port);
|
||||
case AF_INET6:
|
||||
return ntohs(addr->saddr.ipv6.sin6_port);
|
||||
default:
|
||||
abort();
|
||||
}
|
||||
}
|
||||
|
||||
static void destroy_peer(struct peer *peer)
|
||||
{
|
||||
list_del_from(&peer->state->peers, &peer->list);
|
||||
}
|
||||
|
||||
static struct peer *new_peer(struct lightningd_state *state,
|
||||
struct io_conn *conn,
|
||||
int addr_type, int addr_protocol,
|
||||
const char *in_or_out)
|
||||
{
|
||||
struct peer *peer = tal(state, struct peer);
|
||||
char name[INET6_ADDRSTRLEN];
|
||||
|
||||
/* FIXME: Stop listening if too many peers? */
|
||||
list_add(&state->peers, &peer->list);
|
||||
|
||||
peer->state = state;
|
||||
peer->addr.type = addr_type;
|
||||
peer->addr.protocol = addr_protocol;
|
||||
|
||||
/* FIXME: Attach IO logging for this peer. */
|
||||
tal_add_destructor(peer, destroy_peer);
|
||||
|
||||
peer->addr.addrlen = sizeof(peer->addr.saddr);
|
||||
if (getpeername(io_conn_fd(conn), &peer->addr.saddr.s,
|
||||
&peer->addr.addrlen) != 0) {
|
||||
log_unusual(state->base_log,
|
||||
"Could not get address for peer: %s",
|
||||
strerror(errno));
|
||||
return tal_free(peer);
|
||||
}
|
||||
|
||||
if (!inet_ntop(peer->addr.saddr.s.sa_family, &peer->addr.saddr,
|
||||
name, sizeof(name)))
|
||||
strcpy(name, "UNCONVERTABLE-ADDR");
|
||||
|
||||
peer->log = new_log(peer, state->log_record, "%s-%s:%s:%u",
|
||||
log_prefix(state->base_log), in_or_out,
|
||||
name, get_port(&peer->addr));
|
||||
return peer;
|
||||
}
|
||||
|
||||
struct io_plan *peer_connected_out(struct io_conn *conn,
|
||||
struct lightningd_state *state,
|
||||
const char *name, const char *port)
|
||||
{
|
||||
struct peer *peer = new_peer(state, conn, SOCK_STREAM, IPPROTO_TCP,
|
||||
"out");
|
||||
if (!peer) {
|
||||
log_unusual(peer->log, "Failed to make peer for %s:%s",
|
||||
name, port);
|
||||
return io_close(conn);
|
||||
}
|
||||
log_info(peer->log, "Connected out to %s:%s", name, port);
|
||||
return io_write(conn, "Hello!", 6, io_close_cb, NULL);
|
||||
}
|
||||
|
||||
static struct io_plan *peer_connected_in(struct io_conn *conn,
|
||||
struct lightningd_state *state)
|
||||
{
|
||||
struct peer *peer = new_peer(state, conn, SOCK_STREAM, IPPROTO_TCP,
|
||||
"in");
|
||||
if (!peer)
|
||||
return io_close(conn);
|
||||
|
||||
return io_write(conn, "Hello!", 6, io_close_cb, NULL);
|
||||
}
|
||||
|
||||
static int make_listen_fd(struct lightningd_state *state,
|
||||
int domain, void *addr, socklen_t len)
|
||||
{
|
||||
int fd = socket(domain, SOCK_STREAM, 0);
|
||||
if (fd < 0) {
|
||||
log_debug(state->base_log, "Failed to create %u socket: %s",
|
||||
domain, strerror(errno));
|
||||
return -1;
|
||||
}
|
||||
|
||||
if (!addr || bind(fd, addr, len) == 0) {
|
||||
if (listen(fd, 5) == 0)
|
||||
return fd;
|
||||
log_unusual(state->base_log, "Failed to listen on %u socket: %s",
|
||||
domain, strerror(errno));
|
||||
} else
|
||||
log_debug(state->base_log, "Failed to bind on %u socket: %s",
|
||||
domain, strerror(errno));
|
||||
|
||||
close_noerr(fd);
|
||||
return -1;
|
||||
}
|
||||
|
||||
void setup_listeners(struct lightningd_state *state, unsigned int portnum)
|
||||
{
|
||||
struct sockaddr_in addr;
|
||||
struct sockaddr_in6 addr6;
|
||||
socklen_t len;
|
||||
int fd1, fd2;
|
||||
u16 listen_port;
|
||||
|
||||
addr.sin_family = AF_INET;
|
||||
addr.sin_addr.s_addr = INADDR_ANY;
|
||||
addr.sin_port = htons(portnum);
|
||||
|
||||
addr6.sin6_family = AF_INET6;
|
||||
addr6.sin6_addr = in6addr_any;
|
||||
addr6.sin6_port = htons(portnum);
|
||||
|
||||
/* IPv6, since on Linux that (usually) binds to IPv4 too. */
|
||||
fd1 = make_listen_fd(state, AF_INET6, portnum ? &addr6 : NULL,
|
||||
sizeof(addr6));
|
||||
if (fd1 >= 0) {
|
||||
struct sockaddr_in6 in6;
|
||||
|
||||
len = sizeof(in6);
|
||||
if (getsockname(fd1, (void *)&in6, &len) != 0) {
|
||||
log_unusual(state->base_log,
|
||||
"Failed get IPv6 sockname: %s",
|
||||
strerror(errno));
|
||||
close_noerr(fd1);
|
||||
} else {
|
||||
addr.sin_port = in6.sin6_port;
|
||||
listen_port = ntohs(addr.sin_port);
|
||||
log_info(state->base_log,
|
||||
"Creating IPv6 listener on port %u",
|
||||
listen_port);
|
||||
io_new_listener(state, fd1, peer_connected_in, state);
|
||||
}
|
||||
}
|
||||
|
||||
/* Just in case, aim for the same port... */
|
||||
fd2 = make_listen_fd(state, AF_INET,
|
||||
addr.sin_port ? &addr : NULL, sizeof(addr));
|
||||
if (fd2 >= 0) {
|
||||
len = sizeof(addr);
|
||||
if (getsockname(fd2, (void *)&addr, &len) != 0) {
|
||||
log_unusual(state->base_log,
|
||||
"Failed get IPv4 sockname: %s",
|
||||
strerror(errno));
|
||||
close_noerr(fd2);
|
||||
} else {
|
||||
listen_port = ntohs(addr.sin_port);
|
||||
log_info(state->base_log,
|
||||
"Creating IPv4 listener on port %u",
|
||||
listen_port);
|
||||
io_new_listener(state, fd2, peer_connected_in, state);
|
||||
}
|
||||
}
|
||||
|
||||
if (fd1 < 0 && fd2 < 0)
|
||||
fatal("Could not bind to a network address");
|
||||
}
|
||||
43
daemon/peer.h
Normal file
43
daemon/peer.h
Normal file
@@ -0,0 +1,43 @@
|
||||
#ifndef LIGHTNING_DAEMON_PEER_H
|
||||
#define LIGHTNING_DAEMON_PEER_H
|
||||
#include "config.h"
|
||||
#include <ccan/list/list.h>
|
||||
#include <netinet/in.h>
|
||||
#include <netinet/ip.h>
|
||||
#include <sys/socket.h>
|
||||
#include <sys/socket.h>
|
||||
|
||||
/* This can be extended to support other protocols in future. */
|
||||
struct netaddr {
|
||||
int type; /* See socket(2): SOCK_STREAM currently */
|
||||
int protocol; /* See socket(2): 0 currently */
|
||||
socklen_t addrlen;
|
||||
union {
|
||||
struct sockaddr s;
|
||||
struct sockaddr_in ipv4;
|
||||
struct sockaddr_in6 ipv6;
|
||||
} saddr;
|
||||
};
|
||||
|
||||
struct peer {
|
||||
/* state->peers list */
|
||||
struct list_node list;
|
||||
|
||||
/* Global state. */
|
||||
struct lightningd_state *state;
|
||||
|
||||
/* The other end's address. */
|
||||
struct netaddr addr;
|
||||
|
||||
/* What happened. */
|
||||
struct log *log;
|
||||
};
|
||||
|
||||
struct io_conn;
|
||||
struct io_plan *peer_connected_out(struct io_conn *conn,
|
||||
struct lightningd_state *state,
|
||||
const char *name, const char *port);
|
||||
|
||||
void setup_listeners(struct lightningd_state *state, unsigned int portnum);
|
||||
|
||||
#endif /* LIGHTNING_DAEMON_PEER_H */
|
||||
Reference in New Issue
Block a user