authPackage_test.go 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230
  1. /*
  2. * Copyright (c) 2016, Psiphon Inc.
  3. * All rights reserved.
  4. *
  5. * This program is free software: you can redistribute it and/or modify
  6. * it under the terms of the GNU General Public License as published by
  7. * the Free Software Foundation, either version 3 of the License, or
  8. * (at your option) any later version.
  9. *
  10. * This program is distributed in the hope that it will be useful,
  11. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  12. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  13. * GNU General Public License for more details.
  14. *
  15. * You should have received a copy of the GNU General Public License
  16. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  17. *
  18. */
  19. package common
  20. import (
  21. "encoding/base64"
  22. "encoding/json"
  23. "io"
  24. "io/ioutil"
  25. "math/rand"
  26. "os"
  27. "testing"
  28. )
  29. func TestAuthenticatedPackage(t *testing.T) {
  30. signingPublicKey, signingPrivateKey, err := GenerateAuthenticatedDataPackageKeys()
  31. if err != nil {
  32. t.Fatalf("GenerateAuthenticatedDataPackageKeys failed: %s", err)
  33. }
  34. expectedContent := "TestAuthenticatedPackage"
  35. packagePayload, err := WriteAuthenticatedDataPackage(
  36. expectedContent,
  37. signingPublicKey,
  38. signingPrivateKey)
  39. if err != nil {
  40. t.Fatalf("WriteAuthenticatedDataPackage failed: %s", err)
  41. }
  42. tempFileName, err := makeTempFile(packagePayload)
  43. if err != nil {
  44. t.Fatalf("makeTempFile failed: %s", err)
  45. }
  46. defer os.Remove(tempFileName)
  47. wrongSigningPublicKey, _, err := GenerateAuthenticatedDataPackageKeys()
  48. if err != nil {
  49. t.Fatalf("GenerateAuthenticatedDataPackageKeys failed: %s", err)
  50. }
  51. packageJSON, err := Decompress(packagePayload)
  52. if err != nil {
  53. t.Fatalf("Uncompress failed: %s", err)
  54. }
  55. var authDataPackage AuthenticatedDataPackage
  56. err = json.Unmarshal(packageJSON, &authDataPackage)
  57. if err != nil {
  58. t.Fatalf("Unmarshal failed: %s", err)
  59. }
  60. authDataPackage.Data = "TamperedData"
  61. tamperedPackageJSON, err := json.Marshal(&authDataPackage)
  62. if err != nil {
  63. t.Fatalf("Marshal failed: %s", err)
  64. }
  65. tamperedPackagePayload := Compress(tamperedPackageJSON)
  66. tamperedTempFileName, err := makeTempFile(tamperedPackagePayload)
  67. if err != nil {
  68. t.Fatalf("makeTempFile failed: %s", err)
  69. }
  70. defer os.Remove(tempFileName)
  71. t.Run("read package: success", func(t *testing.T) {
  72. content, err := ReadAuthenticatedDataPackage(
  73. packagePayload, true, signingPublicKey)
  74. if err != nil {
  75. t.Fatalf("ReadAuthenticatedDataPackage failed: %s", err)
  76. }
  77. if content != expectedContent {
  78. t.Fatalf(
  79. "unexpected package content: expected %s got %s",
  80. expectedContent, content)
  81. }
  82. })
  83. t.Run("streaming read package: success", func(t *testing.T) {
  84. file, err := os.Open(tempFileName)
  85. if err != nil {
  86. t.Fatalf("Open failed: %s", err)
  87. }
  88. defer file.Close()
  89. contentReader, err := NewAuthenticatedDataPackageReader(
  90. file, signingPublicKey)
  91. if err != nil {
  92. t.Fatalf("NewAuthenticatedDataPackageReader failed: %s", err)
  93. }
  94. content, err := ioutil.ReadAll(contentReader)
  95. if err != nil {
  96. t.Fatalf("ReadAll failed: %s", err)
  97. }
  98. if string(content) != expectedContent {
  99. t.Fatalf(
  100. "unexpected package content: expected %s got %s",
  101. expectedContent, content)
  102. }
  103. })
  104. t.Run("read package: wrong signing key", func(t *testing.T) {
  105. _, err = ReadAuthenticatedDataPackage(
  106. packagePayload, true, wrongSigningPublicKey)
  107. if err == nil {
  108. t.Fatalf("ReadAuthenticatedDataPackage unexpectedly succeeded")
  109. }
  110. })
  111. t.Run("streaming read package: wrong signing key", func(t *testing.T) {
  112. file, err := os.Open(tempFileName)
  113. if err != nil {
  114. t.Fatalf("Open failed: %s", err)
  115. }
  116. defer file.Close()
  117. _, err = NewAuthenticatedDataPackageReader(
  118. file, wrongSigningPublicKey)
  119. if err == nil {
  120. t.Fatalf("NewAuthenticatedDataPackageReader unexpectedly succeeded")
  121. }
  122. })
  123. t.Run("read package: tampered data", func(t *testing.T) {
  124. _, err = ReadAuthenticatedDataPackage(
  125. tamperedPackagePayload, true, signingPublicKey)
  126. if err == nil {
  127. t.Fatalf("ReadAuthenticatedDataPackage unexpectedly succeeded")
  128. }
  129. })
  130. t.Run("streaming read package: tampered data", func(t *testing.T) {
  131. file, err := os.Open(tamperedTempFileName)
  132. if err != nil {
  133. t.Fatalf("Open failed: %s", err)
  134. }
  135. defer file.Close()
  136. _, err = NewAuthenticatedDataPackageReader(
  137. file, signingPublicKey)
  138. if err == nil {
  139. t.Fatalf("NewAuthenticatedDataPackageReader unexpectedly succeeded")
  140. }
  141. })
  142. }
  143. func BenchmarkAuthenticatedPackage(b *testing.B) {
  144. signingPublicKey, signingPrivateKey, err := GenerateAuthenticatedDataPackageKeys()
  145. if err != nil {
  146. b.Fatalf("GenerateAuthenticatedDataPackageKeys failed: %s", err)
  147. }
  148. data := make([]byte, 104857600)
  149. rand.Read(data)
  150. packagePayload, err := WriteAuthenticatedDataPackage(
  151. base64.StdEncoding.EncodeToString(data),
  152. signingPublicKey,
  153. signingPrivateKey)
  154. if err != nil {
  155. b.Fatalf("WriteAuthenticatedDataPackage failed: %s", err)
  156. }
  157. tempFileName, err := makeTempFile(packagePayload)
  158. if err != nil {
  159. b.Fatalf("makeTempFile failed: %s", err)
  160. }
  161. defer os.Remove(tempFileName)
  162. b.Run("read package", func(b *testing.B) {
  163. for i := 0; i < b.N; i++ {
  164. _, err := ReadAuthenticatedDataPackage(
  165. packagePayload, true, signingPublicKey)
  166. if err != nil {
  167. b.Fatalf("ReadAuthenticatedDataPackage failed: %s", err)
  168. }
  169. }
  170. })
  171. b.Run("streaming read package", func(b *testing.B) {
  172. for i := 0; i < b.N; i++ {
  173. file, err := os.Open(tempFileName)
  174. if err != nil {
  175. b.Fatalf("Open failed: %s", err)
  176. }
  177. contentReader, err := NewAuthenticatedDataPackageReader(
  178. file, signingPublicKey)
  179. if err != nil {
  180. file.Close()
  181. b.Fatalf("NewAuthenticatedDataPackageReader failed: %s", err)
  182. }
  183. _, err = io.Copy(ioutil.Discard, contentReader)
  184. if err != nil {
  185. file.Close()
  186. b.Fatalf("Read failed: %s", err)
  187. }
  188. file.Close()
  189. }
  190. })
  191. }
  192. func makeTempFile(data []byte) (string, error) {
  193. file, err := ioutil.TempFile("", "authPackage_test")
  194. if err != nil {
  195. return "", ContextError(err)
  196. }
  197. defer file.Close()
  198. _, err = file.Write(data)
  199. if err != nil {
  200. return "", ContextError(err)
  201. }
  202. return file.Name(), nil
  203. }