gdr / tuntox (public) (License: GPLv3) (since 2017-01-24) (hash sha1)
Tunnel TCP connections over the Tox protocol

/util.c (cdf3ddcf61249fbfa406ac648b1d8cc6c3d9dd89) (5406 bytes) (mode 100644) (type blob)

#include "log.h"
#include "util.h"
#include <arpa/inet.h>
#include <errno.h>
#include <string.h>
#include <tox/tox.h>
#include <stdio.h>
#include <stdlib.h>

void writechecksum(uint8_t *address)
{
    uint8_t *checksum = address + 36;
    uint32_t i;

    for (i = 0; i < 36; ++i)
	checksum[i % 2] ^= address[i];
}

/* From utox/util.c */
void to_hex(char_t *a, const char_t *p, int size)
{
    char_t b, c;
    const char_t *end = p + size;

    while(p != end) {
        b = *p++;

        c = (b & 0xF);
        b = (b >> 4);

        if(b < 10) {
            *a++ = b + '0';
        } else {
            *a++ = b - 10 + 'A';
        }

        if(c < 10) {
            *a++ = c + '0';
        } else {
            *a++ = c  - 10 + 'A';
        }
    }
    *a = '\0';
}

/* From utox/util.c */
void id_to_string(char_t *dest, const char_t *src)
{
    to_hex(dest, src, TOX_ADDRESS_SIZE);
}

/* From utox/util.c */
int string_to_id(char_t *w, char_t *a)
{
    char_t *end = w + TOX_ADDRESS_SIZE;
    while(w != end) {
        char_t c, v;

        c = *a++;
        if(c >= '0' && c <= '9') {
            v = (c - '0') << 4;
        } else if(c >= 'A' && c <= 'F') {
            v = (c - 'A' + 10) << 4;
        } else if(c >= 'a' && c <= 'f') {
            v = (c - 'a' + 10) << 4;
        } else {
            return 0;
        }

        c = *a++;
        if(c >= '0' && c <= '9') {
            v |= (c - '0');
        } else if(c >= 'A' && c <= 'F') {
            v |= (c - 'A' + 10);
        } else if(c >= 'a' && c <= 'f') {
            v |= (c - 'a' + 10);
        } else {
            return 0;
        }

        *w++ = v;
    }

    return 1;
}

/* Parse the -L parameter */
/* 0 = success */
int parse_local_port_forward(char *string, int *local_port, char **hostname, int *remote_port)
{
    char *lport;
    char *host;
    char *rport;

    /* First replace all @ with :, ':' is forbidden in some environments */
    char *p = string;
    while(*p) 
    { 
        if(*p == '@') *p = ':';
        p++;
    }
    
    lport = strtok(string, ":");
    host = strtok(NULL, ":");
    rport = strtok(NULL, ":");

    if(!lport || !host || !rport)
    {
        return -1;
    }

    *local_port = atoi(lport);
    *hostname = host;
    *remote_port = atoi(rport);

    return 0;
}

/* Parse the -W parameter */
/* 0 = success */
int parse_pipe_port_forward(char *string, char **hostname, int *remote_port)
{
    char *host;
    char *rport;

    /* First replace all @ with :, ':' is forbidden in some environments */
    char *p = string;
    while(*p) 
    { 
        if(*p == '@') *p = ':';
        p++;
    }
    
    host = strtok(string, ":");
    rport = strtok(NULL, ":");

    if(!host || !rport)
    {
        return -1;
    }

    *hostname = host;
    *remote_port = atoi(rport);

    return 0;
}

void* file_raw(char *path, uint32_t *size)
{
    FILE *file;
    char *data;
    int len;

    file = fopen(path, "rb");
    if(!file) {
        log_printf(L_WARNING, "File not found (%s)\n", path);
        return NULL;
    }

    fseek(file, 0, SEEK_END);
    len = ftell(file);
    if(len <= 0)
    {
        fclose(file);
        return NULL;
    }
    data = malloc(len);
    if(!data) {
        fclose(file);
        return NULL;
    }

    fseek(file, 0, SEEK_SET);

    if(fread(data, len, 1, file) != 1) {
        log_printf(L_WARNING, "Read error (%s)\n", path);
        fclose(file);
        free(data);
        return NULL;
    }

    fclose(file);

    log_printf(L_DEBUG, "Read %u bytes (%s)\n", len, path);

    if(size) {
        *size = len;
    }
    return data;
}

const char *readable_connection_status(TOX_CONNECTION status)
{
    switch(status)
    {
        case TOX_CONNECTION_NONE:
            return "There is no connection";
        case TOX_CONNECTION_TCP:
            return "A TCP connection has been established (via TCP relay)";
        case TOX_CONNECTION_UDP:
            return "An UDP connection has been established";
        default:
            log_printf(L_WARNING, "Received unknown connection status %d\n", (int)status);
            return "Unknown connection status";
    }
}

/* From https://github.com/TokTok/c-toxcore/blob/master/other/fun/cracker.c */
size_t hex_string_to_bin(const char *hex_string, size_t hex_len, uint8_t *bytes)
{
    size_t i;
    const char *pos = hex_string;
    // make even
    for (i = 0; i < hex_len / 2; ++i, pos += 2) {
        uint8_t val;
        if (sscanf(pos, "%02hhx", &val) != 1) {
            return 0;
        }
        bytes[i] = val;
    }
    if (i * 2 < hex_len) {
        uint8_t val;
        if (sscanf(pos, "%hhx", &val) != 1) {
            return 0;
        }
        bytes[i] = (uint8_t)(val << 4);
        ++i;
    }
    return i;
}

