2015-02-05 16:36:25 +00:00
|
|
|
#include <arpa/inet.h>
|
|
|
|
|
#include <netinet/in.h>
|
|
|
|
|
#include <stdio.h>
|
|
|
|
|
#include <unistd.h>
|
|
|
|
|
|
2015-02-06 16:27:47 +00:00
|
|
|
#include <cassert>
|
2015-02-05 16:36:25 +00:00
|
|
|
#include <iostream>
|
|
|
|
|
|
2015-02-06 16:27:47 +00:00
|
|
|
#include <sodium/crypto_box.h>
|
|
|
|
|
#include <sodium/crypto_secretbox.h>
|
|
|
|
|
#include <sodium/randombytes.h>
|
|
|
|
|
|
2015-02-05 12:55:48 +00:00
|
|
|
#include "crypto.h"
|
|
|
|
|
|
|
|
|
|
|
2015-02-05 13:02:42 +00:00
|
|
|
void CryptoBase::GenKey(std::string* key) {
|
2015-02-06 16:27:47 +00:00
|
|
|
unsigned char buf[crypto_secretbox_KEYBYTES];
|
|
|
|
|
randombytes_buf(buf, crypto_secretbox_KEYBYTES);
|
|
|
|
|
key->assign((char*)buf, crypto_secretbox_KEYBYTES);
|
2015-02-05 13:02:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CryptoBase::GenKeyPair(std::string* secret_key, std::string* public_key) {
|
2015-02-06 16:27:47 +00:00
|
|
|
unsigned char public_key_buf[crypto_box_PUBLICKEYBYTES];
|
|
|
|
|
unsigned char secret_key_buf[crypto_box_PUBLICKEYBYTES];
|
|
|
|
|
assert(crypto_box_keypair(public_key_buf, secret_key_buf) == 0);
|
|
|
|
|
public_key->assign((char*)public_key_buf, crypto_box_PUBLICKEYBYTES);
|
|
|
|
|
secret_key->assign((char*)secret_key_buf, crypto_box_SECRETKEYBYTES);
|
2015-02-05 12:55:48 +00:00
|
|
|
}
|
2015-02-05 16:36:25 +00:00
|
|
|
|
|
|
|
|
|
2015-02-07 17:07:31 +01:00
|
|
|
CryptoPubServer::CryptoPubServer(const std::string& secret_key)
|
2015-02-07 16:44:42 +01:00
|
|
|
: secret_key_(secret_key),
|
|
|
|
|
event_base_(event_base_new()) {
|
|
|
|
|
struct sockaddr_in6 server_addr = {0};
|
|
|
|
|
server_addr.sin6_family = AF_INET6;
|
|
|
|
|
server_addr.sin6_addr = in6addr_any;
|
|
|
|
|
server_addr.sin6_port = htons(4990);
|
|
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
listener_ = evconnlistener_new_bind(event_base_, &CryptoPubServer::OnNewConn_, this, LEV_OPT_REUSEABLE|LEV_OPT_CLOSE_ON_FREE, -1, (struct sockaddr*)&server_addr, sizeof(server_addr));
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CryptoPubServer::~CryptoPubServer() {
|
2015-02-07 16:44:42 +01:00
|
|
|
evconnlistener_free(listener_);
|
|
|
|
|
event_base_free(event_base_);
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
void CryptoPubServer::OnNewConn_(struct evconnlistener* listener, int client_fd, struct sockaddr* client_addr_, int client_addrlen, void* this__) {
|
2015-02-07 16:44:42 +01:00
|
|
|
auto this_ = (CryptoPubServer*)this__;
|
2015-02-07 17:15:09 +01:00
|
|
|
this_->OnNewConn(client_fd, client_addr_, client_addrlen);
|
|
|
|
|
}
|
2015-02-07 16:44:42 +01:00
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
void CryptoPubServer::OnNewConn(int client_fd, struct sockaddr* client_addr_, int client_addrlen) {
|
2015-02-07 16:44:42 +01:00
|
|
|
assert(client_addr_->sa_family == AF_INET6);
|
|
|
|
|
auto client_addr = (struct sockaddr_in6*)client_addr_;
|
2015-02-05 16:36:25 +00:00
|
|
|
|
|
|
|
|
char buf[128];
|
2015-02-07 16:44:42 +01:00
|
|
|
inet_ntop(AF_INET6, &client_addr->sin6_addr, buf, 128);
|
|
|
|
|
std::cerr << "New connection from [" << buf << "]:" << ntohs(client_addr->sin6_port) << std::endl;
|
|
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
auto bev = bufferevent_socket_new(this->event_base_, client_fd, BEV_OPT_CLOSE_ON_FREE);
|
2015-02-07 16:44:42 +01:00
|
|
|
bufferevent_enable(bev, EV_READ);
|
|
|
|
|
bufferevent_disable(bev, EV_WRITE);
|
2015-02-07 17:15:09 +01:00
|
|
|
auto peer = new CryptoPubServerConnection(bev, this->secret_key_);
|
|
|
|
|
bufferevent_setcb(bev, &CryptoPubServerConnection::OnReadable_, NULL, &CryptoPubServerConnection::OnError_, peer);
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CryptoPubServer::Loop() {
|
2015-02-07 16:44:42 +01:00
|
|
|
event_base_dispatch(event_base_);
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
2015-02-07 17:07:31 +01:00
|
|
|
CryptoPubServerConnection::CryptoPubServerConnection(struct bufferevent* bev, const std::string& secret_key)
|
2015-02-07 16:44:42 +01:00
|
|
|
: bev_(bev),
|
2015-02-05 21:57:04 +00:00
|
|
|
secret_key_(secret_key),
|
|
|
|
|
state_(AWAITING_HANDSHAKE) {
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
|
|
|
|
|
2015-02-07 16:44:42 +01:00
|
|
|
CryptoPubServerConnection::~CryptoPubServerConnection() {
|
|
|
|
|
std::cerr << "Connection closed" << std::endl;
|
|
|
|
|
bufferevent_free(bev_);
|
|
|
|
|
}
|
|
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
void CryptoPubServerConnection::OnReadable_(struct bufferevent* bev, void* this__) {
|
2015-02-07 16:44:42 +01:00
|
|
|
auto this_ = (CryptoPubServerConnection*)this__;
|
2015-02-07 17:15:09 +01:00
|
|
|
this_->OnReadable();
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CryptoPubServerConnection::OnReadable() {
|
|
|
|
|
std::cerr << "OnReadable" << std::endl;
|
2015-02-05 16:36:25 +00:00
|
|
|
char buf[128];
|
2015-02-07 17:15:09 +01:00
|
|
|
bufferevent_read(bev_, buf, 128);
|
2015-02-07 16:44:42 +01:00
|
|
|
}
|
|
|
|
|
|
2015-02-07 17:15:09 +01:00
|
|
|
void CryptoPubServerConnection::OnError_(struct bufferevent* bev, const short what, void* this__) {
|
2015-02-07 16:44:42 +01:00
|
|
|
auto this_ = (CryptoPubServerConnection*)this__;
|
2015-02-07 17:15:09 +01:00
|
|
|
this_->OnError(what);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CryptoPubServerConnection::OnError(const short what) {
|
|
|
|
|
std::cerr << "OnError" << std::endl;
|
|
|
|
|
delete this;
|
2015-02-05 16:36:25 +00:00
|
|
|
}
|
2015-02-07 17:07:31 +01:00
|
|
|
|
|
|
|
|
|
|
|
|
|
CryptoPubClient::CryptoPubClient(struct sockaddr* addr, socklen_t addrlen)
|
|
|
|
|
: event_base_(event_base_new()),
|
|
|
|
|
bev_(bufferevent_socket_new(event_base_, -1, BEV_OPT_CLOSE_ON_FREE)) {
|
|
|
|
|
bufferevent_socket_connect(bev_, addr, addrlen);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CryptoPubClient::~CryptoPubClient() {
|
|
|
|
|
bufferevent_free(bev_);
|
|
|
|
|
event_base_free(event_base_);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
CryptoPubClient* CryptoPubClient::FromHostname(const std::string& server_address, const std::string& server_port) {
|
|
|
|
|
struct addrinfo* res;
|
|
|
|
|
int gai_ret = getaddrinfo(server_address.c_str(), server_port.c_str(), NULL, &res);
|
|
|
|
|
if (gai_ret) {
|
|
|
|
|
std::cerr << "Failed to resolve server_address: " << gai_strerror(gai_ret) << std::endl;
|
|
|
|
|
return nullptr;
|
|
|
|
|
}
|
|
|
|
|
auto ret = new CryptoPubClient((struct sockaddr*)res->ai_addr, res->ai_addrlen);
|
|
|
|
|
freeaddrinfo(res);
|
|
|
|
|
return ret;
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
void CryptoPubClient::Loop() {
|
|
|
|
|
event_base_dispatch(event_base_);
|
|
|
|
|
}
|