2 meshlink.c -- Implementation of the MeshLink API.
3 Copyright (C) 2014 Guus Sliepen <guus@meshlink.io>
5 This program is free software; you can redistribute it and/or modify
6 it under the terms of the GNU General Public License as published by
7 the Free Software Foundation; either version 2 of the License, or
8 (at your option) any later version.
10 This program is distributed in the hope that it will be useful,
11 but WITHOUT ANY WARRANTY; without even the implied warranty of
12 MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
13 GNU General Public License for more details.
15 You should have received a copy of the GNU General Public License along
16 with this program; if not, write to the Free Software Foundation, Inc.,
17 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301 USA.
22 #include "meshlink_internal.h"
26 static const char *errstr[] = {
27 [MESHLINK_OK] = "No error",
28 [MESHLINK_ENOMEM] = "Out of memory",
29 [MESHLINK_ENOENT] = "No such node",
32 const char *meshlink_strerror(meshlink_errno_t errno) {
36 static meshlink_handle_t *meshlink_setup(meshlink_handle_t *mesh) {
40 meshlink_handle_t *meshlink_open(const char *confbase, const char *name) {
41 if(!confbase || !*confbase) {
42 fprintf(stderr, "No confbase given!\n");
47 fprintf(stderr, "No name given!\n");
52 fprintf(stderr, "Invalid name given!\n");
56 meshlink_handle_t *mesh = xzalloc(sizeof *mesh);
57 mesh->confbase = xstrdup(confbase);
58 mesh->name = xstrdup(name);
60 char filename[PATH_MAX];
61 snprintf(filename, sizeof filename, "%s" SLASH "meshlink.conf", confbase);
63 FILE *f = fopen(filename, "r");
65 if(!f && errno == ENOENT)
66 return meshlink_setup(mesh);
69 fprintf(stderr, "Could not open %s: %s\n", filename, strerror(errno));
70 return meshlink_close(mesh), NULL;
74 if(!fgets(buf, sizeof buf, f)) {
75 fprintf(stderr, "Could not read line from %s: %s\n", filename, strerror(errno));
77 return meshlink_close(mesh), NULL;
82 size_t len = strlen(buf);
83 if(len && buf[len - 1] == '\n')
85 if(len && buf[len - 1] == '\r')
88 if(strncmp(buf, "Name = ", 7) || !check_id(buf + 7)) {
89 fprintf(stderr, "Could not read Name from %s\n", filename);
90 return meshlink_close(mesh), NULL;
93 if(strcmp(buf + 7, name)) {
94 fprintf(stderr, "Name in %s is %s, not the same as %s\n", filename, buf + 7, name);
96 mesh->name = xstrdup(buf + 7);
99 snprintf(filename, sizeof filename, "%s" SLASH "ed25519_key.priv", mesh->confbase);
100 f = fopen(filename, "r");
102 fprintf(stderr, "Could not open %s: %s\n", filename, strerror(errno));
103 return meshlink_close(mesh), NULL;
106 mesh->self->ecdsa = ecdsa_read_pem_private_key(f);
109 if(!mesh->self->ecdsa) {
110 fprintf(stderr, "Could not read keypair!\n");
111 return meshlink_close(mesh), NULL;
117 bool meshlink_start(meshlink_handle_t *mesh) {
121 void meshlink_stop(meshlink_handle_t *mesh) {
124 void meshlink_close(meshlink_handle_t *mesh) {
127 void meshlink_set_receive_cb(meshlink_handle_t *mesh, meshlink_receive_cb_t cb) {
128 mesh->receive_cb = cb;
131 void meshlink_set_node_status_cb(meshlink_handle_t *mesh, meshlink_node_status_cb_t cb) {
132 mesh->node_status_cb = cb;
135 void meshlink_set_log_cb(meshlink_handle_t *mesh, meshlink_log_level_t level, meshlink_log_cb_t cb) {
137 mesh->log_level = level;
140 bool meshlink_send(meshlink_handle_t *mesh, meshlink_node_t *destination, const void *data, unsigned int len) {
144 meshlink_node_t *meshlink_get_node(meshlink_handle_t *mesh, const char *name) {
148 size_t meshlink_get_all_nodes(meshlink_handle_t *mesh, meshlink_node_t **nodes, size_t nmemb) {
152 char *meshlink_sign(meshlink_handle_t *mesh, const char *data, size_t len) {
156 bool meshlink_verify(meshlink_handle_t *mesh, meshlink_node_t *source, const char *data, size_t len, const char *signature) {
160 char *meshlink_invite(meshlink_handle_t *mesh, const char *name) {
164 bool meshlink_join(meshlink_handle_t *mesh, const char *invitation) {
168 char *meshlink_export(meshlink_handle_t *mesh) {
172 bool meshlink_import(meshlink_handle_t *mesh, const char *data) {
176 void meshlink_blacklist(meshlink_handle_t *mesh, meshlink_node_t *node) {