10 Validation of the Beam Spot calibration
14 from prompt
import ValidationSettings
20 import scipy.linalg
as la
22 import matplotlib.pyplot
as plt
26 from math
import sqrt, frexp, asin
27 from itertools
import groupby
29 from datetime
import datetime, timedelta
30 from ROOT.Belle2
import Unit
33 settings = ValidationSettings(name=
'BeamSpot Calibrations',
35 download_files=[
'stdout'],
40 def getEigenPars(covM):
41 sxx, syy, szz, sxy, sxz, syz = covM
43 mat = np.array([[sxx, sxy, sxz],
48 eigVals = result[0].real
52 idx = eigVals.argsort()[::-1]
53 eigVals = eigVals[idx]
54 eigVecs = eigVecs[:, idx]
56 assert(eigVals[0] > 1e-4)
57 assert(eigVals[1] > 1e-4)
58 assert(eigVals[2] > 1e-4)
60 sxEig =
sqrt(eigVals[1])
61 syEig =
sqrt(eigVals[2])
62 szEig =
sqrt(eigVals[0])
75 angleXZ = Unit.rad / Unit.mrad * asin(eigVecs[0][0] /
sqrt(1 - eigVecs[1][0]**2))
76 angleYZ = Unit.rad / Unit.mrad * asin(eigVecs[1][0])
77 angleXY = Unit.rad / Unit.mrad * asin(eigVecs[1][1] /
sqrt(1 - eigVecs[1][0]**2))
79 return [sxEig, syEig, szEig, angleXZ, angleYZ, angleXY]
83 def getBSvalues(path):
85 with open(path +
'/database.txt')
as fDB:
91 fN = ll[0].replace(
'/',
'_') +
'_rev_' + ll[1] +
'.root'
94 runDict[fN] = (int(r[0]), int(r[1]))
98 fList = glob(path +
'/*.root')
100 bName = os.path.basename(fName)
101 runExp = runDict[bName]
103 f = ROOT.TFile.Open(fName)
104 bsAll = f.Get(
"BeamSpot")
105 assert(bsAll.ClassName() ==
"Belle2::EventDependency")
107 evNums = bsAll.getEventNumbers()
109 for i
in range(len(evNums) + 1):
110 bs = bsAll.getObjectByIndex(i)
111 ipR = bs.getIPPosition()
112 ip = [ipR(i) * Unit.cm / Unit.um
for i
in range(3)]
113 ipeR = bs.getIPPositionCovMatrix()
114 ipe = [
sqrt(ipeR(i, i)) * Unit.cm / Unit.um
for i
in range(3)]
115 covR = bs.getSizeCovMatrix()
116 covM = (covR(0, 0), covR(1, 1), covR(2, 2), covR(0, 1), covR(0, 2), covR(1, 2))
117 covM = [x * (Unit.cm / Unit.um)**2
for x
in covM]
119 tStart = ipeR(0, 1) * 1e20
120 tEnd = ipeR(0, 2) * 1e20
121 tStart = int(frexp(ipeR(0, 1))[0] * 2**53) % 2**32 / 3600.
122 tEnd = int(frexp(ipeR(0, 2))[0] * 2**53) % 2**32 / 3600.
124 eigM = getEigenPars(covM)
125 arr.append((runExp, tStart, tEnd, ip, ipe, covM, eigM))
128 arr = sorted(arr, key=
lambda x: x[1])
135 def printToFile(arr):
137 with open(
'bsData.csv',
'w')
as outFile:
138 outFile.write(
'exp run tStart tEnd xIP yIP zIP xeIP yeIP zeIP sXX sYY sZZ sXY sXZ '
139 +
'sYZ xSizeEig ySizeEig zSizeEig xzAngle yzAngle xyAngle\n')
175 def plotVar(arr, limits, vName, getterV, getterE=None):
184 for i, el
in enumerate(arr):
186 s = datetime.utcfromtimestamp((s + 9) * 3600)
187 e = datetime.utcfromtimestamp((e + 9) * 3600)
190 Vals.append(getterV(el))
191 Vals.append(getterV(el))
193 if getterE
is not None:
194 Errs.append(getterE(el))
195 Errs.append(getterE(el))
198 if i >= len(arr) - 1:
201 dt = (arr[i + 1][1] - arr[i][2]) * 3600
208 gS = datetime.utcfromtimestamp((arr[i][2] + 9) * 3600)
209 gE = datetime.utcfromtimestamp((arr[i + 1][1] + 9) * 3600)
211 tVals.append(gS + (gE - gS) / 2)
213 if getterE
is not None:
217 tGapVals.append(gS - timedelta(seconds=1))
220 tGapVals.append(gE + timedelta(seconds=1))
222 GapVals.append(np.nan)
223 GapVals.append(getterV(arr[i]))
224 GapVals.append(getterV(arr[i + 1]))
225 GapVals.append(np.nan)
227 plt.plot(tVals, Vals, linewidth=2, color=
'C0')
228 plt.plot(tGapVals, GapVals, linewidth=2, color=
'C0', alpha=0.35)
230 Vals = np.array(Vals)
231 Errs = np.array(Errs)
233 if getterE
is not None:
234 plt.fill_between(tVals, Vals - Errs, Vals + Errs, alpha=0.2)
238 plt.ylabel(vName +
' [mrad]')
240 plt.ylabel(vName +
' [um]')
242 loc =
'plots/allData'
243 if limits
is not None:
244 plt.xlim(datetime.strptime(limits[0],
'%Y-%m-%d'), datetime.strptime(limits[1],
'%Y-%m-%d'))
245 loc =
'plots/' + limits[0] +
'to' + limits[1]
248 os.makedirs(loc, exist_ok=
True)
250 plt.savefig(loc +
'/' + vName +
'.png')
254 def plotPullSpectrum(arr, vName, getterV, getterE):
256 valsOrg = np.array([getterV(v)
for v
in arr])
257 errsOrg = np.array([getterE(v)
for v
in arr])
260 indx = np.array([list(g)[0][0]
for k, g
in groupby(zip(np.arange(len(valsOrg)), valsOrg),
lambda x: x[1])])
264 diffs = (vals[1:] - vals[:-1]) / np.hypot(errs[1:], errs[:-1])
267 n1, n2 = scipy.stats.norm.cdf([-1, 1])
269 q1 = np.quantile(diffs, n1)
270 q2 = np.quantile(diffs, n2)
272 plt.hist(diffs, bins=np.linspace(-20, 20, 80), label=f
'q1={round(q1,1)}, q2={round(q2,1)}')
276 loc =
'plots/allData'
277 plt.savefig(loc +
'/' + vName +
'.png')
281 def run_validation(job_path, input_data_path, requested_iov, expert_config):
287 if expert_config !=
'':
288 expert_config = json.loads(expert_config)
291 if expert_config
is not None and 'plotsRanges' in expert_config:
292 allLimits += expert_config[
'plotsRanges']
295 dbFile = glob(f
'{job_path}/**/database.txt', recursive=
True)
296 dbFile = [db
for db
in dbFile
if 'algorithm_output' not in db]
297 assert(len(dbFile) == 1)
299 inputDir = dbFile[:dbFile.rfind(
'/')]
301 arr = getBSvalues(inputDir)
306 plt.figure(figsize=(18, 9))
309 for limits
in allLimits:
310 plotVar(arr, limits,
'xIP',
lambda e: e[3][0],
lambda e: e[4][0])
311 plotVar(arr, limits,
'yIP',
lambda e: e[3][1],
lambda e: e[4][1])
312 plotVar(arr, limits,
'zIP',
lambda e: e[3][2],
lambda e: e[4][2])
314 plotVar(arr, limits,
'xSpotSize',
lambda e:
sqrt(e[5][0]))
315 plotVar(arr, limits,
'ySpotSize',
lambda e:
sqrt(e[5][1]))
316 plotVar(arr, limits,
'zSpotSize',
lambda e:
sqrt(e[5][2]))
318 plotVar(arr, limits,
'xSpotSizeEig',
lambda e: e[6][0])
319 plotVar(arr, limits,
'ySpotSizeEig',
lambda e: e[6][1])
320 plotVar(arr, limits,
'zSpotSizeEig',
lambda e: e[6][2])
322 plotVar(arr, limits,
'xzSpotAngle',
lambda e: e[6][3])
323 plotVar(arr, limits,
'yzSpotAngle',
lambda e: e[6][4])
324 plotVar(arr, limits,
'xySpotAngle',
lambda e: e[6][5])
326 plotPullSpectrum(arr,
'xIPpulls',
lambda e: e[3][0],
lambda e: e[4][0])
327 plotPullSpectrum(arr,
'yIPpulls',
lambda e: e[3][1],
lambda e: e[4][1])
328 plotPullSpectrum(arr,
'zIPpulls',
lambda e: e[3][2],
lambda e: e[4][2])
331 if __name__ ==
"__main__":
double sqrt(double a)
sqrt for double