/* Very stupid test to filter out hostnames */
bool is_valid_ipv4(const char *ip_address)
{
    unsigned int a,b,c,d;
    return sscanf(ip_address,"%u.%u.%u.%u", &a, &b, &c, &d) == 4;
}

bool is_valid_ipv6(const char *ip_address)
{
   struct in6_addr result;
   return (inet_pton(AF_INET6, ip_address, &result) == 1);
}

void save_printable_tox_id(const unsigned char *tox_printable_id, const char *path)
{
   FILE *f = fopen(path, "w");
   if(!f)
   {
      log_printf(L_ERROR, "Could not write to %s: %d %s", path, errno, strerror(errno));
   }
   log_printf(L_DEBUG, "Writing Tox ID to %s", path);
   fputs((char*)tox_printable_id, f);
   fclose(f);
}


Mode Type Size Ref File
100644 blob 268 272c4eb3ad3672621962ce38f8c7472336729ec3 .gitignore
100644 blob 0 e69de29bb2d1d6434b8b29ae775ad8c2e48c5391 .sonarcloud.properties
100644 blob 2088 300c5a7e37f67cb8cdd88261756a10be561d51c5 .travis.yml
100644 blob 1934 9b63571486cca0d558fb18d7826e84f8983217de BUILD.md
100644 blob 612 2ad1c03f9e18223415e56a31bfb2322707ee193f Dockerfile
100644 blob 5248 71a20b47b2798e81647e041cceb3fec887e6019c FAQ.md
100644 blob 36522 d6af5d577e5ccb42e451bffdb59696ae1b8078ab LICENSE.md
100644 blob 1332 67eff94b8d8f56754f643ac90cd76bd0517006b8 Makefile
100644 blob 652 6a0550489a8718388bef0ef61e36f6a0a9d8bfdf Makefile.mac
100644 blob 6215 b1c263d3db0d2728247d8fea652550653f6c6d1e README.md
100644 blob 1989 f1c8658a62d5be3c1726843db101a54df9c52f47 VPN.md
100644 blob 939 d4c2844167d0a3bbe01f06adc650e28ae6d79690 bitbucket-pipelines.yml
100644 blob 77769 3063f74e9864b29839928800f1cfb0469ee74990 cJSON.c
100644 blob 15829 92907a2cd38b1e1f2e099e5cbe2f5c557b70b07b cJSON.h
100644 blob 26918 690abfbce03de953d63e415cc56f1ee831305972 client.c
100644 blob 759 be68f25ae57282e30acf01fd0eb619763045dc1e client.h
040000 tree - 8242200a82b7b6d771a336c7c81b4f76b8e7be84 debian
100644 blob 3389 3198cd41259d1954f69283b4e13ce279b4e04c46 generate_tox_bootstrap.py
100644 blob 467 b5da76802ae2be17bf221890db7e7c0b230beffa gitversion.c
100644 blob 62 a728c2b3232d4c97fa3bd413ca4c2da18ea970e3 gitversion.h
100644 blob 3141 31244349cd221b4e8931f612b3325ae59faa58cb log.c
100644 blob 892 bcd4c9bb1af0a1f1c44b1e7a36c3a5971ba73b34 log.h
100644 blob 169 57385c284c57ab99d21bd53c270ebc04ecd19d31 mach.h
100644 blob 53305 efa22761c346b5f4cfcf50d0c3262b8f76e361c3 main.c
100644 blob 4500 af9f47a54f870914913329ad162b730ea3c73aa1 main.h
040000 tree - 9ea39ebb6fd8fb34f1a28a69d445d099d5001a37 screenshots
040000 tree - b412cd72eccd06bedcb8f98492901c638dd0010b scripts
100644 blob 27171 b1057f5dd5f8786086b29213998b7eeb4af8e68d tox_bootstrap.h
100644 blob 3272 61c6bc0d273f973522940ef64982f8c7a1781973 tox_bootstrap_json.c
100644 blob 124 f6479b9112cbf5aab844fb733a3ab7d9f9a683b0 tox_bootstrap_json.h
100644 blob 12536 75e9dc5ed9399120416e8da5f24d1ccde41cf901 utarray.h
100644 blob 61492 7205c67efa27c66884c8d4d1c8a105d4854a0548 uthash.h
100644 blob 5406 cdf3ddcf61249fbfa406ac648b1d8cc6c3d9dd89 util.c
100644 blob 893 33e1cb381b18a34d413d7379121f9a8e584eaf53 util.h
100644 blob 55882 b5f3f04c104785a57d8280c37c1b19b36068e56e utlist.h
100644 blob 11555 867442c843dbe6bf096a488e3ce9ec6323809f7f utstring.h
Hints:
Before first commit, do not forget to setup your git environment:
git config --global user.name "your_name_here"
git config --global user.email "your@email_here"

Clone this repository using HTTP(S):
git clone https://rocketgit.com/user/gdr/tuntox

Clone this repository using ssh (do not forget to upload a key first):
git clone ssh://rocketgit@ssh.rocketgit.com/user/gdr/tuntox

Clone this repository using git:
git clone git://git.rocketgit.com/user/gdr/tuntox

You are allowed to anonymously push to this repository.
This means that your pushed commits will automatically be transformed into a merge request:
... clone the repository ...
... make some changes and some commits ...
git push origin main