summaryrefslogtreecommitdiff
path: root/src/org/noreply/fancydress/type3/Packet.java
blob: 7738c0a0bc1cbafb67af363625a01fad8dcfb84d (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
/* $Id$ */
package org.noreply.fancydress.type3;

import org.noreply.fancydress.misc.Util;
import org.noreply.fancydress.type3.routing.*;
import org.noreply.fancydress.crypto.*;
import org.noreply.fancydress.status.*;
import java.util.*;

public class Packet {

	private byte[] packet;
	private RoutingForward thisRoute;

	public Packet(
		Path path,
		RoutingDestination address,
		byte[] payload)
	throws Mix3BadArgumentsChainTooLongException
	{
		ForwardLeg leg2 = new ForwardLeg(path.getSecondHalf(), address);
		if (!(leg2.getRoute() instanceof RoutingForward))
			throw new Error("Routing Type of second leg is not a forward type.");
		ForwardLeg leg1 = new ForwardLeg(path.getFirstHalf(), ((RoutingForward)leg2.getRoute()).asSwap());
		makePacket(leg1, leg2, payload);
	}

	public Packet(
		HalfPath path,
		SURB surb,
		byte[] payload)
	throws Mix3BadArgumentsChainTooLongException
	{
		ForwardLeg leg1 = new ForwardLeg(path, surb.getRoute());
		makePacket(leg1, surb, payload);
	}

	public static boolean isPacketVersionSupported(String s) {
		return s.equals(SingleLeg.MAJOR_VERSION + "." + SingleLeg.MINOR_VERSION);
	}

	private void makePacket(ForwardLeg leg1, SingleLeg leg2, byte[] p) {
		byte h1[] = leg1.asOctets();
		byte h2[] = leg2.asOctets();

		if (p.length != 28*1024)
			throw new IllegalArgumentException("Payload needs to be 28 kb long");

		// Phase 1
		if (leg2 instanceof SURB) {
			throw new Error("Not implemented yet");
			// byte[] k = ((SURB) leg2).getEncryptionKey();
			// p = CryptoPrimitives.sprpDecrypt(k, "PAYLOAD ENCRYPT", p);
		} else {
			byte[][] sk = ((ForwardLeg) leg2).getSharedKeys();
			for (int i=sk.length-1; i>=0; i--)
				p = CryptoPrimitives.sprpEncrypt(sk[i], "PAYLOAD ENCRYPT", p);
		}

		// Phase 2
		h2 = CryptoPrimitives.sprpEncrypt(CryptoPrimitives.hash(p), "HIDE HEADER", h2);
		p = CryptoPrimitives.sprpEncrypt(CryptoPrimitives.hash(h2), "HIDE PAYLOAD", p);

		byte[][] sk = leg1.getSharedKeys();
		for (int i=sk.length-1; i>=0; i--) {
			h2 = CryptoPrimitives.sprpEncrypt(sk[i], "HEADER ENCRYPT", h2);
			p = CryptoPrimitives.sprpEncrypt(sk[i], "PAYLOAD ENCRYPT", p);
		}

		packet = Util.concat(Util.concat(h1,h2),p);
		thisRoute = leg1.getRoute();

		if (packet.length != 32*1024)
			throw new Error("packet is not 32k bytes");
	}

	public RoutingForward getRoute() {
		return thisRoute;
	}

	public byte[] asOctets() {
		return packet;
	}
}