client.go 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634
  1. package main
  2. /*
  3. #cgo CFLAGS: -O2
  4. #cgo LDFLAGS: -lcrypto -lm
  5. #include "../c/dpf.h"
  6. #include "../c/okvClient.h"
  7. #include "../c/dpf.c"
  8. #include "../c/okvClient.c"
  9. */
  10. import "C"
  11. //sssssssssss
  12. import (
  13. lib "2PPS/lib"
  14. "bytes"
  15. "crypto/rand"
  16. "crypto/sha256"
  17. "crypto/tls"
  18. "encoding/json"
  19. "fmt"
  20. "math/big"
  21. mr "math/rand"
  22. "net"
  23. "sort"
  24. "sync"
  25. "time"
  26. "unsafe"
  27. "golang.org/x/crypto/nacl/box"
  28. )
  29. type tweet struct {
  30. Topics []string
  31. Text string
  32. }
  33. const leader string = "127.0.0.1:4441"
  34. //needs to be changed at leader/follower/client at the same time
  35. const neededSubscriptions = 1
  36. const numClients = 1
  37. var topicList []string
  38. var archiveTopicList []string
  39. //todo! expand this for multiple clients
  40. var archiveInterests = make([]int, 1)
  41. var sharedSecret [numClients][2][32]byte = createSharedSecret()
  42. var wantsArchive = make([]byte, 1)
  43. var dataLen int = 32
  44. var numThreads int = 12
  45. var dbWriteSize int = 2
  46. var leaderPublicKey *[32]byte
  47. var followerPublicKey *[32]byte
  48. var clientPrivateKey *[32]byte
  49. var clientPublicKey *[32]byte
  50. func main() {
  51. //creates test tweets
  52. tweets := make([][]byte, numClients)
  53. for i := range tweets {
  54. tweets[i] = make([]byte, dataLen)
  55. }
  56. for i := 0; i < numClients; i++ {
  57. var tweet []byte
  58. if i == 0 {
  59. topics := []byte("house; mouse")
  60. text := []byte("I am a house in a mouse;")
  61. tweet = append(tweet, topics...)
  62. tweet = append(tweet, text...)
  63. }
  64. tweet = append(tweet, []byte(";")[0])
  65. length := len(tweet)
  66. for i := length; i < dataLen; i++ {
  67. //todo! replace with random chars
  68. tweet = append(tweet, []byte(";")[0])
  69. }
  70. tweets[i] = tweet
  71. }
  72. wg := &sync.WaitGroup{}
  73. for i := 0; i < numClients; i++ {
  74. wg.Add(1)
  75. go client(tweets[i], i)
  76. }
  77. wg.Wait()
  78. }
  79. func client(tweet []byte, clientNumber int) {
  80. /*
  81. if len(os.Args) != 4 {
  82. fmt.Println("try again with: numThreads, dataLength, numRows")
  83. return
  84. }
  85. //input when executing is follower amount
  86. serverAmount, _ = strconv.Atoi(os.Args[1])
  87. serverAmount++
  88. dataLen, _ = strconv.Atoi(os.Args[2])
  89. numThreads, _ = strconv.Atoi(os.Args[3])
  90. dbSize, _ = strconv.Atoi(os.Args[4])
  91. */
  92. generatedPublicKey, generatedPrivateKey, err := box.GenerateKey(rand.Reader)
  93. if err != nil {
  94. panic(err)
  95. }
  96. clientPrivateKey = generatedPrivateKey
  97. clientPublicKey = generatedPublicKey
  98. C.initializeCipher()
  99. //initializes the connection to the leader
  100. conf := &tls.Config{
  101. InsecureSkipVerify: true,
  102. }
  103. leaderConn, err := tls.Dial("tcp", leader, conf)
  104. if err != nil {
  105. panic(err)
  106. }
  107. leaderConn.SetDeadline(time.Time{})
  108. //receives topics first so client can participate asap
  109. receiveTopicLists(leaderConn)
  110. //gets the public keys of both servers
  111. var tmpLeaderPubKey [32]byte
  112. _, err = leaderConn.Read(tmpLeaderPubKey[:])
  113. if err != nil {
  114. panic(err)
  115. }
  116. leaderPublicKey = &tmpLeaderPubKey
  117. var tmpFollowerPubKey [32]byte
  118. _, err = leaderConn.Read(tmpFollowerPubKey[:])
  119. if err != nil {
  120. panic(err)
  121. }
  122. followerPublicKey = &tmpFollowerPubKey
  123. //sends own public key
  124. _, err = leaderConn.Write(clientPublicKey[:])
  125. if err != nil {
  126. panic(err)
  127. }
  128. //setup ends above
  129. //while client is active he is always connected and has to participate
  130. for {
  131. //gets current phase
  132. phase := make([]byte, 1)
  133. _, err = leaderConn.Read(phase)
  134. if err != nil {
  135. panic(err)
  136. }
  137. fmt.Println("Phase: ", phase[0])
  138. if phase[0] == 1 {
  139. //gets current dbWriteSize from leader
  140. dbWriteSizeBytes := make([]byte, 4)
  141. _, err = leaderConn.Read(dbWriteSizeBytes)
  142. if err != nil {
  143. panic(err)
  144. }
  145. dbWriteSize = byteToInt(dbWriteSizeBytes)
  146. //todo! put into tweet creation
  147. roundAsBytes := make([]byte, 4)
  148. _, err = leaderConn.Read(roundAsBytes)
  149. if err != nil {
  150. panic(err)
  151. }
  152. //request virtualAddress from leader via pirQuery
  153. encryptedQueryLeader, encryptedQueryFollower := createAuditPIRQuery(clientNumber)
  154. sendQuerys(encryptedQueryLeader, encryptedQueryFollower, leaderConn, false)
  155. pos := receiveTweets(sharedSecret[clientNumber], leaderConn, clientNumber, false, true)
  156. fmt.Println(pos)
  157. //prep the query
  158. dataSize := len(tweet)
  159. querySize := make([]byte, 4)
  160. cQuerySize := C.int(byteToInt(querySize))
  161. var dpfQueryA *C.uchar
  162. var dpfQueryB *C.uchar
  163. //change
  164. C.prepQuery(C.int(pos), C.int(dbWriteSize), (*C.uchar)(&tweet[0]), C.int(dataSize), &cQuerySize, &dpfQueryA, &dpfQueryB)
  165. intQuerySize := int(cQuerySize) //byteToInt(querySize)
  166. //write the query
  167. queryAPlaintext := C.GoBytes(unsafe.Pointer(dpfQueryA), C.int(intQuerySize))
  168. //encrypts queryA and appends it to message
  169. var nonce [24]byte
  170. //fill nonce with randomness
  171. _, err = rand.Read(nonce[:])
  172. if err != nil {
  173. panic("couldn't get randomness for nonce!")
  174. }
  175. dpfQueryAEncrypted := box.Seal(nonce[:], queryAPlaintext, &nonce, leaderPublicKey, clientPrivateKey)
  176. //encrypts queryB and appends it to message
  177. queryBPlaintext := C.GoBytes(unsafe.Pointer(dpfQueryB), C.int(intQuerySize))
  178. //fill nonce with randomness
  179. _, err = rand.Read(nonce[:])
  180. if err != nil {
  181. panic("couldn't get randomness for nonce!")
  182. }
  183. dpfQueryBEncrypted := box.Seal(nonce[:], queryBPlaintext, &nonce, followerPublicKey, clientPrivateKey)
  184. //writes the dpfQuery to the leader
  185. dpfLengthBytes := intToByte(len(dpfQueryAEncrypted))
  186. _, err = leaderConn.Write(dpfLengthBytes)
  187. if err != nil {
  188. panic(err)
  189. }
  190. _, err = leaderConn.Write(dpfQueryAEncrypted)
  191. if err != nil {
  192. panic(err)
  193. }
  194. _, err = leaderConn.Write(dpfQueryBEncrypted)
  195. if err != nil {
  196. panic(err)
  197. }
  198. C.free(unsafe.Pointer(dpfQueryA))
  199. C.free(unsafe.Pointer(dpfQueryB))
  200. } else if phase[0] == 3 {
  201. /*
  202. possible Values
  203. 0 : new client
  204. leader expects sharedSecrets, expects pirQuery
  205. 1 : update needed
  206. leader sends topicList, performs local update of sharedSecret, expects pirQuery
  207. 2 : no update needed
  208. nothing
  209. */
  210. subPhase := make([]byte, 1)
  211. _, err := leaderConn.Read(subPhase)
  212. if err != nil {
  213. panic(err)
  214. }
  215. var encryptedQueryLeader, encryptedQueryFollower []byte
  216. //first time participating
  217. if subPhase[0] == 0 {
  218. receiveTopicLists(leaderConn)
  219. encryptedQueryLeader, encryptedQueryFollower = createPIRQuery(int(subPhase[0]), clientNumber)
  220. sendQuerys(encryptedQueryLeader, encryptedQueryFollower, leaderConn, false)
  221. }
  222. //updates the topic list and what client is interested in
  223. if subPhase[0] == 1 {
  224. receiveTopicLists(leaderConn)
  225. //updates local secret
  226. for index := 0; index < 2; index++ {
  227. sharedSecret[clientNumber][index] = sha256.Sum256(sharedSecret[clientNumber][index][:])
  228. }
  229. encryptedQueryLeader, encryptedQueryFollower = createPIRQuery(int(subPhase[0]), clientNumber)
  230. sendQuerys(encryptedQueryLeader, encryptedQueryFollower, leaderConn, false)
  231. }
  232. receiveTweets(sharedSecret[clientNumber], leaderConn, clientNumber, false, false)
  233. if len(archiveTopicList) > 0 {
  234. wantsArchive[0] = 0 //archive test
  235. } else {
  236. wantsArchive[0] = 0
  237. }
  238. _, err = leaderConn.Write(wantsArchive)
  239. if err != nil {
  240. panic(err)
  241. }
  242. if wantsArchive[0] == 1 && len(archiveTopicList) > 0 {
  243. encryptedQueryLeader, encryptedQueryFollower = createPIRQuery(-1, clientNumber)
  244. sendQuerys(encryptedQueryLeader, encryptedQueryFollower, leaderConn, true)
  245. receiveTweets(sharedSecret[clientNumber], leaderConn, clientNumber, true, false)
  246. }
  247. } else {
  248. panic("somethin went wrong")
  249. }
  250. }
  251. }
  252. //creates and sends the pirQuerys for each server
  253. func createPIRQuery(subPhase int, clientNumber int) ([]byte, []byte) {
  254. //later this will be taken from gui, this is only for testing
  255. topicsOfInterest := make([]int, 1)
  256. topicsOfInterest[0] = 1
  257. archiveInterests[0] = 1
  258. tmptopicsOfInterest := make([]int, len(topicsOfInterest))
  259. copy(tmptopicsOfInterest, topicsOfInterest)
  260. tmpNeededSubscriptions := neededSubscriptions
  261. tmpTopicList := make([]string, len(topicList))
  262. copy(tmpTopicList, topicList)
  263. if wantsArchive[0] == 1 && subPhase == -1 {
  264. tmpNeededSubscriptions = len(archiveInterests)
  265. copy(tmptopicsOfInterest, archiveInterests) //archiveInterests from gui
  266. copy(tmpTopicList, archiveTopicList)
  267. }
  268. topicsOfInterestAsBytes := make([][]byte, tmpNeededSubscriptions)
  269. for i := range topicsOfInterestAsBytes {
  270. topicsOfInterestAsBytes[i] = make([]byte, len(tmpTopicList))
  271. }
  272. //creates fake topicsOfInterest if client is boooring
  273. if len(tmptopicsOfInterest) < tmpNeededSubscriptions && subPhase != -1 {
  274. tmptopicsOfInterest = addFakeInterests(neededSubscriptions, len(tmpTopicList), tmptopicsOfInterest, false)
  275. }
  276. for topic, position := range tmptopicsOfInterest {
  277. //change used to be positon-1
  278. topicsOfInterestAsBytes[topic][position] = 1
  279. }
  280. //pirQuery [serverAmount][topicsofinterest][topicAmount]byte
  281. pirQuerys := make([][][]byte, 2)
  282. for i := range pirQuerys {
  283. pirQuerys[i] = make([][]byte, len(tmptopicsOfInterest))
  284. for j := range pirQuerys[i] {
  285. pirQuerys[i][j] = make([]byte, len(tmpTopicList))
  286. }
  287. }
  288. //for leader
  289. //pirQuery will be filled with random bits
  290. for topic := range tmptopicsOfInterest {
  291. for index := range tmpTopicList {
  292. bit, err := rand.Int(rand.Reader, big.NewInt(2))
  293. if err != nil {
  294. panic(err)
  295. }
  296. pirQuerys[0][topic][index] = byte(bit.Int64())
  297. }
  298. }
  299. //creating last manually with result and wanted
  300. //if position random result correct -> 0, not correct -> 1
  301. for topic := range tmptopicsOfInterest {
  302. for index := range tmpTopicList {
  303. if pirQuerys[0][topic][index] == topicsOfInterestAsBytes[topic][index] {
  304. pirQuerys[1][topic][index] = 0
  305. } else {
  306. pirQuerys[1][topic][index] = 1
  307. }
  308. }
  309. }
  310. //flattens the querys to be able to send them more efficently
  311. messagesFlattened := make([][]byte, 2)
  312. //adds the sharedSecret to the first pirQuery when first time participating
  313. if subPhase == 0 {
  314. for server := 0; server < 2; server++ {
  315. messagesFlattened[server] = append(messagesFlattened[server], sharedSecret[clientNumber][server][:]...)
  316. }
  317. }
  318. for server := 0; server < 2; server++ {
  319. for topic := range pirQuerys[server] {
  320. messagesFlattened[server] = append(messagesFlattened[server], pirQuerys[server][topic][:]...)
  321. }
  322. }
  323. var nonce [24]byte
  324. _, err := rand.Read(nonce[:])
  325. if err != nil {
  326. panic("couldn't get randomness for nonce!")
  327. }
  328. encryptedQueryLeader := box.Seal(nonce[:], messagesFlattened[0], &nonce, leaderPublicKey, clientPrivateKey)
  329. _, err = rand.Read(nonce[:])
  330. if err != nil {
  331. panic("couldn't get randomness for nonce!")
  332. }
  333. encryptedQueryFollower := box.Seal(nonce[:], messagesFlattened[1], &nonce, followerPublicKey, clientPrivateKey)
  334. return encryptedQueryLeader, encryptedQueryFollower
  335. }
  336. func sendQuerys(encryptedQueryLeader, encryptedQueryFollower []byte, leaderConn net.Conn, getArchive bool) {
  337. encryptedLength := len(encryptedQueryLeader)
  338. //sends the pirQuerysLength to the leader
  339. _, err := leaderConn.Write(intToByte(encryptedLength))
  340. if err != nil {
  341. panic(err)
  342. }
  343. //sends the pirQuerys to the leader
  344. _, err = leaderConn.Write(encryptedQueryLeader)
  345. if err != nil {
  346. panic(err)
  347. }
  348. _, err = leaderConn.Write(encryptedQueryFollower)
  349. if err != nil {
  350. panic(err)
  351. }
  352. if getArchive {
  353. leaderConn.Write(intToByte(len(archiveInterests)))
  354. if err != nil {
  355. panic(err)
  356. }
  357. }
  358. }
  359. func receiveTweets(sharedSecret [2][32]byte, leaderConn net.Conn, clientNumber int, getArchive, doAuditing bool) int {
  360. tmpNeededSubscriptions := neededSubscriptions
  361. if getArchive {
  362. tmpNeededSubscriptions = len(archiveInterests)
  363. }
  364. if doAuditing {
  365. tmpNeededSubscriptions = 1
  366. }
  367. for i := 0; i < tmpNeededSubscriptions; i++ {
  368. //client receives tweets
  369. tweetsLengthBytes := make([]byte, 4)
  370. _, err := leaderConn.Read(tweetsLengthBytes)
  371. if err != nil {
  372. panic(err)
  373. }
  374. tweetsLength := byteToInt(tweetsLengthBytes)
  375. tweets := make([]byte, tweetsLength)
  376. _, err = leaderConn.Read(tweets)
  377. if err != nil {
  378. panic(err)
  379. }
  380. //expand sharedSecret so it is of right length
  381. expandBy := len(tweets) / 32
  382. expandedSharedSecrets := make([][]byte, 2)
  383. for i := 0; i < 2; i++ {
  384. for j := 0; j < expandBy; j++ {
  385. expandedSharedSecrets[i] = append(expandedSharedSecrets[i], sharedSecret[i][:]...)
  386. }
  387. }
  388. //xors the received messge into the message to display
  389. for i := 0; i < 2; i++ {
  390. lib.Xor(expandedSharedSecrets[i][:], tweets)
  391. }
  392. //tweets can be displayed
  393. fmt.Println("final result: ", string(tweets))
  394. if doAuditing {
  395. return byteToInt(tweets)
  396. }
  397. }
  398. return 0
  399. }
  400. //creates a shared secret for each server
  401. func createSharedSecret() [numClients][2][32]byte {
  402. var tmpSharedSecret [numClients][2][32]byte
  403. for i := 0; i < numClients; i++ {
  404. for j := 0; j < 2; j++ {
  405. _, err := rand.Read(tmpSharedSecret[i][j][:])
  406. if err != nil {
  407. panic("couldn't get randomness for sharedSecret!")
  408. }
  409. }
  410. }
  411. return tmpSharedSecret
  412. }
  413. func createAuditPIRQuery(clientNumber int) ([]byte, []byte) {
  414. //pirQuery [serverAmount][dbWriteSize]byte
  415. pirQuerys := make([][]byte, 2)
  416. for i := range pirQuerys {
  417. pirQuerys[i] = make([]byte, dbWriteSize)
  418. }
  419. //for leader
  420. //pirQuery will be filled with random bits
  421. for index := range pirQuerys[0] {
  422. bit := mr.Intn(2)
  423. pirQuerys[0][index] = byte(bit)
  424. }
  425. copy(pirQuerys[1], pirQuerys[0])
  426. //the positon the virtual address will be taken from
  427. pos := mr.Intn(dbWriteSize)
  428. pirQuerys[0][pos] = 1
  429. pirQuerys[1][pos] = 0
  430. //flattens the querys to be able to send them more efficently
  431. messagesFlattened := make([][]byte, 2)
  432. //adds the sharedSecret to the pirQuery
  433. for server := 0; server < 2; server++ {
  434. messagesFlattened[server] = append(messagesFlattened[server], sharedSecret[clientNumber][server][:]...)
  435. }
  436. for server := 0; server < 2; server++ {
  437. messagesFlattened[server] = append(messagesFlattened[server], pirQuerys[server][:]...)
  438. }
  439. var nonce [24]byte
  440. _, err := rand.Read(nonce[:])
  441. if err != nil {
  442. panic("couldn't get randomness for nonce!")
  443. }
  444. encryptedQueryLeader := box.Seal(nonce[:], messagesFlattened[0], &nonce, leaderPublicKey, clientPrivateKey)
  445. _, err = rand.Read(nonce[:])
  446. if err != nil {
  447. panic("couldn't get randomness for nonce!")
  448. }
  449. encryptedQueryFollower := box.Seal(nonce[:], messagesFlattened[1], &nonce, followerPublicKey, clientPrivateKey)
  450. return encryptedQueryLeader, encryptedQueryFollower
  451. }
  452. //generates a topicOfInterest array with random values
  453. func addFakeInterests(length, max int, topicsOfInterest []int, doAuditing bool) []int {
  454. fakeTopicsOfInterest := make([]int, length)
  455. maxInt := max
  456. //fills the array with unique random ascending values ranging from 0 to max
  457. for i := 0; i < length; i++ {
  458. fakeTopicsOfInterest[i] = mr.Intn(maxInt)
  459. for j := 0; j < i; j++ {
  460. if fakeTopicsOfInterest[i] == fakeTopicsOfInterest[j] {
  461. i--
  462. break
  463. }
  464. }
  465. }
  466. if doAuditing {
  467. sort.Ints(fakeTopicsOfInterest)
  468. return fakeTopicsOfInterest
  469. }
  470. //adds unique and new random numbers to topicOfInterests until length is satisfied
  471. for _, number := range fakeTopicsOfInterest {
  472. if !inList(number, topicsOfInterest) {
  473. topicsOfInterest = append(topicsOfInterest, number)
  474. }
  475. if len(topicsOfInterest) == neededSubscriptions {
  476. break
  477. }
  478. }
  479. sort.Ints(topicsOfInterest)
  480. return topicsOfInterest
  481. }
  482. func inList(number int, list []int) bool {
  483. for _, element := range list {
  484. if element == number {
  485. return true
  486. }
  487. }
  488. return false
  489. }
  490. func receiveTopicLists(leaderConn net.Conn) {
  491. for i := 0; i < 2; i++ {
  492. topicListLength := make([]byte, 4)
  493. _, err := leaderConn.Read(topicListLength)
  494. if err != nil {
  495. panic(err)
  496. }
  497. recTopicList := make([]byte, byteToInt(topicListLength))
  498. _, err = leaderConn.Read(recTopicList[:])
  499. if err != nil {
  500. panic(err)
  501. }
  502. var tmpTopicList []string
  503. arrayReader := bytes.NewReader(recTopicList[:])
  504. json.NewDecoder(arrayReader).Decode(&tmpTopicList)
  505. if i == 0 {
  506. topicList = tmpTopicList
  507. } else {
  508. archiveTopicList = tmpTopicList
  509. }
  510. }
  511. }
  512. func intToByte(myInt int) (retBytes []byte) {
  513. retBytes = make([]byte, 4)
  514. retBytes[3] = byte((myInt >> 24) & 0xff)
  515. retBytes[2] = byte((myInt >> 16) & 0xff)
  516. retBytes[1] = byte((myInt >> 8) & 0xff)
  517. retBytes[0] = byte(myInt & 0xff)
  518. return
  519. }
  520. func byteToInt(myBytes []byte) (x int) {
  521. x = int(myBytes[3])<<24 + int(myBytes[2])<<16 + int(myBytes[1])<<8 + int(myBytes[0])
  522. return
  523. }