-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathBasis.py
340 lines (283 loc) · 13.4 KB
/
Basis.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
#Class to create basis function objects
import numpy as np
from functools import lru_cache
from scipy.special import factorial2 as fac2
from scipy.special import gamma,gammainc
import typing
#Static typing variables
Ang_mom=typing.Tuple[int,int,int]
Dist=typing.Tuple[float,float,float]
Exp_2e=typing.Tuple[float,float,float,float]
Dist_2e=typing.Tuple[Dist,Dist,Dist]
Ang_2e=typing.Tuple[Ang_mom,Ang_mom,Ang_mom,Ang_mom]
class Basis_Function:
def __init__(self,center,coeffs,exponents,shell):
self.center=np.array(center)
self.coeffs=np.array(coeffs)
self.exps=np.array(exponents)
self.shell=shell
self.norms=None
self.N=None
self.normalize()
def normalize(self):
"""Set norms of the primitives and N of the contracted Gaussian"""
order=np.sum(self.shell)
facts=np.prod(fac2([2*s-1 for s in self.shell]))
self.norms=(((2/np.pi)**(3/4))*(2**order)*(facts**(-1/2))*
self.exps**((2*order+3)/4))
pre=(np.pi**1.5)*facts*(2.0**-order)
divisor=np.add.outer(self.exps,self.exps)**-(order+1.5)
normalized=self.norms*self.coeffs
summand=(pre*np.einsum('i,j,ij->',normalized,normalized,divisor))**-.5
self.N=summand
def overlap(self,other,deriv=(0,0,0),multi=(False,False,False)):
"""Determine overlap between contracted Gaussians via Obara-Saika recurrence.
Can do standard overlap, as well as derivatives (specify deriv)
and multipoles (specify deriv and set multi True for the desired components)
"""
distance=self.center-other.center
value=0
#Loop over each contraction
for na,ca,ea in zip(self.norms,self.coeffs,self.exps):
for nb,cb,eb in zip(other.norms,other.coeffs,other.exps):
temp=1
P=(ea*self.center+eb*other.center)/(ea+eb)
x_c=[P[i] if multi[i] else None for i in range(3)]
#Determine x/y/z overlaps based on the shell
for vals in zip(distance,self.shell,other.shell,deriv,x_c):
temp*=ObSa_1e(ea, eb, *vals)
value+=na*ca*nb*cb*temp
return self.N*other.N*value
def Coulomb_1e(self,other,nuc_center):
value = 0
nuc_center=np.array(nuc_center)
# Loop over each contraction
for na, ca, ea in zip(self.norms, self.coeffs, self.exps):
for nb, cb, eb in zip(other.norms, other.coeffs, other.exps):
P = (ea*self.center+eb*other.center)/(ea + eb)
##Distances
PA=tuple(P-self.center)
PB=tuple(P-other.center)
PC=tuple(P-nuc_center)
##Exponent factors
p=ea+eb
mu=ea*eb/p
temp=ObSa_Coulomb(p,mu,PA,PB,PC,self.shell,other.shell)
value+=na*ca*nb*cb*temp
return self.N*other.N*value
#Integral Recursions
@lru_cache()
def ObSa_1e(alpha:float, beta:float, x:float, i=0, j=0, t=0, x_c=None)->float:
"""Obara Saika recurrence for one electron overlap.
Can generate higher angular momentum from the base
case of s-type overlap.
:param alpha: bra exponent
:param x: Distance between bra and ket
:param i: Angular momentum
:param t: Derivative/multipole order
:param x_c: Center of charge (for multipole)
:return: value of overlap integral
"""
p=alpha+beta
mu=(alpha*beta)/p
#Base cases
if i==0 and j==0 and t==0:
return np.sqrt(np.pi/p)*np.exp(-mu*x**2)
elif i<0 or j<0 or t<0:
return 0.0
elif t>0:
if x_c is None:
upper= 2 * alpha * ObSa_1e(alpha, beta, x, i + 1, j, t-1)
lower= -i * ObSa_1e(alpha, beta, x, i-1, j, t-1)
else:
upper= x_c*ObSa_1e(alpha, beta, x, i, j, t-1, x_c)
lower=(i * ObSa_1e(alpha, beta, x, i-1, j, t-1, x_c) +
j * ObSa_1e(alpha, beta, x, i, j-1, t-1, x_c) +
(t-1) * ObSa_1e(alpha, beta, x, i, j, t-2, x_c))/(2*p)
return upper+lower
elif i>0:
upper= (-beta/p) * x * ObSa_1e(alpha, beta, x, i - 1, j, t, x_c)
lower= ((i-1) * ObSa_1e(alpha, beta, x, i - 2, j, t, x_c) +
j * ObSa_1e(alpha, beta, x, i - 1, j - 1, t, x_c))/(2*p)
if x_c is None:
lower+= -2*beta*t*ObSa_1e(alpha, beta, x, i-1, j, t-1)/(2*p)
else:
lower+= t*ObSa_1e(alpha, beta, x, i-1, j, t-1, x_c)/(2*p)
return upper+lower
elif j>0:
upper= (alpha/p) * x * ObSa_1e(alpha, beta, x, i, j-1, t, x_c)
lower= (i * ObSa_1e(alpha, beta, x, i - 1, j - 1, t, x_c) +
(j-1) * ObSa_1e(alpha, beta, x, i, j - 2, t, x_c))/(2*p)
if x_c is None:
lower+= 2*alpha*t*ObSa_1e(alpha, beta, x, i, j-1, t-1)/(2*p)
else:
lower+= t*ObSa_1e(alpha, beta, x, i, j-1, t-1, x_c)/(2*p)
return upper+lower
@lru_cache()
def ObSa_Coulomb(p: int, mu: int, bra_dist: Dist, ket_dist: Dist, nuc_dist: Dist,
bra_ang: Ang_mom = (0,0,0), ket_ang: Ang_mom = (0,0,0), N: int = 0)->float:
"""Evaluate one-electron Coulomb attraction
Uses the Obara-Saika recursion relations to convert angular momentum
to higher orders of the Boys function.
:param p: Sum of bra-ket exponents
:param mu: Product of bra-ket exponents, divided by p
:param bra_dist: Distance from bra to combined Gaussian center
:param bra_ang: Angular momentum of bra
:param N: Boys' function order
"""
##Accumulate as angular momentum is converted to Boy's order N
value=0
if bra_ang==ket_ang==(0,0,0):
pre=(2*np.pi/p)
K=np.prod(np.exp(-mu*(np.array(bra_dist)-np.array(ket_dist))**2))
distance=np.einsum('i,i->',nuc_dist,nuc_dist)
boy=Boys(N,p*distance)
value+= pre*K*boy
elif any(b<0 for b in bra_ang) or any(k<0 for k in ket_ang):
value+=0
else:
##Loop over x/y/z, reducing the angular momentum
for x in range(3):
##Reduce bra momentum first, then ket
args = [p, mu, bra_dist, ket_dist, nuc_dist]
if bra_ang[x]>0:
bra_temp=tuple(b-1 if x==t else b for t,b in enumerate(bra_ang))
value+=(bra_dist[x]*ObSa_Coulomb(*args,bra_temp,ket_ang,N)
-nuc_dist[x]*ObSa_Coulomb(*args,bra_temp,ket_ang,N+1))
ket_temp=tuple(b-1 if x==t else b for t,b in enumerate(ket_ang))
value+=(ket_ang[x]) * (ObSa_Coulomb(*args,bra_temp,ket_temp,N)
-ObSa_Coulomb(*args,bra_temp,ket_temp,N+1))/(2*p)
bra_temp = tuple(b - 1 if x == t else b for t, b in enumerate(bra_temp))
value+=(bra_ang[x]-1)*(ObSa_Coulomb(*args,bra_temp,ket_ang,N)
-ObSa_Coulomb(*args,bra_temp,ket_ang,N+1))/(2*p)
break
elif ket_ang[x]>0:
ket_temp = tuple(b - 1 if x == t else b for t, b in enumerate(ket_ang))
value+=(ket_dist[x]*ObSa_Coulomb(*args,bra_ang,ket_temp,N)
-nuc_dist[x]*ObSa_Coulomb(*args,bra_ang,ket_temp,N+1))
bra_temp = tuple(b - 1 if x == t else b for t, b in enumerate(bra_ang))
value+=(bra_ang[x]) * (ObSa_Coulomb(*args,bra_temp,ket_temp,N)
-ObSa_Coulomb(*args,bra_temp,ket_temp,N+1))/(2*p)
ket_temp = tuple(b - 1 if x == t else b for t, b in enumerate(ket_temp))
value+=(ket_ang[x]-1)*(ObSa_Coulomb(*args,bra_ang,ket_temp,N)
-ObSa_Coulomb(*args,bra_ang,ket_temp,N+1))/(2*p)
break
return value
@lru_cache(maxsize=4096)
def ObSa_2e(expon: Exp_2e,dists: Dist_2e,momentums: Ang_2e,N=0)->float:
"""Evaluates two electron integrals using simplified Obara-Saika
Recursion relations adapted from Chapter 12, Eqs. 238-241 of Modern
Electronic Structure Theory.
:param expon: Exponent of each orbital, left to right, chemist notation
:param dists: Should contain: distance from bra center to 1st orbital [0],
distance from ket center to 3rd orbital [1], and distance between
the bra and ket centers [2].
:param momentums: Momentum of each orbital, left to right, chemist notation
:param N: Boys' function order
:return: Value of two electron integral
"""
value=0
##Unpack exponents
a, b, c, d = expon
p = a + b
q = c + d
#Base case
if all(a==(0,0,0) for a in momentums):
##Form necessary exponent factors
mu=a*b/p
nu=c*d/q
omega=p*q/(p+q)
##Construct base case
pre=(2*np.pi**2.5)/(p*q*(p+q)**.5)
K_ab=np.prod(np.exp(-mu*((p/b)*np.array(dists[0]))**2))
K_cd=np.prod(np.exp(-nu*((q/d)*np.array(dists[1]))**2))
R_sq=np.einsum('i,i->',dists[2],dists[2])
boy=Boys(N,omega*R_sq)
value+= pre*K_ab*K_cd*boy
elif any(a<0 for ang in momentums for a in ang):
value+=0
else:
#Loop over x/y/z, reducing the angular momentum
orb_i, orb_j, orb_k, orb_l = momentums
for x in range(3):
#Transfer momentum from orbital j/l to orbital i/k
if orb_j[x]>0:
minus_j=tuple(b-1 if x == t else b for t, b in enumerate(orb_j))
plus_i=tuple(b+1 if x == t else b for t, b in enumerate(orb_i))
transfer=(plus_i,minus_j,orb_k,orb_l)
reduce=(orb_i,minus_j,orb_k,orb_l)
value+=ObSa_2e(expon,dists,transfer,N=N)
value+=-(p/b)*dists[0][x]*ObSa_2e(expon,dists,reduce,N=N)
break
elif orb_l[x]>0:
minus_l=tuple(b-1 if x == t else b for t, b in enumerate(orb_l))
plus_k=tuple(b+1 if x == t else b for t, b in enumerate(orb_k))
transfer=(orb_i,orb_j,plus_k,minus_l)
reduce=(orb_i,orb_j,orb_k,minus_l)
value+=ObSa_2e(expon, dists, transfer,N=N)
value+=-(q/d)*dists[1][x]*ObSa_2e(expon, dists, reduce,N=N)
break
#Tranfer momentum from elec 2 to elec 1 (i.e. from orb_k to orb_i)
elif orb_k[x]>0:
minus_k = tuple(b-1 if x == t else b for t, b in enumerate(orb_k))
min2_k = tuple(b-2 if x == t else b for t, b in enumerate(orb_k))
plus_i = tuple(b+1 if x == t else b for t, b in enumerate(orb_i))
minus_i = tuple(b-1 if x == t else b for t, b in enumerate(orb_i))
transfer=(plus_i,orb_j,minus_k,orb_l)
reduce=(orb_i,orb_j,minus_k,orb_l)
double_reduce=(orb_i,orb_j,min2_k,orb_l)
both_reduce=(minus_i,orb_j,minus_k,orb_l)
value+=(p*dists[0][x]+q*dists[1][x])*ObSa_2e(expon,dists,reduce,N=N)
value+=(orb_i[x]/2)*ObSa_2e(expon,dists,both_reduce,N=N)
value+=(minus_k[x]/2)*ObSa_2e(expon,dists,double_reduce,N=N)
value+=-p*ObSa_2e(expon,dists,transfer,N=N)
value/=q
break
#Convert remaining angular momentum into higher order Boys functions
elif orb_i[x]>0:
omega=p*q/(p+q)
minus_i = tuple(b - 1 if x == t else b for t, b in enumerate(orb_i))
min2_i = tuple(b - 2 if x == t else b for t, b in enumerate(orb_i))
reduce=(minus_i,orb_j,orb_l,orb_k)
double_reduce=(min2_i,orb_j,orb_l,orb_k)
value+=dists[0][x]*ObSa_2e(expon,dists,reduce,N=N)
value+=-(omega*dists[2][x]/p)*ObSa_2e(expon,dists,reduce,N=N+1)
value+=(minus_i[x]/(2*p))*ObSa_2e(expon,dists,double_reduce,N=N)
value+=-(minus_i[x]*omega/(2*p**2))*ObSa_2e(expon,dists,double_reduce,N=N+1)
break
return value
def Boys(n:int,x:float)->float:
"""Evaluates the Boys' functions
Uses the incomplete Gamma function to evaluate.
:param n: order
:param x: position
"""
if abs(x)<1e-15:
Fn=1/(2*n+1)
else:
Fn=.5*gamma(n+.5)*gammainc(n+.5,x)*(x**-(n+.5))
return Fn
class ChargeDist:
def __init__(self,first: Basis_Function,second: Basis_Function,indices: typing.Tuple[int,int]):
self.first=first
self.second=second
self.indices=indices
def interact(self,other:"ChargeDist"):
value = 0
# Loop over each contraction
for na, ca, ea in zip(self.first.norms, self.first.coeffs, self.first.exps):
for nb, cb, eb in zip(self.second.norms, self.second.coeffs, self.second.exps):
P = (ea * self.first.center + eb * self.second.center)/(ea + eb)
PA= tuple(P-self.first.center)
for nc, cc, ec in zip(other.first.norms, other.first.coeffs, other.first.exps):
for nd, cd, ed in zip(other.second.norms, other.second.coeffs, other.second.exps):
Q = (ec*other.first.center+ed*other.second.center)/(ec + ed)
QC= tuple(Q-other.first.center)
PQ= tuple(P-Q)
exponents=(ea,eb,ec,ed)
distances=(PA,QC,PQ)
momentum=(self.first.shell,self.second.shell,
other.first.shell,other.second.shell)
value+=(na*nb*nc*nd)*(ca*cb*cc*cd)*ObSa_2e(exponents,distances,momentum)
#print(ObSa_2e.cache_info())
return (self.first.N*self.second.N)*(other.first.N*other.second.N)*value