Skip to main content

Structure factor and X-ray scattering from radial distribution functions

Project description

Structure factor and scattering from radial distribution functions

A package to calculate structure factors and x-ray (solution) scattering signals from Radial Distribution Functions (RDFs), sampled from Molecular Dynamics (MD) simulations.

Also includes 3 different finite-size RDF-corrections, as well as a set of Fourier truncation window functions.

Please see (and cite) this work for the necessary background. There is also a pre-print freely available here.

Please read the docs for tutorials on how to use the tools included.

If you want even more details and inspiration, the the example notebook contains more usage-examples, and the this notebook contains all the scripts used to make the plots in the paper.

Example: X-ray Scattering of Water:

from grsq.grsq import rdfset_from_dir

V = 122900.85207774633  # volume of MD box
stoich = {'H_v': 8190, 'O_v': 4095}  # number of atoms in MD box
# Helper function to create RDFSet
rdfs = rdfset_from_dir('tests/data/xray_water/', 
                       volume=V, stoich=stoich)

qvec = np.arange(0, 10, 0.05)  # create q vector
ig_u = rdfs.get_iq(qvec)  # Calculate scattering
rdfs.vdv_correct()  # apply finite-size corrections
ig_c = rdfs.get_iq(qvec)  # Calculate scattering again

fig, ax = plt.subplots(1, 1, figsize=(9, 5)) 
ax.plot(qvec, ig_u, label='Raw')
ax.plot(qvec, ig_c, label='Corrected')
ax.set_xlim([0, 10])
ax.set_xlabel('Q (Å$^{-1})$')
ax.set_ylabel('I(Q)')
ax.legend(loc='best');
fig.tight_layout()

Waterplot{width=85%}

Installation

The package is available from PyPi:

pip install grsq

Or clone this repository and add it to your $PYTHONPATH.

Introduction

For each pairwise RDF you have sampled from your system, you should create an RDF object, in this example between the Oxygens and Hydrogens of the solvent-region:

dat = np.genfromtxt('some_data_file.dat')  # has (r,g(r)) as columns in this example
rdf = RDF(dat[:, 0], dat[:, 1],   # r and g(r)
          'O', 'H',   # atom types
          'solvent', 'solvent',  # atom type regions
          n1=4095, n2=8190,  # number of atoms of each type
          volume=122900.85,  # volume in Å of simulation cell
          qvec=qvec)

There are 2 possible regions for your atom types to exist in: 'solute' and 'solvent'.

From a single RDF object, you can calculate the structure factor:

rdf.structure_factor()

as well as the first (atomic), and 2nd term of the x-ray scattering signal:

rdf.term1() + rdf.term2()

You most likely have a large amount of pairwise RDFs from a system of a solute comprised of various elements in a solvent. To make things simple, we can put them all into an RDFSet instance. How to do so most easily depends on how you store your RDF data, naming, data format, etc...

If you follow the naming convention in the documentation of the helper function rdfset_from_dir, this function can help you do so, as in the water example above.

The RDFSet Class

An OrderedDict of RDF objects, which you have to create yourself, you might e.g. already have created a list_of_rdf_objects:

from grsq.grsq import RDFSet
rdfs = RDFSet()
for rdf in list_of_rdf_objects:
    rdfs.add_rdf(rdf)

Or have a look at:

from grsq.grsq import rdfset_from_dir
?rdfset_from_dir

Having an RDFSet is very convenient, as you get access to a bunch of methods:

  • rdfs.show(): Returns information of the set
  • rdfs.get_iq(qvec=None, cross=False, damping=None) Get the total coherent x-ray scattering signal from entire set. You can specify qvec and Damping to overwrite whatever is already contained in each RDF object.
  • rdfs.get_solute(): Returns the scattering of only the solute-solute atom type RDFs
  • rdfs.get_solvent(): Returns the scattering of only the solvent-solvent atom type RDFs
  • rdfs.get_cross(): Returns the scattering of only the solute-solvent and solvent-solute atom type RDFs
  • rdfs.get_dv(): Returns the excluded volume-scattering

