forked from microsoft/QuantumKatas
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Tests.qs
174 lines (143 loc) · 6.79 KB
/
Tests.qs
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
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
// Copyright (c) Microsoft Corporation.
// Licensed under the MIT license.
//////////////////////////////////////////////////////////////////////
// This file contains testing harness for all tasks.
// You should not modify anything in this file.
// The tasks themselves can be found in Tasks.qs file.
//////////////////////////////////////////////////////////////////////
namespace Quantum.Kata.KeyDistribution {
open Microsoft.Quantum.Arrays;
open Microsoft.Quantum.Intrinsic;
open Microsoft.Quantum.Diagnostics;
open Microsoft.Quantum.Math;
open Microsoft.Quantum.Convert;
open Microsoft.Quantum.Random;
//////////////////////////////////////////////////////////////////
// Part I. Preparation
//////////////////////////////////////////////////////////////////
@Test("QuantumSimulator")
operation T11_DiagonalBasis () : Unit {
for i in 1 .. 5 {
AssertOperationsEqualReferenced(i, DiagonalBasis, DiagonalBasis_Reference);
}
}
@Test("QuantumSimulator")
operation T12_EqualSuperposition () : Unit {
use q = Qubit();
EqualSuperposition(q);
DumpMachine();
AssertMeasurementProbability([PauliZ], [q], One, 0.5, "Measuring should produce 0 and 1 with 50/50 chance.", 1e-5);
Reset(q);
}
//////////////////////////////////////////////////////////////////
// Part II. BB84 Protocol
//////////////////////////////////////////////////////////////////
@Test("QuantumSimulator")
operation T21_RandomArray () : Unit {
// The test checks that the operation does not return always the same array.
let N = 30;
let randomArrays = ForEach(RandomArray, [N, N, N]);
for array in randomArrays {
Fact(Length(array) == N, "Returned array should be of the given length");
}
let randomInts = Mapped(BoolArrayAsInt, randomArrays);
Fact(randomInts[0] != randomInts[1] or randomInts[1] != randomInts[2] or randomInts[0] != randomInts[2],
"Random generation should not return equal arrays. Run again to see if problem goes away");
}
// ------------------------------------------------------
// Helper operation to generate two random arrays
operation GenerateRandomState (N : Int) : (Bool[], Bool[]) {
return (RandomArray_Reference(N), RandomArray_Reference(N));
}
@Test("QuantumSimulator")
operation T22_PrepareAlicesQubits () : Unit {
for N in 2 .. 10 {
let (bases, state) = GenerateRandomState(N);
use qs = Qubit[N];
PrepareAlicesQubits(qs, bases, state);
Adjoint PrepareAlicesQubits_Reference(qs, bases, state);
AssertAllZero(qs);
}
}
// ------------------------------------------------------
@Test("QuantumSimulator")
operation T23_MeasureBobsQubits () : Unit {
for N in 2 .. 10 {
let (bases, state) = GenerateRandomState(N);
use qs = Qubit[N];
// Prepare the input qubits in the given state
PrepareAlicesQubits_Reference(qs, bases, state);
// Measure the qubits in the bases used for preparation - this should return encoded state
let result = MeasureBobsQubits(qs, bases);
Fact(N == Length(result), "The returned array should have the same length as the inputs");
Fact(BoolArrayAsInt(state) == BoolArrayAsInt(result), "Some of the measurements were done in the wrong basis");
}
}
// ------------------------------------------------------
@Test("QuantumSimulator")
operation T24_GenerateSharedKey () : Unit {
for N in 10 .. 30 {
let basesAlice = RandomArray_Reference(N);
let (basesBob, bitsBob) = GenerateRandomState(N);
let expected = GenerateSharedKey_Reference(basesAlice, basesBob, bitsBob);
let result = GenerateSharedKey(basesAlice, basesBob, bitsBob);
Fact(Length(result) == Length(expected), $"Generated key should have length {Length(expected)}");
Fact(BoolArrayAsInt(result) == BoolArrayAsInt(expected), "Unexpected key value {result}, expected {expected}");
}
}
// ------------------------------------------------------
@Test("QuantumSimulator")
operation T25_CheckKeysMatch () : Unit {
// Hard-coded test to validate that the solution checks the right relation with error rate
mutable key1 = ConstantArray(10, false);
mutable key2 = key1 w/ 3 <- true;
mutable errorRate = 15;
mutable result = CheckKeysMatch(key1, key2, errorRate);
// 10% mismatch with 15% error rate should pass
Fact(result, $"Check for {key1} vs {key2} with errorRate {errorRate}% should return {true}, returned {result}");
set key2 w/= 5 <- true;
set result = CheckKeysMatch(key1, key2, errorRate);
// 20% mismatch with 15% error rate should fail
Fact(not result, $"Check for {key1} vs {key2} with errorRate {errorRate}% should return {false}, returned {result}");
for i in 10 .. 30 {
set (key1, key2) = GenerateRandomState(i);
set errorRate = DrawRandomInt(0, 49);
let expected = CheckKeysMatch_Reference(key1, key2, errorRate);
set result = CheckKeysMatch(key1, key2, errorRate);
Fact(expected == result, $"Check for {key1} vs {key2} with errorRate {errorRate}% should return {expected}, returned {result}");
}
}
//////////////////////////////////////////////////////////////////
// Part III. Eavesdropping
//////////////////////////////////////////////////////////////////
@Test("QuantumSimulator")
operation T31_Eavesdrop () : Unit {
use q = Qubit();
// q = 0, Real value: b = rectangular, Input: b = rectangular
let res00 = Eavesdrop(q, false);
Fact(res00 == false, $"Incorrect result: {res00} (expected False)");
// check that the qubit was restored to its state
AssertQubit(Zero, q);
// q = 0, Real value: b = diagonal, Input: b = diagonal
H(q);
let res01 = Eavesdrop(q, true);
Fact(res01 == false, $"Incorrect result: {res01} (expected False)");
// check that the qubit was restored to its state
H(q);
AssertQubit(Zero, q);
// q = 1, Real value: b = rectangular, Input: b = rectangular
X(q);
let res10 = Eavesdrop(q, false);
Fact(res10 == true, $"Incorrect result: {res10} (expected True)");
AssertQubit(One, q);
Reset(q);
// q = 1, Real value: b = diagonal, Input: b = diagonal
X(q);
H(q);
let res11 = Eavesdrop(q, true);
Fact(res11 == true, $"Incorrect result: {res11} (expected True)");
H(q);
AssertQubit(One, q);
Reset(q);
}
}