123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306 |
- package protocols;
- import java.math.BigInteger;
- import java.util.Arrays;
- import org.apache.commons.lang3.ArrayUtils;
- import communication.Communication;
- import crypto.Crypto;
- import exceptions.AccessException;
- import exceptions.NoSuchPartyException;
- import measure.M;
- import measure.P;
- import measure.Timer;
- import oram.Bucket;
- import oram.Forest;
- import oram.Metadata;
- import oram.Tree;
- import oram.Tuple;
- import util.StopWatch;
- import util.Util;
- public class Access extends Protocol {
- public Access(Communication con1, Communication con2) {
- super(con1, con2);
- }
- public OutAccess runE(PreData predata, Tree OTi, byte[] Ni, byte[] Nip1_pr, Timer timer) {
- timer.start(P.ACC, M.online_comp);
- // step 0: get Li from C
- timer.start(P.ACC, M.online_read);
- byte[] Li = new byte[0];
- if (OTi.getTreeIndex() > 0)
- Li = con2.read();
- timer.stop(P.ACC, M.online_read);
- // step 1
- Bucket[] pathBuckets = OTi.getBucketsOnPath(new BigInteger(1, Li).longValue());
- Tuple[] pathTuples = Bucket.bucketsToTuples(pathBuckets);
- for (int i = 0; i < pathTuples.length; i++)
- pathTuples[i].setXor(predata.access_p[i]);
- Object[] objArray = Util.permute(pathTuples, predata.access_sigma);
- pathTuples = Arrays.copyOf(objArray, objArray.length, Tuple[].class);
- // step 3
- byte[] y = null;
- if (OTi.getTreeIndex() == 0)
- y = pathTuples[0].getA();
- else if (OTi.getTreeIndex() < OTi.getH() - 1)
- y = Util.nextBytes(OTi.getABytes(), Crypto.sr);
- else
- y = new byte[OTi.getABytes()];
- if (OTi.getTreeIndex() > 0) {
- byte[][] a = new byte[pathTuples.length][];
- byte[][] m = new byte[pathTuples.length][];
- for (int i = 0; i < pathTuples.length; i++) {
- m[i] = Util.xor(pathTuples[i].getA(), y);
- a[i] = ArrayUtils.addAll(pathTuples[i].getF(), pathTuples[i].getN());
- for (int j = 0; j < Ni.length; j++)
- a[i][a[i].length - 1 - j] ^= Ni[Ni.length - 1 - j];
- }
- SSCOT sscot = new SSCOT(con1, con2);
- sscot.runE(predata, m, a, timer);
- }
- // step 4
- if (OTi.getTreeIndex() < OTi.getH() - 1) {
- int ySegBytes = y.length / OTi.getTwoTauPow();
- byte[][] y_array = new byte[OTi.getTwoTauPow()][];
- for (int i = 0; i < OTi.getTwoTauPow(); i++)
- y_array[i] = Arrays.copyOfRange(y, i * ySegBytes, (i + 1) * ySegBytes);
- SSIOT ssiot = new SSIOT(con1, con2);
- ssiot.runE(predata, y_array, Nip1_pr, timer);
- }
- // step 5
- Tuple Ti = null;
- if (OTi.getTreeIndex() == 0)
- Ti = pathTuples[0];
- else
- Ti = new Tuple(new byte[0], Ni, Li, y);
- OutAccess outaccess = new OutAccess(null, null, null, Ti, pathTuples);
- timer.stop(P.ACC, M.online_comp);
- return outaccess;
- }
- public void runD(PreData predata, Tree OTi, byte[] Ni, byte[] Nip1_pr, Timer timer) {
- timer.start(P.ACC, M.online_comp);
- // step 0: get Li from C
- timer.start(P.ACC, M.online_read);
- byte[] Li = new byte[0];
- if (OTi.getTreeIndex() > 0)
- Li = con2.read();
- timer.stop(P.ACC, M.online_read);
- // step 1
- Bucket[] pathBuckets = OTi.getBucketsOnPath(new BigInteger(1, Li).longValue());
- Tuple[] pathTuples = Bucket.bucketsToTuples(pathBuckets);
- for (int i = 0; i < pathTuples.length; i++)
- pathTuples[i].setXor(predata.access_p[i]);
- Object[] objArray = Util.permute(pathTuples, predata.access_sigma);
- pathTuples = Arrays.copyOf(objArray, objArray.length, Tuple[].class);
- // step 2
- timer.start(P.ACC, M.online_write);
- con2.write(pathTuples);
- con2.write(Ni);
- timer.stop(P.ACC, M.online_write);
- // step 3
- if (OTi.getTreeIndex() > 0) {
- byte[][] b = new byte[pathTuples.length][];
- for (int i = 0; i < pathTuples.length; i++) {
- b[i] = ArrayUtils.addAll(pathTuples[i].getF(), pathTuples[i].getN());
- b[i][0] ^= 1;
- for (int j = 0; j < Ni.length; j++)
- b[i][b[i].length - 1 - j] ^= Ni[Ni.length - 1 - j];
- }
- SSCOT sscot = new SSCOT(con1, con2);
- sscot.runD(predata, b, timer);
- }
- // step 4
- if (OTi.getTreeIndex() < OTi.getH() - 1) {
- SSIOT ssiot = new SSIOT(con1, con2);
- ssiot.runD(predata, Nip1_pr, timer);
- }
- timer.stop(P.ACC, M.online_comp);
- }
- public OutAccess runC(Metadata md, int treeIndex, byte[] Li, Timer timer) {
- timer.start(P.ACC, M.online_comp);
- // step 0: send Li to E and D
- timer.start(P.ACC, M.online_write);
- if (treeIndex > 0) {
- con1.write(Li);
- con2.write(Li);
- }
- timer.stop(P.ACC, M.online_write);
- // step 2
- timer.start(P.ACC, M.online_read);
- Tuple[] pathTuples = con2.readObject();
- byte[] Ni = con2.read();
- timer.stop(P.ACC, M.online_read);
- // step 3
- int j1 = 0;
- byte[] z = null;
- if (treeIndex == 0) {
- z = pathTuples[0].getA();
- } else {
- SSCOT sscot = new SSCOT(con1, con2);
- OutSSCOT je = sscot.runC(timer);
- j1 = je.t;
- byte[] d = pathTuples[j1].getA();
- z = Util.xor(je.m_t, d);
- }
- // step 4
- int j2 = 0;
- byte[] Lip1 = null;
- if (treeIndex < md.getNumTrees() - 1) {
- SSIOT ssiot = new SSIOT(con1, con2);
- OutSSIOT jy = ssiot.runC(timer);
- // step 5
- j2 = jy.t;
- int lSegBytes = md.getABytesOfTree(treeIndex) / md.getTwoTauPow();
- byte[] z_j2 = Arrays.copyOfRange(z, j2 * lSegBytes, (j2 + 1) * lSegBytes);
- Lip1 = Util.xor(jy.m_t, z_j2);
- }
- Tuple Ti = null;
- if (treeIndex == 0) {
- Ti = pathTuples[0];
- } else {
- Ti = new Tuple(new byte[] { 1 }, Ni, new byte[md.getLBytesOfTree(treeIndex)], z);
- pathTuples[j1].getF()[0] = (byte) (1 - pathTuples[j1].getF()[0]);
- Crypto.sr.nextBytes(pathTuples[j1].getN());
- Crypto.sr.nextBytes(pathTuples[j1].getL());
- Crypto.sr.nextBytes(pathTuples[j1].getA());
- }
- OutAccess outaccess = new OutAccess(Lip1, Ti, pathTuples, null, null);
- timer.stop(P.ACC, M.online_comp);
- return outaccess;
- }
- // for testing correctness
- @Override
- public void run(Party party, Metadata md, Forest forest) {
- int records = 5;
- int repeat = 5;
- int tau = md.getTau();
- int numTrees = md.getNumTrees();
- long numInsert = md.getNumInsertRecords();
- int addrBits = md.getAddrBits();
- Timer timer = new Timer();
- StopWatch sw = new StopWatch();
- sanityCheck();
- System.out.println();
- for (int i = 0; i < records; i++) {
- long N = Util.nextLong(numInsert, Crypto.sr);
- for (int j = 0; j < repeat; j++) {
- System.out.println("Test: " + i + " " + j);
- System.out.println("N=" + BigInteger.valueOf(N).toString(2));
- byte[] Li = new byte[0];
- for (int ti = 0; ti < numTrees; ti++) {
- long Ni_value = Util.getSubBits(N, addrBits, addrBits - md.getNBitsOfTree(ti));
- long Nip1_pr_value = Util.getSubBits(N, addrBits - md.getNBitsOfTree(ti),
- Math.max(addrBits - md.getNBitsOfTree(ti) - tau, 0));
- byte[] Ni = Util.longToBytes(Ni_value, md.getNBytesOfTree(ti));
- byte[] Nip1_pr = Util.longToBytes(Nip1_pr_value, (tau + 7) / 8);
- PreData predata = new PreData();
- PreAccess preaccess = new PreAccess(con1, con2);
- if (party == Party.Eddie) {
- Tree OTi = forest.getTree(ti);
- int numTuples = (OTi.getD() - 1) * OTi.getW() + OTi.getStashSize();
- preaccess.runE(predata, OTi, numTuples, timer);
- byte[] sE_Ni = Util.nextBytes(Ni.length, Crypto.sr);
- byte[] sD_Ni = Util.xor(Ni, sE_Ni);
- con1.write(sD_Ni);
- byte[] sE_Nip1_pr = Util.nextBytes(Nip1_pr.length, Crypto.sr);
- byte[] sD_Nip1_pr = Util.xor(Nip1_pr, sE_Nip1_pr);
- con1.write(sD_Nip1_pr);
- sw.start();
- runE(predata, OTi, sE_Ni, sE_Nip1_pr, timer);
- sw.stop();
- if (ti == numTrees - 1)
- con2.write(N);
- } else if (party == Party.Debbie) {
- Tree OTi = forest.getTree(ti);
- preaccess.runD(predata, timer);
- byte[] sD_Ni = con1.read();
- byte[] sD_Nip1_pr = con1.read();
- sw.start();
- runD(predata, OTi, sD_Ni, sD_Nip1_pr, timer);
- sw.stop();
- } else if (party == Party.Charlie) {
- preaccess.runC(timer);
- System.out.println("L" + ti + "=" + new BigInteger(1, Li).toString(2));
- sw.start();
- OutAccess outaccess = runC(md, ti, Li, timer);
- sw.stop();
- Li = outaccess.C_Lip1;
- if (ti == numTrees - 1) {
- N = con1.readObject();
- long data = new BigInteger(1, outaccess.C_Ti.getA()).longValue();
- if (N == data) {
- System.out.println("Access passed");
- System.out.println();
- } else {
- throw new AccessException("Access failed");
- }
- }
- } else {
- throw new NoSuchPartyException(party + "");
- }
- }
- }
- }
- timer.print();
- System.out.println();
- System.out.println(sw.toMS());
- }
- }
|