All methods take qvec and damping as arguments. To use fourier damping functions, parse a damping object to the method:

from grsq.damping import Damping
damping = Damping('lorch', L=r_max)
rdfs.get_solvent(qvec=qvec, damping=damping)

Finite Size corrections

The following finite size corrections are available (see this work for details):

  • rdf.volume_correct(Rl): Apply the volume correction to the RDF, using a spherical $V_l$ volume of radius Rl: $$ g_{lm}^\mathrm{\infty}(r) = \frac{\rho_m}{\rho_\mathrm{eff}} g_{lm}^N(r) = g_{lm}^N(r)\rho_m \frac{V_\mathrm{cell} - V_l}{N_m - \delta_{lm}}, $$

  • rdf.perera_correct(Rl, r_avg): Apply the correction from Perera et al. $$ g^\mathrm{\infty}{lm}(r) = g^N{lm}(r) \left [ 1 + \frac{1 - g^{N,0}}{2} \left( 1 + \tanh \left( \frac{r - \kappa_{lm}}{\alpha_{lm}} \right) \right) \right ]. $$ Where $g^{N,0}$ is obtained by setting r_avg to get the average value of g(r > r_avg), and Rl = $\kappa_{lm} / 2$, to be consistent with the volume correction, as well as the recommendations from the original paper.

  • rdf.vdv_correct(): Apply the Ganguly / van der Vegt correction: $$ g^\mathrm{\infty}\mathrm{lm}(r) = g^N\mathrm{lm}(r) \frac{N_m}{N_m - \left[ (\Delta N(r) + \delta_{lm}) \left( 1 - \frac{(4/3) \pi r^3}{V_\mathrm{cell}} \right)^{-1} \right]} $$

Support methods:
  • rdf.fit(Rl_guess, fit_start=25, fit_stop=50): Fit the excluded volume radius.
    Can be used together with rdf.volume_correct(Rl) to estimate Rl. It minimizes g(r) - 1 from fit_start to fit_stop

Try ?RDF or help(RDF) for more info.

The RDFSet also has the method rdfs.vdv_correct() that apply the van der Vegt correction to all relevant RDFs automatically.

Copyright

Copyright (c) 2022, Asmus Ougaard Dohn

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

grsq-1.1.7.tar.gz (22.3 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

grsq-1.1.7-py3-none-any.whl (17.4 kB view details)

Uploaded Python 3

File details

Details for the file grsq-1.1.7.tar.gz.

File metadata

  • Download URL: grsq-1.1.7.tar.gz
  • Upload date:
  • Size: 22.3 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.16

File hashes

Hashes for grsq-1.1.7.tar.gz
Algorithm Hash digest
SHA256 7d82aca7581ca0bf11618a47c20cb6aac3a16872e03f723129c848e6fb057ca0
MD5 800c4ad9d9709025adaeb8d18b40da79
BLAKE2b-256 717a0defa5abb3d305371f67b5816e68e4006e7983c9568d45824c8cf4c2d2de

See more details on using hashes here.

File details

Details for the file grsq-1.1.7-py3-none-any.whl.

File metadata

  • Download URL: grsq-1.1.7-py3-none-any.whl
  • Upload date:
  • Size: 17.4 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? No
  • Uploaded via: twine/4.0.2 CPython/3.9.16

File hashes

Hashes for grsq-1.1.7-py3-none-any.whl
Algorithm Hash digest
SHA256 77d6963b99f89636fd1b1024afb825fcb2570e18dd194b82219b25703f6d8f2e
MD5 71d7a720d337a49b571fa4b4b854d9dd
BLAKE2b-256 d2da03772f73abb0cf8557faad39ce330c18c5ba6a85abbd53448c92a6d55633

See more details on using hashes here.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page