#include <assert.h> #include <ccan/mem/mem.h> #include <ccan/opt/opt.h> #include <ccan/read_write_all/read_write_all.h> #include <ccan/short_types/short_types.h> #include <ccan/str/hex/hex.h> #include <ccan/tal/grab_file/grab_file.h> #include <ccan/tal/str/str.h> #include <common/amount.h> #include <common/ecdh.h> #include <common/json.h> #include <common/json_helpers.h> #include <common/onion.h> #include <common/sphinx.h> #include <common/utils.h> #include <common/version.h> #include <err.h> #include <secp256k1.h> #include <stdio.h> #include <string.h> #include <unistd.h> /* We don't actually use this, but common/onion needs it */ void ecdh(const struct pubkey *point, struct secret *ss) { abort(); } static void do_generate(int argc, char **argv, const u8 *assocdata, const struct node_id *rvnode_id) { const tal_t *ctx = talz(NULL, tal_t); int num_hops = argc - 2; struct pubkey *path = tal_arr(ctx, struct pubkey, num_hops); u8 rawprivkey[PRIVKEY_LEN]; struct secret session_key; struct secret *shared_secrets; struct sphinx_path *sp; struct sphinx_compressed_onion *comp; u8 *serialized; struct onionpacket *packet; memset(&session_key, 'A', sizeof(struct secret)); sp = sphinx_path_new_with_key(ctx, assocdata, &session_key); sphinx_path_set_rendezvous(sp, rvnode_id); for (int i = 0; i < num_hops; i++) { size_t klen = strcspn(argv[2 + i], "/"); if (hex_data_size(klen) == PRIVKEY_LEN) { if (!hex_decode(argv[2 + i], klen, rawprivkey, PRIVKEY_LEN)) errx(1, "Invalid private key hex '%s'", argv[2 + i]); if (secp256k1_ec_pubkey_create(secp256k1_ctx, &path[i].pubkey, rawprivkey) != 1) errx(1, "Could not decode pubkey"); } else if (hex_data_size(klen) == PUBKEY_CMPR_LEN) { if (!pubkey_from_hexstr(argv[2 + i], klen, &path[i])) errx(1, "Invalid public key hex '%s'", argv[2 + i]); } else { errx(1, "Provided key is neither a pubkey nor a privkey: " "%s\n", argv[2 + i]); } /* /<hex> -> raw hopdata. /tlv -> TLV encoding. */ if (argv[2 + i][klen] != '\0' && argv[2 + i][klen] != 't') { const char *hopstr = argv[2 + i] + klen + 1; u8 *data = tal_hexdata(ctx, hopstr, strlen(hopstr)); if (!data) errx(1, "bad hex after / in %s", argv[1 + i]); sphinx_add_hop(sp, &path[i], data); } else { struct short_channel_id scid; struct amount_msat amt; bool use_tlv = streq(argv[1 + i] + klen, "/tlv"); /* FIXME: support secret and and total_msat */ memset(&scid, i, sizeof(scid)); amt.millisatoshis = i; /* Raw: test code */ if (i == num_hops - 1) sphinx_add_hop(sp, &path[i], take(onion_final_hop(NULL, use_tlv, amt, i, amt, NULL, NULL, NULL))); else sphinx_add_hop(sp, &path[i], take(onion_nonfinal_hop(NULL, use_tlv, &scid, amt, i, NULL, NULL))); } } packet = create_onionpacket(ctx, sp, &shared_secrets); if (rvnode_id != NULL) { comp = sphinx_compress(ctx, packet, sp); serialized = sphinx_compressed_onion_serialize(ctx, comp); printf("Rendezvous onion: %s\n", tal_hex(ctx, serialized)); } else { assert(sphinx_compress(ctx, packet, sp) == NULL); } serialized = serialize_onionpacket(ctx, packet); if (!serialized) errx(1, "Error serializing message."); printf("%s\n", tal_hex(ctx, serialized)); tal_free(ctx); } static struct route_step *decode_with_privkey(const tal_t *ctx, const u8 *onion, char *hexprivkey, const u8 *assocdata) { struct privkey seckey; struct route_step *step; struct onionpacket packet; enum onion_type why_bad; struct secret shared_secret; if (!hex_decode(hexprivkey, strlen(hexprivkey), &seckey, sizeof(seckey))) errx(1, "Invalid private key hex '%s'", hexprivkey); why_bad = parse_onionpacket(onion, TOTAL_PACKET_SIZE, &packet); if (why_bad != 0) errx(1, "Error parsing message: %s", onion_type_name(why_bad)); if (!onion_shared_secret(&shared_secret, &packet, &seckey)) errx(1, "Error creating shared secret."); step = process_onionpacket(ctx, &packet, &shared_secret, assocdata, tal_bytelen(assocdata), true); return step; } static void do_decode(int argc, char **argv, const u8 *assocdata) { const tal_t *ctx = talz(NULL, tal_t); u8 serialized[TOTAL_PACKET_SIZE]; struct route_step *step; if (argc != 4) opt_usage_exit_fail("Expect an filename and privkey with 'decode' method"); /* "-" means stdin, which is NULL for grab_file */ char *hextemp = grab_file(ctx, streq(argv[2], "-") ? NULL : argv[2]); size_t hexlen = strlen(hextemp); // trim trailing whitespace while (isspace(hextemp[hexlen-1])) hexlen--; if (!hex_decode(hextemp, hexlen, serialized, sizeof(serialized))) { errx(1, "Invalid onion hex '%s'", hextemp); } step = decode_with_privkey(ctx, serialized, tal_strdup(ctx, argv[3]), assocdata); if (!step || !step->next) errx(1, "Error processing message."); printf("payload=%s\n", tal_hex(ctx, step->raw_payload)); if (step->nextcase == ONION_FORWARD) { u8 *ser = serialize_onionpacket(ctx, step->next); if (!ser) errx(1, "Error serializing message."); printf("next=%s\n", tal_hex(ctx, ser)); } tal_free(ctx); } static char *opt_set_ad(const char *arg, u8 **assocdata) { *assocdata = tal_hexdata(NULL, arg, strlen(arg)); if (!*assocdata) return "Bad hex string"; return NULL; } static void opt_show_ad(char buf[OPT_SHOW_LEN], u8 *const *assocdata) { hex_encode(*assocdata, tal_bytelen(*assocdata), buf, OPT_SHOW_LEN); } static char *opt_set_node_id(const char *arg, struct node_id *node_id) { node_id_from_hexstr(arg, strlen(arg), node_id); return NULL; } /** * Run an onion encoding/decoding unit-test from a file */ static void runtest(const char *filename) { const tal_t *ctx = tal(NULL, u8); bool valid; char *buffer = grab_file(ctx, filename); const jsmntok_t *toks, *session_key_tok, *associated_data_tok, *gentok, *hopstok, *hop, *payloadtok, *pubkeytok, *typetok, *oniontok, *decodetok; const u8 *associated_data, *session_key_raw, *payload, *serialized, *onion; struct secret session_key, *shared_secrets; struct pubkey pubkey; struct sphinx_path *path; size_t i; struct onionpacket *res; struct route_step *step; char *hexprivkey; toks = json_parse_input(ctx, buffer, strlen(buffer), &valid); if (!valid) errx(1, "File is not a valid JSON file."); gentok = json_get_member(buffer, toks, "generate"); if (!gentok) errx(1, "JSON object does not contain a 'generate' key"); /* Unpack the common parts */ associated_data_tok = json_get_member(buffer, gentok, "associated_data"); session_key_tok = json_get_member(buffer, gentok, "session_key"); associated_data = json_tok_bin_from_hex(ctx, buffer, associated_data_tok); session_key_raw = json_tok_bin_from_hex(ctx, buffer, session_key_tok); memcpy(&session_key, session_key_raw, sizeof(session_key)); path = sphinx_path_new_with_key(ctx, associated_data, &session_key); /* Unpack the hops and build up the path */ hopstok = json_get_member(buffer, gentok, "hops"); json_for_each_arr(i, hop, hopstok) { u8 *full; size_t prepended; payloadtok = json_get_member(buffer, hop, "payload"); typetok = json_get_member(buffer, hop, "type"); pubkeytok = json_get_member(buffer, hop, "pubkey"); payload = json_tok_bin_from_hex(ctx, buffer, payloadtok); json_to_pubkey(buffer, pubkeytok, &pubkey); if (!typetok || json_tok_streq(buffer, typetok, "legacy")) { /* Legacy has a single 0 prepended as "realm" byte */ full = tal_arrz(ctx, u8, 33); memcpy(full + 1, payload, tal_bytelen(payload)); } else { /* TLV has length prepended */ full = tal_arr(ctx, u8, 0); towire_bigsize(&full, tal_bytelen(payload)); prepended = tal_bytelen(full); tal_resize(&full, prepended + tal_bytelen(payload)); memcpy(full + prepended, payload, tal_bytelen(payload)); } sphinx_add_hop(path, &pubkey, full); } res = create_onionpacket(ctx, path, &shared_secrets); serialized = serialize_onionpacket(ctx, res); if (!serialized) errx(1, "Error serializing message."); oniontok = json_get_member(buffer, toks, "onion"); if (oniontok) { onion = json_tok_bin_from_hex(ctx, buffer, oniontok); if (!memeq(onion, tal_bytelen(onion), serialized, tal_bytelen(serialized))) errx(1, "Generated does not match the expected onion: \n" "generated: %s\n" "expected : %s\n", tal_hex(ctx, serialized), tal_hex(ctx, onion)); } printf("Generated onion: %s\n", tal_hex(ctx, serialized)); decodetok = json_get_member(buffer, toks, "decode"); json_for_each_arr(i, hop, decodetok) { enum onion_payload_type type; bool valid; hexprivkey = json_strdup(ctx, buffer, hop); printf("Processing at hop %zu\n", i); step = decode_with_privkey(ctx, serialized, hexprivkey, associated_data); serialized = serialize_onionpacket(ctx, step->next); if (!serialized) errx(1, "Error serializing message."); onion_payload_length(step->raw_payload, tal_bytelen(step->raw_payload), true, &valid, &type); assert(valid); printf(" Type: %d\n", type); printf(" Payload: %s\n", tal_hex(ctx, step->raw_payload)); printf(" Next onion: %s\n", tal_hex(ctx, serialized)); printf(" Next HMAC: %s\n", tal_hexstr(ctx, step->next->hmac.bytes, crypto_auth_hmacsha256_BYTES)); } tal_free(ctx); } static void decompress(char *hexprivkey, char *hexonion) { struct privkey rendezvous_key; size_t onionlen = hex_data_size(strlen(hexonion)); u8 *compressed; struct pubkey ephkey; struct secret shared_secret; struct onionpacket *onion; struct sphinx_compressed_onion *tinyonion; if (!hex_decode(hexprivkey, strlen(hexprivkey), &rendezvous_key, sizeof(rendezvous_key))) errx(1, "Invalid private key hex '%s'", hexprivkey); compressed = tal_arr(NULL, u8, onionlen); if (!hex_decode(hexonion, strlen(hexonion), compressed, onionlen)) errx(1, "Invalid onion hex '%s'", hexonion); if (onionlen < HMAC_SIZE + 1 + PUBKEY_SIZE) errx(1, "Onion is too short to contain the version, ephemeral key and HMAC"); pubkey_from_der(compressed + 1, PUBKEY_SIZE, &ephkey); tinyonion = sphinx_compressed_onion_deserialize(NULL, compressed); if (tinyonion == NULL) errx(1, "Could not deserialize compressed onion"); if (!sphinx_create_shared_secret(&shared_secret, &tinyonion->ephemeralkey, &rendezvous_key.secret)) errx(1, "Could not generate shared secret from ephemeral key %s " "and private key %s", pubkey_to_hexstr(NULL, &ephkey), hexprivkey); onion = sphinx_decompress(NULL, tinyonion, &shared_secret); if (onion == NULL) errx(1, "Could not decompress compressed onion"); printf("Decompressed Onion: %s\n", tal_hex(NULL, serialize_onionpacket(NULL, onion))); } /* Tal wrappers for opt. */ static void *opt_allocfn(size_t size) { return tal_arr_label(NULL, char, size, TAL_LABEL("opt_allocfn", "")); } static void *tal_reallocfn(void *ptr, size_t size) { if (!ptr) return opt_allocfn(size); tal_resize_(&ptr, 1, size, false); return ptr; } static void tal_freefn(void *ptr) { tal_free(ptr); } int main(int argc, char **argv) { setup_locale(); const char *method; u8 *assocdata = NULL; struct node_id rendezvous_id; memset(&rendezvous_id, 0, sizeof(struct node_id)); secp256k1_ctx = secp256k1_context_create(SECP256K1_CONTEXT_VERIFY | SECP256K1_CONTEXT_SIGN); opt_set_alloc(opt_allocfn, tal_reallocfn, tal_freefn); opt_register_arg("--assoc-data", opt_set_ad, opt_show_ad, &assocdata, "Associated data (usu. payment_hash of payment)"); opt_register_arg("--rendezvous-id", opt_set_node_id, NULL, &rendezvous_id, "Node ID of the rendez-vous node"); opt_register_noarg("--help|-h", opt_usage_and_exit, "\n\n\tdecode <onion_file> <privkey>\n" "\tgenerate <pubkey1> <pubkey2> ...\n" "\tgenerate <pubkey1>[/hopdata|/tlv] <pubkey2>[/hopdata|/tlv]\n" "\tgenerate <privkey1>[/hopdata|/tlv] <privkey2>[/hopdata|/tlv]\n" "\truntest <test-filename>\n\n", "Show this message\n\n" "\texample:\n" "\t> onion generate 02c18e7ff9a319983e85094b8c957da5c1230ecb328c1f1c7e88029f1fec2046f8/00000000000000000000000000000f424000000138000000000000000000000000 --assoc-data 44ee26f01e54665937b892f6afbfdfb88df74bcca52d563f088668cf4490aacd > onion.dat\n" "\t> onion decode onion.dat 78302c8edb1b94e662464e99af721054b6ab9d577d3189f933abde57709c5cb8 --assoc-data 44ee26f01e54665937b892f6afbfdfb88df74bcca52d563f088668cf4490aacd\n"); opt_register_version(); opt_early_parse(argc, argv, opt_log_stderr_exit); opt_parse(&argc, argv, opt_log_stderr_exit); if (argc < 2) errx(1, "You must specify a method"); method = argv[1]; if (streq(method, "runtest")) { if (argc != 3) errx(1, "'runtest' requires a filename argument"); runtest(argv[2]); } else if (streq(method, "generate")) { if (memeqzero(&rendezvous_id, sizeof(rendezvous_id))) do_generate(argc, argv, assocdata, NULL); else do_generate(argc, argv, assocdata, &rendezvous_id); } else if (streq(method, "decompress")) { if (argc != 4) { errx(2, "'%s decompress' requires a private key and a " "compressed onion", argv[0]); } decompress(argv[2], argv[3]); } else if (streq(method, "decode")) { do_decode(argc, argv, assocdata); } else { errx(1, "Unrecognized method '%s'", method); } return 0; }