forked from sneakyweasel/quintessence
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathmain.py
363 lines (320 loc) · 11.9 KB
/
main.py
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
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
# import packages
from qiskit import QuantumCircuit, transpile
from qiskit.circuit import Parameter
from qiskit_ionq import IonQProvider
import numpy as np
from flask import Flask, request
import os
import matplotlib.pylab as plt
import matplotlib.cm as cm
import matplotlib.colors as colors
import matplotlib.colorbar as colorbar
from scipy.stats import norm
import openai
import numpy as np
import matplotlib.pylab as plt
from matplotlib import patheffects
#plt.rcParams['path.effects'] = [patheffects.withStroke(linewidth=4, foreground='b')]
from PIL import Image, ImageFilter
import requests
from Plotter import Entropy, PolarPlotmaker
#import json
import time
openai.api_key = "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"
# Load your API key from an environment variable or secret management service
app = Flask(__name__)
# this will be the url
class CircuitSpec:
def __init__(self, start, steps, speed, likelyhood, backend):
self.start = start # starting site
self.steps = steps # number of trotter steps
self.speed = speed
self.likelyhood = likelyhood #that's the K values
self.backend = backend
# Depends on the configuration of the application.
def random_walk(self):
L = len(self.likelyhood)
# Define Parameters
J = Parameter('J')
K_i = []
for ind in range(L):
K_i.append(Parameter('K_'+str(ind)))
# Define Unitary
qc_U = QuantumCircuit(L)
for i in range(L):
qc_U.rz(K_i[i]/2, i)
for i in range(L):
if i % 2 == 0:
qc_U.rxx(-J/2, i % L, (i+1) % L)
qc_U.ryy(-J/2, i % L, (i+1) % L)
for i in range(L):
if i % 2 != 0:
qc_U.rxx(-J, i % L, (i+1) % L)
qc_U.ryy(-J, i % L, (i+1) % L)
for i in range(L):
if i % 2 == 0:
qc_U.rxx(-J/2, i % L, (i+1) % L)
qc_U.ryy(-J/2, i % L, (i+1) % L)
for i in range(L):
qc_U.rz(K_i[i]/2, i)
# Compose Main Circuit with set number of Unitary steps:
qc_main = QuantumCircuit(L)
qc_main.x(self.start)
for step in range(self.steps):
qc_main = qc_main.compose(qc_U, qubits=range(L))
# Add Measurement Circuit
qc_meas = QuantumCircuit(L, L)
qc_meas.measure_all(add_bits=False)
qc_end = qc_main.compose(qc_meas, range(L))
# Bind Parameters
qc_end = qc_end.bind_parameters({J: self.speed})
for ind in range(L):
qc_end = qc_end.bind_parameters({K_i[ind]: self.likelyhood[ind]})
# Transpile and Run
trans_circ = transpile(qc_end, self.backend)
job = self.backend.run(trans_circ)
# Return the counts for the jobs
return job.result().get_counts()
def Clean_Results(counts, n_walkers, verbose = False):
if verbose:
print(f'Number of possible end states: {len(counts)}')
shots = 0
length_qubits = 0
for key in counts.keys():
shots += counts.get(key)
length_qubits = len(key)
walkers = {}
extras = {}
for key in counts.keys():
totes = 0
for l in key:
if l == '1':
totes += 1
else:
pass
if totes == n_walkers:
walkers[key] = counts.get(key)
else:
extras[key] = counts.get(key)
if verbose:
print(f'Number of walkers: {n_walkers}')
print(f'Number of {n_walkers} occurences: {len(walkers)}')
print(f'Number of extra occurences: {len(extras)}')
walkers_prob = {}
error_prob = 0
for key in walkers.keys():
prob = 0
if walkers.get(key) == None:
walkers_prob[key] = 0
else:
walkers_prob[key] = walkers.get(key) / shots
extras_count = 0
for key in extras.keys():
if extras.get(key) == None:
extras_count += 0
else:
extras_count += extras.get(key)
error_prob = extras_count / shots
if verbose:
print(f'Number of {n_walkers} walkers states: {len(walkers)}')
totes_walkers_prob = 0
for key in walkers_prob.keys():
totes_walkers_prob += walkers_prob.get(key)
if verbose:
print(f'Probability of {n_walkers} walkers: {totes_walkers_prob}')
print(f'Probability of errors: {error_prob}')
print(f'(Sanity check) Total Probability: {totes_walkers_prob + error_prob}')
final_states = {}
for key in walkers_prob.keys():
if walkers_prob.get(key) != 0:
final_states[key] = walkers_prob.get(key)
if verbose:
print(f'Number of {n_walkers} walkers states: {len(walkers)}')
print(f'Number of {n_walkers} walkers non-zero probability states: {len(final_states)}')
final_states['Error'] = error_prob
sites_list = []
sites_prob = []
active_sites = []
for key in final_states.keys():
sites_list.append(key)
ordered = sorted(sites_list)
if verbose:
print(f'Ordered sites list: {ordered}')
print('='*149)
for i in ordered:
sites_prob.append(final_states.get(i))
site = '1'
#print(len(sites_list))
active_sites = []
for i in range(len(sites_list)):
#print(np.abs(sites_list[i].rfind(site)-22))
active_sites.append(np.abs(sites_list[i].rfind(site)-length_qubits))
if verbose:
print(f'Ordered sites probabilities: {sites_prob}')
return sites_prob, active_sites
def find_likelyhood_strings(array):
possibilities = [0, "may have gone ", "likely went "]
#this can be changed
list_of_likelyhood = []
for ua in array:
if 0 <= ua < 0.25:
ind = 0
elif 0.25 <= ua < 0.75:
ind = 1
elif 0.75 <= ua < 1.0:
ind = 2
#append to list
list_of_likelyhood.append(possibilities[ind])
return list_of_likelyhood
def wording_entropy(entropy_val, max_val):
possibilities = ["uneventful", "boring", "regular", "exciting", "chaotic"]
normed_entropy = entropy_val/max_val
if 0 <= normed_entropy < 0.1:
ind = 0
elif 0.1 <= normed_entropy < 0.4:
ind = 1
elif 0.4 <= normed_entropy < 0.6:
ind = 2
elif 0.6 <= normed_entropy < 0.9:
ind = 3
elif 0.9 <= normed_entropy < 1.0:
ind = 4
return possibilities[ind]
def gpt_prompt_and_eval(input_places, input_probs, tags, entropy_specifier, initial_state):
#entropy_specifier = "chaotic"
#initial_state = "gutter"
prompt_init = "Mr. Quanta cannot remember how he got here. Tell the story of him trying to remember how he got here in 3 steps and be descriptive. "
prompt_init += "He only remembers a few things, and considers each possible place one at a time. Use grandiose language. Embelish everything and paint a picture with words. Make the descriptions drip with imagery. "
prompt_init += "He had a " + entropy_specifier + " time before awakening at the " + initial_state + ", and before that"
# Probabilities array as text
#input_places = ["bar", "zoo"]
#input_strings = ["unlikely", "likely"]
lvals = len(input_probs)
for i in range(lvals):
if input_probs[i] != 0:
prompt_init += ", he " + input_probs[i] + " went to the " + input_places[tags[i]]
prompt_init += "."
response = openai.Completion.create(model="text-davinci-003", prompt=prompt_init, temperature=0, max_tokens=400)
f = open("./haze-frontend/public/response.txt", "a")
f.write(response.choices[0].text)
f.close()
return response.choices[0].text
def image_from_response(storyline):
#split the text
prepro = storyline.split('\n')
#print(prepro)
processed = []
for i in range(len(prepro)):
if len(prepro[i]) > 20:
processed.append(prepro[i])
pre_prompt = "Digital art in the style of retrowave."
img_urls = []
for step in range(len(processed)):
#print(processed[step])
img = openai.Image.create(prompt=pre_prompt + processed[step], n=1,size="1024x1024")
img_url = img["data"][0]["url"]
print(img_url)
img_urls.append(img_url)
#print("Image " + step + " url:" + img_url)
img_data = requests.get(img_url).content
with open('./haze-frontend/public/pic'+str(step)+'.png', 'wb') as handler:
handler.write(img_data)
time.sleep(5)
#@app.route('/api', methods = ['POST'])
#@app.route('/')
def full_circ_instance(verbose, sim_type, N_qubits, start, steps, activate_ai):
# provider
provider = IonQProvider("tQgNZln2nI3JSOg7hZhRXjSJHYfgrS2S")
if sim_type == 'ideal':
provider.backends()
backend = provider.get_backend("ionq_simulator")
elif sim_type == 'noisy':
pass
elif sim_type == 'hardware':
pass
#open json file.
#places - a list of strings
#probs - a list of strings (a number)
#start - a number (the index)
#data = request.json
#list_places = data.get['places']
#list_probs = data.get['prob']
all_places = [
"A rooftop bar",
"A comedy club",
"A concert venue",
"A music festival",
"A street fair",
"A bowling alley",
"A casino",
"A sports stadium",
"A karaoke bar",
"A restaurant with live music",
"A rooftop terrace",
"A beach bonfire",
"A drive-in movie theater",
"A laser tag arena",
"A trampoline park",
"An escape room",
"A miniature golf course",
"A rock climbing gym",
"A go-kart track",
"A bowling alley",
"A comedy club",
"A concert hall",
"A music festival",
"A street festival",
"A rooftop pool",
"A rooftop garden",
"A lounge",
"A dance club",
"A public square",
"A rooftop yoga class"]
list_places = np.random.choice(all_places, N_qubits, replace = False)
#list_probs = np.random.rand(N_qubits)
list_probs = np.zeros(N_qubits)
list_probs[np.random.randint(N_qubits)] = np.pi/4
K_max = np.pi/4
K_vals = np.zeros(len(list_probs))
#parse into numerical values the probs array
for j in range(len(list_probs)):
K_vals[j] = K_max*int(list_probs[j])/100.0
#start = 4
#start = data.get['start']
#steps = 3
#steps = data.get['steps']
J_val = np.pi/4 #set a default speed value
circuit = CircuitSpec(start, steps, J_val, K_vals, backend)
final_vals = circuit.random_walk()
if verbose==True:
print("\n")
print("Bitstring results and shot number out of 1024 total shots")
print(final_vals)
print("\n")
#feed final_vals into Rob's cleanup function
processed_vals, active_sites = Clean_Results(final_vals, n_walkers = 1)
#print(processed_vals)
list_of_likelyhood = find_likelyhood_strings(processed_vals)
#print(list_of_likelyhood)
#get entropy from results
entropy_specifier = wording_entropy(Entropy(processed_vals), max_val = np.log(len(list_probs)))
#feed resulting array into Gavin's plotting object
#do not take into account the last value for the plot - that's the Errors!
PolarPlotmaker(processed_vals, labels=list_places, figsize = (5,5), dpi = 120, background = None, debug = False, tick_color = 'chartreuse', linelength = 10, pad = 8, save_name = './haze-frontend/public/roseplot', show = False, has_error = True, offwhite_cutoff=170, labelsize = 8)
#this takes into
#list_places
#processed_vals
#saves picture into haze-frontend/public/roseplot.png
if activate_ai:
#feed into openai function
storyline = gpt_prompt_and_eval(list_places, list_of_likelyhood[:-1], active_sites[:-1], entropy_specifier, list_places[start])
print(storyline)
time.sleep(5)
#clean storyline into 3 separated paragraphs
#feed into Dall-E API
#image_from_response(storyline)
if __name__ == "__main__":
#app.run()
#1 call the API
full_circ_instance(verbose = True, sim_type='ideal', N_qubits=12, start = 3, steps = 2, activate_ai=True)
#do function