Skip to main content

Symmetry projectors and modulation-loss (Lambda) estimators for screening molecular crystals for directional dark matter detection.

Project description

symmscreen

Symmetry projectors and modulation-loss (Lambda) estimators for screening molecular crystals for directional dark matter detection.

Companion package to "The Role of Symmetries in Dark Matter Detector Design" (arXiv:..., Benjamin Lillard, Jack D. Shergold, and Juri Smirnov). The paper develops a symmetry-projection framework that predicts how much of a molecule's directional dark matter scattering signal survives crystallisation, using only the crystal's space group and the molecule's point-group symmetry and geometry. This package implements exactly that framework (paper Secs. 3-6): the crystal and molecule point-group projectors, the combined survival operator, and the three Lambda estimators for the fractional modulation-signal loss.

This package is intentionally lightweight, and does not compute electronic structure, form factors, scattering rates, or true modulation amplitudes. See SCarFFF and vsdm for software implementing these calculations.

Installation

pip install symmscreen

Quickstart

The fastest way to get the modulation loss estimate, $\Lambda_\mathrm{coord}$, for a molecular crystal with known coordinates is:

import symmscreen as ss

ss.lambda_coord("your_crystal.cif")

There's a matching one-line function for the quadrupole class of the crystal, and of the molecule itself, too: ss.crystal_quadrupole_class(cif_path), ss.molecule_quadrupole_class(cif_path).

If you want more than one metric for the same crystal, or want to inspect the underlying projector matrices, build a CombinedSurvival object directly instead. This caches the crystal/molecule projectors internally, so nothing is recomputed between calls. Since a CIF gives you the molecule's actual orientation, $\mathcal{T}$, the natural estimators here are the orientation-specific ones, lambda_L and lambda_coord:

from symmscreen import CombinedSurvival

survival = CombinedSurvival.from_cif("your_crystal.cif")

survival.crystal.quadrupole_class      # k in {0, 1, 2, 3, 5}, the quadrupole class of the crystal, Q_k.
survival.molecule.quadrupole_class     # k in {0, 1, 2, 3, 5}, the quadrupole class of the molecule's own symmetry H.
survival.crystal.matrix(l=2)           # Pi_2^(L), the l=2 projector for the crystal symmetry.
survival.molecule.matrix(l=2)          # Pi_2^(H), the l=2 projector for the internal molecule symmetry.
survival.matrix(l=2)                   # C^(2)(T), the combined survival operator.
survival.lambda_L()                    # Lambda^(L)(T), modulation signal loss due to crystallisation alone, for this crystal's specific orientation, T.
survival.lambda_coord()                # Lambda_coord, coordinate-aware estimator of total modulation signal loss.

Or work purely from symmetry labels, with no CIF or coordinates at all. Here the relative orientation, $\mathcal{T}$, is generally unknown, so the natural estimators are the analytic $\mathcal{T}$-averaged quantities, lambda_L_avg and lambda_ideal_avg. These have matching one-line functions that take the same symmetry labels rather than a CIF:

import symmscreen as ss

ss.lambda_L_avg(mol_pg_symbol="D6h", crys_pg_symbol="D2h")
ss.lambda_ideal_avg(mol_pg_symbol="D6h", crys_pg_symbol="D2h")   # ~ sqrt(2)/5, matching the benzene example in the paper.

Or build the combined survival operator and access them from there, as well as the matrices:

from symmscreen import CombinedSurvival

survival = CombinedSurvival.from_symmetry(mol_pg_symbol="D6h", crys_pg_symbol="D2h")
survival.matrix(l=2)          # C^(2), combined survival operator at l=2. Defaults to T=I, with T the relative orientation.
survival.lambda_L_avg()       # <Lambda^(L)>_T, loss due to crystallisation alone, averaged over T.
survival.lambda_ideal_avg()   # <Lambda>_T, ~sqrt(2)/5, matching the benzene example in the paper.

from_symmetry also takes an optional relative rotation T (the paper's $\mathcal{T}$), for when you want a specific embedding rather than the $\mathcal{T}$-averaged estimators above:

from symmscreen import CombinedSurvival
from scipy.spatial.transform import Rotation

T = Rotation.from_euler("z", 90, degrees=True).as_matrix()   # e.g. a 90 degree rotation about z.

survival = CombinedSurvival.from_symmetry(mol_pg_symbol="D6h", crys_pg_symbol="D2h", T=T)
survival.matrix(l=2)  # C^(2)(T), combined survival operator at l=2 for a fixed T.
survival.lambda_L()   # Lambda^(L)(T), orientation-specific crystallistion loss, rather than T-averaged.

$\mathcal{T}$ defaults to the identity if omitted.

Getting a CIF

symmscreen does not contain any example structures, as crystal structure databases are generally not freely redistributable. However, experimentally determined molecular crystal structures are available from:

Citation

If you use this package, please use the following citation:

Bibtex goes here.

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

symmscreen-0.1.0.tar.gz (19.1 kB view details)

Uploaded Source

Built Distribution

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

symmscreen-0.1.0-py3-none-any.whl (17.9 kB view details)

Uploaded Python 3

File details

Details for the file symmscreen-0.1.0.tar.gz.

File metadata

  • Download URL: symmscreen-0.1.0.tar.gz
  • Upload date:
  • Size: 19.1 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for symmscreen-0.1.0.tar.gz
Algorithm Hash digest
SHA256 1f8b0dce5b7be023f449f674938414c84a73e22bb799ccd397ddfa4ecf5a4ab9
MD5 a0cbe836e051ce655ff9a8316bb08a02
BLAKE2b-256 cce93e2fbd129c62c146caffb699db83fa538a6914238cb68f931e794365da6c

See more details on using hashes here.

Provenance

The following attestation bundles were made for symmscreen-0.1.0.tar.gz:

Publisher: publish.yml on jdshergold/symmscreen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file symmscreen-0.1.0-py3-none-any.whl.

File metadata

  • Download URL: symmscreen-0.1.0-py3-none-any.whl
  • Upload date:
  • Size: 17.9 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for symmscreen-0.1.0-py3-none-any.whl
Algorithm Hash digest
SHA256 d1168388b207d05c68c1ae949095bae9b063c622b3dcae1ad1dd30de2205f8bf
MD5 3d51fd4c21713400e1a894ca0072d10b
BLAKE2b-256 f0d3b4275a4b423526b7f67f271a0728750b0db9aaa43a59b1b730333a1e9987

See more details on using hashes here.

Provenance

The following attestation bundles were made for symmscreen-0.1.0-py3-none-any.whl:

Publisher: publish.yml on jdshergold/symmscreen

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

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